blob: 4556d966fd1e6e7f146b1323eae6bb08f0a722ac [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) {
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000061 char buffer[11];
62 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
Michael Wittenab3171d2020-09-29 14:04:22 +000076 len = snprintf(buffer, 10, "%5llu ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +020077 RichString_appendn(str, processColor, buffer, len);
78 } else if (number < 100000) {
79 //2 digit MB, 3 digit KB
Michael Wittenab3171d2020-09-29 14:04:22 +000080 len = snprintf(buffer, 10, "%2llu", number/1000);
Hisham Muhammada939cdf2014-04-24 15:00:09 -030081 RichString_appendn(str, processMegabytesColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000082 number %= 1000;
Michael Wittenab3171d2020-09-29 14:04:22 +000083 len = snprintf(buffer, 10, "%03llu ", number);
Hisham Muhammada939cdf2014-04-24 15:00:09 -030084 RichString_appendn(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;
Michael Wittenab3171d2020-09-29 14:04:22 +000088 len = snprintf(buffer, 10, "%4lluM ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +020089 RichString_appendn(str, processMegabytesColor, buffer, len);
90 } else if (number < 10000 * ONE_K) {
91 //1 digit GB, 3 digit MB
92 number /= ONE_K;
Michael Wittenab3171d2020-09-29 14:04:22 +000093 len = snprintf(buffer, 10, "%1llu", number/1000);
Benny Baumann40441dc2020-09-13 23:50:24 +020094 RichString_appendn(str, processGigabytesColor, buffer, len);
95 number %= 1000;
Michael Wittenab3171d2020-09-29 14:04:22 +000096 len = snprintf(buffer, 10, "%03lluM ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +020097 RichString_appendn(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;
Michael Wittenab3171d2020-09-29 14:04:22 +0000101 len = snprintf(buffer, 10, "%2llu", number/10);
Benny Baumann40441dc2020-09-13 23:50:24 +0200102 RichString_appendn(str, processGigabytesColor, buffer, len);
103 number %= 10;
Michael Wittenab3171d2020-09-29 14:04:22 +0000104 len = snprintf(buffer, 10, ".%1lluG ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +0200105 RichString_appendn(str, processMegabytesColor, buffer, len);
106 } else if (number < 1000 * ONE_M) {
107 //3 digit GB
108 number /= ONE_M;
Michael Wittenab3171d2020-09-29 14:04:22 +0000109 len = snprintf(buffer, 10, "%4lluG ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +0200110 RichString_appendn(str, processGigabytesColor, buffer, len);
Michael Wittenab3171d2020-09-29 14:04:22 +0000111 } else if (number < 10000ULL * ONE_M) {
Benny Baumann40441dc2020-09-13 23:50:24 +0200112 //1 digit TB, 3 digit GB
113 number /= ONE_M;
Michael Wittenab3171d2020-09-29 14:04:22 +0000114 len = snprintf(buffer, 10, "%1llu", number/1000);
Benny Baumann40441dc2020-09-13 23:50:24 +0200115 RichString_appendn(str, largeNumberColor, buffer, len);
116 number %= 1000;
Michael Wittenab3171d2020-09-29 14:04:22 +0000117 len = snprintf(buffer, 10, "%03lluG ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +0200118 RichString_appendn(str, processGigabytesColor, buffer, len);
119 } else {
120 //2 digit TB and above
121 len = snprintf(buffer, 10, "%4.1lfT ", (double)number/ONE_G);
122 RichString_appendn(str, largeNumberColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000123 }
124}
125
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300126void Process_colorNumber(RichString* str, unsigned long long number, bool coloring) {
Hisham Muhammad9b351402011-05-26 16:31:18 +0000127 char buffer[14];
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300128
129 int largeNumberColor = CRT_colors[LARGE_NUMBER];
130 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
131 int processColor = CRT_colors[PROCESS];
132 int processShadowColor = CRT_colors[PROCESS_SHADOW];
133 if (!coloring) {
134 largeNumberColor = CRT_colors[PROCESS];
135 processMegabytesColor = CRT_colors[PROCESS];
136 processShadowColor = CRT_colors[PROCESS];
137 }
138
Michael Wittenab3171d2020-09-29 14:04:22 +0000139 if (number == ULLONG_MAX) {
Hisham Muhammade940aec2017-07-10 20:57:34 -0300140 int len = snprintf(buffer, 13, " no perm ");
141 RichString_appendn(str, CRT_colors[PROCESS_SHADOW], buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800142 } else if (number >= 100000LL * ONE_DECIMAL_T) {
143 xSnprintf(buffer, 13, "%11llu ", number / ONE_DECIMAL_G);
144 RichString_appendn(str, largeNumberColor, buffer, 12);
145 } else if (number >= 100LL * ONE_DECIMAL_T) {
146 xSnprintf(buffer, 13, "%11llu ", number / ONE_DECIMAL_M);
adrien1018f15d55c2018-12-18 21:05:09 +0800147 RichString_appendn(str, largeNumberColor, buffer, 8);
148 RichString_appendn(str, processMegabytesColor, buffer+8, 4);
adrien1018536941f2018-12-30 20:18:35 +0800149 } else if (number >= 10LL * ONE_DECIMAL_G) {
150 xSnprintf(buffer, 13, "%11llu ", number / ONE_DECIMAL_K);
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300151 RichString_appendn(str, largeNumberColor, buffer, 5);
152 RichString_appendn(str, processMegabytesColor, buffer+5, 3);
153 RichString_appendn(str, processColor, buffer+8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000154 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300155 xSnprintf(buffer, 13, "%11llu ", number);
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300156 RichString_appendn(str, largeNumberColor, buffer, 2);
157 RichString_appendn(str, processMegabytesColor, buffer+2, 3);
158 RichString_appendn(str, processColor, buffer+5, 3);
159 RichString_appendn(str, processShadowColor, buffer+8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000160 }
161}
162
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300163void Process_printTime(RichString* str, unsigned long long totalHundredths) {
164 unsigned long long totalSeconds = totalHundredths / 100;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000165
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300166 unsigned long long hours = totalSeconds / 3600;
167 int minutes = (totalSeconds / 60) % 60;
168 int seconds = totalSeconds % 60;
169 int hundredths = totalHundredths - (totalSeconds * 100);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000170 char buffer[11];
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000171 if (hours >= 100) {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300172 xSnprintf(buffer, 10, "%7lluh ", hours);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000173 RichString_append(str, CRT_colors[LARGE_NUMBER], buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000174 } else {
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000175 if (hours) {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300176 xSnprintf(buffer, 10, "%2lluh", hours);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000177 RichString_append(str, CRT_colors[LARGE_NUMBER], buffer);
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300178 xSnprintf(buffer, 10, "%02d:%02d ", minutes, seconds);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000179 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300180 xSnprintf(buffer, 10, "%2d:%02d.%02d ", minutes, seconds, hundredths);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000181 }
182 RichString_append(str, CRT_colors[DEFAULT_COLOR], buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000183 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000184}
185
Christian Göttschea63cfc82020-10-13 14:26:40 +0200186void Process_fillStarttimeBuffer(Process* this) {
187 struct tm date;
188 (void) localtime_r(&this->starttime_ctime, &date);
189 strftime(this->starttime_show, sizeof(this->starttime_show) - 1, (this->starttime_ctime > (time(NULL) - 86400)) ? "%R " : "%b%d ", &date);
190}
191
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200192static inline void Process_writeCommand(const Process* this, int attr, int baseattr, RichString* str) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200193 int start = RichString_size(str), finish = 0;
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200194 const char* comm = this->comm;
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200195
196 if (this->settings->highlightBaseName || !this->settings->showProgramPath) {
197 int i, basename = 0;
198 for (i = 0; i < this->basenameOffset; i++) {
199 if (comm[i] == '/') {
200 basename = i + 1;
201 } else if (comm[i] == ':') {
202 finish = i + 1;
203 break;
Hisham Muhammadf2a190b2014-02-27 17:11:23 -0300204 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000205 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200206 if (!finish) {
Benny Baumann45869512020-11-01 01:09:51 +0100207 if (this->settings->showProgramPath) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200208 start += basename;
Benny Baumann45869512020-11-01 01:09:51 +0100209 } else {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200210 comm += basename;
Benny Baumann45869512020-11-01 01:09:51 +0100211 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200212 finish = this->basenameOffset - basename;
213 }
214 finish += start - 1;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000215 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200216
217 RichString_append(str, attr, comm);
218
Benny Baumann45869512020-11-01 01:09:51 +0100219 if (this->settings->highlightBaseName) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200220 RichString_setAttrn(str, baseattr, start, finish);
Benny Baumann45869512020-11-01 01:09:51 +0100221 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000222}
223
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300224void Process_outputRate(RichString* str, char* buffer, int n, double rate, int coloring) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200225 int largeNumberColor = CRT_colors[LARGE_NUMBER];
226 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
227 int processColor = CRT_colors[PROCESS];
228 if (!coloring) {
229 largeNumberColor = CRT_colors[PROCESS];
230 processMegabytesColor = CRT_colors[PROCESS];
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000231 }
Benny Baumann29ec1152020-09-07 11:53:58 +0200232 if (isnan(rate)) {
Hisham797bcd02016-02-20 02:22:57 -0200233 int len = snprintf(buffer, n, " no perm ");
234 RichString_appendn(str, CRT_colors[PROCESS_SHADOW], buffer, len);
235 } else if (rate < ONE_K) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200236 int len = snprintf(buffer, n, "%7.2f B/s ", rate);
237 RichString_appendn(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800238 } else if (rate < ONE_M) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200239 int len = snprintf(buffer, n, "%7.2f K/s ", rate / ONE_K);
240 RichString_appendn(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800241 } else if (rate < ONE_G) {
242 int len = snprintf(buffer, n, "%7.2f M/s ", rate / ONE_M);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200243 RichString_appendn(str, processMegabytesColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800244 } else if (rate < ONE_T) {
245 int len = snprintf(buffer, n, "%7.2f G/s ", rate / ONE_G);
246 RichString_appendn(str, largeNumberColor, buffer, len);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200247 } else {
adrien1018536941f2018-12-30 20:18:35 +0800248 int len = snprintf(buffer, n, "%7.2f T/s ", rate / ONE_T);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200249 RichString_appendn(str, largeNumberColor, buffer, len);
250 }
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000251}
252
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200253void Process_writeField(const Process* this, RichString* str, ProcessField field) {
Hisham Muhammad2f30cd12014-04-24 15:08:32 -0300254 char buffer[256]; buffer[255] = '\0';
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000255 int attr = CRT_colors[DEFAULT_COLOR];
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000256 int baseattr = CRT_colors[PROCESS_BASENAME];
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000257 int n = sizeof(buffer) - 1;
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200258 bool coloring = this->settings->highlightMegabytes;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000259
260 switch (field) {
Christian Göttsche15eab202020-10-30 17:02:20 +0100261 case PERCENT_CPU:
262 case PERCENT_NORM_CPU: {
263 float cpuPercentage = this->percent_cpu;
264 if (field == PERCENT_NORM_CPU) {
265 cpuPercentage /= this->processList->cpuCount;
266 }
267 if (cpuPercentage > 999.9) {
268 xSnprintf(buffer, n, "%4u ", (unsigned int)cpuPercentage);
269 } else if (cpuPercentage > 99.9) {
270 xSnprintf(buffer, n, "%3u. ", (unsigned int)cpuPercentage);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300271 } else {
Christian Göttsche15eab202020-10-30 17:02:20 +0100272 xSnprintf(buffer, n, "%4.1f ", cpuPercentage);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300273 }
274 break;
275 }
276 case PERCENT_MEM: {
277 if (this->percent_mem > 99.9) {
Daniel Flanagandd334442019-10-31 11:39:12 -0500278 xSnprintf(buffer, n, "100. ");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300279 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300280 xSnprintf(buffer, n, "%4.1f ", this->percent_mem);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300281 }
282 break;
283 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000284 case COMM: {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200285 if (this->settings->highlightThreads && Process_isThread(this)) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000286 attr = CRT_colors[PROCESS_THREAD];
287 baseattr = CRT_colors[PROCESS_THREAD_BASENAME];
288 }
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200289 if (!this->settings->treeView || this->indent == 0) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000290 Process_writeCommand(this, attr, baseattr, str);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000291 return;
292 } else {
293 char* buf = buffer;
294 int maxIndent = 0;
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000295 bool lastItem = (this->indent < 0);
296 int indent = (this->indent < 0 ? -this->indent : this->indent);
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000297
Benny Baumann45869512020-11-01 01:09:51 +0100298 for (int i = 0; i < 32; i++) {
299 if (indent & (1U << i)) {
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000300 maxIndent = i+1;
Benny Baumann45869512020-11-01 01:09:51 +0100301 }
302 }
303
Benny Baumann45cb99d2020-11-20 08:07:56 +0100304 for (int i = 0; i < maxIndent - 1; i++) {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100305 int written, ret;
Benny Baumann45869512020-11-01 01:09:51 +0100306 if (indent & (1 << i)) {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100307 ret = snprintf(buf, n, "%s ", CRT_treeStr[TREE_STR_VERT]);
Benny Baumann45869512020-11-01 01:09:51 +0100308 } else {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100309 ret = snprintf(buf, n, " ");
Benny Baumann45869512020-11-01 01:09:51 +0100310 }
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100311 if (ret < 0 || ret >= n) {
312 written = n;
313 } else {
314 written = ret;
315 }
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000316 buf += written;
317 n -= written;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000318 }
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200319 const char* draw = CRT_treeStr[lastItem ? (this->settings->direction == 1 ? TREE_STR_BEND : TREE_STR_TEND) : TREE_STR_RTEE];
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300320 xSnprintf(buf, n, "%s%s ", draw, this->showChildren ? CRT_treeStr[TREE_STR_SHUT] : CRT_treeStr[TREE_STR_OPEN] );
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000321 RichString_append(str, CRT_colors[PROCESS_TREE], buffer);
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000322 Process_writeCommand(this, attr, baseattr, str);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000323 return;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000324 }
325 }
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300326 case MAJFLT: Process_colorNumber(str, this->majflt, coloring); return;
327 case MINFLT: Process_colorNumber(str, this->minflt, coloring); return;
Christian Göttsche36187742020-10-15 22:37:02 +0200328 case M_RESIDENT: Process_humanNumber(str, this->m_resident * CRT_pageSizeKB, coloring); return;
Christian Göttschefa002c02020-11-20 17:09:34 +0100329 case M_VIRT: Process_humanNumber(str, this->m_virt * CRT_pageSizeKB, coloring); return;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300330 case NICE: {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300331 xSnprintf(buffer, n, "%3ld ", this->nice);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300332 attr = this->nice < 0 ? CRT_colors[PROCESS_HIGH_PRIORITY]
333 : this->nice > 0 ? CRT_colors[PROCESS_LOW_PRIORITY]
334 : attr;
335 break;
336 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300337 case NLWP: xSnprintf(buffer, n, "%4ld ", this->nlwp); break;
338 case PGRP: xSnprintf(buffer, n, Process_pidFormat, this->pgrp); break;
339 case PID: xSnprintf(buffer, n, Process_pidFormat, this->pid); break;
340 case PPID: xSnprintf(buffer, n, Process_pidFormat, this->ppid); break;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300341 case PRIORITY: {
Ivan Kozik07086fc2016-09-06 09:22:38 +0000342 if(this->priority <= -100)
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300343 xSnprintf(buffer, n, " RT ");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300344 else
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300345 xSnprintf(buffer, n, "%3ld ", this->priority);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300346 break;
347 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300348 case PROCESSOR: xSnprintf(buffer, n, "%3d ", Settings_cpuId(this->settings, this->processor)); break;
349 case SESSION: xSnprintf(buffer, n, Process_pidFormat, this->session); break;
350 case STARTTIME: xSnprintf(buffer, n, "%s", this->starttime_show); break;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000351 case STATE: {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300352 xSnprintf(buffer, n, "%c ", this->state);
Valmiky Arquissandas64e0d942014-10-14 02:30:17 +0100353 switch(this->state) {
354 case 'R':
355 attr = CRT_colors[PROCESS_R_STATE];
356 break;
357 case 'D':
358 attr = CRT_colors[PROCESS_D_STATE];
359 break;
Valmiky Arquissandas64e0d942014-10-14 02:30:17 +0100360 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000361 break;
362 }
Daniel Langec34be412018-10-07 11:16:12 +0200363 case ST_UID: xSnprintf(buffer, n, "%5d ", this->st_uid); break;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300364 case TIME: Process_printTime(str, this->time); return;
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300365 case TGID: xSnprintf(buffer, n, Process_pidFormat, this->tgid); break;
366 case TPGID: xSnprintf(buffer, n, Process_pidFormat, this->tpgid); break;
367 case TTY_NR: xSnprintf(buffer, n, "%3u:%3u ", major(this->tty_nr), minor(this->tty_nr)); break;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000368 case USER: {
Christian Göttsche42073ba2020-11-04 17:46:04 +0100369 if (Process_getuid != this->st_uid)
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000370 attr = CRT_colors[PROCESS_SHADOW];
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000371 if (this->user) {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300372 xSnprintf(buffer, n, "%-9s ", this->user);
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000373 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300374 xSnprintf(buffer, n, "%-9d ", this->st_uid);
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000375 }
Hisham Muhammad9b351402011-05-26 16:31:18 +0000376 if (buffer[9] != '\0') {
377 buffer[9] = ' ';
378 buffer[10] = '\0';
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000379 }
380 break;
381 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000382 default:
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300383 xSnprintf(buffer, n, "- ");
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000384 }
385 RichString_append(str, attr, buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000386}
387
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200388void Process_display(const Object* cast, RichString* out) {
389 const Process* this = (const Process*) cast;
390 const ProcessField* fields = this->settings->fields;
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000391 RichString_prune(out);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000392 for (int i = 0; fields[i]; i++)
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300393 As_Process(this)->writeField(this, out, fields[i]);
Benny Baumann45869512020-11-01 01:09:51 +0100394
Christian Göttsche42073ba2020-11-04 17:46:04 +0100395 if (this->settings->shadowOtherUsers && this->st_uid != Process_getuid) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000396 RichString_setAttr(out, CRT_colors[PROCESS_SHADOW]);
Benny Baumann45869512020-11-01 01:09:51 +0100397 }
398
399 if (this->tag == true) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000400 RichString_setAttr(out, CRT_colors[PROCESS_TAG]);
Benny Baumann45869512020-11-01 01:09:51 +0100401 }
402
Daniel Lange09510902020-11-16 12:13:47 +0100403 if (this->settings->highlightChanges) {
404 if (Process_isTomb(this)) {
405 out->highlightAttr = CRT_colors[PROCESS_TOMB];
406 } else if (Process_isNew(this)) {
407 out->highlightAttr = CRT_colors[PROCESS_NEW];
408 }
409 }
410
Hisham Muhammada9c0ea32011-03-22 20:37:08 +0000411 assert(out->chlen > 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000412}
413
Hisham Muhammad6f868b02015-02-20 14:52:10 -0200414void Process_done(Process* this) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000415 assert (this != NULL);
Hisham Muhammadf54a37b2014-05-03 17:49:05 -0300416 free(this->comm);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000417}
418
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530419static const char* Process_getCommandStr(const Process* p) {
420 return p->comm ? p->comm : "";
421}
422
Christian Göttscheba282cf2020-10-05 13:19:50 +0200423const ProcessClass Process_class = {
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300424 .super = {
425 .extends = Class(Object),
426 .display = Process_display,
427 .delete = Process_delete,
428 .compare = Process_compare
429 },
430 .writeField = Process_writeField,
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530431 .getCommandStr = Process_getCommandStr,
Hisham Muhammad00b324b2012-12-05 15:12:20 +0000432};
433
Christian Göttsche4eb44392020-10-21 21:26:05 +0200434void Process_init(Process* this, const struct Settings_* settings) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200435 this->settings = settings;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000436 this->tag = false;
Hisham Muhammad9eb91212010-06-17 19:02:03 +0000437 this->showChildren = true;
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000438 this->show = true;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000439 this->updated = false;
Hisham Muhammadcb297af2014-04-09 17:43:54 -0300440 this->basenameOffset = -1;
Benny Baumann45869512020-11-01 01:09:51 +0100441
Christian Göttsche42073ba2020-11-04 17:46:04 +0100442 if (Process_getuid == (uid_t)-1) {
Benny Baumann45869512020-11-01 01:09:51 +0100443 Process_getuid = getuid();
444 }
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000445}
446
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000447void Process_toggleTag(Process* this) {
448 this->tag = this->tag == true ? false : true;
449}
450
Adam Saponaradde71c62020-10-30 21:56:16 -0400451bool Process_isNew(const Process* this) {
Adam Saponaraa83f5152020-10-31 20:36:53 -0400452 assert(this->processList);
Daniel Lange09510902020-11-16 12:13:47 +0100453 if (this->processList->scanTs >= this->seenTs) {
Adam Saponaraa83f5152020-10-31 20:36:53 -0400454 return this->processList->scanTs - this->seenTs <= this->processList->settings->highlightDelaySecs;
Daniel Lange09510902020-11-16 12:13:47 +0100455 }
Adam Saponaradde71c62020-10-30 21:56:16 -0400456 return false;
457}
458
459bool Process_isTomb(const Process* this) {
Adam Saponaraa83f5152020-10-31 20:36:53 -0400460 return this->tombTs > 0;
Adam Saponaradde71c62020-10-30 21:56:16 -0400461}
462
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000463bool Process_setPriority(Process* this, int priority) {
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300464 CRT_dropPrivileges();
Michael Kleinab3a7c22015-12-07 20:10:09 +0100465 int old_prio = getpriority(PRIO_PROCESS, this->pid);
466 int err = setpriority(PRIO_PROCESS, this->pid, priority);
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300467 CRT_restorePrivileges();
Michael Kleinab3a7c22015-12-07 20:10:09 +0100468 if (err == 0 && old_prio != getpriority(PRIO_PROCESS, this->pid)) {
469 this->nice = priority;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000470 }
Michael Kleinab3a7c22015-12-07 20:10:09 +0100471 return (err == 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000472}
473
Nathan Scott500fb282020-08-20 09:35:24 +1000474bool Process_changePriorityBy(Process* this, Arg delta) {
475 return Process_setPriority(this, this->nice + delta.i);
Hisham Muhammad47e881f2012-10-04 23:59:45 +0000476}
477
Nathan Scott500fb282020-08-20 09:35:24 +1000478bool Process_sendSignal(Process* this, Arg sgn) {
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300479 CRT_dropPrivileges();
Nathan Scott500fb282020-08-20 09:35:24 +1000480 bool ok = (kill(this->pid, sgn.i) == 0);
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300481 CRT_restorePrivileges();
Nathan Scott500fb282020-08-20 09:35:24 +1000482 return ok;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000483}
484
Hisham Muhammad78d09f92014-04-25 19:41:23 -0300485long Process_pidCompare(const void* v1, const void* v2) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200486 const Process* p1 = (const Process*)v1;
487 const Process* p2 = (const Process*)v2;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000488 return (p1->pid - p2->pid);
489}
490
Hisham Muhammad78d09f92014-04-25 19:41:23 -0300491long Process_compare(const void* v1, const void* v2) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200492 const Process *p1, *p2;
493 const Settings *settings = ((const Process*)v1)->settings;
Christian Göttsche397b5c42020-11-04 17:46:24 +0100494 int r;
495
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200496 if (settings->direction == 1) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200497 p1 = (const Process*)v1;
498 p2 = (const Process*)v2;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000499 } else {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200500 p2 = (const Process*)v1;
501 p1 = (const Process*)v2;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000502 }
Christian Göttsche397b5c42020-11-04 17:46:24 +0100503
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200504 switch (settings->sortKey) {
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000505 case PERCENT_CPU:
Christian Göttsche15eab202020-10-30 17:02:20 +0100506 case PERCENT_NORM_CPU:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100507 return SPACESHIP_NUMBER(p2->percent_cpu, p1->percent_cpu);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000508 case PERCENT_MEM:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100509 return SPACESHIP_NUMBER(p2->m_resident, p1->m_resident);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000510 case COMM:
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +0530511 return SPACESHIP_NULLSTR(Process_getCommand(p1), Process_getCommand(p2));
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300512 case MAJFLT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100513 return SPACESHIP_NUMBER(p2->majflt, p1->majflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300514 case MINFLT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100515 return SPACESHIP_NUMBER(p2->minflt, p1->minflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300516 case M_RESIDENT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100517 return SPACESHIP_NUMBER(p2->m_resident, p1->m_resident);
Christian Göttschefa002c02020-11-20 17:09:34 +0100518 case M_VIRT:
519 return SPACESHIP_NUMBER(p2->m_virt, p1->m_virt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300520 case NICE:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100521 return SPACESHIP_NUMBER(p1->nice, p2->nice);
Hisham Muhammadd357c672007-05-21 19:10:53 +0000522 case NLWP:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100523 return SPACESHIP_NUMBER(p1->nlwp, p2->nlwp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300524 case PGRP:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100525 return SPACESHIP_NUMBER(p1->pgrp, p2->pgrp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300526 case PID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100527 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300528 case PPID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100529 return SPACESHIP_NUMBER(p1->ppid, p2->ppid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300530 case PRIORITY:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100531 return SPACESHIP_NUMBER(p1->priority, p2->priority);
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300532 case PROCESSOR:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100533 return SPACESHIP_NUMBER(p1->processor, p2->processor);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300534 case SESSION:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100535 return SPACESHIP_NUMBER(p1->session, p2->session);
536 case STARTTIME:
537 r = SPACESHIP_NUMBER(p1->starttime_ctime, p2->starttime_ctime);
538 return r != 0 ? r : SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300539 case STATE:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100540 return SPACESHIP_NUMBER(Process_sortState(p1->state), Process_sortState(p2->state));
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300541 case ST_UID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100542 return SPACESHIP_NUMBER(p1->st_uid, p2->st_uid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300543 case TIME:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100544 return SPACESHIP_NUMBER(p2->time, p1->time);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300545 case TGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100546 return SPACESHIP_NUMBER(p1->tgid, p2->tgid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300547 case TPGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100548 return SPACESHIP_NUMBER(p1->tpgid, p2->tpgid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300549 case TTY_NR:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100550 return SPACESHIP_NUMBER(p1->tty_nr, p2->tty_nr);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300551 case USER:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100552 return SPACESHIP_NULLSTR(p1->user, p2->user);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000553 default:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100554 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000555 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000556}