blob: a65918f54f8bb1afc6e6c69a07d3d68f4d754f93 [file] [log] [blame]
Hisham Muhammadd6231ba2006-03-04 18:16:49 +00001/*
2htop - Process.c
Hisham Muhammadb1b3f572015-03-21 16:52:54 -03003(C) 2004-2015 Hisham H. Muhammad
Nathan Scott500fb282020-08-20 09:35:24 +10004(C) 2020 Red Hat, Inc. All Rights Reserved.
Daniel Lange079c2ab2020-10-05 09:51:32 +02005Released under the GNU GPLv2, see the COPYING file
Hisham Muhammadd6231ba2006-03-04 18:16:49 +00006in the source distribution for its full text.
7*/
8
Benny Baumann0f526292020-09-19 13:55:23 +02009#include "config.h" // IWYU pragma: keep
Hisham Muhammad272e2d92015-03-16 23:01:48 -030010
Benny Baumann0f526292020-09-19 13:55:23 +020011#include "Process.h"
12
13#include <assert.h>
14#include <limits.h>
15#include <math.h>
16#include <signal.h>
17#include <stdbool.h>
18#include <stdio.h>
19#include <stdlib.h>
Benny Baumann0f526292020-09-19 13:55:23 +020020#include <time.h>
21#include <unistd.h>
22#include <sys/resource.h>
Explorer0935129712018-12-30 12:18:27 +080023
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000024#include "CRT.h"
Christian Göttsche7cf52772020-11-18 14:26:30 +010025#include "Macros.h"
Hisham Muhammadb4f6b112014-11-27 20:10:23 -020026#include "Platform.h"
Christian Göttsche7cf52772020-11-18 14:26:30 +010027#include "ProcessList.h"
Benny Baumann0f526292020-09-19 13:55:23 +020028#include "RichString.h"
29#include "Settings.h"
Benny Baumann872e5422020-10-14 20:21:09 +020030#include "XUtils.h"
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000031
Benny Baumann0f526292020-09-19 13:55:23 +020032#if defined(MAJOR_IN_MKDEV)
Kang-Che Sung (宋岡哲)c01f40e2018-02-26 21:15:05 +080033#include <sys/mkdev.h>
Wataru Ashihara41754e52018-12-15 22:06:00 +090034#elif defined(MAJOR_IN_SYSMACROS)
Kang-Che Sung (宋岡哲)c01f40e2018-02-26 21:15:05 +080035#include <sys/sysmacros.h>
36#endif
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000037
Benny Baumann0f526292020-09-19 13:55:23 +020038
Christian Göttsche42073ba2020-11-04 17:46:04 +010039static uid_t Process_getuid = (uid_t)-1;
Hisham Muhammadeb2803c2006-07-12 01:35:59 +000040
Hishamf1f805f2016-02-10 18:48:04 -020041char Process_pidFormat[20] = "%7d ";
Hisham Muhammad94280102015-08-20 00:32:47 -030042
43static char Process_titleBuffer[20][20];
44
45void Process_setupColumnWidths() {
46 int maxPid = Platform_getMaxPid();
Benny Baumann45869512020-11-01 01:09:51 +010047 if (maxPid == -1)
48 return;
49
Hisham Muhammad94280102015-08-20 00:32:47 -030050 int digits = ceil(log10(maxPid));
51 assert(digits < 20);
52 for (int i = 0; Process_pidColumns[i].label; i++) {
53 assert(i < 20);
Hisham Muhammad09e241f2017-07-27 16:07:50 -030054 xSnprintf(Process_titleBuffer[i], 20, "%*s ", digits, Process_pidColumns[i].label);
Hisham Muhammad94280102015-08-20 00:32:47 -030055 Process_fields[Process_pidColumns[i].id].title = Process_titleBuffer[i];
56 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -030057 xSnprintf(Process_pidFormat, sizeof(Process_pidFormat), "%%%dd ", digits);
Hisham Muhammad94280102015-08-20 00:32:47 -030058}
59
Michael Wittenab3171d2020-09-29 14:04:22 +000060void Process_humanNumber(RichString* str, unsigned long long number, bool coloring) {
Daniel Lange7899ae22020-11-28 17:57:51 +010061 char buffer[10];
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000062 int len;
Daniel Flanagandd334442019-10-31 11:39:12 -050063
Hisham Muhammada939cdf2014-04-24 15:00:09 -030064 int largeNumberColor = CRT_colors[LARGE_NUMBER];
65 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
Benny Baumann40441dc2020-09-13 23:50:24 +020066 int processGigabytesColor = CRT_colors[PROCESS_GIGABYTES];
Hisham Muhammada939cdf2014-04-24 15:00:09 -030067 int processColor = CRT_colors[PROCESS];
68 if (!coloring) {
69 largeNumberColor = CRT_colors[PROCESS];
70 processMegabytesColor = CRT_colors[PROCESS];
Benny Baumann40441dc2020-09-13 23:50:24 +020071 processGigabytesColor = CRT_colors[PROCESS];
Hisham Muhammada939cdf2014-04-24 15:00:09 -030072 }
Daniel Flanagandd334442019-10-31 11:39:12 -050073
Benny Baumann40441dc2020-09-13 23:50:24 +020074 if (number < 1000) {
75 //Plain number, no markings
Daniel Lange7899ae22020-11-28 17:57:51 +010076 len = xSnprintf(buffer, sizeof(buffer), "%5llu ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010077 RichString_appendnAscii(str, processColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020078 } else if (number < 100000) {
79 //2 digit MB, 3 digit KB
Daniel Lange7899ae22020-11-28 17:57:51 +010080 len = xSnprintf(buffer, sizeof(buffer), "%2llu", number/1000);
Christian Göttsche157086e2020-12-04 14:44:57 +010081 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000082 number %= 1000;
Daniel Lange7899ae22020-11-28 17:57:51 +010083 len = xSnprintf(buffer, sizeof(buffer), "%03llu ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010084 RichString_appendnAscii(str, processColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020085 } else if (number < 1000 * ONE_K) {
86 //3 digit MB
87 number /= ONE_K;
Daniel Lange7899ae22020-11-28 17:57:51 +010088 len = xSnprintf(buffer, sizeof(buffer), "%4lluM ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010089 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020090 } else if (number < 10000 * ONE_K) {
91 //1 digit GB, 3 digit MB
92 number /= ONE_K;
Daniel Lange7899ae22020-11-28 17:57:51 +010093 len = xSnprintf(buffer, sizeof(buffer), "%1llu", number/1000);
Christian Göttsche157086e2020-12-04 14:44:57 +010094 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020095 number %= 1000;
Daniel Lange7899ae22020-11-28 17:57:51 +010096 len = xSnprintf(buffer, sizeof(buffer), "%03lluM ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010097 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
Benny Baumanne0e59972020-09-20 19:54:53 +020098 } else if (number < 100000 * ONE_K) {
Benny Baumann40441dc2020-09-13 23:50:24 +020099 //2 digit GB, 1 digit MB
100 number /= 100 * ONE_K;
Daniel Lange7899ae22020-11-28 17:57:51 +0100101 len = xSnprintf(buffer, sizeof(buffer), "%2llu", number/10);
Christian Göttsche157086e2020-12-04 14:44:57 +0100102 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +0200103 number %= 10;
Christian Göttschecd305b42020-11-29 14:14:46 +0100104 len = xSnprintf(buffer, sizeof(buffer), ".%1llu", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100105 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
106 RichString_appendAscii(str, processGigabytesColor, "G ");
Benny Baumann40441dc2020-09-13 23:50:24 +0200107 } else if (number < 1000 * ONE_M) {
108 //3 digit GB
109 number /= ONE_M;
Daniel Lange7899ae22020-11-28 17:57:51 +0100110 len = xSnprintf(buffer, sizeof(buffer), "%4lluG ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100111 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Michael Wittenab3171d2020-09-29 14:04:22 +0000112 } else if (number < 10000ULL * ONE_M) {
Benny Baumann40441dc2020-09-13 23:50:24 +0200113 //1 digit TB, 3 digit GB
114 number /= ONE_M;
Daniel Lange7899ae22020-11-28 17:57:51 +0100115 len = xSnprintf(buffer, sizeof(buffer), "%1llu", number/1000);
Christian Göttsche157086e2020-12-04 14:44:57 +0100116 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +0200117 number %= 1000;
Daniel Lange7899ae22020-11-28 17:57:51 +0100118 len = xSnprintf(buffer, sizeof(buffer), "%03lluG ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100119 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +0200120 } else {
121 //2 digit TB and above
Daniel Lange7899ae22020-11-28 17:57:51 +0100122 len = xSnprintf(buffer, sizeof(buffer), "%4.1lfT ", (double)number/ONE_G);
Christian Göttsche157086e2020-12-04 14:44:57 +0100123 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000124 }
125}
126
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300127void Process_colorNumber(RichString* str, unsigned long long number, bool coloring) {
Daniel Lange7899ae22020-11-28 17:57:51 +0100128 char buffer[13];
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300129
130 int largeNumberColor = CRT_colors[LARGE_NUMBER];
131 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
132 int processColor = CRT_colors[PROCESS];
133 int processShadowColor = CRT_colors[PROCESS_SHADOW];
Benny Baumann0b29e502020-11-28 17:42:02 +0100134
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300135 if (!coloring) {
136 largeNumberColor = CRT_colors[PROCESS];
137 processMegabytesColor = CRT_colors[PROCESS];
138 processShadowColor = CRT_colors[PROCESS];
139 }
140
Michael Wittenab3171d2020-09-29 14:04:22 +0000141 if (number == ULLONG_MAX) {
Christian Göttsche157086e2020-12-04 14:44:57 +0100142 RichString_appendAscii(str, CRT_colors[PROCESS_SHADOW], " N/A ");
adrien1018536941f2018-12-30 20:18:35 +0800143 } else if (number >= 100000LL * ONE_DECIMAL_T) {
Benny Baumann0b29e502020-11-28 17:42:02 +0100144 xSnprintf(buffer, sizeof(buffer), "%11llu ", number / ONE_DECIMAL_G);
Christian Göttsche157086e2020-12-04 14:44:57 +0100145 RichString_appendnAscii(str, largeNumberColor, buffer, 12);
adrien1018536941f2018-12-30 20:18:35 +0800146 } else if (number >= 100LL * ONE_DECIMAL_T) {
Benny Baumann0b29e502020-11-28 17:42:02 +0100147 xSnprintf(buffer, sizeof(buffer), "%11llu ", number / ONE_DECIMAL_M);
Christian Göttsche157086e2020-12-04 14:44:57 +0100148 RichString_appendnAscii(str, largeNumberColor, buffer, 8);
149 RichString_appendnAscii(str, processMegabytesColor, buffer+8, 4);
adrien1018536941f2018-12-30 20:18:35 +0800150 } else if (number >= 10LL * ONE_DECIMAL_G) {
Benny Baumann0b29e502020-11-28 17:42:02 +0100151 xSnprintf(buffer, sizeof(buffer), "%11llu ", number / ONE_DECIMAL_K);
Christian Göttsche157086e2020-12-04 14:44:57 +0100152 RichString_appendnAscii(str, largeNumberColor, buffer, 5);
153 RichString_appendnAscii(str, processMegabytesColor, buffer+5, 3);
154 RichString_appendnAscii(str, processColor, buffer+8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000155 } else {
Benny Baumann0b29e502020-11-28 17:42:02 +0100156 xSnprintf(buffer, sizeof(buffer), "%11llu ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100157 RichString_appendnAscii(str, largeNumberColor, buffer, 2);
158 RichString_appendnAscii(str, processMegabytesColor, buffer+2, 3);
159 RichString_appendnAscii(str, processColor, buffer+5, 3);
160 RichString_appendnAscii(str, processShadowColor, buffer+8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000161 }
162}
163
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300164void Process_printTime(RichString* str, unsigned long long totalHundredths) {
165 unsigned long long totalSeconds = totalHundredths / 100;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000166
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300167 unsigned long long hours = totalSeconds / 3600;
168 int minutes = (totalSeconds / 60) % 60;
169 int seconds = totalSeconds % 60;
170 int hundredths = totalHundredths - (totalSeconds * 100);
Daniel Lange7899ae22020-11-28 17:57:51 +0100171 char buffer[10];
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000172 if (hours >= 100) {
Daniel Lange7899ae22020-11-28 17:57:51 +0100173 xSnprintf(buffer, sizeof(buffer), "%7lluh ", hours);
Christian Göttsche157086e2020-12-04 14:44:57 +0100174 RichString_appendAscii(str, CRT_colors[LARGE_NUMBER], buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000175 } else {
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000176 if (hours) {
Daniel Lange7899ae22020-11-28 17:57:51 +0100177 xSnprintf(buffer, sizeof(buffer), "%2lluh", hours);
Christian Göttsche157086e2020-12-04 14:44:57 +0100178 RichString_appendAscii(str, CRT_colors[LARGE_NUMBER], buffer);
Daniel Lange7899ae22020-11-28 17:57:51 +0100179 xSnprintf(buffer, sizeof(buffer), "%02d:%02d ", minutes, seconds);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000180 } else {
Daniel Lange7899ae22020-11-28 17:57:51 +0100181 xSnprintf(buffer, sizeof(buffer), "%2d:%02d.%02d ", minutes, seconds, hundredths);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000182 }
Christian Göttsche157086e2020-12-04 14:44:57 +0100183 RichString_appendAscii(str, CRT_colors[DEFAULT_COLOR], buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000184 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000185}
186
Christian Göttschea63cfc82020-10-13 14:26:40 +0200187void Process_fillStarttimeBuffer(Process* this) {
188 struct tm date;
189 (void) localtime_r(&this->starttime_ctime, &date);
190 strftime(this->starttime_show, sizeof(this->starttime_show) - 1, (this->starttime_ctime > (time(NULL) - 86400)) ? "%R " : "%b%d ", &date);
191}
192
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200193static inline void Process_writeCommand(const Process* this, int attr, int baseattr, RichString* str) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200194 int start = RichString_size(str), finish = 0;
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200195 const char* comm = this->comm;
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200196
197 if (this->settings->highlightBaseName || !this->settings->showProgramPath) {
198 int i, basename = 0;
199 for (i = 0; i < this->basenameOffset; i++) {
200 if (comm[i] == '/') {
201 basename = i + 1;
202 } else if (comm[i] == ':') {
203 finish = i + 1;
204 break;
Hisham Muhammadf2a190b2014-02-27 17:11:23 -0300205 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000206 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200207 if (!finish) {
Benny Baumann45869512020-11-01 01:09:51 +0100208 if (this->settings->showProgramPath) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200209 start += basename;
Benny Baumann45869512020-11-01 01:09:51 +0100210 } else {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200211 comm += basename;
Benny Baumann45869512020-11-01 01:09:51 +0100212 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200213 finish = this->basenameOffset - basename;
214 }
215 finish += start - 1;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000216 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200217
Christian Göttsche157086e2020-12-04 14:44:57 +0100218 RichString_appendWide(str, attr, comm);
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200219
Benny Baumann45869512020-11-01 01:09:51 +0100220 if (this->settings->highlightBaseName) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200221 RichString_setAttrn(str, baseattr, start, finish);
Benny Baumann45869512020-11-01 01:09:51 +0100222 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000223}
224
Christian Göttsche77ec86a2020-11-24 18:37:13 +0100225void Process_outputRate(RichString* str, char* buffer, size_t n, double rate, int coloring) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200226 int largeNumberColor = CRT_colors[LARGE_NUMBER];
227 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
228 int processColor = CRT_colors[PROCESS];
Benny Baumann0b29e502020-11-28 17:42:02 +0100229
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200230 if (!coloring) {
231 largeNumberColor = CRT_colors[PROCESS];
232 processMegabytesColor = CRT_colors[PROCESS];
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000233 }
Benny Baumann0b29e502020-11-28 17:42:02 +0100234
Benny Baumann29ec1152020-09-07 11:53:58 +0200235 if (isnan(rate)) {
Christian Göttsche157086e2020-12-04 14:44:57 +0100236 RichString_appendAscii(str, CRT_colors[PROCESS_SHADOW], " N/A ");
Hisham797bcd02016-02-20 02:22:57 -0200237 } else if (rate < ONE_K) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200238 int len = snprintf(buffer, n, "%7.2f B/s ", rate);
Christian Göttsche157086e2020-12-04 14:44:57 +0100239 RichString_appendnAscii(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800240 } else if (rate < ONE_M) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200241 int len = snprintf(buffer, n, "%7.2f K/s ", rate / ONE_K);
Christian Göttsche157086e2020-12-04 14:44:57 +0100242 RichString_appendnAscii(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800243 } else if (rate < ONE_G) {
244 int len = snprintf(buffer, n, "%7.2f M/s ", rate / ONE_M);
Christian Göttsche157086e2020-12-04 14:44:57 +0100245 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800246 } else if (rate < ONE_T) {
247 int len = snprintf(buffer, n, "%7.2f G/s ", rate / ONE_G);
Christian Göttsche157086e2020-12-04 14:44:57 +0100248 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200249 } else {
adrien1018536941f2018-12-30 20:18:35 +0800250 int len = snprintf(buffer, n, "%7.2f T/s ", rate / ONE_T);
Christian Göttsche157086e2020-12-04 14:44:57 +0100251 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200252 }
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000253}
254
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200255void Process_writeField(const Process* this, RichString* str, ProcessField field) {
Hisham Muhammad2f30cd12014-04-24 15:08:32 -0300256 char buffer[256]; buffer[255] = '\0';
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000257 int attr = CRT_colors[DEFAULT_COLOR];
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000258 int baseattr = CRT_colors[PROCESS_BASENAME];
Christian Göttsche77ec86a2020-11-24 18:37:13 +0100259 size_t n = sizeof(buffer) - 1;
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200260 bool coloring = this->settings->highlightMegabytes;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000261
262 switch (field) {
Christian Göttsche15eab202020-10-30 17:02:20 +0100263 case PERCENT_CPU:
264 case PERCENT_NORM_CPU: {
265 float cpuPercentage = this->percent_cpu;
266 if (field == PERCENT_NORM_CPU) {
267 cpuPercentage /= this->processList->cpuCount;
268 }
269 if (cpuPercentage > 999.9) {
270 xSnprintf(buffer, n, "%4u ", (unsigned int)cpuPercentage);
271 } else if (cpuPercentage > 99.9) {
272 xSnprintf(buffer, n, "%3u. ", (unsigned int)cpuPercentage);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300273 } else {
Christian Göttsche15eab202020-10-30 17:02:20 +0100274 xSnprintf(buffer, n, "%4.1f ", cpuPercentage);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300275 }
276 break;
277 }
278 case PERCENT_MEM: {
279 if (this->percent_mem > 99.9) {
Daniel Flanagandd334442019-10-31 11:39:12 -0500280 xSnprintf(buffer, n, "100. ");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300281 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300282 xSnprintf(buffer, n, "%4.1f ", this->percent_mem);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300283 }
284 break;
285 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000286 case COMM: {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200287 if (this->settings->highlightThreads && Process_isThread(this)) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000288 attr = CRT_colors[PROCESS_THREAD];
289 baseattr = CRT_colors[PROCESS_THREAD_BASENAME];
290 }
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200291 if (!this->settings->treeView || this->indent == 0) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000292 Process_writeCommand(this, attr, baseattr, str);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000293 return;
294 } else {
295 char* buf = buffer;
296 int maxIndent = 0;
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000297 bool lastItem = (this->indent < 0);
298 int indent = (this->indent < 0 ? -this->indent : this->indent);
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000299
Benny Baumann45869512020-11-01 01:09:51 +0100300 for (int i = 0; i < 32; i++) {
301 if (indent & (1U << i)) {
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000302 maxIndent = i+1;
Benny Baumann45869512020-11-01 01:09:51 +0100303 }
304 }
305
Benny Baumann45cb99d2020-11-20 08:07:56 +0100306 for (int i = 0; i < maxIndent - 1; i++) {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100307 int written, ret;
Benny Baumann45869512020-11-01 01:09:51 +0100308 if (indent & (1 << i)) {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100309 ret = snprintf(buf, n, "%s ", CRT_treeStr[TREE_STR_VERT]);
Benny Baumann45869512020-11-01 01:09:51 +0100310 } else {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100311 ret = snprintf(buf, n, " ");
Benny Baumann45869512020-11-01 01:09:51 +0100312 }
Christian Göttsche77ec86a2020-11-24 18:37:13 +0100313 if (ret < 0 || (size_t)ret >= n) {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100314 written = n;
315 } else {
316 written = ret;
317 }
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000318 buf += written;
319 n -= written;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000320 }
Benny Baumannf6613db2020-12-08 21:24:19 +0100321
322 const char* draw = CRT_treeStr[lastItem ? TREE_STR_BEND : TREE_STR_RTEE];
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300323 xSnprintf(buf, n, "%s%s ", draw, this->showChildren ? CRT_treeStr[TREE_STR_SHUT] : CRT_treeStr[TREE_STR_OPEN] );
Christian Göttsche157086e2020-12-04 14:44:57 +0100324 RichString_appendWide(str, CRT_colors[PROCESS_TREE], buffer);
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000325 Process_writeCommand(this, attr, baseattr, str);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000326 return;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000327 }
328 }
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300329 case MAJFLT: Process_colorNumber(str, this->majflt, coloring); return;
330 case MINFLT: Process_colorNumber(str, this->minflt, coloring); return;
Nathan Scott75e9f9a2020-12-10 11:57:48 +1100331 case M_RESIDENT: Process_humanNumber(str, this->m_resident, coloring); return;
332 case M_VIRT: Process_humanNumber(str, this->m_virt, coloring); return;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300333 case NICE: {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300334 xSnprintf(buffer, n, "%3ld ", this->nice);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300335 attr = this->nice < 0 ? CRT_colors[PROCESS_HIGH_PRIORITY]
336 : this->nice > 0 ? CRT_colors[PROCESS_LOW_PRIORITY]
337 : attr;
338 break;
339 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300340 case NLWP: xSnprintf(buffer, n, "%4ld ", this->nlwp); break;
341 case PGRP: xSnprintf(buffer, n, Process_pidFormat, this->pgrp); break;
342 case PID: xSnprintf(buffer, n, Process_pidFormat, this->pid); break;
343 case PPID: xSnprintf(buffer, n, Process_pidFormat, this->ppid); break;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300344 case PRIORITY: {
Ivan Kozik07086fc2016-09-06 09:22:38 +0000345 if(this->priority <= -100)
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300346 xSnprintf(buffer, n, " RT ");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300347 else
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300348 xSnprintf(buffer, n, "%3ld ", this->priority);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300349 break;
350 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300351 case PROCESSOR: xSnprintf(buffer, n, "%3d ", Settings_cpuId(this->settings, this->processor)); break;
352 case SESSION: xSnprintf(buffer, n, Process_pidFormat, this->session); break;
353 case STARTTIME: xSnprintf(buffer, n, "%s", this->starttime_show); break;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000354 case STATE: {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300355 xSnprintf(buffer, n, "%c ", this->state);
Valmiky Arquissandas64e0d942014-10-14 02:30:17 +0100356 switch(this->state) {
357 case 'R':
358 attr = CRT_colors[PROCESS_R_STATE];
359 break;
360 case 'D':
361 attr = CRT_colors[PROCESS_D_STATE];
362 break;
Valmiky Arquissandas64e0d942014-10-14 02:30:17 +0100363 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000364 break;
365 }
Daniel Langec34be412018-10-07 11:16:12 +0200366 case ST_UID: xSnprintf(buffer, n, "%5d ", this->st_uid); break;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300367 case TIME: Process_printTime(str, this->time); return;
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300368 case TGID: xSnprintf(buffer, n, Process_pidFormat, this->tgid); break;
369 case TPGID: xSnprintf(buffer, n, Process_pidFormat, this->tpgid); break;
370 case TTY_NR: xSnprintf(buffer, n, "%3u:%3u ", major(this->tty_nr), minor(this->tty_nr)); break;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000371 case USER: {
Christian Göttsche42073ba2020-11-04 17:46:04 +0100372 if (Process_getuid != this->st_uid)
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000373 attr = CRT_colors[PROCESS_SHADOW];
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000374 if (this->user) {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300375 xSnprintf(buffer, n, "%-9s ", this->user);
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000376 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300377 xSnprintf(buffer, n, "%-9d ", this->st_uid);
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000378 }
Hisham Muhammad9b351402011-05-26 16:31:18 +0000379 if (buffer[9] != '\0') {
380 buffer[9] = ' ';
381 buffer[10] = '\0';
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000382 }
383 break;
384 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000385 default:
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300386 xSnprintf(buffer, n, "- ");
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000387 }
Christian Göttsche157086e2020-12-04 14:44:57 +0100388 RichString_appendWide(str, attr, buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000389}
390
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200391void Process_display(const Object* cast, RichString* out) {
392 const Process* this = (const Process*) cast;
393 const ProcessField* fields = this->settings->fields;
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000394 RichString_prune(out);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000395 for (int i = 0; fields[i]; i++)
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300396 As_Process(this)->writeField(this, out, fields[i]);
Benny Baumann45869512020-11-01 01:09:51 +0100397
Christian Göttsche42073ba2020-11-04 17:46:04 +0100398 if (this->settings->shadowOtherUsers && this->st_uid != Process_getuid) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000399 RichString_setAttr(out, CRT_colors[PROCESS_SHADOW]);
Benny Baumann45869512020-11-01 01:09:51 +0100400 }
401
402 if (this->tag == true) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000403 RichString_setAttr(out, CRT_colors[PROCESS_TAG]);
Benny Baumann45869512020-11-01 01:09:51 +0100404 }
405
Daniel Lange09510902020-11-16 12:13:47 +0100406 if (this->settings->highlightChanges) {
407 if (Process_isTomb(this)) {
408 out->highlightAttr = CRT_colors[PROCESS_TOMB];
409 } else if (Process_isNew(this)) {
410 out->highlightAttr = CRT_colors[PROCESS_NEW];
411 }
412 }
413
Hisham Muhammada9c0ea32011-03-22 20:37:08 +0000414 assert(out->chlen > 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000415}
416
Hisham Muhammad6f868b02015-02-20 14:52:10 -0200417void Process_done(Process* this) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000418 assert (this != NULL);
Hisham Muhammadf54a37b2014-05-03 17:49:05 -0300419 free(this->comm);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000420}
421
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530422static const char* Process_getCommandStr(const Process* p) {
423 return p->comm ? p->comm : "";
424}
425
Christian Göttscheba282cf2020-10-05 13:19:50 +0200426const ProcessClass Process_class = {
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300427 .super = {
428 .extends = Class(Object),
429 .display = Process_display,
430 .delete = Process_delete,
431 .compare = Process_compare
432 },
433 .writeField = Process_writeField,
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530434 .getCommandStr = Process_getCommandStr,
Hisham Muhammad00b324b2012-12-05 15:12:20 +0000435};
436
Christian Göttsche4eb44392020-10-21 21:26:05 +0200437void Process_init(Process* this, const struct Settings_* settings) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200438 this->settings = settings;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000439 this->tag = false;
Hisham Muhammad9eb91212010-06-17 19:02:03 +0000440 this->showChildren = true;
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000441 this->show = true;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000442 this->updated = false;
Hisham Muhammadcb297af2014-04-09 17:43:54 -0300443 this->basenameOffset = -1;
Benny Baumann45869512020-11-01 01:09:51 +0100444
Christian Göttsche42073ba2020-11-04 17:46:04 +0100445 if (Process_getuid == (uid_t)-1) {
Benny Baumann45869512020-11-01 01:09:51 +0100446 Process_getuid = getuid();
447 }
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000448}
449
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000450void Process_toggleTag(Process* this) {
Christian Göttsche2d231d72020-12-08 22:37:15 +0100451 this->tag = !this->tag;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000452}
453
Adam Saponaradde71c62020-10-30 21:56:16 -0400454bool Process_isNew(const Process* this) {
Adam Saponaraa83f5152020-10-31 20:36:53 -0400455 assert(this->processList);
Daniel Lange09510902020-11-16 12:13:47 +0100456 if (this->processList->scanTs >= this->seenTs) {
Christian Göttsche27b8d812020-12-13 13:18:38 +0100457 return this->processList->scanTs - this->seenTs <= 1000 * this->processList->settings->highlightDelaySecs;
Daniel Lange09510902020-11-16 12:13:47 +0100458 }
Adam Saponaradde71c62020-10-30 21:56:16 -0400459 return false;
460}
461
462bool Process_isTomb(const Process* this) {
Adam Saponaraa83f5152020-10-31 20:36:53 -0400463 return this->tombTs > 0;
Adam Saponaradde71c62020-10-30 21:56:16 -0400464}
465
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000466bool Process_setPriority(Process* this, int priority) {
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300467 CRT_dropPrivileges();
Michael Kleinab3a7c22015-12-07 20:10:09 +0100468 int old_prio = getpriority(PRIO_PROCESS, this->pid);
469 int err = setpriority(PRIO_PROCESS, this->pid, priority);
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300470 CRT_restorePrivileges();
Michael Kleinab3a7c22015-12-07 20:10:09 +0100471 if (err == 0 && old_prio != getpriority(PRIO_PROCESS, this->pid)) {
472 this->nice = priority;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000473 }
Michael Kleinab3a7c22015-12-07 20:10:09 +0100474 return (err == 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000475}
476
Nathan Scott500fb282020-08-20 09:35:24 +1000477bool Process_changePriorityBy(Process* this, Arg delta) {
478 return Process_setPriority(this, this->nice + delta.i);
Hisham Muhammad47e881f2012-10-04 23:59:45 +0000479}
480
Nathan Scott500fb282020-08-20 09:35:24 +1000481bool Process_sendSignal(Process* this, Arg sgn) {
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300482 CRT_dropPrivileges();
Nathan Scott500fb282020-08-20 09:35:24 +1000483 bool ok = (kill(this->pid, sgn.i) == 0);
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300484 CRT_restorePrivileges();
Nathan Scott500fb282020-08-20 09:35:24 +1000485 return ok;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000486}
487
Hisham Muhammad78d09f92014-04-25 19:41:23 -0300488long Process_pidCompare(const void* v1, const void* v2) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200489 const Process* p1 = (const Process*)v1;
490 const Process* p2 = (const Process*)v2;
Christian Göttsche2d231d72020-12-08 22:37:15 +0100491
492 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000493}
494
Hisham Muhammad78d09f92014-04-25 19:41:23 -0300495long Process_compare(const void* v1, const void* v2) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200496 const Process *p1, *p2;
497 const Settings *settings = ((const Process*)v1)->settings;
Christian Göttsche397b5c42020-11-04 17:46:24 +0100498 int r;
499
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200500 if (settings->direction == 1) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200501 p1 = (const Process*)v1;
502 p2 = (const Process*)v2;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000503 } else {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200504 p2 = (const Process*)v1;
505 p1 = (const Process*)v2;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000506 }
Christian Göttsche397b5c42020-11-04 17:46:24 +0100507
Hisham Muhammade8c69942020-12-17 19:08:56 -0300508 ProcessField key = settings->sortKey;
509 if (settings->treeView && settings->treeViewAlwaysByPID) {
510 key = PID;
511 }
512
513 switch (key) {
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000514 case PERCENT_CPU:
Christian Göttsche15eab202020-10-30 17:02:20 +0100515 case PERCENT_NORM_CPU:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100516 return SPACESHIP_NUMBER(p2->percent_cpu, p1->percent_cpu);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000517 case PERCENT_MEM:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100518 return SPACESHIP_NUMBER(p2->m_resident, p1->m_resident);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000519 case COMM:
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530520 return SPACESHIP_NULLSTR(Process_getCommand(p1), Process_getCommand(p2));
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300521 case MAJFLT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100522 return SPACESHIP_NUMBER(p2->majflt, p1->majflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300523 case MINFLT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100524 return SPACESHIP_NUMBER(p2->minflt, p1->minflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300525 case M_RESIDENT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100526 return SPACESHIP_NUMBER(p2->m_resident, p1->m_resident);
Christian Göttschefa002c02020-11-20 17:09:34 +0100527 case M_VIRT:
528 return SPACESHIP_NUMBER(p2->m_virt, p1->m_virt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300529 case NICE:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100530 return SPACESHIP_NUMBER(p1->nice, p2->nice);
Hisham Muhammadd357c672007-05-21 19:10:53 +0000531 case NLWP:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100532 return SPACESHIP_NUMBER(p1->nlwp, p2->nlwp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300533 case PGRP:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100534 return SPACESHIP_NUMBER(p1->pgrp, p2->pgrp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300535 case PID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100536 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300537 case PPID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100538 return SPACESHIP_NUMBER(p1->ppid, p2->ppid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300539 case PRIORITY:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100540 return SPACESHIP_NUMBER(p1->priority, p2->priority);
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300541 case PROCESSOR:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100542 return SPACESHIP_NUMBER(p1->processor, p2->processor);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300543 case SESSION:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100544 return SPACESHIP_NUMBER(p1->session, p2->session);
545 case STARTTIME:
546 r = SPACESHIP_NUMBER(p1->starttime_ctime, p2->starttime_ctime);
547 return r != 0 ? r : SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300548 case STATE:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100549 return SPACESHIP_NUMBER(Process_sortState(p1->state), Process_sortState(p2->state));
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300550 case ST_UID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100551 return SPACESHIP_NUMBER(p1->st_uid, p2->st_uid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300552 case TIME:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100553 return SPACESHIP_NUMBER(p2->time, p1->time);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300554 case TGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100555 return SPACESHIP_NUMBER(p1->tgid, p2->tgid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300556 case TPGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100557 return SPACESHIP_NUMBER(p1->tpgid, p2->tpgid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300558 case TTY_NR:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100559 return SPACESHIP_NUMBER(p1->tty_nr, p2->tty_nr);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300560 case USER:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100561 return SPACESHIP_NULLSTR(p1->user, p2->user);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000562 default:
Hisham Muhammad3d1703f2020-12-17 20:09:55 -0300563 return Process_compareByKey(p1, p2, key);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000564 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000565}