ProcessManager.cpp 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. /*
  2. * Copyright (c) 2024, Andrew Kaster <akaster@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/String.h>
  7. #include <LibCore/EventLoop.h>
  8. #include <LibCore/System.h>
  9. #include <LibWebView/ProcessManager.h>
  10. namespace WebView {
  11. static sig_atomic_t s_received_sigchld = 0;
  12. ProcessType process_type_from_name(StringView name)
  13. {
  14. if (name == "Chrome"sv)
  15. return ProcessType::Chrome;
  16. if (name == "WebContent"sv)
  17. return ProcessType::WebContent;
  18. if (name == "WebWorker"sv)
  19. return ProcessType::WebWorker;
  20. if (name == "SQLServer"sv)
  21. return ProcessType::SQLServer;
  22. if (name == "RequestServer"sv)
  23. return ProcessType::RequestServer;
  24. if (name == "ImageDecoder"sv)
  25. return ProcessType::ImageDecoder;
  26. dbgln("Unknown process type: '{}'", name);
  27. VERIFY_NOT_REACHED();
  28. }
  29. StringView process_name_from_type(ProcessType type)
  30. {
  31. switch (type) {
  32. case ProcessType::Chrome:
  33. return "Chrome"sv;
  34. case ProcessType::WebContent:
  35. return "WebContent"sv;
  36. case ProcessType::WebWorker:
  37. return "WebWorker"sv;
  38. case ProcessType::SQLServer:
  39. return "SQLServer"sv;
  40. case ProcessType::RequestServer:
  41. return "RequestServer"sv;
  42. case ProcessType::ImageDecoder:
  43. return "ImageDecoder"sv;
  44. }
  45. VERIFY_NOT_REACHED();
  46. }
  47. ProcessManager::ProcessManager()
  48. {
  49. }
  50. ProcessManager::~ProcessManager()
  51. {
  52. }
  53. ProcessManager& ProcessManager::the()
  54. {
  55. static ProcessManager s_the;
  56. return s_the;
  57. }
  58. void ProcessManager::initialize()
  59. {
  60. // FIXME: Should we change this to call EventLoop::register_signal?
  61. // Note that only EventLoopImplementationUnix has a working register_signal
  62. struct sigaction action { };
  63. action.sa_flags = SA_RESTART;
  64. action.sa_sigaction = [](int, auto*, auto) {
  65. s_received_sigchld = 1;
  66. };
  67. MUST(Core::System::sigaction(SIGCHLD, &action, nullptr));
  68. the().add_process(WebView::ProcessType::Chrome, getpid());
  69. }
  70. void ProcessManager::add_process(ProcessType type, pid_t pid)
  71. {
  72. dbgln("ProcessManager::add_process({}, {})", process_name_from_type(type), pid);
  73. m_processes.append({ type, pid, 0, 0 });
  74. }
  75. void ProcessManager::remove_process(pid_t pid)
  76. {
  77. m_processes.remove_first_matching([&](auto& info) {
  78. if (info.pid == pid) {
  79. dbgln("ProcessManager: Remove process {} ({})", process_name_from_type(info.type), pid);
  80. return true;
  81. }
  82. return false;
  83. });
  84. }
  85. void ProcessManager::update_all_processes()
  86. {
  87. if (s_received_sigchld) {
  88. s_received_sigchld = 0;
  89. auto result = Core::System::waitpid(-1, WNOHANG);
  90. while (!result.is_error() && result.value().pid > 0) {
  91. auto& [pid, status] = result.value();
  92. if (WIFEXITED(status) || WIFSIGNALED(status)) {
  93. remove_process(pid);
  94. }
  95. result = Core::System::waitpid(-1, WNOHANG);
  96. }
  97. }
  98. // FIXME: Actually gather stats in a platform-specific way
  99. }
  100. String ProcessManager::generate_html()
  101. {
  102. StringBuilder builder;
  103. auto processes = m_processes;
  104. builder.append(R"(
  105. <html>
  106. <head>
  107. <style>
  108. table {
  109. width: 100%;
  110. border-collapse: collapse;
  111. }
  112. th {
  113. text-align: left;
  114. border-bottom: 1px solid #aaa;
  115. }
  116. td, th {
  117. padding: 4px;
  118. border: 1px solid #aaa;
  119. }
  120. tr:nth-child(odd) {
  121. background: #f7f7f7;
  122. }
  123. </style>
  124. </head>
  125. <body>
  126. <table>
  127. <thead>
  128. <tr>
  129. <th>Type</th>
  130. <th>PID</th>
  131. <th>Memory Usage</th>
  132. <th>CPU %</th>
  133. </tr>
  134. </thead>
  135. <tbody>
  136. )"sv);
  137. for (auto& process : processes) {
  138. builder.append("<tr>"sv);
  139. builder.append("<td>"sv);
  140. builder.append(WebView::process_name_from_type(process.type));
  141. builder.append("</td>"sv);
  142. builder.append("<td>"sv);
  143. builder.append(MUST(String::number(process.pid)));
  144. builder.append("</td>"sv);
  145. builder.append("<td>"sv);
  146. builder.append(MUST(String::formatted("{} KB", process.memory_usage_kib)));
  147. builder.append("</td>"sv);
  148. builder.append("<td>"sv);
  149. builder.append(MUST(String::formatted("{:.1f}", process.cpu_percent)));
  150. builder.append("</td>"sv);
  151. builder.append("</tr>"sv);
  152. }
  153. builder.append(R"(
  154. </tbody>
  155. </table>
  156. </body>
  157. </html>
  158. )"sv);
  159. return builder.to_string_without_validation();
  160. }
  161. }