/* ps.c - Show running process statistics. * * Copyright 2013 Sandeep Sharma <sandeep.jack2756@gmail.com> * Copyright 2013 Kyungwan Han <asura321@gmail.com> * * See http://pubs.opengroup.org/onlinepubs/9699919799/utilities/ps.html USE_PS(NEWTOY(ps, ">0o*T", TOYFLAG_BIN)) config PS bool "ps" default n help usage: ps [-o COL1,COL2=HEADER] [-T] Show list of processes -a Show all processes with a tty -A Show all processes -o Select columns for display -T Show threads */ #define FOR_ps #include "toys.h" GLOBALS( struct arg_list *llist_o; unsigned screen_width; void *o; ) #define BUFF_SIZE 1024 struct header_list { struct header_list *next; char *name, *header, *format; int width, position; }; // create list of header attributes taking care of -o (-o ooid=MOM..) // and width of attributes. static void list_add(struct header_list *data, char *c_data) { struct header_list *temp = TT.o, *new; memcpy(new = xmalloc(sizeof(*new)), data, sizeof(*new)); if (c_data) new->header = c_data; else new->header = xstrdup(data->header); if (c_data && (strlen(c_data) > data->width)) new->width = strlen(c_data); if (temp) { while (temp->next) temp = temp->next; temp->next = new; } else TT.o = new; } // parse -o arguments static void parse_o(struct header_list *hdr) { int i; char *ptr, *str, *temp; struct arg_list *node = TT.llist_o; while (node) { char *s = str = xstrdup(node->arg); i = 0; while (str) { if ((ptr = strsep(&str, ","))) { //seprate list if ((temp = strchr(ptr, '='))) { // Handle ppid = MOM *temp++ = 0; while (hdr[i].name) { // search from default header if (!(strcmp(hdr[i].name, ptr))) { //handle condition like ppid = M,OM if (str) ptr = xmprintf("%s,%s", temp, str); else ptr = xmprintf("%s", temp); list_add(hdr+i, ptr); break; } i++; } if (!hdr[i].name) perror_exit("Invalid arg for -o option"); break; } else { while (hdr[i].name) { if (!(strcmp(hdr[i].name, ptr))) { list_add(hdr+i, 0); break; } i++; } if (!hdr[i].name) error_exit("bad -o"); i = 0; } } } free(s); node = node->next; } } //get uid/gid for processes. static void get_uid_gid(char *p, char *id_str, unsigned *id) { FILE *f; if(!p) return; f = xfopen(p, "r"); while (fgets(toybuf, BUFF_SIZE, f)) { if (!strncmp(toybuf, id_str, strlen(id_str))) { sscanf(toybuf, "%*s %u", id); break; } } fclose(f); } //get etime for processes. void get_etime(unsigned long s_time) { unsigned long min; unsigned sec; struct sysinfo info; char *temp; sysinfo(&info); min = s_time/sysconf(_SC_CLK_TCK); min = info.uptime - min; sec = min % 60; min = min / 60; temp = xmprintf("%3lu:%02u", min,sec); xprintf("%*.*s",7,7,temp); free(temp); } //get time attributes for processes. void get_time(unsigned long s_time, unsigned long u_time) { unsigned long min; unsigned sec; char *temp; min = (s_time + u_time)/sysconf(_SC_CLK_TCK); sec = min % 60; min = min / 60; temp = xmprintf("%3lu:%02u", min,sec); xprintf("%*.*s",6,6,temp); free(temp); } // read command line taking care of in between NUL's in command line static void read_cmdline(int fd, char *cmd_ptr) { int size = read(fd, cmd_ptr, BUFF_SIZE); //sizeof(cmd_buf) cmd_ptr[size] = '\0'; while (--size > 0 && cmd_ptr[size] == '\0'); //reach to last char while (size >= 0) { if ((unsigned char)cmd_ptr[size] < ' ') cmd_ptr[size] = ' '; size--; } } // get the processes stats and print the stats // corresponding to header attributes. static void do_ps_line(int pid, int tid) { char *stat_buff = toybuf + BUFF_SIZE, *cmd_buff = toybuf + (2*BUFF_SIZE); char state[4] = {0,}; int tty, tty_major, tty_minor, fd, n, nice, width_counter = 0; struct stat stats; struct passwd *pw; struct group *gr; char *name, *user, *group, *ruser, *rgroup, *ptr; long rss; unsigned long stime, utime, start_time, vsz; unsigned ppid, ruid, rgid, pgid; struct header_list *p = TT.o; sprintf(stat_buff, "/proc/%d", pid); if(stat(stat_buff, &stats)) return; if (tid) { if (snprintf(stat_buff, BUFF_SIZE, "/proc/%d/task/%d/stat", pid, tid) >= BUFF_SIZE) return; if (snprintf(cmd_buff, BUFF_SIZE, "/proc/%d/task/%d/cmdline", pid, tid) >= BUFF_SIZE) return; } else { if (snprintf(stat_buff, BUFF_SIZE, "/proc/%d/stat", pid) >= BUFF_SIZE) return; if (snprintf(cmd_buff, BUFF_SIZE, "/proc/%d/cmdline", pid) >= BUFF_SIZE) return; } fd = xopen(stat_buff, O_RDONLY); n = readall(fd, stat_buff, BUFF_SIZE); xclose(fd); if (n < 0) return; stat_buff[n] = 0; //Null terminate the buffer. ptr = strchr(stat_buff, '('); ptr++; name = ptr; ptr = strrchr(stat_buff, ')'); *ptr = '\0'; //unecessary if? name = xmprintf("[%s]", name); ptr += 2; // goto STATE n = sscanf(ptr, "%c %u %u %*u %d %*s %*s %*s %*s %*s %*s " "%lu %lu %*s %*s %*s %d %*s %*s %lu %lu %ld", &state[0],&ppid, &pgid, &tty, &utime, &stime, &nice,&start_time, &vsz,&rss); if (tid) pid = tid; vsz >>= 10; //Convert into KB rss = rss * 4; //Express in pages tty_major = (tty >> 8) & 0xfff; tty_minor = (tty & 0xff) | ((tty >> 12) & 0xfff00); if (vsz == 0 && state[0] != 'Z') state[1] = 'W'; else state[1] = ' '; if (nice < 0 ) state[2] = '<'; else if (nice) state[2] = 'N'; else state[2] = ' '; if (tid) { if (snprintf(stat_buff, BUFF_SIZE, "/proc/%d/task/%d/status", pid, tid) >= BUFF_SIZE) goto clean; } else { if (snprintf(stat_buff, BUFF_SIZE, "/proc/%d/status", pid) >= BUFF_SIZE) goto clean; } fd = -1; while (p) { int width; width = p->width; width_counter += (width + 1); //how much screen we hv filled, +1, extra space b/w headers switch (p->position) { case 0: pw = getpwuid(stats.st_uid); if (!pw) user = xmprintf("%d",(int)stats.st_uid); else user = xmprintf("%s", pw->pw_name); printf("%-*.*s", width, width, user); free(user); break; case 1: gr = getgrgid(stats.st_gid); if (!gr) group = xmprintf("%d",(int)stats.st_gid); else group = xmprintf("%s", gr->gr_name); printf("%-*.*s", width, width, group); free(group); break; case 2: name[strlen(name) - 1] = '\0'; printf("%-*.*s", width,width, name + 1); name[strlen(name)] = ']'; //Refill it for further process. break; case 3: { int j = 0; width_counter -= width; if(p->next) j = width; //is args is in middle. ( -o pid,args,ppid) else j = (TT.screen_width - width_counter % TT.screen_width); //how much screen left. if (fd == -1) fd = open(cmd_buff, O_RDONLY); //don't want to die else xlseek(fd, 0, SEEK_SET); if (fd < 0) cmd_buff[0] = 0; else read_cmdline(fd, cmd_buff); if (cmd_buff[0]) printf("%-*.*s", j, j, cmd_buff); else printf("%-*.*s", j, j, name); width_counter += width; break; } case 4: printf("%*d", width, pid); break; case 5: printf("%*d", width, ppid); break; case 6: printf("%*d", width, pgid); break; case 7: get_etime(start_time); break; case 8: printf("%*d", width, nice); break; case 9: get_uid_gid(stat_buff, "Gid:", &rgid); gr = getgrgid(rgid); if (!gr) rgroup = xmprintf("%d",(int)stats.st_gid); else rgroup = xmprintf("%s", gr->gr_name); printf("%-*.*s", width, width, rgroup); free(rgroup); break; case 10: get_uid_gid(stat_buff, "Uid:", &ruid); pw = getpwuid(ruid); if (!pw) ruser = xmprintf("%d",(int)stats.st_uid); else ruser = xmprintf("%s", pw->pw_name); printf("%-*.*s", width, width, ruser); free(ruser); break; case 11: get_time(utime, stime); break; case 12: if (tty_major) { char *temp = xmprintf("%d,%d", tty_major,tty_minor); printf("%-*s", width, temp); free(temp); } else printf("%-*s", width, "?"); break; case 13: printf("%*lu", width, vsz); break; case 14: printf("%-*s", width, state); break; case 15: printf("%*lu", width, rss); break; } p = p->next; xputc(' '); //space char } if (fd >= 0) xclose(fd); xputc('\n'); clean: free(name); } // Do stats for threads (for -T option) void do_ps_threads(int pid) { DIR *d; int tid; struct dirent *de; char *tmp = xmprintf("/proc/%d/task",pid); if (!(d = opendir(tmp))) { free(tmp); return; } while ((de = readdir(d))) { if (isdigit(de->d_name[0])) { tid = atoi(de->d_name); if (tid == pid) continue; do_ps_line(pid, tid); } } closedir(d); free(tmp); } void ps_main(void) { DIR *dp; struct dirent *entry; int pid; struct header_list *hdr, def_header[] = { {0, "user", "USER", "%-*s ", 8, 0}, {0, "group", "GROUP", "%-*s ", 8, 1}, {0, "comm", "COMMAND", "%-*s ",16, 2}, {0, "args", "COMMAND", "%-*s ",30, 3}, {0, "pid", "PID", "%*s ", 5, 4}, {0, "ppid","PPID", "%*s ", 5, 5}, {0, "pgid", "PGID", "%*s ", 5, 6}, {0, "etime","ELAPSED", "%*s ", 7, 7}, {0, "nice", "NI", "%*s ", 5, 8}, {0, "rgroup","RGROUP", "%-*s ", 8, 9}, {0, "ruser","RUSER", "%-*s ", 8, 10}, {0, "time", "TIME", "%*s ", 6, 11}, {0, "tty", "TT", "%-*s ", 6, 12}, {0, "vsz","VSZ", "%*s ", 7, 13}, {0, "stat", "STAT", "%-*s ", 4, 14}, {0, "rss", "RSS", "%*s ", 4, 15}, {0,0,0,0,0,0} }; TT.screen_width = 80; //default width terminal_size(&TT.screen_width, NULL); // Default pid, user, time, comm if (!TT.llist_o) { list_add(def_header+4, 0); list_add(def_header, 0); list_add(def_header+11, 0); list_add(def_header+3, 0); } else parse_o(def_header); // ARRAY_LEN(def_header) for (hdr = TT.o; hdr; hdr = hdr->next) printf(hdr->format , hdr->width, hdr->header); xputc('\n'); if (!(dp = opendir("/proc"))) perror_exit("opendir"); while ((entry = readdir(dp))) { if (!isdigit(*entry->d_name)) continue; pid = atoi(entry->d_name); do_ps_line(pid, 0); if (toys.optflags & FLAG_T) do_ps_threads(pid); } closedir(dp); while (CFG_TOYBOX_FREE) { struct header_list *temp = llist_pop(&TT.o); if (!temp) break; free(temp->header); free(temp); } }