ProcessTableModel.cpp 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. #include "ProcessTableModel.h"
  2. #include <fcntl.h>
  3. #include <stdio.h>
  4. enum Column {
  5. PID = 0,
  6. State,
  7. Priority,
  8. Linear,
  9. Physical,
  10. CPU,
  11. Name,
  12. __Count
  13. };
  14. ProcessTableModel::ProcessTableModel()
  15. {
  16. }
  17. ProcessTableModel::~ProcessTableModel()
  18. {
  19. }
  20. int ProcessTableModel::row_count() const
  21. {
  22. return m_processes.size();
  23. }
  24. int ProcessTableModel::column_count() const
  25. {
  26. return Column::__Count;
  27. }
  28. String ProcessTableModel::column_name(int column) const
  29. {
  30. switch (column) {
  31. case Column::PID: return "PID";
  32. case Column::State: return "State";
  33. case Column::Priority: return "Priority";
  34. case Column::Linear: return "Linear";
  35. case Column::Physical: return "Physical";
  36. case Column::CPU: return "CPU";
  37. case Column::Name: return "Name";
  38. default: ASSERT_NOT_REACHED();
  39. }
  40. }
  41. GTableModel::ColumnMetadata ProcessTableModel::column_metadata(int column) const
  42. {
  43. switch (column) {
  44. case Column::PID: return { 30, TextAlignment::CenterRight };
  45. case Column::State: return { 80, TextAlignment::CenterLeft };
  46. case Column::Priority: return { 75, TextAlignment::CenterLeft };
  47. case Column::Linear: return { 70, TextAlignment::CenterRight };
  48. case Column::Physical: return { 70, TextAlignment::CenterRight };
  49. case Column::CPU: return { 30, TextAlignment::CenterRight };
  50. case Column::Name: return { 200, TextAlignment::CenterLeft };
  51. default: ASSERT_NOT_REACHED();
  52. }
  53. }
  54. GModelIndex ProcessTableModel::selected_index() const
  55. {
  56. return { m_selected_row, 0 };
  57. }
  58. void ProcessTableModel::set_selected_index(GModelIndex index)
  59. {
  60. if (!index.is_valid()) {
  61. m_selected_row = -1;
  62. return;
  63. }
  64. if (index.row() >= 0 && index.row() < m_pids.size())
  65. m_selected_row = index.row();
  66. }
  67. static String pretty_byte_size(size_t size)
  68. {
  69. return String::format("%uK", size / 1024);
  70. }
  71. String ProcessTableModel::data(int row, int column) const
  72. {
  73. ASSERT(is_valid({ row, column }));
  74. auto it = m_processes.find(m_pids[row]);
  75. auto& process = *(*it).value;
  76. switch (column) {
  77. case Column::PID: return String::format("%d", process.current_state.pid);
  78. case Column::State: return process.current_state.state;
  79. case Column::Priority: return process.current_state.priority;
  80. case Column::Linear: return pretty_byte_size(process.current_state.linear);
  81. case Column::Physical: return pretty_byte_size(process.current_state.physical);
  82. case Column::CPU: return String::format("%d", (int)process.current_state.cpu_percent);
  83. case Column::Name: return process.current_state.name;
  84. }
  85. ASSERT_NOT_REACHED();
  86. }
  87. void ProcessTableModel::update()
  88. {
  89. FILE* fp = fopen("/proc/all", "r");
  90. if (!fp) {
  91. perror("failed to open /proc/all");
  92. exit(1);
  93. }
  94. unsigned last_sum_nsched = 0;
  95. for (auto& it : m_processes)
  96. last_sum_nsched += it.value->current_state.nsched;
  97. HashTable<pid_t> live_pids;
  98. unsigned sum_nsched = 0;
  99. for (;;) {
  100. char buf[BUFSIZ];
  101. char* ptr = fgets(buf, sizeof(buf), fp);
  102. if (!ptr)
  103. break;
  104. auto parts = String(buf, Chomp).split(',');
  105. if (parts.size() < 17)
  106. break;
  107. bool ok;
  108. pid_t pid = parts[0].to_uint(ok);
  109. ASSERT(ok);
  110. unsigned nsched = parts[1].to_uint(ok);
  111. ASSERT(ok);
  112. ProcessState state;
  113. state.pid = pid;
  114. state.nsched = nsched;
  115. unsigned uid = parts[5].to_uint(ok);
  116. ASSERT(ok);
  117. //state.user = s_usernames->get(uid);
  118. state.user = String::format("%u", uid);
  119. state.priority = parts[16];
  120. state.state = parts[7];
  121. state.name = parts[11];
  122. state.linear = parts[12].to_uint(ok);
  123. ASSERT(ok);
  124. state.physical = parts[13].to_uint(ok);
  125. ASSERT(ok);
  126. {
  127. auto it = m_processes.find(pid);
  128. if (it == m_processes.end())
  129. m_processes.set(pid, make<Process>());
  130. }
  131. auto it = m_processes.find(pid);
  132. ASSERT(it != m_processes.end());
  133. (*it).value->previous_state = (*it).value->current_state;
  134. (*it).value->current_state = state;
  135. live_pids.set(pid);
  136. sum_nsched += nsched;
  137. }
  138. int rc = fclose(fp);
  139. ASSERT(rc == 0);
  140. m_pids.clear();
  141. Vector<pid_t> pids_to_remove;
  142. for (auto& it : m_processes) {
  143. if (!live_pids.contains(it.key)) {
  144. pids_to_remove.append(it.key);
  145. continue;
  146. }
  147. auto& process = *it.value;
  148. dword nsched_diff = process.current_state.nsched - process.previous_state.nsched;
  149. process.current_state.cpu_percent = ((float)nsched_diff * 100) / (float)(sum_nsched - last_sum_nsched);
  150. m_pids.append(it.key);
  151. }
  152. for (auto pid : pids_to_remove)
  153. m_processes.remove(pid);
  154. did_update();
  155. }
  156. pid_t ProcessTableModel::selected_pid() const
  157. {
  158. if (m_selected_row == -1)
  159. return -1;
  160. return m_pids[m_selected_row];
  161. }