mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 10:12:45 +00:00 
			
		
		
		
	 6e19ab2bbc
			
		
	
	
		6e19ab2bbc
		
	
	
	
	
		
			
			We have a new, improved string type coming up in AK (OOM aware, no null state), and while it's going to use UTF-8, the name UTF8String is a mouthful - so let's free up the String name by renaming the existing class. Making the old one have an annoying name will hopefully also help with quick adoption :^)
		
			
				
	
	
		
			202 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			202 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/QuickSort.h>
 | |
| #include <LibCore/ArgsParser.h>
 | |
| #include <LibCore/ProcessStatisticsReader.h>
 | |
| #include <LibCore/System.h>
 | |
| #include <LibMain/Main.h>
 | |
| #include <sys/sysmacros.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| static ErrorOr<DeprecatedString> determine_tty_pseudo_name()
 | |
| {
 | |
|     struct stat tty_stat;
 | |
|     if (fstat(STDIN_FILENO, &tty_stat) < 0) {
 | |
|         int saved_errno = errno;
 | |
|         perror("fstat");
 | |
|         return Error::from_errno(saved_errno);
 | |
|     }
 | |
| 
 | |
|     int tty_device_major = major(tty_stat.st_rdev);
 | |
|     int tty_device_minor = minor(tty_stat.st_rdev);
 | |
| 
 | |
|     if (tty_device_major == 201) {
 | |
|         return DeprecatedString::formatted("pts:{}", tty_device_minor);
 | |
|     }
 | |
| 
 | |
|     if (tty_device_major == 4) {
 | |
|         return DeprecatedString::formatted("tty:{}", tty_device_minor);
 | |
|     }
 | |
|     return "n/a";
 | |
| }
 | |
| 
 | |
| ErrorOr<int> serenity_main(Main::Arguments arguments)
 | |
| {
 | |
|     TRY(Core::System::pledge("stdio rpath tty"));
 | |
| 
 | |
|     auto this_pseudo_tty_name = TRY(determine_tty_pseudo_name());
 | |
| 
 | |
|     TRY(Core::System::pledge("stdio rpath"));
 | |
|     TRY(Core::System::unveil("/sys/kernel/processes", "r"));
 | |
|     TRY(Core::System::unveil("/etc/passwd", "r"));
 | |
|     TRY(Core::System::unveil(nullptr, nullptr));
 | |
| 
 | |
|     enum class Alignment {
 | |
|         Left,
 | |
|         Right,
 | |
|     };
 | |
| 
 | |
|     struct Column {
 | |
|         DeprecatedString title;
 | |
|         Alignment alignment { Alignment::Left };
 | |
|         int width { 0 };
 | |
|         DeprecatedString buffer;
 | |
|     };
 | |
| 
 | |
|     bool every_process_flag = false;
 | |
|     bool full_format_flag = false;
 | |
|     DeprecatedString pid_list;
 | |
| 
 | |
|     Core::ArgsParser args_parser;
 | |
|     args_parser.add_option(every_process_flag, "Show every process", nullptr, 'e');
 | |
|     args_parser.add_option(full_format_flag, "Full format", nullptr, 'f');
 | |
|     args_parser.add_option(pid_list, "A comma-separated list of PIDs. Only processes matching those PIDs will be selected", nullptr, 'q', "pid-list");
 | |
|     args_parser.parse(arguments);
 | |
| 
 | |
|     Vector<Column> columns;
 | |
| 
 | |
|     int uid_column = -1;
 | |
|     int pid_column = -1;
 | |
|     int ppid_column = -1;
 | |
|     int pgid_column = -1;
 | |
|     int sid_column = -1;
 | |
|     int state_column = -1;
 | |
|     int tty_column = -1;
 | |
|     int cmd_column = -1;
 | |
| 
 | |
|     auto add_column = [&](auto title, auto alignment) {
 | |
|         columns.append({ title, alignment, 0, {} });
 | |
|         return columns.size() - 1;
 | |
|     };
 | |
| 
 | |
|     if (full_format_flag) {
 | |
|         uid_column = add_column("UID", Alignment::Left);
 | |
|         pid_column = add_column("PID", Alignment::Right);
 | |
|         ppid_column = add_column("PPID", Alignment::Right);
 | |
|         pgid_column = add_column("PGID", Alignment::Right);
 | |
|         sid_column = add_column("SID", Alignment::Right);
 | |
|         state_column = add_column("STATE", Alignment::Left);
 | |
|         tty_column = add_column("TTY", Alignment::Left);
 | |
|         cmd_column = add_column("CMD", Alignment::Left);
 | |
|     } else {
 | |
|         pid_column = add_column("PID", Alignment::Right);
 | |
|         tty_column = add_column("TTY", Alignment::Left);
 | |
|         cmd_column = add_column("CMD", Alignment::Left);
 | |
|     }
 | |
| 
 | |
|     auto all_processes = Core::ProcessStatisticsReader::get_all();
 | |
|     if (!all_processes.has_value())
 | |
|         return 1;
 | |
| 
 | |
|     auto& processes = all_processes.value().processes;
 | |
| 
 | |
|     if (!pid_list.is_empty()) {
 | |
|         every_process_flag = true;
 | |
|         auto string_parts = pid_list.split_view(',');
 | |
|         Vector<pid_t> selected_pids;
 | |
|         selected_pids.ensure_capacity(string_parts.size());
 | |
| 
 | |
|         for (size_t i = 0; i < string_parts.size(); i++) {
 | |
|             auto pid = string_parts[i].to_int();
 | |
| 
 | |
|             if (!pid.has_value()) {
 | |
|                 warnln("Invalid value for -q: {}", pid_list);
 | |
|                 warnln("Could not parse '{}' as a PID.", string_parts[i]);
 | |
|                 return 1;
 | |
|             }
 | |
| 
 | |
|             selected_pids.append(pid.value());
 | |
|         }
 | |
| 
 | |
|         processes.remove_all_matching([&](auto& a) { return selected_pids.find(a.pid) == selected_pids.end(); });
 | |
| 
 | |
|         auto processes_sort_predicate = [&selected_pids](auto& a, auto& b) {
 | |
|             return selected_pids.find_first_index(a.pid).value() < selected_pids.find_first_index(b.pid).value();
 | |
|         };
 | |
|         quick_sort(processes, processes_sort_predicate);
 | |
|     } else {
 | |
|         quick_sort(processes, [](auto& a, auto& b) { return a.pid < b.pid; });
 | |
|     }
 | |
| 
 | |
|     Vector<Vector<DeprecatedString>> rows;
 | |
|     TRY(rows.try_ensure_capacity(1 + processes.size()));
 | |
| 
 | |
|     Vector<DeprecatedString> header;
 | |
|     TRY(header.try_ensure_capacity(columns.size()));
 | |
|     for (auto& column : columns)
 | |
|         header.unchecked_append(column.title);
 | |
|     rows.append(move(header));
 | |
| 
 | |
|     for (auto const& process : processes) {
 | |
|         auto tty = process.tty;
 | |
|         if (!every_process_flag && tty != this_pseudo_tty_name)
 | |
|             continue;
 | |
| 
 | |
|         auto* state = process.threads.is_empty() ? "Zombie" : process.threads.first().state.characters();
 | |
| 
 | |
|         Vector<DeprecatedString> row;
 | |
|         TRY(row.try_resize(columns.size()));
 | |
| 
 | |
|         if (tty == "")
 | |
|             tty = "n/a";
 | |
| 
 | |
|         if (uid_column != -1)
 | |
|             row[uid_column] = process.username;
 | |
|         if (pid_column != -1)
 | |
|             row[pid_column] = DeprecatedString::number(process.pid);
 | |
|         if (ppid_column != -1)
 | |
|             row[ppid_column] = DeprecatedString::number(process.ppid);
 | |
|         if (pgid_column != -1)
 | |
|             row[pgid_column] = DeprecatedString::number(process.pgid);
 | |
|         if (sid_column != -1)
 | |
|             row[sid_column] = DeprecatedString::number(process.sid);
 | |
|         if (tty_column != -1)
 | |
|             row[tty_column] = tty;
 | |
|         if (state_column != -1)
 | |
|             row[state_column] = state;
 | |
|         if (cmd_column != -1)
 | |
|             row[cmd_column] = process.name;
 | |
| 
 | |
|         TRY(rows.try_append(move(row)));
 | |
|     }
 | |
| 
 | |
|     for (size_t i = 0; i < columns.size(); i++) {
 | |
|         auto& column = columns[i];
 | |
|         for (auto& row : rows)
 | |
|             column.width = max(column.width, static_cast<int>(row[i].length()));
 | |
|     }
 | |
| 
 | |
|     for (auto& row : rows) {
 | |
|         for (size_t i = 0; i < columns.size(); i++) {
 | |
|             auto& column = columns[i];
 | |
|             auto& cell_text = row[i];
 | |
|             if (!column.width) {
 | |
|                 out("{}", cell_text);
 | |
|                 continue;
 | |
|             }
 | |
|             if (column.alignment == Alignment::Right)
 | |
|                 out("{1:>{0}} ", column.width, cell_text);
 | |
|             else
 | |
|                 out("{1:{0}} ", column.width, cell_text);
 | |
|             if (i != columns.size() - 1)
 | |
|                 out(" ");
 | |
|         }
 | |
|         outln();
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 |