1 /*- 2 * Top users/processes display for Unix 3 * 4 * This program may be freely redistributed, 5 * but this entire comment MUST remain intact. 6 * 7 * Copyright (c) 1984, 1989, William LeFebvre, Rice University 8 * Copyright (c) 1989 - 1994, William LeFebvre, Northwestern University 9 * Copyright (c) 1994, 1995, William LeFebvre, Argonne National Laboratory 10 * Copyright (c) 1996, William LeFebvre, Group sys Consulting 11 * 12 * $FreeBSD$ 13 */ 14 15 #include <sys/types.h> 16 #include <sys/time.h> 17 #include <sys/cdefs.h> 18 #include <sys/limits.h> 19 #include <sys/resource.h> 20 #include <sys/select.h> 21 #include <sys/signal.h> 22 23 #include <assert.h> 24 #include <err.h> 25 #include <errno.h> 26 #include <getopt.h> 27 #include <jail.h> 28 #include <stdbool.h> 29 #include <stdio.h> 30 #include <stdlib.h> 31 #include <signal.h> 32 #include <string.h> 33 #include <unistd.h> 34 35 #include "commands.h" 36 #include "display.h" /* interface to display package */ 37 #include "screen.h" /* interface to screen package */ 38 #include "top.h" 39 #include "machine.h" 40 #include "utils.h" 41 #include "username.h" 42 43 /* Size of the stdio buffer given to stdout */ 44 #define Buffersize 2048 45 46 char copyright[] = 47 "Copyright (c) 1984 through 1996, William LeFebvre"; 48 49 typedef void sigret_t; 50 51 /* The buffer that stdio will use */ 52 static char stdoutbuf[Buffersize]; 53 54 static int fmt_flags = 0; 55 int pcpu_stats = false; 56 57 /* signal handling routines */ 58 static sigret_t leave(int); 59 static sigret_t tstop(int); 60 static sigret_t top_winch(int); 61 62 static volatile sig_atomic_t leaveflag; 63 static volatile sig_atomic_t tstopflag; 64 static volatile sig_atomic_t winchflag; 65 66 /* values which need to be accessed by signal handlers */ 67 static int max_topn; /* maximum displayable processes */ 68 69 /* miscellaneous things */ 70 struct process_select ps; 71 pid_t mypid; 72 73 /* pointers to display routines */ 74 static void (*d_loadave)(int mpid, double *avenrun) = i_loadave; 75 static void (*d_procstates)(int total, int *brkdn) = i_procstates; 76 static void (*d_cpustates)(int *states) = i_cpustates; 77 static void (*d_memory)(int *stats) = i_memory; 78 static void (*d_arc)(int *stats) = i_arc; 79 static void (*d_carc)(int *stats) = i_carc; 80 static void (*d_swap)(int *stats) = i_swap; 81 static void (*d_message)(void) = i_message; 82 static void (*d_header)(const char *text) = i_header; 83 static void (*d_process)(int line, char *thisline) = i_process; 84 85 static void reset_display(void); 86 87 static const struct option longopts[] = { 88 { "cpu-display-mode", no_argument, NULL, 'C' }, /* differs from orignal */ 89 /* D reserved */ 90 { "thread", no_argument, NULL, 'H' }, 91 { "idle-procs", no_argument, NULL, 'I' }, 92 { "jail", required_argument, NULL, 'J' }, 93 { "per-cpu", no_argument, NULL, 'P' }, 94 { "system-procs", no_argument, NULL, 'S' }, 95 { "thread-id", no_argument, NULL, 'T' }, /* differs from orignal */ 96 { "user", required_argument, NULL, 'U' }, 97 { "all", no_argument, NULL, 'a' }, 98 { "batch", no_argument, NULL, 'b' }, 99 /* c reserved */ 100 { "displays", required_argument, NULL, 'd' }, 101 { "interactive", no_argument, NULL, 'i' }, 102 { "jail-id", no_argument, NULL, 'j' }, 103 { "display-mode", required_argument, NULL, 'm' }, 104 /* n is identical to batch */ 105 { "sort-order", required_argument, NULL, 'o' }, 106 { "pid", required_argument, NULL, 'p' }, 107 { "quick", no_argument, NULL, 'q' }, 108 { "delay", required_argument, NULL, 's' }, 109 { "threads", no_argument, NULL, 't' }, 110 { "uids", no_argument, NULL, 'u' }, 111 { "version", no_argument, NULL, 'v' }, 112 { "swap", no_argument, NULL, 'w' }, 113 { "system-idle-procs", no_argument, NULL, 'z' } 114 }; 115 116 static void 117 reset_uids(void) 118 { 119 for (size_t i = 0; i < TOP_MAX_UIDS; ++i) 120 ps.uid[i] = -1; 121 } 122 123 static int 124 add_uid(int uid) 125 { 126 size_t i = 0; 127 128 /* Add the uid if there's room */ 129 for (; i < TOP_MAX_UIDS; ++i) 130 { 131 if (ps.uid[i] == -1 || ps.uid[i] == uid) 132 { 133 ps.uid[i] = uid; 134 break; 135 } 136 } 137 138 return (i == TOP_MAX_UIDS); 139 } 140 141 static void 142 rem_uid(int uid) 143 { 144 size_t i = 0; 145 size_t where = TOP_MAX_UIDS; 146 147 /* Look for the user to remove - no problem if it's not there */ 148 for (; i < TOP_MAX_UIDS; ++i) 149 { 150 if (ps.uid[i] == -1) 151 break; 152 if (ps.uid[i] == uid) 153 where = i; 154 } 155 156 /* Make sure we don't leave a hole in the middle */ 157 if (where != TOP_MAX_UIDS) 158 { 159 ps.uid[where] = ps.uid[i-1]; 160 ps.uid[i-1] = -1; 161 } 162 } 163 164 static int 165 handle_user(char *buf, size_t buflen) 166 { 167 int rc = 0; 168 int uid = -1; 169 char *buf2 = buf; 170 171 new_message(MT_standout, "Username to show (+ for all): "); 172 if (readline(buf, buflen, false) <= 0) 173 { 174 clear_message(); 175 return (rc); 176 } 177 178 if (buf[0] == '+' || buf[0] == '-') 179 { 180 if (buf[1] == '\0') 181 { 182 reset_uids(); 183 goto end; 184 } 185 else 186 ++buf2; 187 } 188 189 if ((uid = userid(buf2)) == -1) 190 { 191 new_message(MT_standout, " %s: unknown user", buf2); 192 rc = 1; 193 goto end; 194 } 195 196 if (buf2 == buf) 197 { 198 reset_uids(); 199 ps.uid[0] = uid; 200 goto end; 201 } 202 203 if (buf[0] == '+') 204 { 205 if (add_uid(uid)) 206 { 207 new_message(MT_standout, " too many users, reset with '+'"); 208 rc = 1; 209 goto end; 210 } 211 } 212 else 213 rem_uid(uid); 214 215 end: 216 putchar('\r'); 217 return (rc); 218 } 219 220 int 221 main(int argc, char *argv[]) 222 { 223 int i; 224 int active_procs; 225 226 struct system_info system_info; 227 struct statics statics; 228 void * processes; 229 230 static char tempbuf1[50]; 231 static char tempbuf2[50]; 232 sigset_t old_sigmask, new_sigmask; 233 int topn = Infinity; 234 double delay = 2; 235 int displays = 0; /* indicates unspecified */ 236 int sel_ret = 0; 237 time_t curr_time; 238 char *(*get_userid)(int) = username; 239 const char *uname_field = "USERNAME"; 240 const char *header_text; 241 char *env_top; 242 const char **preset_argv; 243 int preset_argc = 0; 244 const char **av = NULL; 245 int ac = -1; 246 bool do_unames = true; 247 char interactive = 2; 248 char warnings = 0; 249 char topn_specified = false; 250 char ch; 251 char no_command = 1; 252 struct timeval timeout; 253 char *order_name = NULL; 254 int order_index = 0; 255 fd_set readfds; 256 char *nptr; 257 258 /* set the buffer for stdout */ 259 #ifdef DEBUG 260 extern FILE *debug; 261 debug = fopen("debug.run", "w"); 262 setbuffer(stdout, NULL, 0); 263 #else 264 setbuffer(stdout, stdoutbuf, Buffersize); 265 #endif 266 267 mypid = getpid(); 268 269 /* get our name */ 270 /* initialize some selection options */ 271 ps.idle = true; 272 ps.self = true; 273 ps.system = false; 274 reset_uids(); 275 ps.thread = false; 276 ps.wcpu = 1; 277 ps.jid = -1; 278 ps.jail = false; 279 ps.swap = false; 280 ps.kidle = true; 281 ps.pid = -1; 282 ps.command = NULL; 283 ps.thread_id = false; 284 285 /* get preset options from the environment */ 286 if ((env_top = getenv("TOP")) != NULL) 287 { 288 av = preset_argv = argparse(env_top, &preset_argc); 289 ac = preset_argc; 290 291 /* set the dummy argument to an explanatory message, in case 292 getopt encounters a bad argument */ 293 preset_argv[0] = "while processing environment"; 294 } 295 296 /* process options */ 297 do { 298 /* if we're done doing the presets, then process the real arguments */ 299 if (preset_argc == 0) 300 { 301 ac = argc; 302 av = argv; 303 304 /* this should keep getopt happy... */ 305 optind = 1; 306 } 307 308 while ((i = getopt_long(ac, av, "CSIHPabijJ:nquvzs:d:U:m:o:p:Ttw", longopts, NULL)) != EOF) 309 { 310 switch(i) 311 { 312 case 'v': /* show version number */ 313 errx(0, "version FreeBSD"); 314 break; 315 316 case 'u': /* toggle uid/username display */ 317 do_unames = !do_unames; 318 break; 319 320 case 'U': /* display only username's processes */ 321 if ((ps.uid[0] = userid(optarg)) == -1) 322 { 323 errx(1, "%s: unknown user\n", optarg); 324 } 325 break; 326 327 case 'S': /* show system processes */ 328 ps.system = true; 329 break; 330 331 case 'I': /* show idle processes */ 332 ps.idle = !ps.idle; 333 break; 334 335 case 'i': /* go interactive regardless */ 336 interactive = 1; 337 break; 338 339 case 'n': /* batch, or non-interactive */ 340 case 'b': 341 interactive = 0; 342 break; 343 344 case 'a': 345 fmt_flags ^= FMT_SHOWARGS; 346 break; 347 348 case 'd': /* number of displays to show */ 349 if ((i = atoiwi(optarg)) == Invalid || i == 0) 350 { 351 warnx("warning: display count should be positive -- option ignored"); 352 warnings++; 353 } 354 else 355 { 356 displays = i; 357 } 358 break; 359 case 'p': { 360 unsigned long long num; 361 const char *errstr; 362 363 num = strtonum(optarg, 0, INT_MAX, &errstr); 364 if (errstr != NULL || !find_pid(num)) { 365 fprintf(stderr, "%s: unknown pid\n", optarg); 366 exit(1); 367 } 368 ps.pid = (pid_t)num; 369 ps.system = true; 370 break; 371 } 372 373 case 's': 374 delay = strtod(optarg, &nptr); 375 if (nptr == optarg) { 376 warnx("warning: invalid delay"); 377 delay = 2; 378 warnings++; 379 } 380 if (delay < 0) { 381 warnx("warning: seconds delay should be positive -- using default"); 382 delay = 2; 383 warnings++; 384 } 385 386 break; 387 388 case 'q': /* be quick about it */ 389 errno = 0; 390 i = setpriority(PRIO_PROCESS, 0, PRIO_MIN); 391 if (i == -1 && errno != 0) { 392 warnx("warning: `-q' option failed (%m)"); 393 warnings++; 394 } 395 break; 396 397 case 'm': /* select display mode */ 398 if (strcmp(optarg, "io") == 0) { 399 displaymode = DISP_IO; 400 } else if (strcmp(optarg, "cpu") == 0) { 401 displaymode = DISP_CPU; 402 } else { 403 errx(1, "warning: `-m' option can only take args 'io' or 'cpu'"); 404 } 405 break; 406 407 case 'o': /* select sort order */ 408 order_name = optarg; 409 break; 410 411 case 't': 412 ps.self = !ps.self; 413 break; 414 415 case 'C': 416 ps.wcpu = !ps.wcpu; 417 break; 418 419 case 'H': 420 ps.thread = !ps.thread; 421 break; 422 423 case 'T': 424 ps.thread_id = !ps.thread_id; 425 break; 426 427 case 'j': 428 ps.jail = !ps.jail; 429 break; 430 431 case 'J': /* display only jail's processes */ 432 if ((ps.jid = jail_getid(optarg)) == -1) 433 { 434 fprintf(stderr, "%s: unknown jail\n", optarg); 435 exit(1); 436 } 437 ps.jail = 1; 438 break; 439 440 case 'P': 441 pcpu_stats = !pcpu_stats; 442 break; 443 444 case 'w': 445 ps.swap = 1; 446 break; 447 448 case 'z': 449 ps.kidle = !ps.kidle; 450 break; 451 452 default: 453 errx(1, 454 "[-abCHIijnPqStuvwz] [-d count] [-m io | cpu] [-o field] [-p pid]\n" 455 " [-s time] [-J jail] [-U username] [number]"); 456 } 457 } 458 459 /* get count of top processes to display (if any) */ 460 if (optind < ac) 461 { 462 if ((topn = atoiwi(av[optind])) == Invalid) 463 { 464 warnx("warning: process display count should be non-negative -- using default"); 465 warnings++; 466 } 467 else 468 { 469 topn_specified = true; 470 } 471 } 472 473 /* tricky: remember old value of preset_argc & set preset_argc = 0 */ 474 i = preset_argc; 475 preset_argc = 0; 476 477 /* repeat only if we really did the preset arguments */ 478 } while (i != 0); 479 480 /* set constants for username/uid display correctly */ 481 if (!do_unames) 482 { 483 uname_field = " UID "; 484 get_userid = itoa7; 485 } 486 487 /* initialize the kernel memory interface */ 488 if (machine_init(&statics) == -1) 489 { 490 exit(1); 491 } 492 493 /* determine sorting order index, if necessary */ 494 if (order_name != NULL) 495 { 496 if ((order_index = string_index(order_name, statics.order_names)) == -1) 497 { 498 const char * const *pp; 499 500 warnx("'%s' is not a recognized sorting order.", order_name); 501 fprintf(stderr, "\tTry one of these:"); 502 pp = statics.order_names; 503 while (*pp != NULL) 504 { 505 fprintf(stderr, " %s", *pp++); 506 } 507 fputc('\n', stderr); 508 exit(1); 509 } 510 } 511 512 /* initialize termcap */ 513 init_termcap(interactive); 514 515 /* get the string to use for the process area header */ 516 header_text = format_header(uname_field); 517 518 /* initialize display interface */ 519 if ((max_topn = display_init(&statics)) == -1) 520 { 521 errx(4, "can't allocate sufficient memory"); 522 } 523 524 /* print warning if user requested more processes than we can display */ 525 if (topn > max_topn) 526 { 527 warnx("warning: this terminal can only display %d processes.", max_topn); 528 warnings++; 529 } 530 531 /* adjust for topn == Infinity */ 532 if (topn == Infinity) 533 { 534 /* 535 * For smart terminals, infinity really means everything that can 536 * be displayed, or Largest. 537 * On dumb terminals, infinity means every process in the system! 538 * We only really want to do that if it was explicitly specified. 539 * This is always the case when "Default_TOPN != Infinity". But if 540 * topn wasn't explicitly specified and we are on a dumb terminal 541 * and the default is Infinity, then (and only then) we use 542 * "Nominal_TOPN" instead. 543 */ 544 topn = smart_terminal ? Largest : 545 (topn_specified ? Largest : Nominal_TOPN); 546 } 547 548 /* set header display accordingly */ 549 display_header(topn > 0); 550 551 /* determine interactive state */ 552 if (interactive == 2) 553 { 554 interactive = smart_terminal; 555 } 556 557 /* if # of displays not specified, fill it in */ 558 if (displays == 0) 559 { 560 displays = smart_terminal ? Infinity : 1; 561 } 562 563 /* hold interrupt signals while setting up the screen and the handlers */ 564 565 sigemptyset(&new_sigmask); 566 sigaddset(&new_sigmask, SIGINT); 567 sigaddset(&new_sigmask, SIGQUIT); 568 sigaddset(&new_sigmask, SIGTSTP); 569 sigprocmask(SIG_BLOCK, &new_sigmask, &old_sigmask); 570 init_screen(); 571 signal(SIGINT, leave); 572 signal(SIGQUIT, leave); 573 signal(SIGTSTP, tstop); 574 signal(SIGWINCH, top_winch); 575 sigprocmask(SIG_SETMASK, &old_sigmask, NULL); 576 if (warnings) 577 { 578 fputs("....", stderr); 579 fflush(stderr); 580 sleep(3 * warnings); 581 fputc('\n', stderr); 582 } 583 584 restart: 585 586 /* 587 * main loop -- repeat while display count is positive or while it 588 * indicates infinity (by being -1) 589 */ 590 591 while ((displays == -1) || (displays-- > 0)) 592 { 593 int (*compare)(const void * const, const void * const); 594 595 596 /* get the current stats */ 597 get_system_info(&system_info); 598 599 compare = compares[order_index]; 600 601 /* get the current set of processes */ 602 processes = 603 get_process_info(&system_info, &ps, compare); 604 605 /* display the load averages */ 606 (*d_loadave)(system_info.last_pid, 607 system_info.load_avg); 608 609 /* display the current time */ 610 /* this method of getting the time SHOULD be fairly portable */ 611 time(&curr_time); 612 i_uptime(&system_info.boottime, &curr_time); 613 i_timeofday(&curr_time); 614 615 /* display process state breakdown */ 616 (*d_procstates)(system_info.p_total, 617 system_info.procstates); 618 (*d_cpustates)(system_info.cpustates); 619 620 /* display memory stats */ 621 (*d_memory)(system_info.memory); 622 (*d_arc)(system_info.arc); 623 (*d_carc)(system_info.carc); 624 625 /* display swap stats */ 626 (*d_swap)(system_info.swap); 627 628 /* handle message area */ 629 (*d_message)(); 630 631 /* update the header area */ 632 (*d_header)(header_text); 633 634 if (topn > 0) 635 { 636 /* determine number of processes to actually display */ 637 /* this number will be the smallest of: active processes, 638 number user requested, number current screen accomodates */ 639 active_procs = system_info.p_pactive; 640 if (active_procs > topn) 641 { 642 active_procs = topn; 643 } 644 if (active_procs > max_topn) 645 { 646 active_procs = max_topn; 647 } 648 649 /* now show the top "n" processes. */ 650 for (i = 0; i < active_procs; i++) 651 { 652 (*d_process)(i, format_next_process(processes, get_userid, 653 fmt_flags)); 654 } 655 } 656 else 657 { 658 i = 0; 659 } 660 661 /* do end-screen processing */ 662 u_endscreen(i); 663 664 /* now, flush the output buffer */ 665 if (fflush(stdout) != 0) 666 { 667 new_message(MT_standout, " Write error on stdout"); 668 putchar('\r'); 669 quit(1); 670 } 671 672 /* only do the rest if we have more displays to show */ 673 if (displays) 674 { 675 /* switch out for new display on smart terminals */ 676 if (smart_terminal) 677 { 678 if (overstrike) 679 { 680 reset_display(); 681 } 682 else 683 { 684 d_loadave = u_loadave; 685 d_procstates = u_procstates; 686 d_cpustates = u_cpustates; 687 d_memory = u_memory; 688 d_arc = u_arc; 689 d_carc = u_carc; 690 d_swap = u_swap; 691 d_message = u_message; 692 d_header = u_header; 693 d_process = u_process; 694 } 695 } 696 697 no_command = true; 698 if (!interactive) 699 { 700 usleep(delay * 1e6); 701 if (leaveflag) { 702 end_screen(); 703 exit(0); 704 } 705 } 706 else while (no_command) 707 { 708 /* assume valid command unless told otherwise */ 709 no_command = false; 710 711 /* set up arguments for select with timeout */ 712 FD_ZERO(&readfds); 713 FD_SET(0, &readfds); /* for standard input */ 714 timeout.tv_sec = delay; 715 timeout.tv_usec = 0; 716 717 if (leaveflag) { 718 end_screen(); 719 exit(0); 720 } 721 722 if (tstopflag) { 723 /* move to the lower left */ 724 end_screen(); 725 fflush(stdout); 726 727 /* default the signal handler action */ 728 signal(SIGTSTP, SIG_DFL); 729 730 /* unblock the signal and send ourselves one */ 731 sigsetmask(sigblock(0) & ~(1 << (SIGTSTP - 1))); 732 kill(0, SIGTSTP); 733 734 /* reset the signal handler */ 735 signal(SIGTSTP, tstop); 736 737 /* reinit screen */ 738 reinit_screen(); 739 reset_display(); 740 tstopflag = 0; 741 goto restart; 742 } 743 744 if (winchflag) { 745 /* reascertain the screen dimensions */ 746 get_screensize(); 747 748 /* tell display to resize */ 749 max_topn = display_resize(); 750 751 /* reset the signal handler */ 752 signal(SIGWINCH, top_winch); 753 754 reset_display(); 755 winchflag = 0; 756 goto restart; 757 } 758 759 /* wait for either input or the end of the delay period */ 760 sel_ret = select(2, &readfds, NULL, NULL, &timeout); 761 if (sel_ret < 0 && errno != EINTR) 762 quit(0); 763 if (sel_ret > 0) 764 { 765 int newval; 766 const char *errmsg; 767 const struct command *cptr; 768 769 /* something to read -- clear the message area first */ 770 clear_message(); 771 772 /* now read it and convert to command strchr */ 773 /* (use "change" as a temporary to hold strchr) */ 774 if (read(0, &ch, 1) != 1) 775 { 776 /* read error: either 0 or -1 */ 777 new_message(MT_standout, " Read error on stdin"); 778 putchar('\r'); 779 quit(1); 780 } 781 if (ch == '\r' || ch == '\n') { 782 continue; 783 } 784 cptr = all_commands; 785 while (cptr->c != '\0') { 786 if (cptr->c == ch) { 787 break; 788 } 789 cptr++; 790 } 791 if (cptr->c == '\0') { 792 new_message(MT_standout, " Command not understood"); 793 putchar('\r'); 794 no_command = true; 795 } 796 if (overstrike && !cptr->available_to_dumb) 797 { 798 new_message(MT_standout, 799 " Command cannot be handled by this terminal"); 800 putchar('\r'); 801 no_command = true; 802 } 803 if (!no_command) { 804 switch(cptr->id) 805 { 806 case CMD_redraw: /* redraw screen */ 807 reset_display(); 808 break; 809 810 case CMD_update: /* merely update display */ 811 break; 812 813 case CMD_quit: 814 quit(0); 815 break; 816 817 case CMD_help: 818 reset_display(); 819 top_clear(); 820 show_help(); 821 top_standout("Hit any key to continue: "); 822 fflush(stdout); 823 read(0, &ch, 1); 824 break; 825 826 case CMD_errors: /* show errors */ 827 if (error_count() == 0) 828 { 829 new_message(MT_standout, 830 " Currently no errors to report."); 831 putchar('\r'); 832 no_command = true; 833 } 834 else 835 { 836 reset_display(); 837 top_clear(); 838 show_errors(); 839 top_standout("Hit any key to continue: "); 840 fflush(stdout); 841 read(0, &ch, 1); 842 } 843 break; 844 845 case CMD_number: 846 new_message(MT_standout, 847 "Number of processes to show: "); 848 newval = readline(tempbuf1, 8, true); 849 if (newval > -1) 850 { 851 if (newval > max_topn) 852 { 853 new_message(MT_standout | MT_delayed, 854 " This terminal can only display %d processes.", 855 max_topn); 856 putchar('\r'); 857 } 858 859 if (newval == 0) 860 { 861 /* inhibit the header */ 862 display_header(false); 863 } 864 else if (newval > topn && topn == 0) 865 { 866 /* redraw the header */ 867 display_header(true); 868 d_header = i_header; 869 } 870 topn = newval; 871 } 872 break; 873 874 case CMD_delay: /* new seconds delay */ 875 new_message(MT_standout, "Seconds to delay: "); 876 if ((i = readline(tempbuf1, 8, true)) > -1) 877 { 878 if ((delay = i) == 0) 879 { 880 delay = 1; 881 } 882 } 883 clear_message(); 884 break; 885 886 case CMD_displays: /* change display count */ 887 new_message(MT_standout, 888 "Displays to show (currently %s): ", 889 displays == -1 ? "infinite" : 890 itoa(displays)); 891 if ((i = readline(tempbuf1, 10, true)) > 0) 892 { 893 displays = i; 894 } 895 else if (i == 0) 896 { 897 quit(0); 898 } 899 clear_message(); 900 break; 901 902 case CMD_kill: /* kill program */ 903 new_message(0, "kill "); 904 if (readline(tempbuf2, sizeof(tempbuf2), false) > 0) 905 { 906 if ((errmsg = kill_procs(tempbuf2)) != NULL) 907 { 908 new_message(MT_standout, "%s", errmsg); 909 putchar('\r'); 910 no_command = true; 911 } 912 } 913 else 914 { 915 clear_message(); 916 } 917 break; 918 919 case CMD_renice: /* renice program */ 920 new_message(0, "renice "); 921 if (readline(tempbuf2, sizeof(tempbuf2), false) > 0) 922 { 923 if ((errmsg = renice_procs(tempbuf2)) != NULL) 924 { 925 new_message(MT_standout, "%s", errmsg); 926 putchar('\r'); 927 no_command = true; 928 } 929 } 930 else 931 { 932 clear_message(); 933 } 934 break; 935 936 case CMD_idletog: 937 ps.idle = !ps.idle; 938 new_message(MT_standout | MT_delayed, 939 " %sisplaying idle processes.", 940 ps.idle ? "D" : "Not d"); 941 putchar('\r'); 942 break; 943 944 case CMD_selftog: 945 ps.self = !ps.self; 946 new_message(MT_standout | MT_delayed, 947 " %sisplaying self.", 948 (ps.self) ? "D" : "Not d"); 949 putchar('\r'); 950 break; 951 952 case CMD_user: 953 if (handle_user(tempbuf2, sizeof(tempbuf2))) 954 no_command = true; 955 break; 956 957 case CMD_thrtog: 958 ps.thread = !ps.thread; 959 new_message(MT_standout | MT_delayed, 960 " Displaying threads %s", 961 ps.thread ? "separately" : "as a count"); 962 header_text = format_header(uname_field); 963 reset_display(); 964 putchar('\r'); 965 break; 966 967 case CMD_toggletid: 968 ps.thread_id = !ps.thread_id; 969 new_message(MT_standout | MT_delayed, 970 " Displaying %s", 971 ps.thread_id ? "tid" : "pid"); 972 header_text = format_header(uname_field); 973 reset_display(); 974 putchar('\r'); 975 break; 976 977 case CMD_wcputog: 978 ps.wcpu = !ps.wcpu; 979 new_message(MT_standout | MT_delayed, 980 " Displaying %s CPU", 981 ps.wcpu ? "weighted" : "raw"); 982 header_text = format_header(uname_field); 983 reset_display(); 984 putchar('\r'); 985 break; 986 case CMD_viewtog: 987 displaymode = displaymode == DISP_IO ? DISP_CPU : DISP_IO; 988 new_message(MT_standout | MT_delayed, 989 " Displaying %s statistics.", 990 displaymode == DISP_IO ? "IO" : "CPU"); 991 header_text = format_header(uname_field); 992 display_header(true); 993 d_header = i_header; 994 reset_display(); 995 break; 996 case CMD_viewsys: 997 ps.system = !ps.system; 998 new_message(MT_standout | MT_delayed, 999 " %sisplaying system processes.", 1000 ps.system ? "D" : "Not d"); 1001 break; 1002 case CMD_showargs: 1003 fmt_flags ^= FMT_SHOWARGS; 1004 new_message(MT_standout | MT_delayed, 1005 " %sisplaying process arguments.", 1006 fmt_flags & FMT_SHOWARGS ? "D" : "Not d"); 1007 break; 1008 case CMD_order: 1009 new_message(MT_standout, 1010 "Order to sort: "); 1011 if (readline(tempbuf2, sizeof(tempbuf2), false) > 0) 1012 { 1013 if ((i = string_index(tempbuf2, statics.order_names)) == -1) 1014 { 1015 new_message(MT_standout, 1016 " %s: unrecognized sorting order", tempbuf2); 1017 no_command = true; 1018 } 1019 else 1020 { 1021 order_index = i; 1022 } 1023 putchar('\r'); 1024 } 1025 else 1026 { 1027 clear_message(); 1028 } 1029 break; 1030 case CMD_jidtog: 1031 ps.jail = !ps.jail; 1032 new_message(MT_standout | MT_delayed, 1033 " %sisplaying jail ID.", 1034 ps.jail ? "D" : "Not d"); 1035 header_text = format_header(uname_field); 1036 reset_display(); 1037 putchar('\r'); 1038 break; 1039 1040 case CMD_jail: 1041 new_message(MT_standout, 1042 "Jail to show (+ for all): "); 1043 if (readline(tempbuf2, sizeof(tempbuf2), false) > 0) 1044 { 1045 if (tempbuf2[0] == '+' && 1046 tempbuf2[1] == '\0') 1047 { 1048 ps.jid = -1; 1049 } 1050 else if ((i = jail_getid(tempbuf2)) == -1) 1051 { 1052 new_message(MT_standout, 1053 " %s: unknown jail", tempbuf2); 1054 no_command = true; 1055 } 1056 else 1057 { 1058 ps.jid = i; 1059 } 1060 if (ps.jail == 0) { 1061 ps.jail = 1; 1062 new_message(MT_standout | 1063 MT_delayed, " Displaying jail " 1064 "ID."); 1065 header_text = 1066 format_header(uname_field); 1067 reset_display(); 1068 } 1069 putchar('\r'); 1070 } 1071 else 1072 { 1073 clear_message(); 1074 } 1075 break; 1076 1077 case CMD_kidletog: 1078 ps.kidle = !ps.kidle; 1079 new_message(MT_standout | MT_delayed, 1080 " %sisplaying system idle process.", 1081 ps.kidle ? "D" : "Not d"); 1082 putchar('\r'); 1083 break; 1084 case CMD_pcputog: 1085 pcpu_stats = !pcpu_stats; 1086 new_message(MT_standout | MT_delayed, 1087 " Displaying %sCPU statistics.", 1088 pcpu_stats ? "per-" : "global "); 1089 toggle_pcpustats(); 1090 max_topn = display_updatecpus(&statics); 1091 reset_display(); 1092 putchar('\r'); 1093 break; 1094 case CMD_swaptog: 1095 ps.swap = !ps.swap; 1096 new_message(MT_standout | MT_delayed, 1097 " %sisplaying per-process swap usage.", 1098 ps.swap ? "D" : "Not d"); 1099 header_text = format_header(uname_field); 1100 reset_display(); 1101 putchar('\r'); 1102 break; 1103 case CMD_pid: 1104 new_message(MT_standout, 1105 "Process id to show (+ for all): "); 1106 if (readline(tempbuf2, sizeof(tempbuf2), false) > 0) { 1107 if (tempbuf2[0] == '+' && 1108 tempbuf2[1] == '\0') { 1109 ps.pid = (pid_t)-1; 1110 } else { 1111 unsigned long long num; 1112 const char *errstr; 1113 1114 num = strtonum(tempbuf2, 0, INT_MAX, 1115 &errstr); 1116 if (errstr != NULL || !find_pid(num)) { 1117 new_message(MT_standout, 1118 " %s: unknown pid", 1119 tempbuf2); 1120 no_command = true; 1121 } else { 1122 ps.pid = (pid_t)num; 1123 } 1124 } 1125 putchar('\r'); 1126 } else 1127 clear_message(); 1128 break; 1129 case CMD_NONE: 1130 assert(false && "reached switch without command"); 1131 } 1132 } 1133 } 1134 1135 /* flush out stuff that may have been written */ 1136 fflush(stdout); 1137 } 1138 } 1139 } 1140 1141 #ifdef DEBUG 1142 fclose(debug); 1143 #endif 1144 quit(0); 1145 } 1146 1147 /* 1148 * reset_display() - reset all the display routine pointers so that entire 1149 * screen will get redrawn. 1150 */ 1151 1152 static void 1153 reset_display(void) 1154 { 1155 d_loadave = i_loadave; 1156 d_procstates = i_procstates; 1157 d_cpustates = i_cpustates; 1158 d_memory = i_memory; 1159 d_arc = i_arc; 1160 d_carc = i_carc; 1161 d_swap = i_swap; 1162 d_message = i_message; 1163 d_header = i_header; 1164 d_process = i_process; 1165 } 1166 1167 /* 1168 * signal handlers 1169 */ 1170 1171 static sigret_t 1172 leave(int i __unused) /* exit under normal conditions -- INT handler */ 1173 { 1174 1175 leaveflag = 1; 1176 } 1177 1178 static sigret_t 1179 tstop(int i __unused) /* SIGTSTP handler */ 1180 { 1181 1182 tstopflag = 1; 1183 } 1184 1185 static sigret_t 1186 top_winch(int i __unused) /* SIGWINCH handler */ 1187 { 1188 1189 winchflag = 1; 1190 } 1191 1192 void __dead2 1193 quit(int status) /* exit under duress */ 1194 { 1195 end_screen(); 1196 exit(status); 1197 } 1198