ProcessManager.cpp 5.3 KB

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