blob: 6ca383fe1606c68ac8aafaa52100b3bde9ef31dd [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>
20#include <string.h>
21#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"
Hisham Muhammadb4f6b112014-11-27 20:10:23 -020026#include "Platform.h"
Benny Baumann0f526292020-09-19 13:55:23 +020027#include "RichString.h"
28#include "Settings.h"
Benny Baumann872e5422020-10-14 20:21:09 +020029#include "XUtils.h"
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000030
Benny Baumann0f526292020-09-19 13:55:23 +020031#if defined(MAJOR_IN_MKDEV)
Kang-Che Sung (宋岡哲)c01f40e2018-02-26 21:15:05 +080032#include <sys/mkdev.h>
Wataru Ashihara41754e52018-12-15 22:06:00 +090033#elif defined(MAJOR_IN_SYSMACROS)
Kang-Che Sung (宋岡哲)c01f40e2018-02-26 21:15:05 +080034#include <sys/sysmacros.h>
35#endif
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000036
Benny Baumann0f526292020-09-19 13:55:23 +020037
Christian Göttsche42073ba2020-11-04 17:46:04 +010038static uid_t Process_getuid = (uid_t)-1;
Hisham Muhammadeb2803c2006-07-12 01:35:59 +000039
Hishamf1f805f2016-02-10 18:48:04 -020040char Process_pidFormat[20] = "%7d ";
Hisham Muhammad94280102015-08-20 00:32:47 -030041
42static char Process_titleBuffer[20][20];
43
44void Process_setupColumnWidths() {
45 int maxPid = Platform_getMaxPid();
Benny Baumann45869512020-11-01 01:09:51 +010046 if (maxPid == -1)
47 return;
48
Hisham Muhammad94280102015-08-20 00:32:47 -030049 int digits = ceil(log10(maxPid));
50 assert(digits < 20);
51 for (int i = 0; Process_pidColumns[i].label; i++) {
52 assert(i < 20);
Hisham Muhammad09e241f2017-07-27 16:07:50 -030053 xSnprintf(Process_titleBuffer[i], 20, "%*s ", digits, Process_pidColumns[i].label);
Hisham Muhammad94280102015-08-20 00:32:47 -030054 Process_fields[Process_pidColumns[i].id].title = Process_titleBuffer[i];
55 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -030056 xSnprintf(Process_pidFormat, sizeof(Process_pidFormat), "%%%dd ", digits);
Hisham Muhammad94280102015-08-20 00:32:47 -030057}
58
Michael Wittenab3171d2020-09-29 14:04:22 +000059void Process_humanNumber(RichString* str, unsigned long long number, bool coloring) {
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000060 char buffer[11];
61 int len;
Daniel Flanagandd334442019-10-31 11:39:12 -050062
Hisham Muhammada939cdf2014-04-24 15:00:09 -030063 int largeNumberColor = CRT_colors[LARGE_NUMBER];
64 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
Benny Baumann40441dc2020-09-13 23:50:24 +020065 int processGigabytesColor = CRT_colors[PROCESS_GIGABYTES];
Hisham Muhammada939cdf2014-04-24 15:00:09 -030066 int processColor = CRT_colors[PROCESS];
67 if (!coloring) {
68 largeNumberColor = CRT_colors[PROCESS];
69 processMegabytesColor = CRT_colors[PROCESS];
Benny Baumann40441dc2020-09-13 23:50:24 +020070 processGigabytesColor = CRT_colors[PROCESS];
Hisham Muhammada939cdf2014-04-24 15:00:09 -030071 }
Daniel Flanagandd334442019-10-31 11:39:12 -050072
Benny Baumann40441dc2020-09-13 23:50:24 +020073 if (number < 1000) {
74 //Plain number, no markings
Michael Wittenab3171d2020-09-29 14:04:22 +000075 len = snprintf(buffer, 10, "%5llu ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +020076 RichString_appendn(str, processColor, buffer, len);
77 } else if (number < 100000) {
78 //2 digit MB, 3 digit KB
Benny Baumann61e14d42020-10-31 23:28:02 +010079 len = snprintf(buffer, 10, "%2llu", number / 1000);
Hisham Muhammada939cdf2014-04-24 15:00:09 -030080 RichString_appendn(str, processMegabytesColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +000081 number %= 1000;
Michael Wittenab3171d2020-09-29 14:04:22 +000082 len = snprintf(buffer, 10, "%03llu ", number);
Hisham Muhammada939cdf2014-04-24 15:00:09 -030083 RichString_appendn(str, processColor, buffer, len);
Benny Baumann40441dc2020-09-13 23:50:24 +020084 } else if (number < 1000 * ONE_K) {
85 //3 digit MB
86 number /= ONE_K;
Michael Wittenab3171d2020-09-29 14:04:22 +000087 len = snprintf(buffer, 10, "%4lluM ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +020088 RichString_appendn(str, processMegabytesColor, buffer, len);
89 } else if (number < 10000 * ONE_K) {
90 //1 digit GB, 3 digit MB
91 number /= ONE_K;
Benny Baumann61e14d42020-10-31 23:28:02 +010092 len = snprintf(buffer, 10, "%1llu", number / 1000);
Benny Baumann40441dc2020-09-13 23:50:24 +020093 RichString_appendn(str, processGigabytesColor, buffer, len);
94 number %= 1000;
Michael Wittenab3171d2020-09-29 14:04:22 +000095 len = snprintf(buffer, 10, "%03lluM ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +020096 RichString_appendn(str, processMegabytesColor, buffer, len);
Benny Baumanne0e59972020-09-20 19:54:53 +020097 } else if (number < 100000 * ONE_K) {
Benny Baumann40441dc2020-09-13 23:50:24 +020098 //2 digit GB, 1 digit MB
99 number /= 100 * ONE_K;
Benny Baumann61e14d42020-10-31 23:28:02 +0100100 len = snprintf(buffer, 10, "%2llu", number / 10);
Benny Baumann40441dc2020-09-13 23:50:24 +0200101 RichString_appendn(str, processGigabytesColor, buffer, len);
102 number %= 10;
Michael Wittenab3171d2020-09-29 14:04:22 +0000103 len = snprintf(buffer, 10, ".%1lluG ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +0200104 RichString_appendn(str, processMegabytesColor, buffer, len);
105 } else if (number < 1000 * ONE_M) {
106 //3 digit GB
107 number /= ONE_M;
Michael Wittenab3171d2020-09-29 14:04:22 +0000108 len = snprintf(buffer, 10, "%4lluG ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +0200109 RichString_appendn(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;
Benny Baumann61e14d42020-10-31 23:28:02 +0100113 len = snprintf(buffer, 10, "%1llu", number / 1000);
Benny Baumann40441dc2020-09-13 23:50:24 +0200114 RichString_appendn(str, largeNumberColor, buffer, len);
115 number %= 1000;
Michael Wittenab3171d2020-09-29 14:04:22 +0000116 len = snprintf(buffer, 10, "%03lluG ", number);
Benny Baumann40441dc2020-09-13 23:50:24 +0200117 RichString_appendn(str, processGigabytesColor, buffer, len);
118 } else {
119 //2 digit TB and above
Benny Baumann61e14d42020-10-31 23:28:02 +0100120 len = snprintf(buffer, 10, "%4.1lfT ", (double)number / ONE_G);
Benny Baumann40441dc2020-09-13 23:50:24 +0200121 RichString_appendn(str, largeNumberColor, buffer, len);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000122 }
123}
124
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300125void Process_colorNumber(RichString* str, unsigned long long number, bool coloring) {
Hisham Muhammad9b351402011-05-26 16:31:18 +0000126 char buffer[14];
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300127
128 int largeNumberColor = CRT_colors[LARGE_NUMBER];
129 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
130 int processColor = CRT_colors[PROCESS];
131 int processShadowColor = CRT_colors[PROCESS_SHADOW];
132 if (!coloring) {
133 largeNumberColor = CRT_colors[PROCESS];
134 processMegabytesColor = CRT_colors[PROCESS];
135 processShadowColor = CRT_colors[PROCESS];
136 }
137
Michael Wittenab3171d2020-09-29 14:04:22 +0000138 if (number == ULLONG_MAX) {
Hisham Muhammade940aec2017-07-10 20:57:34 -0300139 int len = snprintf(buffer, 13, " no perm ");
140 RichString_appendn(str, CRT_colors[PROCESS_SHADOW], buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800141 } else if (number >= 100000LL * ONE_DECIMAL_T) {
142 xSnprintf(buffer, 13, "%11llu ", number / ONE_DECIMAL_G);
143 RichString_appendn(str, largeNumberColor, buffer, 12);
144 } else if (number >= 100LL * ONE_DECIMAL_T) {
145 xSnprintf(buffer, 13, "%11llu ", number / ONE_DECIMAL_M);
adrien1018f15d55c2018-12-18 21:05:09 +0800146 RichString_appendn(str, largeNumberColor, buffer, 8);
Benny Baumann61e14d42020-10-31 23:28:02 +0100147 RichString_appendn(str, processMegabytesColor, buffer + 8, 4);
adrien1018536941f2018-12-30 20:18:35 +0800148 } else if (number >= 10LL * ONE_DECIMAL_G) {
149 xSnprintf(buffer, 13, "%11llu ", number / ONE_DECIMAL_K);
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300150 RichString_appendn(str, largeNumberColor, buffer, 5);
Benny Baumann61e14d42020-10-31 23:28:02 +0100151 RichString_appendn(str, processMegabytesColor, buffer + 5, 3);
152 RichString_appendn(str, processColor, buffer + 8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000153 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300154 xSnprintf(buffer, 13, "%11llu ", number);
Hisham Muhammada939cdf2014-04-24 15:00:09 -0300155 RichString_appendn(str, largeNumberColor, buffer, 2);
Benny Baumann61e14d42020-10-31 23:28:02 +0100156 RichString_appendn(str, processMegabytesColor, buffer + 2, 3);
157 RichString_appendn(str, processColor, buffer + 5, 3);
158 RichString_appendn(str, processShadowColor, buffer + 8, 4);
Hisham Muhammad9b351402011-05-26 16:31:18 +0000159 }
160}
161
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300162void Process_printTime(RichString* str, unsigned long long totalHundredths) {
163 unsigned long long totalSeconds = totalHundredths / 100;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000164
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300165 unsigned long long hours = totalSeconds / 3600;
166 int minutes = (totalSeconds / 60) % 60;
167 int seconds = totalSeconds % 60;
168 int hundredths = totalHundredths - (totalSeconds * 100);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000169 char buffer[11];
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000170 if (hours >= 100) {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300171 xSnprintf(buffer, 10, "%7lluh ", hours);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000172 RichString_append(str, CRT_colors[LARGE_NUMBER], buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000173 } else {
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000174 if (hours) {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300175 xSnprintf(buffer, 10, "%2lluh", hours);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000176 RichString_append(str, CRT_colors[LARGE_NUMBER], buffer);
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300177 xSnprintf(buffer, 10, "%02d:%02d ", minutes, seconds);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000178 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300179 xSnprintf(buffer, 10, "%2d:%02d.%02d ", minutes, seconds, hundredths);
Hisham Muhammad9c44f582011-12-14 23:29:07 +0000180 }
181 RichString_append(str, CRT_colors[DEFAULT_COLOR], buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000182 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000183}
184
Christian Göttschea63cfc82020-10-13 14:26:40 +0200185void Process_fillStarttimeBuffer(Process* this) {
186 struct tm date;
187 (void) localtime_r(&this->starttime_ctime, &date);
188 strftime(this->starttime_show, sizeof(this->starttime_show) - 1, (this->starttime_ctime > (time(NULL) - 86400)) ? "%R " : "%b%d ", &date);
189}
190
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200191static inline void Process_writeCommand(const Process* this, int attr, int baseattr, RichString* str) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200192 int start = RichString_size(str), finish = 0;
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200193 const char* comm = this->comm;
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200194
195 if (this->settings->highlightBaseName || !this->settings->showProgramPath) {
196 int i, basename = 0;
197 for (i = 0; i < this->basenameOffset; i++) {
198 if (comm[i] == '/') {
199 basename = i + 1;
200 } else if (comm[i] == ':') {
201 finish = i + 1;
202 break;
Hisham Muhammadf2a190b2014-02-27 17:11:23 -0300203 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000204 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200205 if (!finish) {
Benny Baumann45869512020-11-01 01:09:51 +0100206 if (this->settings->showProgramPath) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200207 start += basename;
Benny Baumann45869512020-11-01 01:09:51 +0100208 } else {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200209 comm += basename;
Benny Baumann45869512020-11-01 01:09:51 +0100210 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200211 finish = this->basenameOffset - basename;
212 }
213 finish += start - 1;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000214 }
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200215
216 RichString_append(str, attr, comm);
217
Benny Baumann45869512020-11-01 01:09:51 +0100218 if (this->settings->highlightBaseName) {
Tobias Geerinckx-Rice293eec42015-07-29 21:14:29 +0200219 RichString_setAttrn(str, baseattr, start, finish);
Benny Baumann45869512020-11-01 01:09:51 +0100220 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000221}
222
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300223void Process_outputRate(RichString* str, char* buffer, int n, double rate, int coloring) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200224 int largeNumberColor = CRT_colors[LARGE_NUMBER];
225 int processMegabytesColor = CRT_colors[PROCESS_MEGABYTES];
226 int processColor = CRT_colors[PROCESS];
227 if (!coloring) {
228 largeNumberColor = CRT_colors[PROCESS];
229 processMegabytesColor = CRT_colors[PROCESS];
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000230 }
Benny Baumann29ec1152020-09-07 11:53:58 +0200231 if (isnan(rate)) {
Hisham797bcd02016-02-20 02:22:57 -0200232 int len = snprintf(buffer, n, " no perm ");
233 RichString_appendn(str, CRT_colors[PROCESS_SHADOW], buffer, len);
234 } else if (rate < ONE_K) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200235 int len = snprintf(buffer, n, "%7.2f B/s ", rate);
236 RichString_appendn(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800237 } else if (rate < ONE_M) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200238 int len = snprintf(buffer, n, "%7.2f K/s ", rate / ONE_K);
239 RichString_appendn(str, processColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800240 } else if (rate < ONE_G) {
241 int len = snprintf(buffer, n, "%7.2f M/s ", rate / ONE_M);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200242 RichString_appendn(str, processMegabytesColor, buffer, len);
adrien1018536941f2018-12-30 20:18:35 +0800243 } else if (rate < ONE_T) {
244 int len = snprintf(buffer, n, "%7.2f G/s ", rate / ONE_G);
245 RichString_appendn(str, largeNumberColor, buffer, len);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200246 } else {
adrien1018536941f2018-12-30 20:18:35 +0800247 int len = snprintf(buffer, n, "%7.2f T/s ", rate / ONE_T);
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200248 RichString_appendn(str, largeNumberColor, buffer, len);
249 }
Hisham Muhammad2338ad52008-03-14 18:50:49 +0000250}
251
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200252void Process_writeField(const Process* this, RichString* str, ProcessField field) {
Hisham Muhammad2f30cd12014-04-24 15:08:32 -0300253 char buffer[256]; buffer[255] = '\0';
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000254 int attr = CRT_colors[DEFAULT_COLOR];
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000255 int baseattr = CRT_colors[PROCESS_BASENAME];
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000256 int n = sizeof(buffer) - 1;
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200257 bool coloring = this->settings->highlightMegabytes;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000258
259 switch (field) {
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300260 case PERCENT_CPU: {
261 if (this->percent_cpu > 999.9) {
Daniel Flanagandd334442019-10-31 11:39:12 -0500262 xSnprintf(buffer, n, "%4u ", (unsigned int)this->percent_cpu);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300263 } else if (this->percent_cpu > 99.9) {
Daniel Flanagandd334442019-10-31 11:39:12 -0500264 xSnprintf(buffer, n, "%3u. ", (unsigned int)this->percent_cpu);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300265 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300266 xSnprintf(buffer, n, "%4.1f ", this->percent_cpu);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300267 }
268 break;
269 }
270 case PERCENT_MEM: {
271 if (this->percent_mem > 99.9) {
Daniel Flanagandd334442019-10-31 11:39:12 -0500272 xSnprintf(buffer, n, "100. ");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300273 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300274 xSnprintf(buffer, n, "%4.1f ", this->percent_mem);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300275 }
276 break;
277 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000278 case COMM: {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200279 if (this->settings->highlightThreads && Process_isThread(this)) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000280 attr = CRT_colors[PROCESS_THREAD];
281 baseattr = CRT_colors[PROCESS_THREAD_BASENAME];
282 }
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200283 if (!this->settings->treeView || this->indent == 0) {
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000284 Process_writeCommand(this, attr, baseattr, str);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000285 return;
286 } else {
287 char* buf = buffer;
288 int maxIndent = 0;
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000289 bool lastItem = (this->indent < 0);
290 int indent = (this->indent < 0 ? -this->indent : this->indent);
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000291
Benny Baumann45869512020-11-01 01:09:51 +0100292 for (int i = 0; i < 32; i++) {
293 if (indent & (1U << i)) {
Benny Baumannb23f8232020-10-31 22:14:27 +0100294 maxIndent = i + 1;
Benny Baumann45869512020-11-01 01:09:51 +0100295 }
296 }
297
Benny Baumannb23f8232020-10-31 22:14:27 +0100298 for (int i = 0; i < maxIndent - 1; i++) {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100299 int written, ret;
Benny Baumann45869512020-11-01 01:09:51 +0100300 if (indent & (1 << i)) {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100301 ret = snprintf(buf, n, "%s ", CRT_treeStr[TREE_STR_VERT]);
Benny Baumann45869512020-11-01 01:09:51 +0100302 } else {
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100303 ret = snprintf(buf, n, " ");
Benny Baumann45869512020-11-01 01:09:51 +0100304 }
Hisham Muhammad90518bf2019-02-10 00:36:34 +0100305 if (ret < 0 || ret >= n) {
306 written = n;
307 } else {
308 written = ret;
309 }
Hisham Muhammadca6b9232011-11-03 22:12:12 +0000310 buf += written;
311 n -= written;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000312 }
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200313 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 -0300314 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 +0000315 RichString_append(str, CRT_colors[PROCESS_TREE], buffer);
Hisham Muhammad93f091c2008-03-08 23:39:48 +0000316 Process_writeCommand(this, attr, baseattr, str);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000317 return;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000318 }
319 }
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300320 case MAJFLT: Process_colorNumber(str, this->majflt, coloring); return;
321 case MINFLT: Process_colorNumber(str, this->minflt, coloring); return;
Christian Göttsche36187742020-10-15 22:37:02 +0200322 case M_RESIDENT: Process_humanNumber(str, this->m_resident * CRT_pageSizeKB, coloring); return;
323 case M_SIZE: Process_humanNumber(str, this->m_size * CRT_pageSizeKB, coloring); return;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300324 case NICE: {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300325 xSnprintf(buffer, n, "%3ld ", this->nice);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300326 attr = this->nice < 0 ? CRT_colors[PROCESS_HIGH_PRIORITY]
327 : this->nice > 0 ? CRT_colors[PROCESS_LOW_PRIORITY]
328 : attr;
329 break;
330 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300331 case NLWP: xSnprintf(buffer, n, "%4ld ", this->nlwp); break;
332 case PGRP: xSnprintf(buffer, n, Process_pidFormat, this->pgrp); break;
333 case PID: xSnprintf(buffer, n, Process_pidFormat, this->pid); break;
334 case PPID: xSnprintf(buffer, n, Process_pidFormat, this->ppid); break;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300335 case PRIORITY: {
Benny Baumann374edb92020-10-31 20:52:20 +0100336 if (this->priority <= -100)
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300337 xSnprintf(buffer, n, " RT ");
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300338 else
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300339 xSnprintf(buffer, n, "%3ld ", this->priority);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300340 break;
341 }
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300342 case PROCESSOR: xSnprintf(buffer, n, "%3d ", Settings_cpuId(this->settings, this->processor)); break;
343 case SESSION: xSnprintf(buffer, n, Process_pidFormat, this->session); break;
344 case STARTTIME: xSnprintf(buffer, n, "%s", this->starttime_show); break;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000345 case STATE: {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300346 xSnprintf(buffer, n, "%c ", this->state);
Valmiky Arquissandas64e0d942014-10-14 02:30:17 +0100347 switch(this->state) {
348 case 'R':
349 attr = CRT_colors[PROCESS_R_STATE];
350 break;
351 case 'D':
352 attr = CRT_colors[PROCESS_D_STATE];
353 break;
Valmiky Arquissandas64e0d942014-10-14 02:30:17 +0100354 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000355 break;
356 }
Daniel Langec34be412018-10-07 11:16:12 +0200357 case ST_UID: xSnprintf(buffer, n, "%5d ", this->st_uid); break;
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300358 case TIME: Process_printTime(str, this->time); return;
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300359 case TGID: xSnprintf(buffer, n, Process_pidFormat, this->tgid); break;
360 case TPGID: xSnprintf(buffer, n, Process_pidFormat, this->tpgid); break;
361 case TTY_NR: xSnprintf(buffer, n, "%3u:%3u ", major(this->tty_nr), minor(this->tty_nr)); break;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000362 case USER: {
Christian Göttsche42073ba2020-11-04 17:46:04 +0100363 if (Process_getuid != this->st_uid)
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000364 attr = CRT_colors[PROCESS_SHADOW];
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000365 if (this->user) {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300366 xSnprintf(buffer, n, "%-9s ", this->user);
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000367 } else {
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300368 xSnprintf(buffer, n, "%-9d ", this->st_uid);
Hisham Muhammadeb2803c2006-07-12 01:35:59 +0000369 }
Hisham Muhammad9b351402011-05-26 16:31:18 +0000370 if (buffer[9] != '\0') {
371 buffer[9] = ' ';
372 buffer[10] = '\0';
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000373 }
374 break;
375 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000376 default:
Hisham Muhammad09e241f2017-07-27 16:07:50 -0300377 xSnprintf(buffer, n, "- ");
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000378 }
379 RichString_append(str, attr, buffer);
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000380}
381
Christian Göttsche79ad39c2020-10-06 12:28:11 +0200382void Process_display(const Object* cast, RichString* out) {
383 const Process* this = (const Process*) cast;
384 const ProcessField* fields = this->settings->fields;
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000385 RichString_prune(out);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000386 for (int i = 0; fields[i]; i++)
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300387 As_Process(this)->writeField(this, out, fields[i]);
Benny Baumann45869512020-11-01 01:09:51 +0100388
Christian Göttsche42073ba2020-11-04 17:46:04 +0100389 if (this->settings->shadowOtherUsers && this->st_uid != Process_getuid) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000390 RichString_setAttr(out, CRT_colors[PROCESS_SHADOW]);
Benny Baumann45869512020-11-01 01:09:51 +0100391 }
392
393 if (this->tag == true) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000394 RichString_setAttr(out, CRT_colors[PROCESS_TAG]);
Benny Baumann45869512020-11-01 01:09:51 +0100395 }
396
Hisham Muhammada9c0ea32011-03-22 20:37:08 +0000397 assert(out->chlen > 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000398}
399
Hisham Muhammad6f868b02015-02-20 14:52:10 -0200400void Process_done(Process* this) {
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000401 assert (this != NULL);
Hisham Muhammadf54a37b2014-05-03 17:49:05 -0300402 free(this->comm);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000403}
404
Christian Göttscheba282cf2020-10-05 13:19:50 +0200405const ProcessClass Process_class = {
Hisham Muhammad4c24a9b2015-03-31 23:23:10 -0300406 .super = {
407 .extends = Class(Object),
408 .display = Process_display,
409 .delete = Process_delete,
410 .compare = Process_compare
411 },
412 .writeField = Process_writeField,
Hisham Muhammad00b324b2012-12-05 15:12:20 +0000413};
414
Christian Göttsche4eb44392020-10-21 21:26:05 +0200415void Process_init(Process* this, const struct Settings_* settings) {
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200416 this->settings = settings;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000417 this->tag = false;
Hisham Muhammad9eb91212010-06-17 19:02:03 +0000418 this->showChildren = true;
Hisham Muhammadd8e14802010-11-22 12:40:20 +0000419 this->show = true;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000420 this->updated = false;
Hisham Muhammadcb297af2014-04-09 17:43:54 -0300421 this->basenameOffset = -1;
Benny Baumann45869512020-11-01 01:09:51 +0100422
Christian Göttsche42073ba2020-11-04 17:46:04 +0100423 if (Process_getuid == (uid_t)-1) {
Benny Baumann45869512020-11-01 01:09:51 +0100424 Process_getuid = getuid();
425 }
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000426}
427
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000428void Process_toggleTag(Process* this) {
429 this->tag = this->tag == true ? false : true;
430}
431
432bool Process_setPriority(Process* this, int priority) {
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300433 CRT_dropPrivileges();
Michael Kleinab3a7c22015-12-07 20:10:09 +0100434 int old_prio = getpriority(PRIO_PROCESS, this->pid);
435 int err = setpriority(PRIO_PROCESS, this->pid, priority);
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300436 CRT_restorePrivileges();
Michael Kleinab3a7c22015-12-07 20:10:09 +0100437 if (err == 0 && old_prio != getpriority(PRIO_PROCESS, this->pid)) {
438 this->nice = priority;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000439 }
Michael Kleinab3a7c22015-12-07 20:10:09 +0100440 return (err == 0);
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000441}
442
Nathan Scott500fb282020-08-20 09:35:24 +1000443bool Process_changePriorityBy(Process* this, Arg delta) {
444 return Process_setPriority(this, this->nice + delta.i);
Hisham Muhammad47e881f2012-10-04 23:59:45 +0000445}
446
Nathan Scott500fb282020-08-20 09:35:24 +1000447bool Process_sendSignal(Process* this, Arg sgn) {
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300448 CRT_dropPrivileges();
Nathan Scott500fb282020-08-20 09:35:24 +1000449 bool ok = (kill(this->pid, sgn.i) == 0);
Hisham Muhammad543d65c2017-07-26 15:40:55 -0300450 CRT_restorePrivileges();
Nathan Scott500fb282020-08-20 09:35:24 +1000451 return ok;
Hisham Muhammadda23c8c2008-03-09 08:58:38 +0000452}
453
Hisham Muhammad78d09f92014-04-25 19:41:23 -0300454long Process_pidCompare(const void* v1, const void* v2) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200455 const Process* p1 = (const Process*)v1;
456 const Process* p2 = (const Process*)v2;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000457 return (p1->pid - p2->pid);
458}
459
Hisham Muhammad78d09f92014-04-25 19:41:23 -0300460long Process_compare(const void* v1, const void* v2) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200461 const Process *p1, *p2;
462 const Settings *settings = ((const Process*)v1)->settings;
Christian Göttsche397b5c42020-11-04 17:46:24 +0100463 int r;
464
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200465 if (settings->direction == 1) {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200466 p1 = (const Process*)v1;
467 p2 = (const Process*)v2;
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000468 } else {
Christian Göttsche18b1e9f2020-09-23 14:15:51 +0200469 p2 = (const Process*)v1;
470 p1 = (const Process*)v2;
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000471 }
Christian Göttsche397b5c42020-11-04 17:46:24 +0100472
Hisham Muhammad3383d8e2015-01-21 23:27:31 -0200473 switch (settings->sortKey) {
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000474 case PERCENT_CPU:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100475 return SPACESHIP_NUMBER(p2->percent_cpu, p1->percent_cpu);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000476 case PERCENT_MEM:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100477 return SPACESHIP_NUMBER(p2->m_resident, p1->m_resident);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000478 case COMM:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100479 return SPACESHIP_NULLSTR(p1->comm, p2->comm);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300480 case MAJFLT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100481 return SPACESHIP_NUMBER(p2->majflt, p1->majflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300482 case MINFLT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100483 return SPACESHIP_NUMBER(p2->minflt, p1->minflt);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300484 case M_RESIDENT:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100485 return SPACESHIP_NUMBER(p2->m_resident, p1->m_resident);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300486 case M_SIZE:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100487 return SPACESHIP_NUMBER(p2->m_size, p1->m_size);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300488 case NICE:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100489 return SPACESHIP_NUMBER(p1->nice, p2->nice);
Hisham Muhammadd357c672007-05-21 19:10:53 +0000490 case NLWP:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100491 return SPACESHIP_NUMBER(p1->nlwp, p2->nlwp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300492 case PGRP:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100493 return SPACESHIP_NUMBER(p1->pgrp, p2->pgrp);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300494 case PID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100495 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300496 case PPID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100497 return SPACESHIP_NUMBER(p1->ppid, p2->ppid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300498 case PRIORITY:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100499 return SPACESHIP_NUMBER(p1->priority, p2->priority);
Hisham Muhammad272e2d92015-03-16 23:01:48 -0300500 case PROCESSOR:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100501 return SPACESHIP_NUMBER(p1->processor, p2->processor);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300502 case SESSION:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100503 return SPACESHIP_NUMBER(p1->session, p2->session);
504 case STARTTIME:
505 r = SPACESHIP_NUMBER(p1->starttime_ctime, p2->starttime_ctime);
506 return r != 0 ? r : SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300507 case STATE:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100508 return SPACESHIP_NUMBER(Process_sortState(p1->state), Process_sortState(p2->state));
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300509 case ST_UID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100510 return SPACESHIP_NUMBER(p1->st_uid, p2->st_uid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300511 case TIME:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100512 return SPACESHIP_NUMBER(p2->time, p1->time);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300513 case TGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100514 return SPACESHIP_NUMBER(p1->tgid, p2->tgid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300515 case TPGID:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100516 return SPACESHIP_NUMBER(p1->tpgid, p2->tpgid);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300517 case TTY_NR:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100518 return SPACESHIP_NUMBER(p1->tty_nr, p2->tty_nr);
Hisham Muhammadbe1700c2015-03-16 01:43:04 -0300519 case USER:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100520 return SPACESHIP_NULLSTR(p1->user, p2->user);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000521 default:
Christian Göttsche397b5c42020-11-04 17:46:24 +0100522 return SPACESHIP_NUMBER(p1->pid, p2->pid);
Hisham Muhammad5d48ab82006-07-11 06:13:32 +0000523 }
Hisham Muhammadd6231ba2006-03-04 18:16:49 +0000524}