blob: d04cce76e7aad2f810ad56eab41ce2569a915d3f [file] [log] [blame]
Hisham Muhammad7ee26462009-06-02 19:28:19 +00001/*
2htop - OpenFilesScreen.c
3(C) 2005-2006 Hisham H. Muhammad
Daniel Lange94ad1112021-09-22 11:33:00 +02004Released under the GNU GPLv2+, see the COPYING file
Hisham Muhammad7ee26462009-06-02 19:28:19 +00005in the source distribution for its full text.
6*/
7
Benny Baumann0f526292020-09-19 13:55:23 +02008#include "config.h" // IWYU pragma: keep
9
Hisham Muhammad84281bd2011-12-26 21:35:57 +000010#include "OpenFilesScreen.h"
11
Tobias Stoeckmannd0d9f202022-01-11 22:25:29 +010012#include <errno.h>
Hisham Muhammad7ee26462009-06-02 19:28:19 +000013#include <fcntl.h>
Peter VandeHaarddfacb82021-09-03 00:20:15 -040014#include <inttypes.h>
Benny Baumanne56089e2023-11-28 15:15:03 +010015#include <stdbool.h>
Benny Baumann0f526292020-09-19 13:55:23 +020016#include <stdio.h>
17#include <stdlib.h>
Benny Baumann0f526292020-09-19 13:55:23 +020018#include <unistd.h>
Hisham Muhammad7ee26462009-06-02 19:28:19 +000019#include <sys/types.h>
20#include <sys/wait.h>
Peter VandeHaarddfacb82021-09-03 00:20:15 -040021#include <sys/stat.h>
Hisham Muhammad7ee26462009-06-02 19:28:19 +000022
Benny Baumann0f526292020-09-19 13:55:23 +020023#include "Macros.h"
24#include "Panel.h"
25#include "ProvideCurses.h"
26#include "Vector.h"
27#include "XUtils.h"
28
Hisham Muhammad7ee26462009-06-02 19:28:19 +000029
Benny Baumannc845b792023-12-29 14:01:10 +010030// cf. getIndexForType; must be larger than the maximum value returned.
31#define LSOF_DATACOL_COUNT 8
32
Christian Göttschef8b9ced2020-10-07 19:02:23 +020033typedef struct OpenFiles_Data_ {
Benny Baumannc845b792023-12-29 14:01:10 +010034 char* data[LSOF_DATACOL_COUNT];
Christian Göttschef8b9ced2020-10-07 19:02:23 +020035} OpenFiles_Data;
Hisham Muhammad7ee26462009-06-02 19:28:19 +000036
Christian Göttschef8b9ced2020-10-07 19:02:23 +020037typedef struct OpenFiles_ProcessData_ {
38 OpenFiles_Data data;
39 int error;
Benny Baumannc845b792023-12-29 14:01:10 +010040 int cols[LSOF_DATACOL_COUNT];
Christian Göttschef8b9ced2020-10-07 19:02:23 +020041 struct OpenFiles_FileData_* files;
42} OpenFiles_ProcessData;
43
44typedef struct OpenFiles_FileData_ {
45 OpenFiles_Data data;
46 struct OpenFiles_FileData_* next;
47} OpenFiles_FileData;
48
49static size_t getIndexForType(char type) {
50 switch (type) {
Benny Baumann6aa9ef22023-11-23 12:22:02 +010051 case 'f':
52 return 0;
53 case 'a':
54 return 1;
55 case 'D':
56 return 2;
57 case 'i':
58 return 3;
59 case 'n':
60 return 4;
61 case 's':
62 return 5;
63 case 't':
64 return 6;
65 case 'o':
66 return 7;
Christian Göttschef8b9ced2020-10-07 19:02:23 +020067 }
68
69 /* should never reach here */
70 abort();
71}
72
73static const char* getDataForType(const OpenFiles_Data* data, char type) {
74 size_t index = getIndexForType(type);
75 return data->data[index] ? data->data[index] : "";
76}
77
78OpenFilesScreen* OpenFilesScreen_new(const Process* process) {
Nathan Scott0f751e92023-08-22 16:11:05 +100079 OpenFilesScreen* this = xCalloc(1, sizeof(OpenFilesScreen));
Hisham Muhammad466d4da2016-01-12 06:00:58 -020080 Object_setClass(this, Class(OpenFilesScreen));
Benny Baumann45869512020-11-01 01:09:51 +010081 if (Process_isThread(process)) {
Nathan Scott0f751e92023-08-22 16:11:05 +100082 this->pid = Process_getThreadGroup(process);
Benny Baumann45869512020-11-01 01:09:51 +010083 } else {
Nathan Scott0f751e92023-08-22 16:11:05 +100084 this->pid = Process_getPid(process);
Benny Baumann45869512020-11-01 01:09:51 +010085 }
Peter VandeHaar27a18832021-09-03 00:16:43 -040086 return (OpenFilesScreen*) InfoScreen_init(&this->super, process, NULL, LINES - 2, " FD TYPE MODE DEVICE SIZE OFFSET NODE NAME");
Hisham Muhammad7ee26462009-06-02 19:28:19 +000087}
88
Hisham Muhammad466d4da2016-01-12 06:00:58 -020089void OpenFilesScreen_delete(Object* this) {
90 free(InfoScreen_done((InfoScreen*)this));
Hisham Muhammad7ee26462009-06-02 19:28:19 +000091}
92
Christian Göttschef8b9ced2020-10-07 19:02:23 +020093static void OpenFilesScreen_draw(InfoScreen* this) {
Narendran Gopalakrishnan09fe94d2020-10-17 16:24:45 +053094 InfoScreen_drawTitled(this, "Snapshot of files open in process %d - %s", ((OpenFilesScreen*)this)->pid, Process_getCommand(this->process));
Hisham Muhammad7ee26462009-06-02 19:28:19 +000095}
96
Hisham Muhammad7d7118b2011-03-22 22:09:42 +000097static OpenFiles_ProcessData* OpenFilesScreen_getProcessData(pid_t pid) {
Hishamb54d2dd2016-02-02 15:53:02 +010098 OpenFiles_ProcessData* pdata = xCalloc(1, sizeof(OpenFiles_ProcessData));
наб4bcea762023-12-23 22:46:44 +010099 pdata->cols[getIndexForType('s')] = 8;
100 pdata->cols[getIndexForType('o')] = 8;
101 pdata->cols[getIndexForType('i')] = 8;
Benny Baumann443a9432020-09-15 22:07:52 +0200102
103 int fdpair[2] = {0, 0};
wangqra042cfe2018-03-16 22:39:03 +0800104 if (pipe(fdpair) == -1) {
105 pdata->error = 1;
Hisham Muhammad2a734052012-11-10 00:31:37 +0000106 return pdata;
Hisham Muhammad7dcd2be2009-10-16 20:15:01 +0000107 }
Benny Baumann443a9432020-09-15 22:07:52 +0200108
wangqra042cfe2018-03-16 22:39:03 +0800109 pid_t child = fork();
110 if (child == -1) {
Benny Baumann443a9432020-09-15 22:07:52 +0200111 close(fdpair[1]);
112 close(fdpair[0]);
wangqra042cfe2018-03-16 22:39:03 +0800113 pdata->error = 1;
114 return pdata;
115 }
Benny Baumann443a9432020-09-15 22:07:52 +0200116
wangqra042cfe2018-03-16 22:39:03 +0800117 if (child == 0) {
118 close(fdpair[0]);
119 dup2(fdpair[1], STDOUT_FILENO);
120 close(fdpair[1]);
121 int fdnull = open("/dev/null", O_WRONLY);
Benny Baumann45869512020-11-01 01:09:51 +0100122 if (fdnull < 0) {
wangqra042cfe2018-03-16 22:39:03 +0800123 exit(1);
Benny Baumann45869512020-11-01 01:09:51 +0100124 }
125
wangqra042cfe2018-03-16 22:39:03 +0800126 dup2(fdnull, STDERR_FILENO);
127 close(fdnull);
Benny Baumann69210002020-09-15 22:02:57 +0200128 char buffer[32] = {0};
129 xSnprintf(buffer, sizeof(buffer), "%d", pid);
Explorer0944091702022-05-17 19:15:37 +0800130 // Use of NULL in variadic functions must have a pointer cast.
131 // The NULL constant is not required by standard to have a pointer type.
Benny Baumann6aa9ef22023-11-23 12:22:02 +0100132 execlp("lsof", "lsof", "-P", "-o", "-p", buffer, "-F", (char*)NULL);
wangqra042cfe2018-03-16 22:39:03 +0800133 exit(127);
134 }
135 close(fdpair[1]);
Benny Baumann443a9432020-09-15 22:07:52 +0200136
137 OpenFiles_Data* item = &(pdata->data);
138 OpenFiles_FileData* fdata = NULL;
Peter VandeHaarddfacb82021-09-03 00:20:15 -0400139 bool lsofIncludesFileSize = false;
Benny Baumann443a9432020-09-15 22:07:52 +0200140
Explorer09a46b3f02024-04-17 16:45:46 +0800141 FILE* fp = fdopen(fdpair[0], "r");
142 if (!fp) {
Christian Göttsche1efddaf2020-09-15 19:56:52 +0200143 pdata->error = 1;
144 return pdata;
145 }
Hisham Muhammad0fa03322016-06-19 18:55:35 -0300146 for (;;) {
Explorer09a46b3f02024-04-17 16:45:46 +0800147 char* line = String_readLine(fp);
Hisham Muhammad0fa03322016-06-19 18:55:35 -0300148 if (!line) {
Hisham Muhammad282f16c2010-03-03 21:10:51 +0000149 break;
150 }
Benny Baumann443a9432020-09-15 22:07:52 +0200151
Hisham Muhammad0fa03322016-06-19 18:55:35 -0300152 unsigned char cmd = line[0];
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200153 switch (cmd) {
Benny Baumann6aa9ef22023-11-23 12:22:02 +0100154 case 'f': /* file descriptor */
155 {
156 OpenFiles_FileData* nextFile = xCalloc(1, sizeof(OpenFiles_FileData));
157 if (fdata == NULL) {
158 pdata->files = nextFile;
159 } else {
160 fdata->next = nextFile;
161 }
162 fdata = nextFile;
163 item = &(fdata->data);
164 } /* FALLTHRU */
165 case 'a': /* file access mode */
166 case 'D': /* file's major/minor device number */
167 case 'i': /* file's inode number */
168 case 'n': /* file name, comment, Internet address */
169 case 's': /* file's size */
170 case 't': /* file's type */
171 {
172 size_t index = getIndexForType(cmd);
Peter VandeHaar5beef3e2021-09-03 00:18:30 -0400173 free_and_xStrdup(&item->data[index], line + 1);
наб4bcea762023-12-23 22:46:44 +0100174 size_t dlen = strlen(item->data[index]);
175 if (dlen > (size_t)pdata->cols[index]) {
176 pdata->cols[index] = (int)CLAMP(dlen, 0, INT16_MAX);
177 }
Benny Baumann6aa9ef22023-11-23 12:22:02 +0100178 break;
Peter VandeHaar5beef3e2021-09-03 00:18:30 -0400179 }
Benny Baumann6aa9ef22023-11-23 12:22:02 +0100180 case 'o': /* file's offset */
181 {
182 size_t index = getIndexForType(cmd);
183 if (String_startsWith(line + 1, "0t")) {
184 free_and_xStrdup(&item->data[index], line + 3);
185 } else {
186 free_and_xStrdup(&item->data[index], line + 1);
187 }
наб4bcea762023-12-23 22:46:44 +0100188 size_t dlen = strlen(item->data[index]);
189 if (dlen > (size_t)pdata->cols[index]) {
190 pdata->cols[index] = (int)CLAMP(dlen, 0, INT16_MAX);
191 }
Benny Baumann6aa9ef22023-11-23 12:22:02 +0100192 break;
193 }
194 case 'c': /* process command name */
195 case 'd': /* file's device character code */
196 case 'g': /* process group ID */
197 case 'G': /* file flags */
198 case 'k': /* link count */
199 case 'l': /* file's lock status */
200 case 'L': /* process login name */
201 case 'p': /* process ID */
202 case 'P': /* protocol name */
203 case 'R': /* parent process ID */
204 case 'T': /* TCP/TPI information, identified by prefixes */
205 case 'u': /* process user ID */
206 /* ignore */
207 break;
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200208 }
Peter VandeHaarddfacb82021-09-03 00:20:15 -0400209
210 if (cmd == 's')
211 lsofIncludesFileSize = true;
212
Hisham Muhammad0fa03322016-06-19 18:55:35 -0300213 free(line);
Hisham Muhammad7ee26462009-06-02 19:28:19 +0000214 }
Explorer09a46b3f02024-04-17 16:45:46 +0800215 fclose(fp);
Benny Baumann443a9432020-09-15 22:07:52 +0200216
wangqra042cfe2018-03-16 22:39:03 +0800217 int wstatus;
Tobias Stoeckmannd0d9f202022-01-11 22:25:29 +0100218 while (waitpid(child, &wstatus, 0) == -1)
219 if (errno != EINTR) {
220 pdata->error = 1;
221 return pdata;
222 }
Benny Baumann443a9432020-09-15 22:07:52 +0200223
Benny Baumann45869512020-11-01 01:09:51 +0100224 if (!WIFEXITED(wstatus)) {
wangqra042cfe2018-03-16 22:39:03 +0800225 pdata->error = 1;
Benny Baumann45869512020-11-01 01:09:51 +0100226 } else {
wangqra042cfe2018-03-16 22:39:03 +0800227 pdata->error = WEXITSTATUS(wstatus);
Benny Baumann45869512020-11-01 01:09:51 +0100228 }
Benny Baumann443a9432020-09-15 22:07:52 +0200229
Peter VandeHaarddfacb82021-09-03 00:20:15 -0400230 /* We got all information we need; no post-processing needed */
231 if (lsofIncludesFileSize)
232 return pdata;
233
234 /* On linux, `lsof -o -F` omits SIZE, so add it back. */
235 /* On macOS, `lsof -o -F` includes SIZE, so this block isn't needed. If no open files have a filesize, this will still run, unfortunately. */
236 size_t fileSizeIndex = getIndexForType('s');
237 for (fdata = pdata->files; fdata != NULL; fdata = fdata->next) {
238 item = &fdata->data;
239 const char* filename = getDataForType(item, 'n');
240
Explorer097017b862024-04-17 16:50:34 +0800241 struct stat sb;
242 if (stat(filename, &sb) == 0) {
Peter VandeHaarddfacb82021-09-03 00:20:15 -0400243 char fileSizeBuf[21]; /* 20 (long long) + 1 (NULL) */
Explorer097017b862024-04-17 16:50:34 +0800244 xSnprintf(fileSizeBuf, sizeof(fileSizeBuf), "%"PRIu64, (uint64_t)sb.st_size); /* sb.st_size is long long on macOS, long on linux */
Peter VandeHaarddfacb82021-09-03 00:20:15 -0400245 free_and_xStrdup(&item->data[fileSizeIndex], fileSizeBuf);
246 }
247 }
248
Hisham Muhammad2a734052012-11-10 00:31:37 +0000249 return pdata;
Hisham Muhammad7ee26462009-06-02 19:28:19 +0000250}
251
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200252static void OpenFiles_Data_clear(OpenFiles_Data* data) {
253 for (size_t i = 0; i < ARRAYSIZE(data->data); i++)
254 free(data->data[i]);
Hisham Muhammad1de87622014-04-22 18:45:47 -0300255}
256
Benny Baumann338baa42023-12-29 12:11:24 +0100257static void OpenFilesScreen_scan(InfoScreen* super) {
258 Panel* panel = super->display;
Hisham Muhammad2a734052012-11-10 00:31:37 +0000259 int idx = Panel_getSelectedIndex(panel);
Hisham Muhammad7ee26462009-06-02 19:28:19 +0000260 Panel_prune(panel);
Benny Baumann338baa42023-12-29 12:11:24 +0100261 OpenFiles_ProcessData* pdata = OpenFilesScreen_getProcessData(((OpenFilesScreen*)super)->pid);
Hisham Muhammad2a734052012-11-10 00:31:37 +0000262 if (pdata->error == 127) {
Benny Baumann338baa42023-12-29 12:11:24 +0100263 InfoScreen_addLine(super, "Could not execute 'lsof'. Please make sure it is available in your $PATH.");
Hisham Muhammad2a734052012-11-10 00:31:37 +0000264 } else if (pdata->error == 1) {
Benny Baumann338baa42023-12-29 12:11:24 +0100265 InfoScreen_addLine(super, "Failed listing open files.");
Hisham Muhammad7ee26462009-06-02 19:28:19 +0000266 } else {
наб4bcea762023-12-23 22:46:44 +0100267 char hdrbuf[128] = {0};
268 snprintf(hdrbuf, sizeof(hdrbuf), "%5.5s %-7.7s %-4.4s %6.6s %*s %*s %*s %s",
269 "FD", "TYPE", "MODE", "DEVICE",
270 pdata->cols[getIndexForType('s')], "SIZE",
271 pdata->cols[getIndexForType('o')], "OFFSET",
272 pdata->cols[getIndexForType('i')], "NODE",
273 "NAME"
274 );
275 Panel_setHeader(panel, hdrbuf);
276
Hisham Muhammad2a734052012-11-10 00:31:37 +0000277 OpenFiles_FileData* fdata = pdata->files;
278 while (fdata) {
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200279 OpenFiles_Data* data = &fdata->data;
Benny Baumann958057f2023-07-28 12:56:00 +0200280 char* entry = NULL;
наб4bcea762023-12-23 22:46:44 +0100281 xAsprintf(&entry, "%5.5s %-7.7s %-4.4s %6.6s %*s %*s %*s %s",
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200282 getDataForType(data, 'f'),
283 getDataForType(data, 't'),
284 getDataForType(data, 'a'),
285 getDataForType(data, 'D'),
наб4bcea762023-12-23 22:46:44 +0100286 pdata->cols[getIndexForType('s')],
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200287 getDataForType(data, 's'),
наб4bcea762023-12-23 22:46:44 +0100288 pdata->cols[getIndexForType('o')],
Peter VandeHaar27a18832021-09-03 00:16:43 -0400289 getDataForType(data, 'o'),
наб4bcea762023-12-23 22:46:44 +0100290 pdata->cols[getIndexForType('i')],
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200291 getDataForType(data, 'i'),
292 getDataForType(data, 'n'));
Benny Baumann338baa42023-12-29 12:11:24 +0100293 InfoScreen_addLine(super, entry);
Benny Baumann958057f2023-07-28 12:56:00 +0200294 free(entry);
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200295 OpenFiles_Data_clear(data);
Hisham Muhammad2a734052012-11-10 00:31:37 +0000296 OpenFiles_FileData* old = fdata;
297 fdata = fdata->next;
Hisham Muhammad7ee26462009-06-02 19:28:19 +0000298 free(old);
299 }
Hisham Muhammad1de87622014-04-22 18:45:47 -0300300 OpenFiles_Data_clear(&pdata->data);
Hisham Muhammad7ee26462009-06-02 19:28:19 +0000301 }
Hisham Muhammad2a734052012-11-10 00:31:37 +0000302 free(pdata);
Benny Baumann338baa42023-12-29 12:11:24 +0100303 Vector_insertionSort(super->lines);
Hisham Muhammad7ca10812011-11-18 06:08:56 +0000304 Vector_insertionSort(panel->items);
Hisham Muhammad02a30bf2010-02-25 01:43:18 +0000305 Panel_setSelected(panel, idx);
Hisham Muhammad7ee26462009-06-02 19:28:19 +0000306}
Christian Göttschef8b9ced2020-10-07 19:02:23 +0200307
308const InfoScreenClass OpenFilesScreen_class = {
309 .super = {
310 .extends = Class(Object),
311 .delete = OpenFilesScreen_delete
312 },
313 .scan = OpenFilesScreen_scan,
314 .draw = OpenFilesScreen_draw
315};