htop.c 29.9 KB
Newer Older
Hisham Muhammad's avatar
Hisham Muhammad committed
1
2
/*
htop - htop.c
3
(C) 2004-2011 Hisham H. Muhammad
Hisham Muhammad's avatar
Hisham Muhammad committed
4
5
6
7
8
Released under the GNU GPL, see the COPYING file
in the source distribution for its full text.
*/

#include "ProcessList.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
9

Hisham Muhammad's avatar
Hisham Muhammad committed
10
#include "CRT.h"
11
#include "Panel.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
12
13
14
15
16
17
#include "UsersTable.h"
#include "RichString.h"
#include "Settings.h"
#include "ScreenManager.h"
#include "FunctionBar.h"
#include "ListItem.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
18
19
#include "String.h"
#include "ColumnsPanel.h"
20
21
#include "CategoriesPanel.h"
#include "SignalsPanel.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
22
#include "TraceScreen.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
23
#include "OpenFilesScreen.h"
24
#include "AffinityPanel.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
25

Hisham Muhammad's avatar
Hisham Muhammad committed
26
27
28
29
30
31
32
33
34
35
36
37
#include <unistd.h>
#include <math.h>
#include <ctype.h>
#include <stdbool.h>
#include <stdlib.h>
#include <locale.h>
#include <getopt.h>
#include <pwd.h>
#include <string.h>
#include <sys/param.h>
#include <sys/time.h>

Hisham Muhammad's avatar
Hisham Muhammad committed
38
39
40
41
//#link m

#define INCSEARCH_MAX 40

42
#define COPYRIGHT "(C) 2004-2011 Hisham Muhammad"
43

44
static void printVersionFlag() {
45
46
47
   fputs("htop " VERSION " - " COPYRIGHT "\n"
         "Released under the GNU GPL.\n\n",
         stdout);
Hisham Muhammad's avatar
Hisham Muhammad committed
48
49
50
   exit(0);
}

51
static void printHelpFlag() {
52
53
54
55
56
57
58
59
60
61
62
63
64
   fputs("htop " VERSION " - " COPYRIGHT "\n"
         "Released under the GNU GPL.\n\n"
         "-C --no-color         Use a monochrome color scheme\n"
         "-d --delay=DELAY      Set the delay between updates, in tenths of seconds\n"
         "-h --help             Print this help screen\n"
         "-s --sort-key=COLUMN  Sort by COLUMN (try --sort-key=help for a list)\n"
         "-u --user=USERNAME    Show only processes of a given user\n"
         "-v --version          Print version info\n"
         "\n"
         "Long options may be passed with a single dash.\n\n"
         "Press F1 inside htop for online help.\n"
         "See 'man htop' for more information.\n",
         stdout);
Hisham Muhammad's avatar
Hisham Muhammad committed
65
66
67
   exit(0);
}

68
static void showHelp(ProcessList* pl) {
Hisham Muhammad's avatar
Hisham Muhammad committed
69
70
   clear();
   attrset(CRT_colors[HELP_BOLD]);
71
72
73
74

   for (int i = 0; i < LINES-1; i++)
      mvhline(i, 0, ' ', COLS);

75
   mvaddstr(0, 0, "htop " VERSION " - " COPYRIGHT);
Hisham Muhammad's avatar
Hisham Muhammad committed
76
77
78
79
80
81
   mvaddstr(1, 0, "Released under the GNU GPL. See 'man' page for more info.");

   attrset(CRT_colors[DEFAULT_COLOR]);
   mvaddstr(3, 0, "CPU usage bar: ");
   #define addattrstr(a,s) attrset(a);addstr(s)
   addattrstr(CRT_colors[BAR_BORDER], "[");
82
   if (pl->detailedCPUTime) {
83
84
85
86
      addattrstr(CRT_colors[CPU_NICE], "low"); addstr("/");
      addattrstr(CRT_colors[CPU_NORMAL], "normal"); addstr("/");
      addattrstr(CRT_colors[CPU_KERNEL], "kernel"); addstr("/");
      addattrstr(CRT_colors[CPU_IRQ], "irq"); addstr("/");
87
      addattrstr(CRT_colors[CPU_SOFTIRQ], "soft-irq"); addstr("/");
88
89
90
      addattrstr(CRT_colors[CPU_IOWAIT], "io-wait"); addstr("/");
      addattrstr(CRT_colors[CPU_STEAL], "steal"); addstr("/");
      addattrstr(CRT_colors[CPU_GUEST], "guest");
91
92
93
94
      addattrstr(CRT_colors[BAR_SHADOW], " used%");
   } else {
      addattrstr(CRT_colors[CPU_NICE], "low-priority"); addstr("/");
      addattrstr(CRT_colors[CPU_NORMAL], "normal"); addstr("/");
95
96
97
      addattrstr(CRT_colors[CPU_KERNEL], "kernel"); addstr("/");
      addattrstr(CRT_colors[CPU_STEAL], "virtualiz");
      addattrstr(CRT_colors[BAR_SHADOW], "               used%");
98
   }
Hisham Muhammad's avatar
Hisham Muhammad committed
99
100
101
102
103
104
105
   addattrstr(CRT_colors[BAR_BORDER], "]");
   attrset(CRT_colors[DEFAULT_COLOR]);
   mvaddstr(4, 0, "Memory bar:    ");
   addattrstr(CRT_colors[BAR_BORDER], "[");
   addattrstr(CRT_colors[MEMORY_USED], "used"); addstr("/");
   addattrstr(CRT_colors[MEMORY_BUFFERS], "buffers"); addstr("/");
   addattrstr(CRT_colors[MEMORY_CACHE], "cache");
106
   addattrstr(CRT_colors[BAR_SHADOW], "                            used/total");
Hisham Muhammad's avatar
Hisham Muhammad committed
107
108
109
110
111
   addattrstr(CRT_colors[BAR_BORDER], "]");
   attrset(CRT_colors[DEFAULT_COLOR]);
   mvaddstr(5, 0, "Swap bar:      ");
   addattrstr(CRT_colors[BAR_BORDER], "[");
   addattrstr(CRT_colors[SWAP], "used");
112
   addattrstr(CRT_colors[BAR_SHADOW], "                                          used/total");
Hisham Muhammad's avatar
Hisham Muhammad committed
113
114
   addattrstr(CRT_colors[BAR_BORDER], "]");
   attrset(CRT_colors[DEFAULT_COLOR]);
115
   mvaddstr(6,0, "Type and layout of header meters are configurable in the setup screen.");
116
117
118
119
   if (CRT_colorScheme == COLORSCHEME_MONOCHROME) {
      mvaddstr(7, 0, "In monochrome, meters are displayed through different chars, in order: |#*@$%&");
   }
   mvaddstr( 8, 0, " Status: R: running; S: sleeping; T: traced/stopped; Z: zombie; D: disk sleep");
120
121
122
   mvaddstr( 9, 0, " Arrows: scroll process list             F5 t: tree view");
   mvaddstr(10, 0, " Digits: incremental PID search             u: show processes of a single user");
   mvaddstr(11, 0, "   F3 /: incremental name search            H: hide/show user threads");
Hisham Muhammad's avatar
Hisham Muhammad committed
123
   mvaddstr(12, 0, "   F4 \\: incremental name filtering         K: hide/show kernel threads");
124
   mvaddstr(13, 0, "  Space: tag processes                      F: cursor follows process");
Hisham Muhammad's avatar
Hisham Muhammad committed
125
   mvaddstr(14, 0, "      U: untag all processes              + -: expand/collapse tree");
126
   mvaddstr(15, 0, "   F9 k: kill process/tagged processes      P: sort by CPU%");
Hisham Muhammad's avatar
Hisham Muhammad committed
127
128
   mvaddstr(16, 0, "   ] F7: higher priority (root only)        M: sort by MEM%");
   mvaddstr(17, 0, "   [ F8: lower priority (+ nice)            T: sort by TIME");
129
#if (HAVE_LIBHWLOC || HAVE_NATIVE_AFFINITY)
130
   if (pl->cpuCount > 1)
Hisham Muhammad's avatar
Hisham Muhammad committed
131
      mvaddstr(18, 0, "      a: set CPU affinity                   I: invert sort order");
132
   else
Hisham Muhammad's avatar
Hisham Muhammad committed
133
#endif
Hisham Muhammad's avatar
Hisham Muhammad committed
134
      mvaddstr(18, 0, "                                            I: invert sort order");
135
   mvaddstr(19, 0, "   F2 S: setup                           F6 >: select sort column");
Hisham Muhammad's avatar
Hisham Muhammad committed
136
   mvaddstr(20, 0, "   F1 h: show this help screen              l: list open files with lsof");
137
   mvaddstr(21, 0, "  F10 q: quit                               s: trace syscalls with strace");
Hisham Muhammad's avatar
Hisham Muhammad committed
138
139

   attrset(CRT_colors[HELP_BOLD]);
140
141
142
   mvaddstr( 9, 0, " Arrows"); mvaddstr( 9,40, " F5 t");
   mvaddstr(10, 0, " Digits"); mvaddstr(10,40, "    u");
   mvaddstr(11, 0, "   F3 /"); mvaddstr(11,40, "    H");
Hisham Muhammad's avatar
Hisham Muhammad committed
143
   mvaddstr(12, 0, "   F4 \\"); mvaddstr(12,40, "    K");
144
   mvaddstr(13, 0, "  Space"); mvaddstr(13,40, "    F");
Hisham Muhammad's avatar
Hisham Muhammad committed
145
   mvaddstr(14, 0, "      U"); mvaddstr(14,40, "  + -");
146
   mvaddstr(15, 0, "   F9 k"); mvaddstr(15,40, "    P");
147
148
   mvaddstr(16, 0, "   ] F7"); mvaddstr(16,40, "    M");
   mvaddstr(17, 0, "   [ F8"); mvaddstr(17,40, "    T");
149
                               mvaddstr(18,40, " F4 I");
150
#if (HAVE_LIBHWLOC || HAVE_NATIVE_AFFINITY)
151
   if (pl->cpuCount > 1)
152
      mvaddstr(18, 0, "      a:");
Hisham Muhammad's avatar
Hisham Muhammad committed
153
#endif
154
   mvaddstr(19, 0, "   F2 S"); mvaddstr(19,40, " F6 >");
155
   mvaddstr(20, 0, " ? F1 h"); mvaddstr(20,40, "    l");
156
   mvaddstr(21, 0, "  F10 q"); mvaddstr(21,40, "    s");
Hisham Muhammad's avatar
Hisham Muhammad committed
157
158
159
160
161
162
163
164
165
166
   attrset(CRT_colors[DEFAULT_COLOR]);

   attrset(CRT_colors[HELP_BOLD]);
   mvaddstr(23,0, "Press any key to return.");
   attrset(CRT_colors[DEFAULT_COLOR]);
   refresh();
   CRT_readKey();
   clear();
}

Hisham Muhammad's avatar
Hisham Muhammad committed
167
static const char* CategoriesFunctions[] = {"      ", "      ", "      ", "      ", "      ", "      ", "      ", "      ", "      ", "Done  ", NULL};
168

169
170
static void Setup_run(Settings* settings, const Header* header) {
   ScreenManager* scr = ScreenManager_new(0, header->height, 0, -1, HORIZONTAL, header, true);
Hisham Muhammad's avatar
Hisham Muhammad committed
171
   CategoriesPanel* panelCategories = CategoriesPanel_new(settings, scr);
Hisham Muhammad's avatar
Hisham Muhammad committed
172
   ScreenManager_add(scr, (Panel*) panelCategories, FunctionBar_new(CategoriesFunctions, NULL, NULL), 16);
Hisham Muhammad's avatar
Hisham Muhammad committed
173
174
   CategoriesPanel_makeMetersPage(panelCategories);
   Panel* panelFocus;
Hisham Muhammad's avatar
Hisham Muhammad committed
175
   int ch;
Hisham Muhammad's avatar
Hisham Muhammad committed
176
   ScreenManager_run(scr, &panelFocus, &ch);
Hisham Muhammad's avatar
Hisham Muhammad committed
177
178
179
   ScreenManager_delete(scr);
}

Hisham Muhammad's avatar
Hisham Muhammad committed
180
static bool changePriority(Panel* panel, int delta) {
181
   bool ok = true;
Hisham Muhammad's avatar
Hisham Muhammad committed
182
   bool anyTagged = false;
Hisham Muhammad's avatar
Hisham Muhammad committed
183
   for (int i = 0; i < Panel_size(panel); i++) {
Hisham Muhammad's avatar
Hisham Muhammad committed
184
      Process* p = (Process*) Panel_get(panel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
185
      if (p->tag) {
186
         ok = Process_setPriority(p, p->nice + delta) && ok;
Hisham Muhammad's avatar
Hisham Muhammad committed
187
188
189
190
         anyTagged = true;
      }
   }
   if (!anyTagged) {
Hisham Muhammad's avatar
Hisham Muhammad committed
191
      Process* p = (Process*) Panel_getSelected(panel);
192
      if (p) ok = Process_setPriority(p, p->nice + delta) && ok;
Hisham Muhammad's avatar
Hisham Muhammad committed
193
   }
194
195
   if (!ok)
      beep();
Hisham Muhammad's avatar
Hisham Muhammad committed
196
197
198
   return anyTagged;
}

199
static Object* pickFromVector(Panel* panel, Panel* list, int x, int y, const char** keyLabels, FunctionBar* prevBar, Header* header) {
Hisham Muhammad's avatar
Hisham Muhammad committed
200
   const char* fuKeys[] = {"Enter", "Esc", NULL};
201
   int fuEvents[] = {13, 27};
202
   if (!list->eventHandler)
203
      Panel_setEventHandler(list, Panel_selectByTyping);
204
   ScreenManager* scr = ScreenManager_new(0, y, 0, -1, HORIZONTAL, header, false);
205
   scr->allowFocusChange = false;
Hisham Muhammad's avatar
Hisham Muhammad committed
206
   ScreenManager_add(scr, list, FunctionBar_new(keyLabels, fuKeys, fuEvents), x - 1);
Hisham Muhammad's avatar
Hisham Muhammad committed
207
208
   ScreenManager_add(scr, panel, NULL, -1);
   Panel* panelFocus;
Hisham Muhammad's avatar
Hisham Muhammad committed
209
   int ch;
Hisham Muhammad's avatar
Hisham Muhammad committed
210
   ScreenManager_run(scr, &panelFocus, &ch);
Hisham Muhammad's avatar
Hisham Muhammad committed
211
   ScreenManager_delete(scr);
Hisham Muhammad's avatar
Hisham Muhammad committed
212
213
   Panel_move(panel, 0, y);
   Panel_resize(panel, COLS, LINES-y-1);
Hisham Muhammad's avatar
Hisham Muhammad committed
214
   FunctionBar_draw(prevBar, NULL);
Hisham Muhammad's avatar
Hisham Muhammad committed
215
   if (panelFocus == list && ch == 13) {
216
      return Panel_getSelected(list);
Hisham Muhammad's avatar
Hisham Muhammad committed
217
218
219
220
   }
   return NULL;
}

Hisham Muhammad's avatar
Hisham Muhammad committed
221
static void addUserToVector(int key, void* userCast, void* panelCast) {
Hisham Muhammad's avatar
Hisham Muhammad committed
222
   char* user = (char*) userCast;
Hisham Muhammad's avatar
Hisham Muhammad committed
223
224
   Panel* panel = (Panel*) panelCast;
   Panel_add(panel, (Object*) ListItem_new(user, key));
Hisham Muhammad's avatar
Hisham Muhammad committed
225
226
}

227
static bool setUserOnly(const char* userName, bool* userOnly, uid_t* userId) {
Hisham Muhammad's avatar
Hisham Muhammad committed
228
229
230
231
   struct passwd* user = getpwnam(userName);
   if (user) {
      *userOnly = true;
      *userId = user->pw_uid;
232
      return true;
Hisham Muhammad's avatar
Hisham Muhammad committed
233
   }
234
   return false;
Hisham Muhammad's avatar
Hisham Muhammad committed
235
236
}

237
238
239
240
241
static inline void setSortKey(ProcessList* pl, ProcessField sortKey, Panel* panel, Settings* settings) {
   pl->sortKey = sortKey;
   pl->direction = 1;
   pl->treeView = false;
   settings->changed = true;
242
   ProcessList_printHeader(pl, Panel_getHeader(panel));
243
244
}

Hisham Muhammad's avatar
Hisham Muhammad committed
245
246
247
248
249
250
251
252
253
254
255
typedef struct IncBuffer_ {
   char buffer[INCSEARCH_MAX];
   int index;
   FunctionBar* bar;
} IncBuffer;

static void IncBuffer_reset(IncBuffer* inc) {
   inc->index = 0;
   inc->buffer[0] = 0;
}

Hisham Muhammad's avatar
Hisham Muhammad committed
256
257
258
259
260
int main(int argc, char** argv) {

   int delay = -1;
   bool userOnly = false;
   uid_t userId = 0;
261
   int usecolors = 1;
262
   TreeType treeType = TREE_TYPE_AUTO;
263
264
265
266
267
268
269
270
271
272
273
274
275

   int opt, opti=0;
   static struct option long_opts[] =
   {
      {"help",     no_argument,         0, 'h'},
      {"version",  no_argument,         0, 'v'},
      {"delay",    required_argument,   0, 'd'},
      {"sort-key", required_argument,   0, 's'},
      {"user",     required_argument,   0, 'u'},
      {"no-color", no_argument,         0, 'C'},
      {"no-colour",no_argument,         0, 'C'},
      {0,0,0,0}
   };
276
   int sortKey = 0;
Hisham Muhammad's avatar
Hisham Muhammad committed
277

278
279
280
   char *lc_ctype = getenv("LC_CTYPE");
   if(lc_ctype != NULL)
      setlocale(LC_CTYPE, lc_ctype);
281
282
   else if ((lc_ctype = getenv("LC_ALL")))
      setlocale(LC_CTYPE, lc_ctype);
283
   else
284
      setlocale(LC_CTYPE, "");
285

286
   /* Parse arguments */
287
   while ((opt = getopt_long(argc, argv, "hvCs:d:u:", long_opts, &opti))) {
288
289
290
291
292
293
294
295
296
      if (opt == EOF) break;
      switch (opt) {
         case 'h':
            printHelpFlag();
            break;
         case 'v':
            printVersionFlag();
            break;
         case 's':
Hisham Muhammad's avatar
Hisham Muhammad committed
297
            if (strcmp(optarg, "help") == 0) {
298
299
300
301
302
303
304
305
306
307
308
309
               for (int j = 1; j < LAST_PROCESSFIELD; j++)
                  printf ("%s\n", Process_fieldNames[j]);
               exit(0);
            }

            sortKey = ColumnsPanel_fieldNameToIndex(optarg);
            if (sortKey == -1) {
               fprintf(stderr, "Error: invalid column \"%s\".\n", optarg);
               exit(1);
            }
            break;
         case 'd':
310
311
312
313
314
315
316
            if (sscanf(optarg, "%d", &delay) == 1) {
               if (delay < 1) delay = 1;
               if (delay > 100) delay = 100;
            } else {
               fprintf(stderr, "Error: invalid delay value \"%s\".\n", optarg);
               exit(1);
            }
317
318
            break;
         case 'u':
319
320
321
322
            if (!setUserOnly(optarg, &userOnly, &userId)) {
               fprintf(stderr, "Error: invalid user \"%s\".\n", optarg);
               exit(1);
            }
323
324
325
326
            break;
         case 'C':
            usecolors=0;
            break;
327
328
         default:
            exit(1);
Hisham Muhammad's avatar
Hisham Muhammad committed
329
330
      }
   }
331
332


333
334
335
336
   if (access(PROCDIR, R_OK) != 0) {
      fprintf(stderr, "Error: could not read procfs (compiled to look in %s).\n", PROCDIR);
      exit(1);
   }
Hisham Muhammad's avatar
Hisham Muhammad committed
337
338
339
340
341

   int quit = 0;
   int refreshTimeout = 0;
   int resetRefreshTimeout = 5;
   bool doRefresh = true;
Hisham Muhammad's avatar
Hisham Muhammad committed
342
   bool doRecalculate = false;
Hisham Muhammad's avatar
Hisham Muhammad committed
343
344
   Settings* settings;
   
Hisham Muhammad's avatar
Hisham Muhammad committed
345
   Panel* killPanel = NULL;
Hisham Muhammad's avatar
Hisham Muhammad committed
346
   
Hisham Muhammad's avatar
Hisham Muhammad committed
347
348
349
350
   ProcessList* pl = NULL;
   UsersTable* ut = UsersTable_new();

   pl = ProcessList_new(ut);
351
   Process_getMaxPid();
Hisham Muhammad's avatar
Hisham Muhammad committed
352
353
   
   Header* header = Header_new(pl);
354
   settings = Settings_new(pl, header, pl->cpuCount);
Hisham Muhammad's avatar
Hisham Muhammad committed
355
356
357
358
359
   int headerHeight = Header_calculateHeight(header);

   // FIXME: move delay code to settings
   if (delay != -1)
      settings->delay = delay;
360
361
   if (!usecolors) 
      settings->colorScheme = COLORSCHEME_MONOCHROME;
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389

   if (treeType == TREE_TYPE_AUTO) {
#ifdef HAVE_LIBNCURSESW
      char *locale = setlocale(LC_ALL, NULL);
      if (locale == NULL || locale[0] == '\0')
         locale = setlocale(LC_CTYPE, NULL);
      if (locale != NULL &&
          (strstr(locale, "UTF-8") ||
           strstr(locale, "utf-8") ||
           strstr(locale, "UTF8")  ||
           strstr(locale, "utf8")))
         treeType = TREE_TYPE_UTF8;
      else
         treeType = TREE_TYPE_ASCII;
#else
      treeType = TREE_TYPE_ASCII;
#endif
   }
   switch (treeType) {
   default:
   case TREE_TYPE_ASCII:
      pl->treeStr = ProcessList_treeStrAscii;
      break;
   case TREE_TYPE_UTF8:
      pl->treeStr = ProcessList_treeStrUtf8;
      break;
   }

Hisham Muhammad's avatar
Hisham Muhammad committed
390
   CRT_init(settings->delay, settings->colorScheme);
391
392
393

   Panel* panel = Panel_new(0, headerHeight, COLS, LINES - headerHeight - 2, PROCESS_CLASS, false, NULL);
   ProcessList_setPanel(pl, panel);
Hisham Muhammad's avatar
Hisham Muhammad committed
394
   
395
396
397
398
399
   if (sortKey > 0) {
      pl->sortKey = sortKey;
      pl->treeView = false;
      pl->direction = 1;
   }
400
   ProcessList_printHeader(pl, Panel_getHeader(panel));
Hisham Muhammad's avatar
Hisham Muhammad committed
401
402
403
404
405
406

   IncBuffer incSearch, incFilter;
   bool filtering = false;

   memset(&incSearch, 0, sizeof(IncBuffer));
   const char* searchFunctions[] = {"Next  ", "Cancel ", " Search: ", NULL};
Hisham Muhammad's avatar
Hisham Muhammad committed
407
   const char* searchKeys[] = {"F3", "Esc", "  "};
Hisham Muhammad's avatar
Hisham Muhammad committed
408
   int searchEvents[] = {KEY_F(3), 27, ERR};
Hisham Muhammad's avatar
Hisham Muhammad committed
409
410
411
412
413
414
415
   incSearch.bar = FunctionBar_new(searchFunctions, searchKeys, searchEvents);

   memset(&incFilter, 0, sizeof(IncBuffer));
   const char* filterFunctions[] = {"Done  ", "Clear ", " Filter: ", NULL};
   const char* filterKeys[] = {"Enter", "Esc", "  "};
   int filterEvents[] = {13, 27, ERR};
   incFilter.bar = FunctionBar_new(filterFunctions, filterKeys, filterEvents);
Hisham Muhammad's avatar
Hisham Muhammad committed
416
   
Hisham Muhammad's avatar
Hisham Muhammad committed
417
418
419
   IncBuffer* incMode = NULL;
   
   const char* defaultFunctions[] = {"Help  ", "Setup ", "Search", "Filter", "Tree  ",
Hisham Muhammad's avatar
Hisham Muhammad committed
420
421
       "SortBy", "Nice -", "Nice +", "Kill  ", "Quit  ", NULL};
   FunctionBar* defaultBar = FunctionBar_new(defaultFunctions, NULL, NULL);
Hisham Muhammad's avatar
Hisham Muhammad committed
422
423
424
425
426
427
428
429
430
431
432
433
434
435

   ProcessList_scan(pl);
   usleep(75000);
   
   FunctionBar_draw(defaultBar, NULL);
   
   int acc = 0;
   bool follow = false;
 
   struct timeval tv;
   double newTime = 0.0;
   double oldTime = 0.0;
   bool recalculate;

436
   int ch = ERR;
Hisham Muhammad's avatar
Hisham Muhammad committed
437
438
439
440
441
442
   int closeTimeout = 0;

   while (!quit) {
      gettimeofday(&tv, NULL);
      newTime = ((double)tv.tv_sec * 10) + ((double)tv.tv_usec / 100000);
      recalculate = (newTime - oldTime > CRT_delay);
443
444
      Process* p = (Process*)Panel_getSelected(panel);
      int following = (follow && p) ? p->pid : -1;
Hisham Muhammad's avatar
Hisham Muhammad committed
445
446
447
      if (recalculate)
         oldTime = newTime;
      if (doRefresh) {
Hisham Muhammad's avatar
Hisham Muhammad committed
448
         if (recalculate || doRecalculate) {
Hisham Muhammad's avatar
Hisham Muhammad committed
449
            ProcessList_scan(pl);
Hisham Muhammad's avatar
Hisham Muhammad committed
450
451
            doRecalculate = false;
         }
452
         if (refreshTimeout == 0 || pl->treeView) {
Hisham Muhammad's avatar
Hisham Muhammad committed
453
454
455
            ProcessList_sort(pl);
            refreshTimeout = 1;
         }
456
         ProcessList_rebuildPanel(pl, true, following, userOnly, userId, filtering, incFilter.buffer);
Hisham Muhammad's avatar
Hisham Muhammad committed
457
458
459
460
461
      }
      doRefresh = true;
      
      Header_draw(header);

Hisham Muhammad's avatar
Hisham Muhammad committed
462
      Panel_draw(panel, true);
Hisham Muhammad's avatar
Hisham Muhammad committed
463
      int prev = ch;
464
465
      if (incMode)
         move(LINES-1, CRT_cursorX);
Hisham Muhammad's avatar
Hisham Muhammad committed
466
467
468
      ch = getch();

      if (ch == ERR) {
Hisham Muhammad's avatar
Hisham Muhammad committed
469
         if (!incMode)
Hisham Muhammad's avatar
Hisham Muhammad committed
470
471
472
            refreshTimeout--;
         if (prev == ch && !recalculate) {
            closeTimeout++;
473
            if (closeTimeout == 100) {
Hisham Muhammad's avatar
Hisham Muhammad committed
474
               break;
475
            }
Hisham Muhammad's avatar
Hisham Muhammad committed
476
477
478
479
480
         } else
            closeTimeout = 0;
         continue;
      }

Hisham Muhammad's avatar
Hisham Muhammad committed
481
      if (incMode) {
Hisham Muhammad's avatar
Hisham Muhammad committed
482
         doRefresh = false;
483
         int size = Panel_size(panel);
Hisham Muhammad's avatar
Hisham Muhammad committed
484
         if (ch == KEY_F(3)) {
485
            if (Panel_size(panel) == 0) continue;
Hisham Muhammad's avatar
Hisham Muhammad committed
486
            int here = Panel_getSelectedIndex(panel);
Hisham Muhammad's avatar
Hisham Muhammad committed
487
488
489
490
            int i = here+1;
            while (i != here) {
               if (i == size)
                  i = 0;
491
               Process* p = (Process*) Panel_get(panel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
492
               if (String_contains_i(p->comm, incMode->buffer)) {
Hisham Muhammad's avatar
Hisham Muhammad committed
493
                  Panel_setSelected(panel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
494
495
496
497
498
                  break;
               }
               i++;
            }
            continue;
Hisham Muhammad's avatar
Hisham Muhammad committed
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
         } else if (isprint((char)ch) && (incMode->index < INCSEARCH_MAX)) {
            incMode->buffer[incMode->index] = ch;
            incMode->index++;
            incMode->buffer[incMode->index] = 0;
            if (incMode == &incFilter) {
               doRefresh = true;
               if (incFilter.index == 1) filtering = true;
            }
         } else if ((ch == KEY_BACKSPACE || ch == 127) && (incMode->index > 0)) {
            incMode->index--;
            incMode->buffer[incMode->index] = 0;
            if (incMode == &incFilter) {
               doRefresh = true;
               if (incFilter.index == 0) {
                  filtering = false;
                  IncBuffer_reset(incMode);
               }
            }
Hisham Muhammad's avatar
Hisham Muhammad committed
517
         } else {
Hisham Muhammad's avatar
Hisham Muhammad committed
518
519
520
521
522
523
524
525
            if (incMode == &incFilter) {
               doRefresh = true;
               if (ch == 27) {
                  filtering = false;
                  IncBuffer_reset(incMode);
               }
            }
            incMode = NULL;
Hisham Muhammad's avatar
Hisham Muhammad committed
526
527
528
529
530
            FunctionBar_draw(defaultBar, NULL);
            continue;
         }

         bool found = false;
531
532
         for (int i = 0; i < size; i++) {
            Process* p = (Process*) Panel_get(panel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
533
            if (String_contains_i(p->comm, incSearch.buffer)) {
Hisham Muhammad's avatar
Hisham Muhammad committed
534
               Panel_setSelected(panel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
535
536
537
538
539
               found = true;
               break;
            }
         }
         if (found)
Hisham Muhammad's avatar
Hisham Muhammad committed
540
            FunctionBar_draw(incMode->bar, incMode->buffer);
Hisham Muhammad's avatar
Hisham Muhammad committed
541
         else
Hisham Muhammad's avatar
Hisham Muhammad committed
542
            FunctionBar_drawAttr(incMode->bar, incMode->buffer, CRT_colors[FAILED_SEARCH]);
Hisham Muhammad's avatar
Hisham Muhammad committed
543
544
545
         continue;
      }
      if (isdigit((char)ch)) {
546
         if (Panel_size(panel) == 0) continue;
Hisham Muhammad's avatar
Hisham Muhammad committed
547
         pid_t pid = ch-48 + acc;
Hisham Muhammad's avatar
Hisham Muhammad committed
548
549
         for (int i = 0; i < ProcessList_size(pl) && ((Process*) Panel_getSelected(panel))->pid != pid; i++)
            Panel_setSelected(panel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
550
         acc = pid * 10;
551
         if (acc > 10000000)
Hisham Muhammad's avatar
Hisham Muhammad committed
552
553
554
555
556
557
558
559
560
561
            acc = 0;
         continue;
      } else {
         acc = 0;
      }

      if (ch == KEY_MOUSE) {
         MEVENT mevent;
         int ok = getmouse(&mevent);
         if (ok == OK) {
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
            if (mevent.bstate & BUTTON1_CLICKED) {
               if (mevent.y == panel->y) {
                  int x = panel->scrollH + mevent.x + 1;
                  ProcessField field = ProcessList_keyAt(pl, x);
                  if (field == pl->sortKey) {
                     ProcessList_invertSortOrder(pl);
                     pl->treeView = false;
                  } else {
                     setSortKey(pl, field, panel, settings);
                  }
                  refreshTimeout = 0;
                  continue;
               } else if (mevent.y >= panel->y + 1 && mevent.y < LINES - 1) {
                  Panel_setSelected(panel, mevent.y - panel->y + panel->scrollV - 1);
                  doRefresh = false;
                  refreshTimeout = resetRefreshTimeout;
                  follow = true;
                  continue;
               } if (mevent.y == LINES - 1) {
                  FunctionBar* bar;
Hisham Muhammad's avatar
Hisham Muhammad committed
582
                  if (incMode) bar = incMode->bar;
583
584
                  else bar = defaultBar;
                  ch = FunctionBar_synthesizeEvent(bar, mevent.x);
585
               }
586
587
588
589
590
591
            } else if (mevent.bstate & BUTTON4_CLICKED) {
               ch = KEY_UP;
            #if NCURSES_MOUSE_VERSION > 1
            } else if (mevent.bstate & BUTTON5_CLICKED) {
               ch = KEY_DOWN;
            #endif
Hisham Muhammad's avatar
Hisham Muhammad committed
592
593
594
595
596
597
            }
         }
      }

      switch (ch) {
      case KEY_RESIZE:
Hisham Muhammad's avatar
Hisham Muhammad committed
598
         Panel_resize(panel, COLS, LINES-headerHeight-1);
Hisham Muhammad's avatar
Hisham Muhammad committed
599
600
         if (incMode)
            FunctionBar_draw(incMode->bar, incMode->buffer);
Hisham Muhammad's avatar
Hisham Muhammad committed
601
602
603
604
605
606
         else
            FunctionBar_draw(defaultBar, NULL);
         break;
      case 'M':
      {
         refreshTimeout = 0;
607
         setSortKey(pl, PERCENT_MEM, panel, settings);
Hisham Muhammad's avatar
Hisham Muhammad committed
608
609
610
611
612
         break;
      }
      case 'T':
      {
         refreshTimeout = 0;
613
         setSortKey(pl, TIME, panel, settings);
Hisham Muhammad's avatar
Hisham Muhammad committed
614
615
616
617
         break;
      }
      case 'U':
      {
Hisham Muhammad's avatar
Hisham Muhammad committed
618
         for (int i = 0; i < Panel_size(panel); i++) {
Hisham Muhammad's avatar
Hisham Muhammad committed
619
            Process* p = (Process*) Panel_get(panel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
620
621
622
623
624
625
626
627
            p->tag = false;
         }
         doRefresh = true;
         break;
      }
      case 'P':
      {
         refreshTimeout = 0;
628
         setSortKey(pl, PERCENT_CPU, panel, settings);
Hisham Muhammad's avatar
Hisham Muhammad committed
629
630
631
632
         break;
      }
      case KEY_F(1):
      case 'h':
633
      case '?':
Hisham Muhammad's avatar
Hisham Muhammad committed
634
      {
635
         showHelp(pl);
Hisham Muhammad's avatar
Hisham Muhammad committed
636
637
638
639
640
641
642
643
644
645
646
647
648
         FunctionBar_draw(defaultBar, NULL);
         refreshTimeout = 0;
         break;
      }
      case '\014': // Ctrl+L
      {
         clear();
         FunctionBar_draw(defaultBar, NULL);
         refreshTimeout = 0;
         break;
      }
      case ' ':
      {
Hisham Muhammad's avatar
Hisham Muhammad committed
649
         Process* p = (Process*) Panel_getSelected(panel);
650
         if (!p) break;
Hisham Muhammad's avatar
Hisham Muhammad committed
651
         Process_toggleTag(p);
Hisham Muhammad's avatar
Hisham Muhammad committed
652
         Panel_onKey(panel, KEY_DOWN);
Hisham Muhammad's avatar
Hisham Muhammad committed
653
654
655
656
         break;
      }
      case 's':
      {
657
658
659
         Process* p = (Process*) Panel_getSelected(panel);
         if (!p) break;
         TraceScreen* ts = TraceScreen_new(p);
Hisham Muhammad's avatar
Hisham Muhammad committed
660
661
662
663
664
665
666
667
         TraceScreen_run(ts);
         TraceScreen_delete(ts);
         clear();
         FunctionBar_draw(defaultBar, NULL);
         refreshTimeout = 0;
         CRT_enableDelay();
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
668
669
      case 'l':
      {
670
671
672
         Process* p = (Process*) Panel_getSelected(panel);
         if (!p) break;
         OpenFilesScreen* ts = OpenFilesScreen_new(p);
Hisham Muhammad's avatar
Hisham Muhammad committed
673
674
675
676
677
678
679
680
         OpenFilesScreen_run(ts);
         OpenFilesScreen_delete(ts);
         clear();
         FunctionBar_draw(defaultBar, NULL);
         refreshTimeout = 0;
         CRT_enableDelay();
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
681
682
683
684
      case 'S':
      case 'C':
      case KEY_F(2):
      {
685
         Setup_run(settings, header);
Hisham Muhammad's avatar
Hisham Muhammad committed
686
         // TODO: shouldn't need this, colors should be dynamic
687
         ProcessList_printHeader(pl, Panel_getHeader(panel));
Hisham Muhammad's avatar
Hisham Muhammad committed
688
         headerHeight = Header_calculateHeight(header);
Hisham Muhammad's avatar
Hisham Muhammad committed
689
690
         Panel_move(panel, 0, headerHeight);
         Panel_resize(panel, COLS, LINES-headerHeight-1);
Hisham Muhammad's avatar
Hisham Muhammad committed
691
692
693
694
695
696
697
698
699
700
701
         FunctionBar_draw(defaultBar, NULL);
         refreshTimeout = 0;
         break;
      }
      case 'F':
      {
         follow = true;
         continue;
      }
      case 'u':
      {
702
         Panel* usersPanel = Panel_new(0, 0, 0, 0, LISTITEM_CLASS, true, ListItem_compare);
Hisham Muhammad's avatar
Hisham Muhammad committed
703
         Panel_setHeader(usersPanel, "Show processes of:");
Hisham Muhammad's avatar
Hisham Muhammad committed
704
         UsersTable_foreach(ut, addUserToVector, usersPanel);
705
         Vector_insertionSort(usersPanel->items);
Hisham Muhammad's avatar
Hisham Muhammad committed
706
         ListItem* allUsers = ListItem_new("All users", -1);
Hisham Muhammad's avatar
Hisham Muhammad committed
707
         Panel_insert(usersPanel, 0, (Object*) allUsers);
Hisham Muhammad's avatar
Hisham Muhammad committed
708
         const char* fuFunctions[] = {"Show    ", "Cancel ", NULL};
709
         ListItem* picked = (ListItem*) pickFromVector(panel, usersPanel, 20, headerHeight, fuFunctions, defaultBar, header);
Hisham Muhammad's avatar
Hisham Muhammad committed
710
711
712
713
714
715
716
         if (picked) {
            if (picked == allUsers) {
               userOnly = false;
            } else {
               setUserOnly(ListItem_getRef(picked), &userOnly, &userId);
            }
         }
Hisham Muhammad's avatar
Hisham Muhammad committed
717
         Panel_delete((Object*)usersPanel);
Hisham Muhammad's avatar
Hisham Muhammad committed
718
719
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
720
721
722
723
724
      case '+':
      case '=':
      case '-':
      {
         Process* p = (Process*) Panel_getSelected(panel);
725
         if (!p) break;
Hisham Muhammad's avatar
Hisham Muhammad committed
726
727
728
729
730
         p->showChildren = !p->showChildren;
         refreshTimeout = 0;
         doRecalculate = true;
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
731
732
733
      case KEY_F(9):
      case 'k':
      {
Hisham Muhammad's avatar
Hisham Muhammad committed
734
735
         if (!killPanel) {
            killPanel = (Panel*) SignalsPanel_new(0, 0, 0, 0);
Hisham Muhammad's avatar
Hisham Muhammad committed
736
         }
737
         bool anyTagged = false;
738
         pid_t selectedPid = 0;
739
740
741
742
743
744
745
746
747
         for (int i = 0; i < Panel_size(panel); i++) {
            Process* p = (Process*) Panel_get(panel, i);
            if (p->tag) {
               anyTagged = true;
               break;
            }
         }
         if (!anyTagged) {
            Process* p = (Process*) Panel_getSelected(panel);
748
            if (p) selectedPid = p->pid;
749
         }
750
         if (selectedPid == 0) break;
Hisham Muhammad's avatar
Hisham Muhammad committed
751
         SignalsPanel_reset((SignalsPanel*) killPanel);
Hisham Muhammad's avatar
Hisham Muhammad committed
752
         const char* fuFunctions[] = {"Send  ", "Cancel ", NULL};
Hisham Muhammad's avatar
Hisham Muhammad committed
753
         ListItem* sgn = (ListItem*) pickFromVector(panel, killPanel, 15, headerHeight, fuFunctions, defaultBar, header);
Hisham Muhammad's avatar
Hisham Muhammad committed
754
         if (sgn) {
Hisham Muhammad's avatar
Hisham Muhammad committed
755
            if (sgn->key != 0) {
Hisham Muhammad's avatar
Hisham Muhammad committed
756
757
               Panel_setHeader(panel, "Sending...");
               Panel_draw(panel, true);
Hisham Muhammad's avatar
Hisham Muhammad committed
758
               refresh();
759
760
761
762
763
764
765
               if (anyTagged) {
                  for (int i = 0; i < Panel_size(panel); i++) {
                     Process* p = (Process*) Panel_get(panel, i);
                     if (p->tag) {
                        Process_sendSignal(p, sgn->key);
                        anyTagged = true;
                     }
Hisham Muhammad's avatar
Hisham Muhammad committed
766
                  }
767
               } else {
Hisham Muhammad's avatar
Hisham Muhammad committed
768
                  Process* p = (Process*) Panel_getSelected(panel);
769
770
                  if (p->pid == selectedPid)
                     Process_sendSignal(p, sgn->key);
Hisham Muhammad's avatar
Hisham Muhammad committed
771
772
773
774
               }
               napms(500);
            }
         }
775
         ProcessList_printHeader(pl, Panel_getHeader(panel));
Hisham Muhammad's avatar
Hisham Muhammad committed
776
777
778
         refreshTimeout = 0;
         break;
      }
779
#if (HAVE_LIBHWLOC || HAVE_NATIVE_AFFINITY)
780
781
      case 'a':
      {
782
         if (pl->cpuCount == 1)
783
784
            break;

785
786
787
         Process* p = (Process*) Panel_getSelected(panel);
         if (!p) break;
         Affinity* affinity = Process_getAffinity(p);
788
         if (!affinity) break;
789
790
         Panel* affinityPanel = AffinityPanel_new(pl, affinity);
         Affinity_delete(affinity);
791

Hisham Muhammad's avatar
Hisham Muhammad committed
792
         const char* fuFunctions[] = {"Set    ", "Cancel ", NULL};
793
         void* set = pickFromVector(panel, affinityPanel, 15, headerHeight, fuFunctions, defaultBar, header);
794
         if (set) {
795
            Affinity* affinity = AffinityPanel_getAffinity(affinityPanel);
796
797
            bool anyTagged = false;
            bool ok = true;
Hisham Muhammad's avatar
Hisham Muhammad committed
798
            for (int i = 0; i < Panel_size(panel); i++) {
799
800
               Process* p = (Process*) Panel_get(panel, i);
               if (p->tag) {
801
                  ok = Process_setAffinity(p, affinity) && ok;
802
803
                  anyTagged = true;
               }
804
            }
805
806
            if (!anyTagged) {
               Process* p = (Process*) Panel_getSelected(panel);
807
               if (p) ok = Process_setAffinity(p, affinity) && ok;
808
809
810
            }
            if (!ok)
               beep();
811
            Affinity_delete(affinity);
812
         }
813
         Panel_delete((Object*)affinityPanel);
814
         ProcessList_printHeader(pl, Panel_getHeader(panel));
815
816
817
         refreshTimeout = 0;
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
818
#endif
Hisham Muhammad's avatar
Hisham Muhammad committed
819
820
821
822
823
824
825
826
827
828
829
      case KEY_F(10):
      case 'q':
         quit = 1;
         break;
      case '<':
      case ',':
      case KEY_F(18):
      case '>':
      case '.':
      case KEY_F(6):
      {
830
         Panel* sortPanel = Panel_new(0, 0, 0, 0, LISTITEM_CLASS, true, ListItem_compare);
Hisham Muhammad's avatar
Hisham Muhammad committed
831
         Panel_setHeader(sortPanel, "Sort by");
Hisham Muhammad's avatar
Hisham Muhammad committed
832
         const char* fuFunctions[] = {"Sort  ", "Cancel ", NULL};
Hisham Muhammad's avatar
Hisham Muhammad committed
833
834
         ProcessField* fields = pl->fields;
         for (int i = 0; fields[i]; i++) {
835
            char* name = String_trim(Process_fieldTitles[fields[i]]);
Hisham Muhammad's avatar
Hisham Muhammad committed
836
            Panel_add(sortPanel, (Object*) ListItem_new(name, fields[i]));
Hisham Muhammad's avatar
Hisham Muhammad committed
837
            if (fields[i] == pl->sortKey)
Hisham Muhammad's avatar
Hisham Muhammad committed
838
               Panel_setSelected(sortPanel, i);
Hisham Muhammad's avatar
Hisham Muhammad committed
839
840
            free(name);
         }
841
         ListItem* field = (ListItem*) pickFromVector(panel, sortPanel, 15, headerHeight, fuFunctions, defaultBar, header);
Hisham Muhammad's avatar
Hisham Muhammad committed
842
843
         if (field) {
            settings->changed = true;
844
845
            setSortKey(pl, field->key, panel, settings);
         } else {
846
            ProcessList_printHeader(pl, Panel_getHeader(panel));
Hisham Muhammad's avatar
Hisham Muhammad committed
847
         }
Hisham Muhammad's avatar
Hisham Muhammad committed
848
         ((Object*)sortPanel)->delete((Object*)sortPanel);
Hisham Muhammad's avatar
Hisham Muhammad committed
849
850
851
852
853
854
855
856
857
858
859
860
861
         refreshTimeout = 0;
         break;
      }
      case 'I':
      {
         refreshTimeout = 0;
         settings->changed = true;
         ProcessList_invertSortOrder(pl);
         break;
      }
      case KEY_F(8):
      case '[':
      {
Hisham Muhammad's avatar
Hisham Muhammad committed
862
         doRefresh = changePriority(panel, 1);
Hisham Muhammad's avatar
Hisham Muhammad committed
863
864
865
866
867
         break;
      }
      case KEY_F(7):
      case ']':
      {
Hisham Muhammad's avatar
Hisham Muhammad committed
868
         doRefresh = changePriority(panel, -1);
Hisham Muhammad's avatar
Hisham Muhammad committed
869
870
871
872
         break;
      }
      case KEY_F(3):
      case '/':
Hisham Muhammad's avatar
Hisham Muhammad committed
873
874
875
         incMode = &incSearch;
         IncBuffer_reset(incMode);
         FunctionBar_draw(incSearch.bar, incSearch.buffer);
Hisham Muhammad's avatar
Hisham Muhammad committed
876
         break;
Hisham Muhammad's avatar
Hisham Muhammad committed
877
878
879
880
881
882
883
      case KEY_F(4):
      case '\\':
         incMode = &incFilter;
         refreshTimeout = 0;
         doRefresh = true;
         FunctionBar_draw(incFilter.bar, incFilter.buffer);
         continue;
Hisham Muhammad's avatar
Hisham Muhammad committed
884
885
886
887
      case 't':
      case KEY_F(5):
         refreshTimeout = 0;
         pl->treeView = !pl->treeView;
888
         ProcessList_expandTree(pl);
Hisham Muhammad's avatar
Hisham Muhammad committed
889
890
891
         settings->changed = true;
         break;
      case 'H':
Hisham Muhammad's avatar
Hisham Muhammad committed
892
         doRecalculate = true;
Hisham Muhammad's avatar
Hisham Muhammad committed
893
         refreshTimeout = 0;
894
895
         pl->hideUserlandThreads = !pl->hideUserlandThreads;
         pl->hideThreads = pl->hideUserlandThreads;
Hisham Muhammad's avatar
Hisham Muhammad committed
896
897
898
         settings->changed = true;
         break;
      case 'K':
Hisham Muhammad's avatar
Hisham Muhammad committed
899
         doRecalculate = true;
Hisham Muhammad's avatar
Hisham Muhammad committed
900
901
902
903
904
905
906
         refreshTimeout = 0;
         pl->hideKernelThreads = !pl->hideKernelThreads;
         settings->changed = true;
         break;
      default:
         doRefresh = false;
         refreshTimeout = resetRefreshTimeout;
Hisham Muhammad's avatar
Hisham Muhammad committed
907
         Panel_onKey(panel, ch);
Hisham Muhammad's avatar
Hisham Muhammad committed
908
909
910
911
912
913
914
915
916
917
918
919
920
921
         break;
      }
      follow = false;
   }
   attron(CRT_colors[RESET_COLOR]);
   mvhline(LINES-1, 0, ' ', COLS);
   attroff(CRT_colors[RESET_COLOR]);
   refresh();
   
   CRT_done();
   if (settings->changed)
      Settings_write(settings);
   Header_delete(header);
   ProcessList_delete(pl);
Hisham Muhammad's avatar
Hisham Muhammad committed
922
923
   FunctionBar_delete((Object*)incFilter.bar);
   FunctionBar_delete((Object*)incSearch.bar);
Hisham Muhammad's avatar
Hisham Muhammad committed
924
   FunctionBar_delete((Object*)defaultBar);
Hisham Muhammad's avatar
Hisham Muhammad committed
925
   Panel_delete((Object*)panel);
Hisham Muhammad's avatar
Hisham Muhammad committed
926
927
   if (killPanel)
      ((Object*)killPanel)->delete((Object*)killPanel);
Hisham Muhammad's avatar
Hisham Muhammad committed
928
929
930
931
   UsersTable_delete(ut);
   Settings_delete(settings);
   return 0;
}