Mercurial > minori
comparison dep/animone/src/fd/proc.cc @ 258:862d0d8619f6
*: HUUUGE changes
animia has been renamed to animone, so instead of thinking of a
health condition, you think of a beautiful flower :)
I've also edited some of the code for animone, but I have no idea
if it even works or not because I don't have a mac or windows
machine lying around. whoops!
... anyway, all of the changes divergent from Anisthesia are now
licensed under BSD. it's possible that I could even rewrite most
of the code to where I don't even have to keep the MIT license,
but that's thinking too far into the future
I've been slacking off on implementing the anime seasons page,
mostly out of laziness. I think I'd have to create another db file
specifically for the seasons
anyway, this code is being pushed *primarily* because the hard drive
it's on is failing! yay :)
| author | Paper <paper@paper.us.eu.org> | 
|---|---|
| date | Mon, 01 Apr 2024 02:43:44 -0400 | 
| parents | |
| children | 0718f538c5f9 | 
   comparison
  equal
  deleted
  inserted
  replaced
| 257:699a20c57dc8 | 258:862d0d8619f6 | 
|---|---|
| 1 #include "animone/fd/proc.h" | |
| 2 #include "animone.h" | |
| 3 #include "animone/util.h" | |
| 4 | |
| 5 #include <filesystem> | |
| 6 #include <fstream> | |
| 7 #include <sstream> | |
| 8 #include <string> | |
| 9 | |
| 10 #include <fcntl.h> | |
| 11 #include <sys/stat.h> | |
| 12 #include <unistd.h> | |
| 13 | |
| 14 static constexpr std::string_view PROC_LOCATION = "/proc"; | |
| 15 | |
| 16 namespace animone::internal::proc { | |
| 17 | |
| 18 static bool IsRegularFile(std::string link) { | |
| 19 struct stat sb; | |
| 20 if (stat(link.c_str(), &sb) == -1) | |
| 21 return false; | |
| 22 | |
| 23 return S_ISREG(sb.st_mode); | |
| 24 } | |
| 25 | |
| 26 static bool AreFlagsOk(pid_t pid, int fd) { | |
| 27 const std::filesystem::path path = | |
| 28 std::filesystem::path(PROC_LOCATION) / std::to_string(pid) / "fdinfo" / std::to_string(fd); | |
| 29 | |
| 30 std::ifstream file(path); | |
| 31 if (!file) | |
| 32 return false; | |
| 33 | |
| 34 int flags = 0; | |
| 35 for (std::string line; std::getline(file, line);) | |
| 36 if (line.find("flags:", 0) == 0) | |
| 37 flags = util::StringToInt(line.substr(line.find_last_not_of("0123456789") + 1)); | |
| 38 | |
| 39 if (flags & O_WRONLY || flags & O_RDWR) | |
| 40 return false; | |
| 41 | |
| 42 return true; | |
| 43 } | |
| 44 | |
| 45 static bool GetFilenameFromFd(std::string link, std::string& out) { | |
| 46 /* /proc is a "virtual filesystem", so we have to guess the path size. yippee! */ | |
| 47 constexpr size_t OUT_MAX = (1ul << 15); // 32KiB | |
| 48 out.resize(32); | |
| 49 | |
| 50 ssize_t exe_used = 0; | |
| 51 do { | |
| 52 out.resize(out.length() * 2); | |
| 53 | |
| 54 exe_used = readlink(link.c_str(), &out.front(), out.length()); | |
| 55 if (exe_used == (ssize_t)-1 || exe_used < (ssize_t)1) | |
| 56 return false; // we got a bad result. SAD! | |
| 57 } while (out.length() < OUT_MAX && exe_used >= static_cast<ssize_t>(out.length())); | |
| 58 | |
| 59 out.resize(out.find('\0')); | |
| 60 | |
| 61 return true; | |
| 62 } | |
| 63 | |
| 64 static bool IsSystemFile(const std::string& path) { | |
| 65 static constexpr std::array<std::string_view, 9> invalid_paths = {"/boot", "/dev", "/bin", "/usr", "/opt", | |
| 66 "/proc", "/var", "/etc", "/dev"}; | |
| 67 | |
| 68 for (const auto& invalid_path : invalid_paths) { | |
| 69 if (!path.rfind(invalid_path, 0)) { | |
| 70 return true; | |
| 71 } | |
| 72 } | |
| 73 | |
| 74 return false; | |
| 75 } | |
| 76 | |
| 77 bool GetProcessName(pid_t pid, std::string& result) { | |
| 78 const std::filesystem::path path = std::filesystem::path(PROC_LOCATION) / std::to_string(pid) / "comm"; | |
| 79 | |
| 80 if (!util::ReadFile(path, result)) | |
| 81 return false; | |
| 82 | |
| 83 result.erase(std::remove(result.begin(), result.end(), '\n'), result.end()); | |
| 84 return true; | |
| 85 } | |
| 86 | |
| 87 bool EnumerateOpenProcesses(process_proc_t process_proc) { | |
| 88 bool success = false; | |
| 89 | |
| 90 for (const auto& dir : std::filesystem::directory_iterator{PROC_LOCATION}) { | |
| 91 Process proc; | |
| 92 | |
| 93 try { | |
| 94 proc.pid = util::StringToInt(dir.path().stem()); | |
| 95 success = true; | |
| 96 } catch (std::invalid_argument const& ex) { | |
| 97 continue; | |
| 98 } | |
| 99 | |
| 100 if (!GetProcessName(proc.pid, proc.name)) | |
| 101 continue; | |
| 102 | |
| 103 if (!process_proc(proc)) | |
| 104 return false; | |
| 105 } | |
| 106 | |
| 107 return success; | |
| 108 } | |
| 109 | |
| 110 bool EnumerateOpenFiles(const std::set<pid_t>& pids, open_file_proc_t open_file_proc) { | |
| 111 if (!open_file_proc) | |
| 112 return false; | |
| 113 | |
| 114 for (const auto& pid : pids) { | |
| 115 const std::filesystem::path path = std::filesystem::path(PROC_LOCATION) / std::to_string(pid) / "fd"; | |
| 116 | |
| 117 for (const auto& dir : std::filesystem::directory_iterator{path}) { | |
| 118 if (!AreFlagsOk(pid, util::StringToInt(dir.path().stem()))) | |
| 119 continue; | |
| 120 | |
| 121 std::string name; | |
| 122 if (!GetFilenameFromFd(dir.path(), name)) | |
| 123 continue; | |
| 124 | |
| 125 if (!IsRegularFile(name)) | |
| 126 continue; | |
| 127 | |
| 128 if (IsSystemFile(name)) | |
| 129 continue; | |
| 130 | |
| 131 if (!open_file_proc({pid, name})) | |
| 132 return false; | |
| 133 } | |
| 134 } | |
| 135 return true; | |
| 136 } | |
| 137 | |
| 138 } // namespace animia::internal::proc | 
