blob: a70d391cd17efdf65f00d45abe08c2a60aad6a2a [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>
mayurdahibhate1b74dfe2021-04-29 20:42:43 +053020#include <string.h>
Benny Baumann0f526292020-09-19 13:55:23 +020021#include <time.h>
22#include <unistd.h>
23#include <sys/resource.h>
Explorer0935129712018-12-30 12:18:27 +080024
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000025#include "CRT.h"
Christian Göttsche7cf52772020-11-18 14:26:30 +010026#include "Macros.h"
Hisham Muhammadb4f6b112014-11-27 20:10:23 -020027#include "Platform.h"
Christian Göttsche7cf52772020-11-18 14:26:30 +010028#include "ProcessList.h"
Benny Baumann0f526292020-09-19 13:55:23 +020029#include "RichString.h"
30#include "Settings.h"
Benny Baumann872e5422020-10-14 20:21:09 +020031#include "XUtils.h"
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000032
Benny Baumann0f526292020-09-19 13:55:23 +020033#if defined(MAJOR_IN_MKDEV)
Kang-Che Sung (宋岡哲)c01f40e2018-02-26 21:15:05 +080034#include <sys/mkdev.h>
Kang-Che Sung (宋岡哲)c01f40e2018-02-26 21:15:05 +080035#endif
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000036
Benny Baumann0f526292020-09-19 13:55:23 +020037
Benny Baumannaa8552b2021-04-18 19:25:56 +020038/* Used to identify kernel threads in Comm and Exe columns */
39static const char *const kthreadID = "KTHREAD";
40
Christian Göttsche42073ba2020-11-04 17:46:04 +010041static uid_t Process_getuid = (uid_t)-1;
Hisham Muhammadeb2803c2006-07-12 01:35:59 +000042
Christian Göttsche9f68c8d2020-12-15 19:44:52 +010043int Process_pidDigits = 7;
Hisham Muhammad94280102015-08-20 00:32:47 -030044
45void Process_setupColumnWidths() {
46 int maxPid = Platform_getMaxPid();
Benny Baumann45869512020-11-01 01:09:51 +010047 if (maxPid == -1)
48 return;
49
Christian Göttsche9f68c8d2020-12-15 19:44:52 +010050 Process_pidDigits = ceil(log10(maxPid));
51 assert(Process_pidDigits <= PROCESS_MAX_PID_DIGITS);
Hisham Muhammad94280102015-08-20 00:32:47 -030052}
53
Christian Göttscheb41e4d92021-04-14 20:16:16 +020054void Process_printBytes(RichString* str, unsigned long long number, bool coloring) {
55 char buffer[16];
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000056 int len;
Daniel Flanagandd334442019-10-31 11:39:12 -050057
Christian Göttschefdaa15b2021-01-27 15:11:48 +010058 int largeNumberColor = coloring ? CRT_colors[LARGE_NUMBER] : CRT_colors[PROCESS];
59 int processMegabytesColor = coloring ? CRT_colors[PROCESS_MEGABYTES] : CRT_colors[PROCESS];
60 int processGigabytesColor = coloring ? CRT_colors[PROCESS_GIGABYTES] : CRT_colors[PROCESS];
61 int shadowColor = coloring ? CRT_colors[PROCESS_SHADOW] : CRT_colors[PROCESS];
Hisham Muhammada939cdf2014-04-24 15:00:09 -030062 int processColor = CRT_colors[PROCESS];
Daniel Flanagandd334442019-10-31 11:39:12 -050063
Christian Göttschefdaa15b2021-01-27 15:11:48 +010064 if (number == ULLONG_MAX) {
65 //Invalid number
66 RichString_appendAscii(str, shadowColor, " N/A ");
Christian Göttscheb41e4d92021-04-14 20:16:16 +020067 return;
68 }
69
70 number /= ONE_K;
71
72 if (number < 1000) {
Benny Baumann40441dc2020-09-13 23:50:24 +020073 //Plain number, no markings
Daniel Lange7899ae22020-11-28 17:57:51 +010074 len = xSnprintf(buffer, sizeof(buffer), "%5llu ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010075 RichString_appendnAscii(str, processColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020076 } else if (number < 100000) {
77 //2 digit MB, 3 digit KB
Daniel Lange7899ae22020-11-28 17:57:51 +010078 len = xSnprintf(buffer, sizeof(buffer), "%2llu", number/1000);
Christian Göttsche157086e2020-12-04 14:44:57 +010079 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000080 number %= 1000;
Daniel Lange7899ae22020-11-28 17:57:51 +010081 len = xSnprintf(buffer, sizeof(buffer), "%03llu ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010082 RichString_appendnAscii(str, processColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020083 } else if (number < 1000 * ONE_K) {
84 //3 digit MB
85 number /= ONE_K;
Daniel Lange7899ae22020-11-28 17:57:51 +010086 len = xSnprintf(buffer, sizeof(buffer), "%4lluM ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010087 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020088 } else if (number < 10000 * ONE_K) {
89 //1 digit GB, 3 digit MB
90 number /= ONE_K;
Daniel Lange7899ae22020-11-28 17:57:51 +010091 len = xSnprintf(buffer, sizeof(buffer), "%1llu", number/1000);
Christian Göttsche157086e2020-12-04 14:44:57 +010092 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020093 number %= 1000;
Daniel Lange7899ae22020-11-28 17:57:51 +010094 len = xSnprintf(buffer, sizeof(buffer), "%03lluM ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +010095 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
Benny Baumanne0e59972020-09-20 19:54:53 +020096 } else if (number < 100000 * ONE_K) {
Benny Baumann40441dc2020-09-13 23:50:24 +020097 //2 digit GB, 1 digit MB
98 number /= 100 * ONE_K;
Daniel Lange7899ae22020-11-28 17:57:51 +010099 len = xSnprintf(buffer, sizeof(buffer), "%2llu", number/10);
Christian Göttsche157086e2020-12-04 14:44:57 +0100100 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +0200101 number %= 10;
Christian Göttschecd305b42020-11-29 14:14:46 +0100102 len = xSnprintf(buffer, sizeof(buffer), ".%1llu", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100103 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
104 RichString_appendAscii(str, processGigabytesColor, "G ");
Benny Baumann40441dc2020-09-13 23:50:24 +0200105 } else if (number < 1000 * ONE_M) {
106 //3 digit GB
107 number /= ONE_M;
Daniel Lange7899ae22020-11-28 17:57:51 +0100108 len = xSnprintf(buffer, sizeof(buffer), "%4lluG ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100109 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Michael Wittenab3171d2020-09-29 14:04:22 +0000110 } else if (number < 10000ULL * ONE_M) {
Benny Baumann40441dc2020-09-13 23:50:24 +0200111 //1 digit TB, 3 digit GB
112 number /= ONE_M;
Daniel Lange7899ae22020-11-28 17:57:51 +0100113 len = xSnprintf(buffer, sizeof(buffer), "%1llu", number/1000);
Christian Göttsche157086e2020-12-04 14:44:57 +0100114 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +0200115 number %= 1000;
Daniel Lange7899ae22020-11-28 17:57:51 +0100116 len = xSnprintf(buffer, sizeof(buffer), "%03lluG ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100117 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200118 } else if (number < 100000 * ONE_M) {
119 //2 digit TB, 1 digit GB
120 number /= 100 * ONE_M;
121 len = xSnprintf(buffer, sizeof(buffer), "%2llu", number/10);
122 RichString_appendnAscii(str, largeNumberColor, buffer, len);
123 number %= 10;
124 len = xSnprintf(buffer, sizeof(buffer), ".%1llu", number);
125 RichString_appendnAscii(str, processGigabytesColor, buffer, len);
126 RichString_appendAscii(str, largeNumberColor, "T ");
127 } else if (number < 10000ULL * ONE_G) {
128 //3 digit TB or 1 digit PB, 3 digit TB
129 number /= ONE_G;
130 len = xSnprintf(buffer, sizeof(buffer), "%4lluT ", number);
131 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +0200132 } else {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200133 //2 digit PB and above
134 len = xSnprintf(buffer, sizeof(buffer), "%4.1lfP ", (double)number/ONE_T);
Christian Göttsche157086e2020-12-04 14:44:57 +0100135 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000136 }
137}
138
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200139void Process_printKBytes(RichString* str, unsigned long long number, bool coloring) {
140 if (number == ULLONG_MAX)
141 Process_printBytes(str, ULLONG_MAX, coloring);
142 else
143 Process_printBytes(str, number * ONE_K, coloring);
144}
145
146void Process_printCount(RichString* str, unsigned long long number, bool coloring) {
Daniel Lange7899ae22020-11-28 17:57:51 +0100147 char buffer[13];
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300148
Christian Göttschefee744a2021-01-27 15:11:46 +0100149 int largeNumberColor = coloring ? CRT_colors[LARGE_NUMBER] : CRT_colors[PROCESS];
150 int processMegabytesColor = coloring ? CRT_colors[PROCESS_MEGABYTES] : CRT_colors[PROCESS];
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300151 int processColor = CRT_colors[PROCESS];
Christian Göttschefee744a2021-01-27 15:11:46 +0100152 int processShadowColor = coloring ? CRT_colors[PROCESS_SHADOW] : CRT_colors[PROCESS];
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300153
Michael Wittenab3171d2020-09-29 14:04:22 +0000154 if (number == ULLONG_MAX) {
Christian Göttsche157086e2020-12-04 14:44:57 +0100155 RichString_appendAscii(str, CRT_colors[PROCESS_SHADOW], " N/A ");
adrien1018536941f2018-12-30 20:18:35 +0800156 } else if (number >= 100000LL * ONE_DECIMAL_T) {
Benny Baumann0b29e502020-11-28 17:42:02 +0100157 xSnprintf(buffer, sizeof(buffer), "%11llu ", number / ONE_DECIMAL_G);
Christian Göttsche157086e2020-12-04 14:44:57 +0100158 RichString_appendnAscii(str, largeNumberColor, buffer, 12);
adrien1018536941f2018-12-30 20:18:35 +0800159 } else if (number >= 100LL * ONE_DECIMAL_T) {
Benny Baumann0b29e502020-11-28 17:42:02 +0100160 xSnprintf(buffer, sizeof(buffer), "%11llu ", number / ONE_DECIMAL_M);
Christian Göttsche157086e2020-12-04 14:44:57 +0100161 RichString_appendnAscii(str, largeNumberColor, buffer, 8);
162 RichString_appendnAscii(str, processMegabytesColor, buffer+8, 4);
adrien1018536941f2018-12-30 20:18:35 +0800163 } else if (number >= 10LL * ONE_DECIMAL_G) {
Benny Baumann0b29e502020-11-28 17:42:02 +0100164 xSnprintf(buffer, sizeof(buffer), "%11llu ", number / ONE_DECIMAL_K);
Christian Göttsche157086e2020-12-04 14:44:57 +0100165 RichString_appendnAscii(str, largeNumberColor, buffer, 5);
166 RichString_appendnAscii(str, processMegabytesColor, buffer+5, 3);
167 RichString_appendnAscii(str, processColor, buffer+8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000168 } else {
Benny Baumann0b29e502020-11-28 17:42:02 +0100169 xSnprintf(buffer, sizeof(buffer), "%11llu ", number);
Christian Göttsche157086e2020-12-04 14:44:57 +0100170 RichString_appendnAscii(str, largeNumberColor, buffer, 2);
171 RichString_appendnAscii(str, processMegabytesColor, buffer+2, 3);
172 RichString_appendnAscii(str, processColor, buffer+5, 3);
173 RichString_appendnAscii(str, processShadowColor, buffer+8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000174 }
175}
176
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200177void Process_printTime(RichString* str, unsigned long long totalHundredths, bool coloring) {
178 char buffer[10];
179 int len;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000180
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200181 unsigned long long totalSeconds = totalHundredths / 100;
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300182 unsigned long long hours = totalSeconds / 3600;
David Zarzyckif3d9eca2021-04-10 08:02:59 -0400183 unsigned long long days = totalSeconds / 86400;
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300184 int minutes = (totalSeconds / 60) % 60;
185 int seconds = totalSeconds % 60;
186 int hundredths = totalHundredths - (totalSeconds * 100);
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200187
188 int yearColor = coloring ? CRT_colors[LARGE_NUMBER] : CRT_colors[PROCESS];
189 int dayColor = coloring ? CRT_colors[PROCESS_GIGABYTES] : CRT_colors[PROCESS];
190 int hourColor = coloring ? CRT_colors[PROCESS_MEGABYTES] : CRT_colors[PROCESS];
191 int defColor = CRT_colors[PROCESS];
192
193 if (days >= /* Ignore leapyears */365) {
194 int years = days / 365;
195 int daysLeft = days - 365 * years;
196
197 if (daysLeft >= 100) {
198 len = xSnprintf(buffer, sizeof(buffer), "%3dy", years);
199 RichString_appendnAscii(str, yearColor, buffer, len);
200 len = xSnprintf(buffer, sizeof(buffer), "%3dd ", daysLeft);
201 RichString_appendnAscii(str, dayColor, buffer, len);
202 } else if (daysLeft >= 10) {
203 len = xSnprintf(buffer, sizeof(buffer), "%4dy", years);
204 RichString_appendnAscii(str, yearColor, buffer, len);
205 len = xSnprintf(buffer, sizeof(buffer), "%2dd ", daysLeft);
206 RichString_appendnAscii(str, dayColor, buffer, len);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000207 } else {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200208 len = xSnprintf(buffer, sizeof(buffer), "%5dy", years);
209 RichString_appendnAscii(str, yearColor, buffer, len);
210 len = xSnprintf(buffer, sizeof(buffer), "%1dd ", daysLeft);
211 RichString_appendnAscii(str, dayColor, buffer, len);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000212 }
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200213 } else if (days >= 100) {
214 int hoursLeft = hours - days * 24;
215
216 if (hoursLeft >= 10) {
217 len = xSnprintf(buffer, sizeof(buffer), "%4llud", days);
218 RichString_appendnAscii(str, dayColor, buffer, len);
219 len = xSnprintf(buffer, sizeof(buffer), "%2dh ", hoursLeft);
220 RichString_appendnAscii(str, hourColor, buffer, len);
221 } else {
222 len = xSnprintf(buffer, sizeof(buffer), "%5llud", days);
223 RichString_appendnAscii(str, dayColor, buffer, len);
224 len = xSnprintf(buffer, sizeof(buffer), "%1dh ", hoursLeft);
225 RichString_appendnAscii(str, hourColor, buffer, len);
226 }
227 } else if (hours >= 100) {
228 int minutesLeft = totalSeconds / 60 - hours * 60;
229
230 if (minutesLeft >= 10) {
231 len = xSnprintf(buffer, sizeof(buffer), "%4lluh", hours);
232 RichString_appendnAscii(str, hourColor, buffer, len);
233 len = xSnprintf(buffer, sizeof(buffer), "%2dm ", minutesLeft);
234 RichString_appendnAscii(str, defColor, buffer, len);
235 } else {
236 len = xSnprintf(buffer, sizeof(buffer), "%5lluh", hours);
237 RichString_appendnAscii(str, hourColor, buffer, len);
238 len = xSnprintf(buffer, sizeof(buffer), "%1dm ", minutesLeft);
239 RichString_appendnAscii(str, defColor, buffer, len);
240 }
241 } else if (hours > 0) {
242 len = xSnprintf(buffer, sizeof(buffer), "%2lluh", hours);
243 RichString_appendnAscii(str, hourColor, buffer, len);
244 len = xSnprintf(buffer, sizeof(buffer), "%02d:%02d ", minutes, seconds);
245 RichString_appendnAscii(str, defColor, buffer, len);
246 } else {
247 len = xSnprintf(buffer, sizeof(buffer), "%2d:%02d.%02d ", minutes, seconds, hundredths);
248 RichString_appendnAscii(str, defColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000249 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000250}
251
Christian Göttschea63cfc82020-10-13 14:26:40 +0200252void Process_fillStarttimeBuffer(Process* this) {
253 struct tm date;
254 (void) localtime_r(&this->starttime_ctime, &date);
255 strftime(this->starttime_show, sizeof(this->starttime_show) - 1, (this->starttime_ctime > (time(NULL) - 86400)) ? "%R " : "%b%d ", &date);
256}
257
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200258/*
259 * TASK_COMM_LEN is defined to be 16 for /proc/[pid]/comm in man proc(5), but it is
260 * not available in an userspace header - so define it.
261 *
262 * Note: This is taken from LINUX headers, but implicitly taken for other platforms
263 * for sake of brevity.
264 *
265 * Note: when colorizing a basename with the comm prefix, the entire basename
266 * (not just the comm prefix) is colorized for better readability, and it is
267 * implicit that only upto (TASK_COMM_LEN - 1) could be comm.
268 */
269#define TASK_COMM_LEN 16
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200270
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200271static bool findCommInCmdline(const char *comm, const char *cmdline, int cmdlineBasenameStart, int *pCommStart, int *pCommEnd) {
272 /* Try to find procComm in tokenized cmdline - this might in rare cases
273 * mis-identify a string or fail, if comm or cmdline had been unsuitably
274 * modified by the process */
275 const char *tokenBase;
276 size_t tokenLen;
277 const size_t commLen = strlen(comm);
278
279 if (cmdlineBasenameStart < 0)
280 return false;
281
282 for (const char *token = cmdline + cmdlineBasenameStart; *token;) {
283 for (tokenBase = token; *token && *token != '\n'; ++token) {
284 if (*token == '/') {
285 tokenBase = token + 1;
Hisham Muhammadf2a190b2014-02-27 17:11:23 -0300286 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000287 }
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200288 tokenLen = token - tokenBase;
289
290 if ((tokenLen == commLen || (tokenLen > commLen && commLen == (TASK_COMM_LEN - 1))) &&
291 strncmp(tokenBase, comm, commLen) == 0) {
292 *pCommStart = tokenBase - cmdline;
293 *pCommEnd = token - cmdline;
294 return true;
295 }
296
297 if (*token) {
298 do {
299 ++token;
300 } while (*token && '\n' == *token);
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200301 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000302 }
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200303 return false;
304}
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200305
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200306static int matchCmdlinePrefixWithExeSuffix(const char *cmdline, int cmdlineBaseOffset, const char *exe, int exeBaseOffset, int exeBaseLen) {
307 int matchLen; /* matching length to be returned */
308 char delim; /* delimiter following basename */
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200309
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200310 /* cmdline prefix is an absolute path: it must match whole exe. */
311 if (cmdline[0] == '/') {
312 matchLen = exeBaseLen + exeBaseOffset;
313 if (strncmp(cmdline, exe, matchLen) == 0) {
314 delim = cmdline[matchLen];
315 if (delim == 0 || delim == '\n' || delim == ' ') {
316 return matchLen;
317 }
318 }
319 return 0;
320 }
321
322 /* cmdline prefix is a relative path: We need to first match the basename at
323 * cmdlineBaseOffset and then reverse match the cmdline prefix with the exe
324 * suffix. But there is a catch: Some processes modify their cmdline in ways
325 * that make htop's identification of the basename in cmdline unreliable.
326 * For e.g. /usr/libexec/gdm-session-worker modifies its cmdline to
327 * "gdm-session-worker [pam/gdm-autologin]" and htop ends up with
328 * proccmdlineBasenameEnd at "gdm-autologin]". This issue could arise with
329 * chrome as well as it stores in cmdline its concatenated argument vector,
330 * without NUL delimiter between the arguments (which may contain a '/')
331 *
332 * So if needed, we adjust cmdlineBaseOffset to the previous (if any)
333 * component of the cmdline relative path, and retry the procedure. */
334 bool delimFound; /* if valid basename delimiter found */
335 do {
336 /* match basename */
337 matchLen = exeBaseLen + cmdlineBaseOffset;
338 if (cmdlineBaseOffset < exeBaseOffset &&
339 strncmp(cmdline + cmdlineBaseOffset, exe + exeBaseOffset, exeBaseLen) == 0) {
340 delim = cmdline[matchLen];
341 if (delim == 0 || delim == '\n' || delim == ' ') {
342 int i, j;
343 /* reverse match the cmdline prefix and exe suffix */
344 for (i = cmdlineBaseOffset - 1, j = exeBaseOffset - 1;
345 i >= 0 && j >= 0 && cmdline[i] == exe[j]; --i, --j)
346 ;
347
348 /* full match, with exe suffix being a valid relative path */
349 if (i < 0 && j >= 0 && exe[j] == '/')
350 return matchLen;
351 }
352 }
353
354 /* Try to find the previous potential cmdlineBaseOffset - it would be
355 * preceded by '/' or nothing, and delimited by ' ' or '\n' */
356 for (delimFound = false, cmdlineBaseOffset -= 2; cmdlineBaseOffset > 0; --cmdlineBaseOffset) {
357 if (delimFound) {
358 if (cmdline[cmdlineBaseOffset - 1] == '/') {
359 break;
360 }
361 } else if (cmdline[cmdlineBaseOffset] == ' ' || cmdline[cmdlineBaseOffset] == '\n') {
362 delimFound = true;
363 }
364 }
365 } while (delimFound);
366
367 return 0;
368}
369
370/* stpcpy, but also converts newlines to spaces */
371static inline char *stpcpyWithNewlineConversion(char *dstStr, const char *srcStr) {
372 for (; *srcStr; ++srcStr) {
373 *dstStr++ = (*srcStr == '\n') ? ' ' : *srcStr;
374 }
375 *dstStr = 0;
376 return dstStr;
377}
378
379/*
380 * This function makes the merged Command string. It also stores the offsets of the
381 * basename, comm w.r.t the merged Command string - these offsets will be used by
382 * Process_writeCommand() for coloring. The merged Command string is also
383 * returned by Process_getCommandStr() for searching, sorting and filtering.
384 */
385void Process_makeCommandStr(Process *this) {
386 ProcessMergedCommand *mc = &this->mergedCommand;
387 const Settings *settings = this->settings;
388
389 bool showMergedCommand = settings->showMergedCommand;
390 bool showProgramPath = settings->showProgramPath;
391 bool searchCommInCmdline = settings->findCommInCmdline;
392 bool stripExeFromCmdline = settings->stripExeFromCmdline;
393
Benny Baumanna61a2e62021-04-18 18:10:04 +0200394 /* Nothing to do to (Re)Generate the Command string, if the process is:
395 * - a kernel thread, or
396 * - a zombie from before being under htop's watch, or
397 * - a user thread and showThreadNames is not set */
398 if (Process_isKernelThread(this))
399 return;
400 if (this->state == 'Z' && !this->mergedCommand.str)
401 return;
402 if (Process_isUserlandThread(this) && settings->showThreadNames)
403 return;
404
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200405 /* this->mergedCommand.str needs updating only if its state or contents changed.
406 * Its content is based on the fields cmdline, comm, and exe. */
407 if (
408 mc->prevMergeSet == showMergedCommand &&
409 mc->prevPathSet == showProgramPath &&
410 mc->prevCommSet == searchCommInCmdline &&
411 mc->prevCmdlineSet == stripExeFromCmdline &&
412 !mc->cmdlineChanged &&
413 !mc->commChanged &&
414 !mc->exeChanged
415 ) {
416 return;
417 }
418
419 /* The field separtor "│" has been chosen such that it will not match any
420 * valid string used for searching or filtering */
421 const char *SEPARATOR = CRT_treeStr[TREE_STR_VERT];
422 const int SEPARATOR_LEN = strlen(SEPARATOR);
423
424 /* Check for any changed fields since we last built this string */
425 if (mc->cmdlineChanged || mc->commChanged || mc->exeChanged) {
426 free(mc->str);
427 /* Accommodate the column text, two field separators and terminating NUL */
Benny Baumann7ef58f22021-05-17 23:15:24 +0200428 size_t maxLen = 2 * SEPARATOR_LEN + 1;
429 maxLen += this->cmdline ? strlen(this->cmdline) : strlen("(zombie)");
430 maxLen += this->procComm ? strlen(this->procComm) : 0;
431 maxLen += this->procExe ? strlen(this->procExe) : 0;
432
433 mc->str = xCalloc(1, maxLen);
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200434 }
435
436 /* Preserve the settings used in this run */
437 mc->prevMergeSet = showMergedCommand;
438 mc->prevPathSet = showProgramPath;
439 mc->prevCommSet = searchCommInCmdline;
440 mc->prevCmdlineSet = stripExeFromCmdline;
441
442 /* Mark everything as unchanged */
443 mc->cmdlineChanged = false;
444 mc->commChanged = false;
445 mc->exeChanged = false;
446
447 /* Reset all locations that need extra handling when actually displaying */
448 mc->highlightCount = 0;
449 memset(mc->highlights, 0, sizeof(mc->highlights));
450
451 size_t mbMismatch = 0;
Benny Baumann9a781552021-05-15 21:54:46 +0200452 #define WRITE_HIGHLIGHT(_offset, _length, _attr, _flags) \
453 do { \
454 /* Check if we still have capacity */ \
455 assert(mc->highlightCount < ARRAYSIZE(mc->highlights)); \
456 if (mc->highlightCount >= ARRAYSIZE(mc->highlights)) \
457 continue; \
458 \
459 mc->highlights[mc->highlightCount].offset = str - strStart + (_offset) - mbMismatch; \
460 mc->highlights[mc->highlightCount].length = _length; \
461 mc->highlights[mc->highlightCount].attr = _attr; \
462 mc->highlights[mc->highlightCount].flags = _flags; \
463 mc->highlightCount++; \
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200464 } while (0)
465
Benny Baumann9a781552021-05-15 21:54:46 +0200466 #define WRITE_SEPARATOR \
467 do { \
468 WRITE_HIGHLIGHT(0, 1, CRT_colors[FAILED_READ], CMDLINE_HIGHLIGHT_FLAG_SEPARATOR); \
469 mbMismatch += SEPARATOR_LEN - 1; \
470 str = stpcpy(str, SEPARATOR); \
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200471 } while (0)
472
473 const int baseAttr = Process_isThread(this) ? CRT_colors[PROCESS_THREAD_BASENAME] : CRT_colors[PROCESS_BASENAME];
474 const int commAttr = Process_isThread(this) ? CRT_colors[PROCESS_THREAD_COMM] : CRT_colors[PROCESS_COMM];
475 const int delAttr = CRT_colors[FAILED_READ];
476
477 /* Establish some shortcuts to data we need */
478 const char *cmdline = this->cmdline;
479 const char *procComm = this->procComm;
480 const char *procExe = this->procExe;
481
482 char *strStart = mc->str;
483 char *str = strStart;
484
485 int cmdlineBasenameStart = this->cmdlineBasenameStart;
486 int cmdlineBasenameEnd = this->cmdlineBasenameEnd;
487
488 if (!cmdline) {
489 cmdlineBasenameStart = 0;
490 cmdlineBasenameEnd = 0;
491 cmdline = "(zombie)";
492 }
493
494 assert(cmdlineBasenameStart >= 0);
495 assert(cmdlineBasenameStart <= (int)strlen(cmdline));
496
497 if (!showMergedCommand || !procExe || !procComm) { /* fall back to cmdline */
498 if (showMergedCommand && !procExe && procComm && strlen(procComm)) { /* Prefix column with comm */
499 if (strncmp(cmdline + cmdlineBasenameStart, procComm, MINIMUM(TASK_COMM_LEN - 1, strlen(procComm))) != 0) {
500 WRITE_HIGHLIGHT(0, strlen(procComm), commAttr, CMDLINE_HIGHLIGHT_FLAG_COMM);
501 str = stpcpy(str, procComm);
502
503 WRITE_SEPARATOR;
504 }
505 }
506
Benny Baumann2824e292021-05-15 21:55:14 +0200507 if (cmdlineBasenameEnd > cmdlineBasenameStart)
508 WRITE_HIGHLIGHT(showProgramPath ? cmdlineBasenameStart : 0, cmdlineBasenameEnd - cmdlineBasenameStart, baseAttr, CMDLINE_HIGHLIGHT_FLAG_BASENAME);
509 (void)stpcpyWithNewlineConversion(str, cmdline + (showProgramPath ? 0 : cmdlineBasenameStart));
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200510
511 return;
512 }
513
514 int exeLen = strlen(this->procExe);
515 int exeBasenameOffset = this->procExeBasenameOffset;
516 int exeBasenameLen = exeLen - exeBasenameOffset;
517
518 assert(exeBasenameOffset >= 0);
519 assert(exeBasenameOffset <= (int)strlen(procExe));
520
521 bool haveCommInExe = false;
522 if (procExe && procComm) {
523 haveCommInExe = strncmp(procExe + exeBasenameOffset, procComm, TASK_COMM_LEN - 1) == 0;
524 }
525
526 /* Start with copying exe */
527 if (showProgramPath) {
528 if (haveCommInExe)
529 WRITE_HIGHLIGHT(exeBasenameOffset, exeBasenameLen, commAttr, CMDLINE_HIGHLIGHT_FLAG_COMM);
530 WRITE_HIGHLIGHT(exeBasenameOffset, exeBasenameLen, baseAttr, CMDLINE_HIGHLIGHT_FLAG_BASENAME);
531 if (this->procExeDeleted)
532 WRITE_HIGHLIGHT(exeBasenameOffset, exeBasenameLen, delAttr, CMDLINE_HIGHLIGHT_FLAG_DELETED);
533 str = stpcpy(str, procExe);
534 } else {
535 if (haveCommInExe)
536 WRITE_HIGHLIGHT(0, exeBasenameLen, commAttr, CMDLINE_HIGHLIGHT_FLAG_COMM);
537 WRITE_HIGHLIGHT(0, exeBasenameLen, baseAttr, CMDLINE_HIGHLIGHT_FLAG_BASENAME);
538 if (this->procExeDeleted)
539 WRITE_HIGHLIGHT(0, exeBasenameLen, delAttr, CMDLINE_HIGHLIGHT_FLAG_DELETED);
540 str = stpcpy(str, procExe + exeBasenameOffset);
541 }
542
543 bool haveCommInCmdline = false;
544 int commStart = 0;
545 int commEnd = 0;
546
547 /* Try to match procComm with procExe's basename: This is reliable (predictable) */
548 if (searchCommInCmdline) {
549 /* commStart/commEnd will be adjusted later along with cmdline */
550 haveCommInCmdline = findCommInCmdline(procComm, cmdline, cmdlineBasenameStart, &commStart, &commEnd);
551 }
552
553 int matchLen = matchCmdlinePrefixWithExeSuffix(cmdline, cmdlineBasenameStart, procExe, exeBasenameOffset, exeBasenameLen);
554
555 bool haveCommField = false;
556
557 if (!haveCommInExe && !haveCommInCmdline && procComm) {
558 WRITE_SEPARATOR;
559 WRITE_HIGHLIGHT(0, strlen(procComm), commAttr, CMDLINE_HIGHLIGHT_FLAG_COMM);
560 str = stpcpy(str, procComm);
561 haveCommField = true;
562 }
563
564 if (matchLen) {
565 /* strip the matched exe prefix */
566 cmdline += matchLen;
567
568 commStart -= matchLen;
569 commEnd -= matchLen;
570 }
571
572 if (!matchLen || (haveCommField && *cmdline)) {
573 /* cmdline will be a separate field */
574 WRITE_SEPARATOR;
575 }
576
577 if (!haveCommInExe && haveCommInCmdline && !haveCommField)
578 WRITE_HIGHLIGHT(commStart, commEnd - commStart, commAttr, CMDLINE_HIGHLIGHT_FLAG_COMM);
579
580 /* Display cmdline if it hasn't been consumed by procExe */
581 if (*cmdline)
582 (void)stpcpyWithNewlineConversion(str, cmdline);
583
584 #undef WRITE_SEPARATOR
585 #undef WRITE_HIGHLIGHT
586}
587
588void Process_writeCommand(const Process* this, int attr, int baseAttr, RichString* str) {
589 (void)baseAttr;
590
591 const ProcessMergedCommand *mc = &this->mergedCommand;
592
593 int strStart = RichString_size(str);
594
595 const bool highlightBaseName = this->settings->highlightBaseName;
596 const bool highlightSeparator = true;
Benny Baumannbf07c712020-12-19 16:46:00 +0100597 const bool highlightDeleted = this->settings->highlightDeletedExe;
Benny Baumannbcb18ef2021-04-10 13:31:39 +0200598
599 if (!this->mergedCommand.str) {
600 int len = 0;
601 const char* cmdline = this->cmdline;
602
603 if (highlightBaseName || !this->settings->showProgramPath) {
604 int basename = 0;
605 for (int i = 0; i < this->cmdlineBasenameEnd; i++) {
606 if (cmdline[i] == '/') {
607 basename = i + 1;
608 } else if (cmdline[i] == ':') {
609 len = i + 1;
610 break;
611 }
612 }
613 if (len == 0) {
614 if (this->settings->showProgramPath) {
615 strStart += basename;
616 } else {
617 cmdline += basename;
618 }
619 len = this->cmdlineBasenameEnd - basename;
620 }
621 }
622
623 RichString_appendWide(str, attr, cmdline);
624
625 if (this->settings->highlightBaseName) {
626 RichString_setAttrn(str, baseAttr, strStart, len);
627 }
628
629 return;
630 }
631
632 RichString_appendWide(str, attr, this->mergedCommand.str);
633
634 for (size_t i = 0, hlCount = CLAMP(mc->highlightCount, 0, ARRAYSIZE(mc->highlights)); i < hlCount; i++) {
635 const ProcessCmdlineHighlight *hl = &mc->highlights[i];
636
637 if (!hl->length)
638 continue;
639
640 if (hl->flags & CMDLINE_HIGHLIGHT_FLAG_SEPARATOR)
641 if (!highlightSeparator)
642 continue;
643
644 if (hl->flags & CMDLINE_HIGHLIGHT_FLAG_BASENAME)
645 if (!highlightBaseName)
646 continue;
647
648 if (hl->flags & CMDLINE_HIGHLIGHT_FLAG_DELETED)
649 if (!highlightDeleted)
650 continue;
651
652 RichString_setAttrn(str, hl->attr, strStart + hl->offset, hl->length);
Benny Baumann45869512020-11-01 01:09:51 +0100653 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000654}
655
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200656void Process_printRate(RichString* str, double rate, bool coloring) {
657 char buffer[16];
658
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200659 int largeNumberColor = CRT_colors[LARGE_NUMBER];
660 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
661 int processColor = CRT_colors[PROCESS];
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200662 int shadowColor = CRT_colors[PROCESS_SHADOW];
Benny Baumann0b29e502020-11-28 17:42:02 +0100663
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200664 if (!coloring) {
665 largeNumberColor = CRT_colors[PROCESS];
666 processMegabytesColor = CRT_colors[PROCESS];
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000667 }
Benny Baumann0b29e502020-11-28 17:42:02 +0100668
Benny Baumann29ec1152020-09-07 11:53:58 +0200669 if (isnan(rate)) {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200670 RichString_appendAscii(str, shadowColor, " N/A ");
671 } else if (rate < 0.005) {
672 int len = snprintf(buffer, sizeof(buffer), "%7.2f B/s ", rate);
673 RichString_appendnAscii(str, shadowColor, buffer, len);
Hisham797bcd02016-02-20 02:22:57 -0200674 } else if (rate < ONE_K) {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200675 int len = snprintf(buffer, sizeof(buffer), "%7.2f B/s ", rate);
Christian Göttsche157086e2020-12-04 14:44:57 +0100676 RichString_appendnAscii(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800677 } else if (rate < ONE_M) {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200678 int len = snprintf(buffer, sizeof(buffer), "%7.2f K/s ", rate / ONE_K);
Christian Göttsche157086e2020-12-04 14:44:57 +0100679 RichString_appendnAscii(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800680 } else if (rate < ONE_G) {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200681 int len = snprintf(buffer, sizeof(buffer), "%7.2f M/s ", rate / ONE_M);
Christian Göttsche157086e2020-12-04 14:44:57 +0100682 RichString_appendnAscii(str, processMegabytesColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800683 } else if (rate < ONE_T) {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200684 int len = snprintf(buffer, sizeof(buffer), "%7.2f G/s ", rate / ONE_G);
685 RichString_appendnAscii(str, largeNumberColor, buffer, len);
686 } else if (rate < ONE_P) {
687 int len = snprintf(buffer, sizeof(buffer), "%7.2f T/s ", rate / ONE_T);
Christian Göttsche157086e2020-12-04 14:44:57 +0100688 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200689 } else {
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200690 int len = snprintf(buffer, sizeof(buffer), "%7.2f P/s ", rate / ONE_P);
Christian Göttsche157086e2020-12-04 14:44:57 +0100691 RichString_appendnAscii(str, largeNumberColor, buffer, len);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200692 }
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000693}
694
Christian Göttschea5db1392021-01-10 15:57:46 +0100695void Process_printLeftAlignedField(RichString* str, int attr, const char* content, unsigned int width) {
Christian Göttsche08ac22d2021-01-14 09:59:11 +0100696 int columns = width;
697 RichString_appendnWideColumns(str, attr, content, strlen(content), &columns);
698 RichString_appendChr(str, attr, ' ', width + 1 - columns);
Christian Göttschea5db1392021-01-10 15:57:46 +0100699}
700
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200701void Process_writeField(const Process* this, RichString* str, ProcessField field) {
Christian Göttschefee744a2021-01-27 15:11:46 +0100702 char buffer[256];
703 size_t n = sizeof(buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000704 int attr = CRT_colors[DEFAULT_COLOR];
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200705 bool coloring = this->settings->highlightMegabytes;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000706
707 switch (field) {
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000708 case COMM: {
Christian Göttschefee744a2021-01-27 15:11:46 +0100709 int baseattr = CRT_colors[PROCESS_BASENAME];
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200710 if (this->settings->highlightThreads && Process_isThread(this)) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000711 attr = CRT_colors[PROCESS_THREAD];
712 baseattr = CRT_colors[PROCESS_THREAD_BASENAME];
713 }
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200714 if (!this->settings->treeView || this->indent == 0) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000715 Process_writeCommand(this, attr, baseattr, str);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000716 return;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000717 }
Christian Göttschefee744a2021-01-27 15:11:46 +0100718
719 char* buf = buffer;
720 int maxIndent = 0;
721 bool lastItem = (this->indent < 0);
722 int indent = (this->indent < 0 ? -this->indent : this->indent);
723
724 for (int i = 0; i < 32; i++) {
725 if (indent & (1U << i)) {
726 maxIndent = i+1;
727 }
728 }
729
730 for (int i = 0; i < maxIndent - 1; i++) {
731 int written, ret;
732 if (indent & (1 << i)) {
733 ret = xSnprintf(buf, n, "%s ", CRT_treeStr[TREE_STR_VERT]);
734 } else {
735 ret = xSnprintf(buf, n, " ");
736 }
737 if (ret < 0 || (size_t)ret >= n) {
738 written = n;
739 } else {
740 written = ret;
741 }
742 buf += written;
743 n -= written;
744 }
745
746 const char* draw = CRT_treeStr[lastItem ? TREE_STR_BEND : TREE_STR_RTEE];
747 xSnprintf(buf, n, "%s%s ", draw, this->showChildren ? CRT_treeStr[TREE_STR_SHUT] : CRT_treeStr[TREE_STR_OPEN] );
748 RichString_appendWide(str, CRT_colors[PROCESS_TREE], buffer);
749 Process_writeCommand(this, attr, baseattr, str);
750 return;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000751 }
Benny Baumannaa8552b2021-04-18 19:25:56 +0200752 case PROC_COMM: {
753 const char* procComm;
754 if (this->procComm) {
755 attr = CRT_colors[Process_isUserlandThread(this) ? PROCESS_THREAD_COMM : PROCESS_COMM];
756 procComm = this->procComm;
757 } else {
758 attr = CRT_colors[PROCESS_SHADOW];
759 procComm = Process_isKernelThread(this) ? kthreadID : "N/A";
760 }
761
762 Process_printLeftAlignedField(str, attr, procComm, TASK_COMM_LEN - 1);
763 return;
764 }
765 case PROC_EXE: {
766 const char* procExe;
767 if (this->procExe) {
768 attr = CRT_colors[Process_isUserlandThread(this) ? PROCESS_THREAD_BASENAME : PROCESS_BASENAME];
Benny Baumannbf07c712020-12-19 16:46:00 +0100769 if (this->procExeDeleted && this->settings->highlightDeletedExe)
Benny Baumannaa8552b2021-04-18 19:25:56 +0200770 attr = CRT_colors[FAILED_READ];
771 procExe = this->procExe + this->procExeBasenameOffset;
772 } else {
773 attr = CRT_colors[PROCESS_SHADOW];
774 procExe = Process_isKernelThread(this) ? kthreadID : "N/A";
775 }
776
777 Process_printLeftAlignedField(str, attr, procExe, TASK_COMM_LEN - 1);
778 return;
779 }
Christian Göttsche550a1412021-05-02 13:29:39 +0200780 case ELAPSED: Process_printTime(str, /* convert to hundreds of a second */ this->processList->realtimeMs / 10 - 100 * this->starttime_ctime, coloring); return;
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200781 case MAJFLT: Process_printCount(str, this->majflt, coloring); return;
782 case MINFLT: Process_printCount(str, this->minflt, coloring); return;
783 case M_RESIDENT: Process_printKBytes(str, this->m_resident, coloring); return;
784 case M_VIRT: Process_printKBytes(str, this->m_virt, coloring); return;
Christian Göttschefee744a2021-01-27 15:11:46 +0100785 case NICE:
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300786 xSnprintf(buffer, n, "%3ld ", this->nice);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300787 attr = this->nice < 0 ? CRT_colors[PROCESS_HIGH_PRIORITY]
788 : this->nice > 0 ? CRT_colors[PROCESS_LOW_PRIORITY]
Christian Göttschefee744a2021-01-27 15:11:46 +0100789 : CRT_colors[PROCESS_SHADOW];
790 break;
791 case NLWP:
792 if (this->nlwp == 1)
793 attr = CRT_colors[PROCESS_SHADOW];
794
795 xSnprintf(buffer, n, "%4ld ", this->nlwp);
796 break;
797 case PERCENT_CPU:
798 case PERCENT_NORM_CPU: {
799 float cpuPercentage = this->percent_cpu;
800 if (field == PERCENT_NORM_CPU) {
801 cpuPercentage /= this->processList->cpuCount;
802 }
Christian Göttsche979aca92021-03-02 21:59:56 +0100803 if (cpuPercentage > 999.9F) {
Christian Göttschefee744a2021-01-27 15:11:46 +0100804 xSnprintf(buffer, n, "%4u ", (unsigned int)cpuPercentage);
Christian Göttsche979aca92021-03-02 21:59:56 +0100805 } else if (cpuPercentage > 99.9F) {
Christian Göttschefee744a2021-01-27 15:11:46 +0100806 xSnprintf(buffer, n, "%3u. ", (unsigned int)cpuPercentage);
807 } else {
Christian Göttsche979aca92021-03-02 21:59:56 +0100808 if (cpuPercentage < 0.05F)
Christian Göttschefee744a2021-01-27 15:11:46 +0100809 attr = CRT_colors[PROCESS_SHADOW];
810
811 xSnprintf(buffer, n, "%4.1f ", cpuPercentage);
812 }
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300813 break;
814 }
Christian Göttschefee744a2021-01-27 15:11:46 +0100815 case PERCENT_MEM:
Christian Göttsche979aca92021-03-02 21:59:56 +0100816 if (this->percent_mem > 99.9F) {
Christian Göttschefee744a2021-01-27 15:11:46 +0100817 xSnprintf(buffer, n, "100. ");
818 } else {
Christian Göttsche979aca92021-03-02 21:59:56 +0100819 if (this->percent_mem < 0.05F)
Christian Göttschefee744a2021-01-27 15:11:46 +0100820 attr = CRT_colors[PROCESS_SHADOW];
821
822 xSnprintf(buffer, n, "%4.1f ", this->percent_mem);
823 }
824 break;
Christian Göttsche9f68c8d2020-12-15 19:44:52 +0100825 case PGRP: xSnprintf(buffer, n, "%*d ", Process_pidDigits, this->pgrp); break;
826 case PID: xSnprintf(buffer, n, "%*d ", Process_pidDigits, this->pid); break;
827 case PPID: xSnprintf(buffer, n, "%*d ", Process_pidDigits, this->ppid); break;
Christian Göttschefee744a2021-01-27 15:11:46 +0100828 case PRIORITY:
829 if (this->priority <= -100)
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300830 xSnprintf(buffer, n, " RT ");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300831 else
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300832 xSnprintf(buffer, n, "%3ld ", this->priority);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300833 break;
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300834 case PROCESSOR: xSnprintf(buffer, n, "%3d ", Settings_cpuId(this->settings, this->processor)); break;
Christian Göttsche9f68c8d2020-12-15 19:44:52 +0100835 case SESSION: xSnprintf(buffer, n, "%*d ", Process_pidDigits, this->session); break;
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300836 case STARTTIME: xSnprintf(buffer, n, "%s", this->starttime_show); break;
Christian Göttschefee744a2021-01-27 15:11:46 +0100837 case STATE:
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300838 xSnprintf(buffer, n, "%c ", this->state);
Christian Göttschefee744a2021-01-27 15:11:46 +0100839 switch (this->state) {
840 case 'R':
841 attr = CRT_colors[PROCESS_R_STATE];
842 break;
843 case 'D':
844 attr = CRT_colors[PROCESS_D_STATE];
845 break;
846 case 'I':
847 case 'S':
848 attr = CRT_colors[PROCESS_SHADOW];
849 break;
Valmiky Arquissandas64e0d942014-10-14 02:30:17 +0100850 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000851 break;
Daniel Langec34be412018-10-07 11:16:12 +0200852 case ST_UID: xSnprintf(buffer, n, "%5d ", this->st_uid); break;
Christian Göttscheb41e4d92021-04-14 20:16:16 +0200853 case TIME: Process_printTime(str, this->time, coloring); return;
Christian Göttschefee744a2021-01-27 15:11:46 +0100854 case TGID:
855 if (this->tgid == this->pid)
856 attr = CRT_colors[PROCESS_SHADOW];
857
858 xSnprintf(buffer, n, "%*d ", Process_pidDigits, this->tgid);
859 break;
Christian Göttsche9f68c8d2020-12-15 19:44:52 +0100860 case TPGID: xSnprintf(buffer, n, "%*d ", Process_pidDigits, this->tpgid); break;
Christian Göttsche9a822152021-03-21 19:40:56 +0100861 case TTY:
862 if (!this->tty_name) {
Christian Göttschea3c82852021-01-27 15:11:42 +0100863 attr = CRT_colors[PROCESS_SHADOW];
Christian Göttsche9a822152021-03-21 19:40:56 +0100864 xSnprintf(buffer, n, "(no tty) ");
Christian Göttschea3c82852021-01-27 15:11:42 +0100865 } else {
Christian Göttsche9a822152021-03-21 19:40:56 +0100866 const char* name = String_startsWith(this->tty_name, "/dev/") ? (this->tty_name + strlen("/dev/")) : this->tty_name;
867 xSnprintf(buffer, n, "%-8s ", name);
Christian Göttschea3c82852021-01-27 15:11:42 +0100868 }
869 break;
Christian Göttschefee744a2021-01-27 15:11:46 +0100870 case USER:
Christian Göttsche42073ba2020-11-04 17:46:04 +0100871 if (Process_getuid != this->st_uid)
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000872 attr = CRT_colors[PROCESS_SHADOW];
Christian Göttsche5359eae2020-12-23 12:21:29 +0100873
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000874 if (this->user) {
Christian Göttschea5db1392021-01-10 15:57:46 +0100875 Process_printLeftAlignedField(str, attr, this->user, 9);
Christian Göttsche5359eae2020-12-23 12:21:29 +0100876 return;
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000877 }
Christian Göttsche5359eae2020-12-23 12:21:29 +0100878
879 xSnprintf(buffer, n, "%-9d ", this->st_uid);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000880 break;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000881 default:
Christian Göttsche615fc932021-04-18 15:52:28 +0200882 assert(0 && "Process_writeField: default key reached"); /* should never be reached */
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300883 xSnprintf(buffer, n, "- ");
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000884 }
Christian Göttschea2be57d2021-04-14 20:54:38 +0200885 RichString_appendAscii(str, attr, buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000886}
887
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200888void Process_display(const Object* cast, RichString* out) {
889 const Process* this = (const Process*) cast;
890 const ProcessField* fields = this->settings->fields;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000891 for (int i = 0; fields[i]; i++)
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300892 As_Process(this)->writeField(this, out, fields[i]);
Benny Baumann45869512020-11-01 01:09:51 +0100893
Christian Göttsche42073ba2020-11-04 17:46:04 +0100894 if (this->settings->shadowOtherUsers && this->st_uid != Process_getuid) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000895 RichString_setAttr(out, CRT_colors[PROCESS_SHADOW]);
Benny Baumann45869512020-11-01 01:09:51 +0100896 }
897
898 if (this->tag == true) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000899 RichString_setAttr(out, CRT_colors[PROCESS_TAG]);
Benny Baumann45869512020-11-01 01:09:51 +0100900 }
901
Daniel Lange09510902020-11-16 12:13:47 +0100902 if (this->settings->highlightChanges) {
903 if (Process_isTomb(this)) {
904 out->highlightAttr = CRT_colors[PROCESS_TOMB];
905 } else if (Process_isNew(this)) {
906 out->highlightAttr = CRT_colors[PROCESS_NEW];
907 }
908 }
909
Christian Göttsche3f99c2d2021-03-12 16:46:04 +0100910 assert(RichString_size(out) > 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000911}
912
Hisham Muhammad6f868b02015-02-20 14:52:10 -0200913void Process_done(Process* this) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000914 assert (this != NULL);
Benny Baumann02431c42020-12-19 16:21:08 +0100915 free(this->cmdline);
Benny Baumannd74e8b72021-01-30 15:31:59 +0100916 free(this->procComm);
917 free(this->procExe);
Benny Baumanncdb660a2021-04-10 11:10:50 +0200918 free(this->mergedCommand.str);
Christian Göttsche9a822152021-03-21 19:40:56 +0100919 free(this->tty_name);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000920}
921
Benny Baumannc0d02022021-04-24 12:06:49 +0200922/* This function returns the string displayed in Command column, so that sorting
923 * happens on what is displayed - whether comm, full path, basename, etc.. So
924 * this follows Process_writeField(COMM) and Process_writeCommand */
925const char *Process_getCommandStr(const Process *this) {
926 if ((Process_isUserlandThread(this) && this->settings->showThreadNames) || !this->mergedCommand.str) {
927 return this->cmdline;
928 }
929
930 return this->mergedCommand.str;
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530931}
932
Christian Göttscheba282cf2020-10-05 13:19:50 +0200933const ProcessClass Process_class = {
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300934 .super = {
935 .extends = Class(Object),
936 .display = Process_display,
937 .delete = Process_delete,
938 .compare = Process_compare
939 },
940 .writeField = Process_writeField,
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530941 .getCommandStr = Process_getCommandStr,
Hisham Muhammad00b324b2012-12-05 15:12:20 +0000942};
943
Christian Göttsche3035e292021-01-26 18:41:04 +0100944void Process_init(Process* this, const Settings* settings) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200945 this->settings = settings;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000946 this->tag = false;
Hisham Muhammad9eb91212010-06-17 19:02:03 +0000947 this->showChildren = true;
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000948 this->show = true;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000949 this->updated = false;
Benny Baumann94a52cb2021-04-10 11:46:57 +0200950 this->cmdlineBasenameEnd = -1;
Benny Baumann45869512020-11-01 01:09:51 +0100951
Christian Göttsche42073ba2020-11-04 17:46:04 +0100952 if (Process_getuid == (uid_t)-1) {
Benny Baumann45869512020-11-01 01:09:51 +0100953 Process_getuid = getuid();
954 }
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000955}
956
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000957void Process_toggleTag(Process* this) {
Christian Göttsche2d231d72020-12-08 22:37:15 +0100958 this->tag = !this->tag;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000959}
960
Adam Saponaradde71c62020-10-30 21:56:16 -0400961bool Process_isNew(const Process* this) {
Adam Saponaraa83f5152020-10-31 20:36:53 -0400962 assert(this->processList);
Nathan Scott356488a2021-03-30 15:55:48 +1100963 if (this->processList->monotonicMs >= this->seenStampMs) {
964 return this->processList->monotonicMs - this->seenStampMs <= 1000 * (uint64_t)this->processList->settings->highlightDelaySecs;
Daniel Lange09510902020-11-16 12:13:47 +0100965 }
Adam Saponaradde71c62020-10-30 21:56:16 -0400966 return false;
967}
968
969bool Process_isTomb(const Process* this) {
Nathan Scott356488a2021-03-30 15:55:48 +1100970 return this->tombStampMs > 0;
Adam Saponaradde71c62020-10-30 21:56:16 -0400971}
972
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000973bool Process_setPriority(Process* this, int priority) {
Christian Göttsche36880cd2021-01-21 20:27:37 +0100974 if (Settings_isReadonly())
975 return false;
976
Michael Kleinab3a7c22015-12-07 20:10:09 +0100977 int old_prio = getpriority(PRIO_PROCESS, this->pid);
978 int err = setpriority(PRIO_PROCESS, this->pid, priority);
Benny Baumann82157f52021-02-16 19:44:59 +0100979
Michael Kleinab3a7c22015-12-07 20:10:09 +0100980 if (err == 0 && old_prio != getpriority(PRIO_PROCESS, this->pid)) {
981 this->nice = priority;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000982 }
Michael Kleinab3a7c22015-12-07 20:10:09 +0100983 return (err == 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000984}
985
Nathan Scott500fb282020-08-20 09:35:24 +1000986bool Process_changePriorityBy(Process* this, Arg delta) {
987 return Process_setPriority(this, this->nice + delta.i);
Hisham Muhammad47e881f2012-10-04 23:59:45 +0000988}
989
Nathan Scott500fb282020-08-20 09:35:24 +1000990bool Process_sendSignal(Process* this, Arg sgn) {
Benny Baumann82157f52021-02-16 19:44:59 +0100991 return kill(this->pid, sgn.i) == 0;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000992}
993
Christian Göttsche90ea3ac2020-12-23 13:02:32 +0100994int Process_pidCompare(const void* v1, const void* v2) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200995 const Process* p1 = (const Process*)v1;
996 const Process* p2 = (const Process*)v2;
Christian Göttsche2d231d72020-12-08 22:37:15 +0100997
998 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000999}
1000
Christian Göttsche90ea3ac2020-12-23 13:02:32 +01001001int Process_compare(const void* v1, const void* v2) {
Daniel Lange074703b2021-01-21 20:57:34 +01001002 const Process *p1 = (const Process*)v1;
1003 const Process *p2 = (const Process*)v2;
Christian Göttsche397b5c42020-11-04 17:46:24 +01001004
Daniel Lange074703b2021-01-21 20:57:34 +01001005 const Settings *settings = p1->settings;
Christian Göttsche397b5c42020-11-04 17:46:24 +01001006
Hisham Muhammad23272602020-12-18 11:03:31 -03001007 ProcessField key = Settings_getActiveSortKey(settings);
Hisham Muhammade8c69942020-12-17 19:08:56 -03001008
Christian Göttsche90ea3ac2020-12-23 13:02:32 +01001009 int result = Process_compareByKey(p1, p2, key);
Benny Baumann77db2402020-12-18 22:12:26 +01001010
1011 // Implement tie-breaker (needed to make tree mode more stable)
1012 if (!result)
Daniel Lange074703b2021-01-21 20:57:34 +01001013 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Benny Baumann77db2402020-12-18 22:12:26 +01001014
Christian Göttschebea7f8e2021-03-12 16:44:46 +01001015 return (Settings_getActiveDirection(settings) == 1) ? result : -result;
Benny Baumann77db2402020-12-18 22:12:26 +01001016}
1017
Christian Göttsched5de1bc2021-01-27 15:11:44 +01001018static uint8_t stateCompareValue(char state) {
1019 switch (state) {
1020
1021 case 'S':
1022 return 10;
1023
1024 case 'I':
1025 return 9;
1026
1027 case 'X':
1028 return 8;
1029
1030 case 'Z':
1031 return 7;
1032
1033 case 't':
1034 return 6;
1035
1036 case 'T':
1037 return 5;
1038
1039 case 'L':
1040 return 4;
1041
1042 case 'D':
1043 return 3;
1044
1045 case 'R':
1046 return 2;
1047
1048 case '?':
1049 return 1;
1050
1051 default:
1052 return 0;
1053 }
1054}
1055
Christian Göttsche90ea3ac2020-12-23 13:02:32 +01001056int Process_compareByKey_Base(const Process* p1, const Process* p2, ProcessField key) {
Benny Baumann77db2402020-12-18 22:12:26 +01001057 int r;
1058
Christian Göttsche89473cc2020-12-15 19:44:48 +01001059 switch (key) {
Hisham Muhammad5d48ab82006-07-11 06:13:32 +00001060 case PERCENT_CPU:
Christian Göttsche15eab202020-10-30 17:02:20 +01001061 case PERCENT_NORM_CPU:
Daniel Lange4531b312021-01-21 14:27:23 +01001062 return SPACESHIP_NUMBER(p1->percent_cpu, p2->percent_cpu);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +00001063 case PERCENT_MEM:
Daniel Lange4531b312021-01-21 14:27:23 +01001064 return SPACESHIP_NUMBER(p1->m_resident, p2->m_resident);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +00001065 case COMM:
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +05301066 return SPACESHIP_NULLSTR(Process_getCommand(p1), Process_getCommand(p2));
Benny Baumannaa8552b2021-04-18 19:25:56 +02001067 case PROC_COMM: {
1068 const char *comm1 = p1->procComm ? p1->procComm : (Process_isKernelThread(p1) ? kthreadID : "");
1069 const char *comm2 = p2->procComm ? p2->procComm : (Process_isKernelThread(p2) ? kthreadID : "");
1070 return SPACESHIP_NULLSTR(comm1, comm2);
1071 }
1072 case PROC_EXE: {
1073 const char *exe1 = p1->procExe ? (p1->procExe + p1->procExeBasenameOffset) : (Process_isKernelThread(p1) ? kthreadID : "");
1074 const char *exe2 = p2->procExe ? (p2->procExe + p2->procExeBasenameOffset) : (Process_isKernelThread(p2) ? kthreadID : "");
1075 return SPACESHIP_NULLSTR(exe1, exe2);
1076 }
Christian Göttsche550a1412021-05-02 13:29:39 +02001077 case ELAPSED:
1078 r = -SPACESHIP_NUMBER(p1->starttime_ctime, p2->starttime_ctime);
1079 return r != 0 ? r : SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001080 case MAJFLT:
Daniel Lange4531b312021-01-21 14:27:23 +01001081 return SPACESHIP_NUMBER(p1->majflt, p2->majflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001082 case MINFLT:
Daniel Lange4531b312021-01-21 14:27:23 +01001083 return SPACESHIP_NUMBER(p1->minflt, p2->minflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001084 case M_RESIDENT:
Daniel Lange4531b312021-01-21 14:27:23 +01001085 return SPACESHIP_NUMBER(p1->m_resident, p2->m_resident);
Christian Göttschefa002c02020-11-20 17:09:34 +01001086 case M_VIRT:
Daniel Lange4531b312021-01-21 14:27:23 +01001087 return SPACESHIP_NUMBER(p1->m_virt, p2->m_virt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001088 case NICE:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001089 return SPACESHIP_NUMBER(p1->nice, p2->nice);
Hisham Muhammadd357c672007-05-21 19:10:53 +00001090 case NLWP:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001091 return SPACESHIP_NUMBER(p1->nlwp, p2->nlwp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001092 case PGRP:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001093 return SPACESHIP_NUMBER(p1->pgrp, p2->pgrp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001094 case PID:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001095 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001096 case PPID:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001097 return SPACESHIP_NUMBER(p1->ppid, p2->ppid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001098 case PRIORITY:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001099 return SPACESHIP_NUMBER(p1->priority, p2->priority);
Hisham Muhammad272e2d92015-03-16 23:01:48 -03001100 case PROCESSOR:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001101 return SPACESHIP_NUMBER(p1->processor, p2->processor);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001102 case SESSION:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001103 return SPACESHIP_NUMBER(p1->session, p2->session);
1104 case STARTTIME:
1105 r = SPACESHIP_NUMBER(p1->starttime_ctime, p2->starttime_ctime);
1106 return r != 0 ? r : SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001107 case STATE:
Christian Göttsched5de1bc2021-01-27 15:11:44 +01001108 return SPACESHIP_NUMBER(stateCompareValue(p1->state), stateCompareValue(p2->state));
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001109 case ST_UID:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001110 return SPACESHIP_NUMBER(p1->st_uid, p2->st_uid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001111 case TIME:
Daniel Lange4531b312021-01-21 14:27:23 +01001112 return SPACESHIP_NUMBER(p1->time, p2->time);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001113 case TGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001114 return SPACESHIP_NUMBER(p1->tgid, p2->tgid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001115 case TPGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001116 return SPACESHIP_NUMBER(p1->tpgid, p2->tpgid);
Christian Göttsche9a822152021-03-21 19:40:56 +01001117 case TTY:
1118 /* Order no tty last */
1119 return SPACESHIP_DEFAULTSTR(p1->tty_name, p2->tty_name, "\x7F");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -03001120 case USER:
Christian Göttsche397b5c42020-11-04 17:46:24 +01001121 return SPACESHIP_NULLSTR(p1->user, p2->user);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +00001122 default:
Christian Göttsche615fc932021-04-18 15:52:28 +02001123 assert(0 && "Process_compareByKey_Base: default key reached"); /* should never be reached */
Benny Baumann77db2402020-12-18 22:12:26 +01001124 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +00001125 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +00001126}
Christian Göttsche05fb6812021-05-18 22:29:25 +02001127
1128void Process_updateComm(Process* this, const char* comm) {
1129 if (!this->procComm && !comm)
1130 return;
1131
1132 if (this->procComm && comm && String_eq(this->procComm, comm))
1133 return;
1134
1135 free(this->procComm);
1136 this->procComm = comm ? xStrdup(comm) : NULL;
1137 this->mergedCommand.commChanged = true;
1138}
1139
1140static int skipPotentialPath(const char* cmdline, int end) {
1141 if (cmdline[0] != '/')
1142 return 0;
1143
1144 int slash = 0;
1145 for (int i = 1; i < end; i++) {
1146 if (cmdline[i] == '/' && cmdline[i+1] != '\0') {
1147 slash = i + 1;
1148 continue;
1149 }
1150
1151 if (cmdline[i] == ' ' && cmdline[i-1] != '\\')
1152 return slash;
1153
1154 if (cmdline[i] == ':' && cmdline[i+1] == ' ')
1155 return slash;
1156 }
1157
1158 return slash;
1159}
1160
1161void Process_updateCmdline(Process* this, const char* cmdline, int basenameStart, int basenameEnd) {
1162 assert(basenameStart >= 0);
1163 assert((cmdline && basenameStart < (int)strlen(cmdline)) || (!cmdline && basenameStart == 0));
Christian Göttschec408add2021-05-23 15:53:23 +02001164 assert((basenameEnd > basenameStart) || (basenameEnd == 0 && basenameStart == 0));
Christian Göttsche05fb6812021-05-18 22:29:25 +02001165 assert((cmdline && basenameEnd <= (int)strlen(cmdline)) || (!cmdline && basenameEnd == 0));
1166
1167 if (!this->cmdline && !cmdline)
1168 return;
1169
1170 if (this->cmdline && cmdline && String_eq(this->cmdline, cmdline))
1171 return;
1172
1173 free(this->cmdline);
1174 this->cmdline = cmdline ? xStrdup(cmdline) : NULL;
1175 this->cmdlineBasenameStart = (basenameStart || !cmdline) ? basenameStart : skipPotentialPath(cmdline, basenameEnd);
1176 this->cmdlineBasenameEnd = basenameEnd;
1177 this->mergedCommand.cmdlineChanged = true;
1178}
1179
1180void Process_updateExe(Process* this, const char* exe) {
1181 if (!this->procExe && !exe)
1182 return;
1183
1184 if (this->procExe && exe && String_eq(this->procExe, exe))
1185 return;
1186
1187 free(this->procExe);
1188 if (exe) {
1189 this->procExe = xStrdup(exe);
1190 const char* lastSlash = strrchr(exe, '/');
1191 this->procExeBasenameOffset = (lastSlash && *(lastSlash + 1) != '\0' && lastSlash != exe) ? (lastSlash - exe + 1) : 0;
1192 } else {
1193 this->procExe = NULL;
1194 this->procExeBasenameOffset = 0;
1195 }
1196 this->mergedCommand.exeChanged = true;
1197}