|
|
1.1 ! root 1: static char *sccsid = "@(#)ps.c 4.7 (Berkeley) 10/20/80"; ! 2: /* ! 3: * ps; VAX 4BSD version ! 4: */ ! 5: ! 6: #include <stdio.h> ! 7: #include <ctype.h> ! 8: #include <nlist.h> ! 9: #include <pwd.h> ! 10: #include <sys/param.h> ! 11: #include <sys/tty.h> ! 12: #include <sys/dir.h> ! 13: #include <sys/user.h> ! 14: #include <sys/proc.h> ! 15: #include <sys/pte.h> ! 16: #include <sys/vm.h> ! 17: #include <sys/text.h> ! 18: #include <sys/stat.h> ! 19: #include <math.h> ! 20: ! 21: struct nlist nl[] = { ! 22: { "_proc" }, ! 23: #define X_PROC 0 ! 24: { "_Usrptmap" }, ! 25: #define X_USRPTMA 1 ! 26: { "_usrpt" }, ! 27: #define X_USRPT 2 ! 28: { "_text" }, ! 29: #define X_TEXT 3 ! 30: { "_nswap" }, ! 31: #define X_NSWAP 4 ! 32: { "_maxslp" }, ! 33: #define X_MAXSLP 5 ! 34: { "_ccpu" }, ! 35: #define X_CCPU 6 ! 36: { "_ecmx" }, ! 37: #define X_ECMX 7 ! 38: { 0 }, ! 39: }; ! 40: ! 41: struct savcom { ! 42: union { ! 43: struct lsav *lp; ! 44: float u_pctcpu; ! 45: struct vsav *vp; ! 46: int s_ssiz; ! 47: } sun; ! 48: struct asav *ap; ! 49: } savcom[NPROC]; ! 50: ! 51: struct asav { ! 52: char *a_cmdp; ! 53: int a_flag; ! 54: short a_stat, a_uid, a_pid, a_nice, a_pri, a_slptime, a_time; ! 55: size_t a_size, a_rss, a_tsiz, a_txtrss; ! 56: short a_xccount; ! 57: char a_tty[DIRSIZ+1]; ! 58: dev_t a_ttyd; ! 59: time_t a_cpu; ! 60: }; ! 61: ! 62: char *lhdr; ! 63: struct lsav { ! 64: short l_ppid; ! 65: char l_cpu; ! 66: int l_addr; ! 67: caddr_t l_wchan; ! 68: }; ! 69: ! 70: char *uhdr; ! 71: char *shdr; ! 72: ! 73: char *vhdr; ! 74: struct vsav { ! 75: u_int v_majflt; ! 76: size_t v_swrss, v_txtswrss; ! 77: float v_pctcpu; ! 78: }; ! 79: ! 80: struct proc proc[8]; /* 8 = a few, for less syscalls */ ! 81: struct proc *mproc; ! 82: struct text *text; ! 83: ! 84: int paduser1; /* avoid hardware mem clobbering botch */ ! 85: union { ! 86: struct user user; ! 87: char upages[UPAGES][NBPG]; ! 88: } user; ! 89: #define u user.user ! 90: int paduser2; /* avoid hardware mem clobbering botch */ ! 91: ! 92: #define clear(x) ((int)x & 0x7fffffff) ! 93: ! 94: int chkpid; ! 95: int aflg, cflg, eflg, gflg, kflg, lflg, sflg, uflg, vflg, xflg; ! 96: char *tptr; ! 97: char *gettty(), *getcmd(), *getname(), *savestr(), *alloc(), *state(); ! 98: double pcpu(), pmem(); ! 99: int pscomp(); ! 100: int nswap, maxslp; ! 101: double ccpu; ! 102: int ecmx; ! 103: struct pte *Usrptma, *usrpt; ! 104: ! 105: struct ttys { ! 106: char name[DIRSIZ+1]; ! 107: dev_t ttyd; ! 108: struct ttys *next; ! 109: struct ttys *cand; ! 110: } *allttys, *cand[16]; ! 111: ! 112: struct savcom savcom[NPROC]; ! 113: int npr; ! 114: ! 115: int cmdstart; ! 116: int twidth; ! 117: char *kmemf, *memf, *swapf, *nlistf; ! 118: int kmem, mem, swap; ! 119: int rawcpu, sumcpu; ! 120: ! 121: int pcbpf; ! 122: int argaddr; ! 123: extern char _sobuf[]; ! 124: ! 125: main(argc, argv) ! 126: char **argv; ! 127: { ! 128: register int i, j; ! 129: register char *ap; ! 130: int uid; ! 131: off_t procp; ! 132: ! 133: if (chdir("/dev") < 0) { ! 134: perror("/dev"); ! 135: exit(1); ! 136: } ! 137: twidth = 80; ! 138: setbuf(stdout, _sobuf); ! 139: argc--, argv++; ! 140: if (argc > 0) { ! 141: ap = argv[0]; ! 142: while (*ap) switch (*ap++) { ! 143: ! 144: case 'C': ! 145: rawcpu++; ! 146: break; ! 147: case 'S': ! 148: sumcpu++; ! 149: break; ! 150: case 'a': ! 151: aflg++; ! 152: break; ! 153: case 'c': ! 154: cflg = !cflg; ! 155: break; ! 156: case 'e': ! 157: eflg++; ! 158: break; ! 159: case 'g': ! 160: gflg++; ! 161: break; ! 162: case 'k': ! 163: kflg++; ! 164: break; ! 165: case 'l': ! 166: lflg++; ! 167: break; ! 168: case 's': ! 169: sflg++; ! 170: break; ! 171: case 't': ! 172: if (*ap) ! 173: tptr = ap; ! 174: aflg++; ! 175: gflg++; ! 176: if (*tptr == '?') ! 177: xflg++; ! 178: while (*ap) ! 179: ap++; ! 180: break; ! 181: case 'u': ! 182: uflg++; ! 183: break; ! 184: case 'v': ! 185: cflg = 1; ! 186: vflg++; ! 187: break; ! 188: case 'w': ! 189: if (twidth == 80) ! 190: twidth = 132; ! 191: else ! 192: twidth = BUFSIZ; ! 193: break; ! 194: case 'x': ! 195: xflg++; ! 196: break; ! 197: default: ! 198: if (!isdigit(ap[-1])) ! 199: break; ! 200: chkpid = atoi(--ap); ! 201: *ap = 0; ! 202: aflg++; ! 203: xflg++; ! 204: break; ! 205: } ! 206: } ! 207: openfiles(argc, argv); ! 208: getkvars(argc, argv); ! 209: getdev(); ! 210: uid = getuid(); ! 211: printhdr(); ! 212: procp = nl[X_PROC].n_value; ! 213: for (i=0; i<NPROC; i += 8) { ! 214: lseek(kmem, (char *)procp, 0); ! 215: j = NPROC - i; ! 216: if (j > 8) ! 217: j = 8; ! 218: j *= sizeof (struct proc); ! 219: if (read(kmem, (char *)proc, j) != j) ! 220: cantread("proc table", kmemf); ! 221: procp += j; ! 222: for (j = j / sizeof (struct proc) - 1; j >= 0; j--) { ! 223: mproc = &proc[j]; ! 224: if (mproc->p_stat == 0 || ! 225: mproc->p_pgrp == 0 && xflg == 0) ! 226: continue; ! 227: if (tptr == 0 && gflg == 0 && xflg == 0 && ! 228: mproc->p_ppid == 1 && (mproc->p_flag&SDETACH) == 0) ! 229: continue; ! 230: if (uid != mproc->p_uid && aflg==0 || ! 231: chkpid != 0 && chkpid != mproc->p_pid) ! 232: continue; ! 233: if (vflg && gflg == 0 && xflg == 0) { ! 234: if (mproc->p_stat == SZOMB || ! 235: mproc->p_flag&SWEXIT) ! 236: continue; ! 237: if (mproc->p_slptime > MAXSLP && ! 238: (mproc->p_stat == SSLEEP || ! 239: mproc->p_stat == SSTOP)) ! 240: continue; ! 241: } ! 242: save(); ! 243: } ! 244: } ! 245: qsort(savcom, npr, sizeof(savcom[0]), pscomp); ! 246: for (i=0; i<npr; i++) { ! 247: register struct savcom *sp = &savcom[i]; ! 248: if (lflg) ! 249: lpr(sp); ! 250: else if (vflg) ! 251: vpr(sp); ! 252: else if (uflg) ! 253: upr(sp); ! 254: else ! 255: spr(sp); ! 256: if (sp->ap->a_flag & SWEXIT) ! 257: printf(" <exiting>"); ! 258: else if (sp->ap->a_stat == SZOMB) ! 259: printf(" <defunct>"); ! 260: else if (sp->ap->a_pid == 0) ! 261: printf(" swapper"); ! 262: else if (sp->ap->a_pid == 2) ! 263: printf(" pagedaemon"); ! 264: else ! 265: printf(" %.*s", twidth - cmdstart - 2, sp->ap->a_cmdp); ! 266: printf("\n"); ! 267: } ! 268: exit(npr == 0); ! 269: } ! 270: ! 271: openfiles(argc, argv) ! 272: char **argv; ! 273: { ! 274: ! 275: kmemf = "kmem"; ! 276: if (kflg) ! 277: kmemf = argc > 1 ? argv[1] : "/vmcore"; ! 278: kmem = open(kmemf, 0); ! 279: if (kmem < 0) { ! 280: perror(kmemf); ! 281: exit(1); ! 282: } ! 283: if (kflg) { ! 284: mem = kmem; ! 285: memf = kmemf; ! 286: } else { ! 287: memf = "mem"; ! 288: mem = open(memf, 0); ! 289: if (mem < 0) { ! 290: perror(memf); ! 291: exit(1); ! 292: } ! 293: } ! 294: swapf = argc>2 ? argv[2]: "drum"; ! 295: swap = open(swapf, 0); ! 296: if (swap < 0) { ! 297: perror(swapf); ! 298: exit(1); ! 299: } ! 300: } ! 301: ! 302: getkvars(argc, argv) ! 303: char **argv; ! 304: { ! 305: register struct nlist *nlp; ! 306: ! 307: nlistf = argc > 3 ? argv[3] : "/vmunix"; ! 308: nlist(nlistf, nl); ! 309: if (nl[0].n_type == 0) { ! 310: fprintf(stderr, "%s: No namelist\n", nlistf); ! 311: exit(1); ! 312: } ! 313: if (kflg) ! 314: for (nlp = nl; nlp < &nl[sizeof (nl)/sizeof (nl[0])]; nlp++) ! 315: nlp->n_value = clear(nlp->n_value); ! 316: Usrptma = (struct pte *)nl[X_USRPTMA].n_value; ! 317: usrpt = (struct pte *)nl[X_USRPT].n_value; ! 318: lseek(kmem, (long)nl[X_NSWAP].n_value, 0); ! 319: if (read(kmem, &nswap, sizeof (nswap)) != sizeof (nswap)) { ! 320: cantread("nswap", kmemf); ! 321: exit(1); ! 322: } ! 323: lseek(kmem, (long)nl[X_MAXSLP].n_value, 0); ! 324: if (read(kmem, &maxslp, sizeof (maxslp)) != sizeof (maxslp)) { ! 325: cantread("maxslp", kmemf); ! 326: exit(1); ! 327: } ! 328: lseek(kmem, (long)nl[X_CCPU].n_value, 0); ! 329: if (read(kmem, &ccpu, sizeof (ccpu)) != sizeof (ccpu)) { ! 330: cantread("ccpu", kmemf); ! 331: exit(1); ! 332: } ! 333: lseek(kmem, (long)nl[X_ECMX].n_value, 0); ! 334: if (read(kmem, &ecmx, sizeof (ecmx)) != sizeof (ecmx)) { ! 335: cantread("ecmx", kmemf); ! 336: exit(1); ! 337: } ! 338: if (uflg || vflg) { ! 339: text = (struct text *)alloc(NTEXT * sizeof (struct text)); ! 340: if (text == 0) { ! 341: fprintf(stderr, "no room for text table\n"); ! 342: exit(1); ! 343: } ! 344: lseek(kmem, (long)nl[X_TEXT].n_value, 0); ! 345: if (read(kmem, (char *)text, NTEXT * sizeof (struct text)) ! 346: != NTEXT * sizeof (struct text)) { ! 347: cantread("text table", kmemf); ! 348: exit(1); ! 349: } ! 350: } ! 351: } ! 352: ! 353: printhdr() ! 354: { ! 355: char *hdr; ! 356: ! 357: if (sflg+lflg+vflg+uflg > 1) { ! 358: fprintf(stderr, "ps: specify only one of s,l,v and u\n"); ! 359: exit(1); ! 360: } ! 361: hdr = lflg ? lhdr : (vflg ? vhdr : (uflg ? uhdr : shdr)); ! 362: if (lflg+vflg+uflg+sflg == 0) ! 363: hdr += strlen("SSIZ "); ! 364: cmdstart = strlen(hdr); ! 365: printf("%s COMMAND\n", hdr); ! 366: fflush(stdout); ! 367: } ! 368: ! 369: cantread(what, fromwhat) ! 370: char *what, *fromwhat; ! 371: { ! 372: ! 373: fprintf(stderr, "ps: error reading %s from %s", what, fromwhat); ! 374: } ! 375: ! 376: struct direct dbuf; ! 377: int dialbase; ! 378: ! 379: getdev() ! 380: { ! 381: register FILE *df; ! 382: register struct ttys *dp; ! 383: ! 384: dialbase = -1; ! 385: if ((df = fopen(".", "r")) == NULL) { ! 386: fprintf(stderr, "Can't open . in /dev\n"); ! 387: exit(1); ! 388: } ! 389: while (fread((char *)&dbuf, sizeof(dbuf), 1, df) == 1) { ! 390: if (dbuf.d_ino == 0) ! 391: continue; ! 392: maybetty(dp); ! 393: } ! 394: fclose(df); ! 395: } ! 396: ! 397: /* ! 398: * Attempt to avoid stats by guessing minor device ! 399: * numbers from tty names. Console is known, ! 400: * know that r(hp|up|mt) are unlikely as are different mem's, ! 401: * floppy, null, tty, etc. ! 402: */ ! 403: maybetty() ! 404: { ! 405: register char *cp = dbuf.d_name; ! 406: register struct ttys *dp; ! 407: int x; ! 408: struct stat stb; ! 409: ! 410: switch (cp[0]) { ! 411: ! 412: case 'c': ! 413: if (!strcmp(cp, "console")) { ! 414: x = 0; ! 415: goto donecand; ! 416: } ! 417: /* cu[la]? are possible!?! don't rule them out */ ! 418: break; ! 419: ! 420: case 'd': ! 421: if (!strcmp(cp, "drum")) ! 422: return (0); ! 423: break; ! 424: ! 425: case 'f': ! 426: if (!strcmp(cp, "floppy")) ! 427: return (0); ! 428: break; ! 429: ! 430: case 'k': ! 431: cp++; ! 432: if (*cp == 'U') ! 433: cp++; ! 434: goto trymem; ! 435: ! 436: case 'r': ! 437: cp++; ! 438: if (*cp == 'r' || *cp == 'u' || *cp == 'h') ! 439: cp++; ! 440: #define is(a,b) cp[0] == 'a' && cp[1] == 'b' ! 441: if (is(r,p) || is(u,p) || is(r,k) || is(r,m) || is(m,t)) { ! 442: cp += 2; ! 443: if (isdigit(*cp) && cp[2] == 0) ! 444: return (0); ! 445: } ! 446: break; ! 447: ! 448: case 'm': ! 449: trymem: ! 450: if (cp[0] == 'm' && cp[1] == 'e' && cp[2] == 'm' && cp[3] == 0) ! 451: return (0); ! 452: break; ! 453: ! 454: case 'n': ! 455: if (!strcmp(cp, "null")) ! 456: return (0); ! 457: break; ! 458: ! 459: case 'v': ! 460: if ((cp[1] == 'a' || cp[1] == 'p') && isdigit(cp[2]) && ! 461: cp[3] == 0) ! 462: return (0); ! 463: break; ! 464: } ! 465: mightbe: ! 466: cp = dbuf.d_name; ! 467: while (cp < &dbuf.d_name[DIRSIZ] && *cp) ! 468: cp++; ! 469: --cp; ! 470: x = 0; ! 471: if (cp[-1] == 'd') { ! 472: if (dialbase == -1) { ! 473: if (stat("ttyd0", &stb) == 0) ! 474: dialbase = stb.st_rdev & 017; ! 475: else ! 476: dialbase = -2; ! 477: } ! 478: if (dialbase == -2) ! 479: x = 0; ! 480: else ! 481: x = 11; ! 482: } ! 483: if (cp > dbuf.d_name && isdigit(cp[-1]) && isdigit(*cp)) ! 484: x += 10 * (cp[-1] - ' ') + cp[0] - '0'; ! 485: else if (*cp >= 'a' && *cp <= 'f') ! 486: x += 10 + *cp - 'a'; ! 487: else if (isdigit(*cp)) ! 488: x += *cp - '0'; ! 489: else ! 490: x = -1; ! 491: donecand: ! 492: dp = (struct ttys *)alloc(sizeof (struct ttys)); ! 493: strncpy(dp->name, dbuf.d_name, DIRSIZ); ! 494: dp->next = allttys; ! 495: dp->ttyd = -1; ! 496: allttys = dp; ! 497: if (x == -1) ! 498: return; ! 499: x &= 017; ! 500: dp->cand = cand[x]; ! 501: cand[x] = dp; ! 502: } ! 503: ! 504: char * ! 505: gettty() ! 506: { ! 507: register char *p; ! 508: register struct ttys *dp; ! 509: struct stat stb; ! 510: int x; ! 511: ! 512: if (u.u_ttyp == 0) ! 513: return("?"); ! 514: x = u.u_ttyd & 017; ! 515: for (dp = cand[x]; dp; dp = dp->cand) { ! 516: if (dp->ttyd == -1) { ! 517: if (stat(dp->name, &stb) == 0 && ! 518: (stb.st_mode&S_IFMT)==S_IFCHR) ! 519: dp->ttyd = stb.st_rdev; ! 520: else ! 521: dp->ttyd = -2; ! 522: } ! 523: if (dp->ttyd == u.u_ttyd) ! 524: goto found; ! 525: } ! 526: /* ick */ ! 527: for (dp = allttys; dp; dp = dp->next) { ! 528: if (dp->ttyd == -1) { ! 529: if (stat(dp->name, &stb) == 0) ! 530: dp->ttyd = stb.st_rdev; ! 531: else ! 532: dp->ttyd = -2; ! 533: } ! 534: if (dp->ttyd == u.u_ttyd) ! 535: goto found; ! 536: } ! 537: return ("?"); ! 538: found: ! 539: p = dp->name; ! 540: if (p[0]=='t' && p[1]=='t' && p[2]=='y') ! 541: p += 3; ! 542: return (p); ! 543: } ! 544: ! 545: save() ! 546: { ! 547: register struct savcom *sp; ! 548: register struct asav *ap; ! 549: register char *cp; ! 550: register struct text *xp; ! 551: char *ttyp, *cmdp; ! 552: ! 553: if (mproc->p_stat != SZOMB && getu() == 0) ! 554: return; ! 555: ttyp = gettty(); ! 556: if (xflg == 0 && ttyp[0] == '?' || tptr && strcmpn(tptr, ttyp, 2)) ! 557: return; ! 558: sp = &savcom[npr]; ! 559: cmdp = getcmd(); ! 560: if (cmdp == 0) ! 561: return; ! 562: sp->ap = ap = (struct asav *)alloc(sizeof (struct asav)); ! 563: sp->ap->a_cmdp = cmdp; ! 564: #define e(a,b) ap->a = mproc->b ! 565: e(a_flag, p_flag); e(a_stat, p_stat); e(a_nice, p_nice); ! 566: e(a_uid, p_uid); e(a_pid, p_pid); e(a_pri, p_pri); ! 567: e(a_slptime, p_slptime); e(a_time, p_time); ! 568: ap->a_tty[0] = ttyp[0]; ! 569: ap->a_tty[1] = ttyp[1] ? ttyp[1] : ' '; ! 570: if (ap->a_stat == SZOMB) { ! 571: register struct xproc *xp = (struct xproc *)mproc; ! 572: ! 573: ap->a_cpu = xp->xp_vm.vm_utime + xp->xp_vm.vm_stime; ! 574: } else { ! 575: ap->a_size = mproc->p_dsize + mproc->p_ssize; ! 576: e(a_rss, p_rssize); ! 577: ap->a_ttyd = u.u_ttyd; ! 578: ap->a_cpu = u.u_vm.vm_utime + u.u_vm.vm_stime; ! 579: if (sumcpu) ! 580: ap->a_cpu += u.u_cvm.vm_utime + u.u_cvm.vm_stime; ! 581: if (mproc->p_textp && text) { ! 582: xp = &text[mproc->p_textp - ! 583: (struct text *)nl[X_TEXT].n_value]; ! 584: ap->a_tsiz = xp->x_size; ! 585: ap->a_txtrss = xp->x_rssize; ! 586: ap->a_xccount = xp->x_ccount; ! 587: } ! 588: } ! 589: #undef e ! 590: ap->a_cpu /= HZ; ! 591: if (lflg) { ! 592: register struct lsav *lp; ! 593: ! 594: sp->sun.lp = lp = (struct lsav *)alloc(sizeof (struct lsav)); ! 595: #define e(a,b) lp->a = mproc->b ! 596: e(l_ppid, p_ppid); e(l_cpu, p_cpu); ! 597: if (ap->a_stat != SZOMB) ! 598: e(l_wchan, p_wchan); ! 599: #undef e ! 600: lp->l_addr = pcbpf; ! 601: } else if (vflg) { ! 602: register struct vsav *vp; ! 603: ! 604: sp->sun.vp = vp = (struct vsav *)alloc(sizeof (struct vsav)); ! 605: #define e(a,b) vp->a = mproc->b ! 606: if (ap->a_stat != SZOMB) { ! 607: e(v_swrss, p_swrss); ! 608: vp->v_majflt = u.u_vm.vm_majflt; ! 609: if (mproc->p_textp) ! 610: vp->v_txtswrss = xp->x_swrss; ! 611: } ! 612: vp->v_pctcpu = pcpu(); ! 613: #undef e ! 614: } else if (uflg) ! 615: sp->sun.u_pctcpu = pcpu(); ! 616: else if (sflg) { ! 617: if (ap->a_stat != SZOMB) { ! 618: for (cp = (char *)u.u_stack; ! 619: cp < &user.upages[UPAGES][NBPG]; ) ! 620: if (*cp++) ! 621: break; ! 622: sp->sun.s_ssiz = (&user.upages[UPAGES][NBPG] - cp); ! 623: } ! 624: } ! 625: npr++; ! 626: } ! 627: ! 628: double ! 629: pmem(ap) ! 630: register struct asav *ap; ! 631: { ! 632: double fracmem; ! 633: int szptudot; ! 634: ! 635: if ((ap->a_flag&SLOAD) == 0) ! 636: fracmem = 0.0; ! 637: else { ! 638: szptudot = UPAGES + clrnd(ctopt(ap->a_size+ap->a_tsiz)); ! 639: fracmem = ((float)ap->a_rss+szptudot)/CLSIZE/ecmx; ! 640: if (ap->a_xccount) ! 641: fracmem += ((float)ap->a_txtrss)/CLSIZE/ ! 642: ap->a_xccount/ecmx; ! 643: } ! 644: return (100.0 * fracmem); ! 645: } ! 646: ! 647: double ! 648: pcpu() ! 649: { ! 650: time_t time; ! 651: ! 652: time = mproc->p_time; ! 653: if (time == 0 || (mproc->p_flag&SLOAD) == 0) ! 654: return (0.0); ! 655: if (rawcpu) ! 656: return (100.0 * mproc->p_pctcpu); ! 657: return (100.0 * mproc->p_pctcpu / (1.0 - exp(time * log(ccpu)))); ! 658: } ! 659: ! 660: getu() ! 661: { ! 662: struct pte *pteaddr, apte; ! 663: int pad1; /* avoid hardware botch */ ! 664: struct pte arguutl[UPAGES+CLSIZE]; ! 665: int pad2; /* avoid hardware botch */ ! 666: register int i; ! 667: int ncl, size; ! 668: ! 669: size = sflg ? ctob(UPAGES) : sizeof (struct user); ! 670: if ((mproc->p_flag & SLOAD) == 0) { ! 671: lseek(swap, ctob(mproc->p_swaddr), 0); ! 672: if (read(swap, (char *)&user.user, size) != size) { ! 673: fprintf(stderr, "ps: cant read u for pid %d from %s\n", ! 674: mproc->p_pid, swapf); ! 675: return (0); ! 676: } ! 677: pcbpf = 0; ! 678: argaddr = 0; ! 679: return (1); ! 680: } ! 681: pteaddr = &Usrptma[btokmx(mproc->p_p0br) + mproc->p_szpt - 1]; ! 682: lseek(kmem, kflg ? clear(pteaddr) : (int)pteaddr, 0); ! 683: if (read(kmem, (char *)&apte, sizeof(apte)) != sizeof(apte)) { ! 684: printf("ps: cant read indir pte to get u for pid %d from %s\n", ! 685: mproc->p_pid, swapf); ! 686: return (0); ! 687: } ! 688: lseek(mem, ! 689: ctob(apte.pg_pfnum+1) - (UPAGES+CLSIZE) * sizeof (struct pte), 0); ! 690: if (read(mem, (char *)arguutl, sizeof(arguutl)) != sizeof(arguutl)) { ! 691: printf("ps: cant read page table for u of pid %d from %s\n", ! 692: mproc->p_pid, swapf); ! 693: return (0); ! 694: } ! 695: if (arguutl[0].pg_fod == 0 && arguutl[0].pg_pfnum) ! 696: argaddr = ctob(arguutl[0].pg_pfnum); ! 697: else ! 698: argaddr = 0; ! 699: pcbpf = arguutl[CLSIZE].pg_pfnum; ! 700: ncl = (size + NBPG*CLSIZE - 1) / (NBPG*CLSIZE); ! 701: while (--ncl >= 0) { ! 702: i = ncl * CLSIZE; ! 703: lseek(mem, ctob(arguutl[CLSIZE+i].pg_pfnum), 0); ! 704: if (read(mem, user.upages[i], CLSIZE*NBPG) != CLSIZE*NBPG) { ! 705: printf("ps: cant read page %d of u of pid %d from %s\n", ! 706: arguutl[CLSIZE+i].pg_pfnum, mproc->p_pid, memf); ! 707: return(0); ! 708: } ! 709: } ! 710: return (1); ! 711: } ! 712: ! 713: char * ! 714: getcmd() ! 715: { ! 716: char cmdbuf[BUFSIZ]; ! 717: int pad1; /* avoid hardware botch */ ! 718: union { ! 719: char argc[CLSIZE*NBPG]; ! 720: int argi[CLSIZE*NBPG/sizeof (int)]; ! 721: } argspac; ! 722: int pad2; /* avoid hardware botch */ ! 723: register char *cp; ! 724: register int *ip; ! 725: char c; ! 726: int nbad; ! 727: struct dblock db; ! 728: ! 729: if (mproc->p_stat == SZOMB || mproc->p_flag&(SSYS|SWEXIT)) ! 730: return (""); ! 731: if (cflg) { ! 732: strncpy(cmdbuf, u.u_comm, sizeof (u.u_comm)); ! 733: return (savestr(cmdbuf)); ! 734: } ! 735: if ((mproc->p_flag & SLOAD) == 0 || argaddr == 0) { ! 736: vstodb(0, CLSIZE, &u.u_smap, &db, 1); ! 737: lseek(swap, ctob(db.db_base), 0); ! 738: if (read(swap, (char *)&argspac, sizeof(argspac)) ! 739: != sizeof(argspac)) ! 740: goto bad; ! 741: } else { ! 742: lseek(mem, argaddr, 0); ! 743: if (read(mem, (char *)&argspac, sizeof (argspac)) ! 744: != sizeof (argspac)) ! 745: goto bad; ! 746: } ! 747: ip = &argspac.argi[CLSIZE*NBPG/sizeof (int)]; ! 748: ip -= 2; /* last arg word and .long 0 */ ! 749: while (*--ip) ! 750: if (ip == argspac.argi) ! 751: goto retucomm; ! 752: *(char *)ip = ' '; ! 753: ip++; ! 754: nbad = 0; ! 755: for (cp = (char *)ip; cp < &argspac.argc[CLSIZE*NBPG]; cp++) { ! 756: c = *cp & 0177; ! 757: if (c == 0) ! 758: *cp = ' '; ! 759: else if (c < ' ' || c > 0176) { ! 760: if (++nbad >= 5*(eflg+1)) { ! 761: *cp++ = ' '; ! 762: break; ! 763: } ! 764: *cp = '?'; ! 765: } else if (eflg == 0 && c == '=') { ! 766: while (*--cp != ' ') ! 767: if (cp <= (char *)ip) ! 768: break; ! 769: break; ! 770: } ! 771: } ! 772: *cp = 0; ! 773: while (*--cp == ' ') ! 774: *cp = 0; ! 775: cp = (char *)ip; ! 776: strncpy(cmdbuf, cp, &argspac.argc[CLSIZE*NBPG] - cp); ! 777: if (cp[0] == '-' || cp[0] == '?' || cp[0] <= ' ') { ! 778: strcat(cmdbuf, " ("); ! 779: strncat(cmdbuf, u.u_comm, sizeof(u.u_comm)); ! 780: strcat(cmdbuf, ")"); ! 781: } ! 782: /* ! 783: if (xflg == 0 && gflg == 0 && tptr == 0 && cp[0] == '-') ! 784: return (0); ! 785: */ ! 786: return (savestr(cmdbuf)); ! 787: ! 788: bad: ! 789: fprintf(stderr, "ps: error locating command name for pid %d\n", ! 790: mproc->p_pid); ! 791: retucomm: ! 792: strcpy(cmdbuf, " ("); ! 793: strncat(cmdbuf, u.u_comm, sizeof (u.u_comm)); ! 794: strcat(cmdbuf, ")"); ! 795: return (savestr(cmdbuf)); ! 796: } ! 797: ! 798: char *lhdr = ! 799: " F UID PID PPID CP PRI NI ADDR SZ RSS WCHAN STAT TT TIME"; ! 800: lpr(sp) ! 801: struct savcom *sp; ! 802: { ! 803: register struct asav *ap = sp->ap; ! 804: register struct lsav *lp = sp->sun.lp; ! 805: ! 806: printf("%6x%4d%6u%6u%3d%4d%3d%5x%4d%5d", ! 807: ap->a_flag, ap->a_uid, ! 808: ap->a_pid, lp->l_ppid, lp->l_cpu&0377, ap->a_pri-PZERO, ! 809: ap->a_nice-NZERO, lp->l_addr, ap->a_size/2, ap->a_rss/2); ! 810: printf(lp->l_wchan ? " %5x" : " ", (int)lp->l_wchan&0xfffff); ! 811: printf(" %4.4s ", state(ap)); ! 812: ptty(ap->a_tty); ! 813: ptime(ap); ! 814: } ! 815: ! 816: ptty(tp) ! 817: char *tp; ! 818: { ! 819: ! 820: printf("%-2.2s", tp); ! 821: } ! 822: ! 823: ptime(ap) ! 824: struct asav *ap; ! 825: { ! 826: ! 827: printf("%3ld:%02ld", ap->a_cpu / HZ, ap->a_cpu % HZ); ! 828: } ! 829: ! 830: char *uhdr = ! 831: "USER PID %CPU %MEM SZ RSS TT STAT TIME"; ! 832: upr(sp) ! 833: struct savcom *sp; ! 834: { ! 835: register struct asav *ap = sp->ap; ! 836: int vmsize, rmsize; ! 837: ! 838: vmsize = (ap->a_size + ap->a_tsiz)/2; ! 839: rmsize = ap->a_rss/2; ! 840: if (ap->a_xccount) ! 841: rmsize += ap->a_txtrss/ap->a_xccount/2; ! 842: printf("%-8.8s %5d%5.1f%5.1f%5d%5d", ! 843: getname(ap->a_uid), ap->a_pid, sp->sun.u_pctcpu, pmem(ap), ! 844: vmsize, rmsize); ! 845: putchar(' '); ! 846: ptty(ap->a_tty); ! 847: printf(" %4.4s", state(ap)); ! 848: ptime(ap); ! 849: } ! 850: ! 851: char *vhdr = ! 852: " PID TT STAT TIME SL RE PAGEIN SIZE RSS SRS TSIZ TRS %CPU %MEM"; ! 853: vpr(sp) ! 854: struct savcom *sp; ! 855: { ! 856: register struct vsav *vp = sp->sun.vp; ! 857: register struct asav *ap = sp->ap; ! 858: ! 859: printf("%5u ", ap->a_pid); ! 860: ptty(ap->a_tty); ! 861: printf(" %4.4s", state(ap)); ! 862: ptime(ap); ! 863: printf("%3d%3d%7d%5d%5d%5d%5d%4d%5.1f%5.1f", ! 864: ap->a_slptime, ap->a_time > 99 ? 99 : ap->a_time, vp->v_majflt, ! 865: ap->a_size/2, ap->a_rss/2, vp->v_swrss/2, ! 866: ap->a_tsiz/2, ap->a_txtrss/2, vp->v_pctcpu, pmem(ap)); ! 867: } ! 868: ! 869: char *shdr = ! 870: "SSIZ PID TT STAT TIME"; ! 871: spr(sp) ! 872: struct savcom *sp; ! 873: { ! 874: register struct asav *ap = sp->ap; ! 875: ! 876: if (sflg) ! 877: printf("%4d ", sp->sun.s_ssiz); ! 878: printf("%5u", ap->a_pid); ! 879: putchar(' '); ! 880: ptty(ap->a_tty); ! 881: printf(" %4.4s", state(ap)); ! 882: ptime(ap); ! 883: } ! 884: ! 885: char * ! 886: state(ap) ! 887: register struct asav *ap; ! 888: { ! 889: char stat, load, nice, anom; ! 890: static char res[5]; ! 891: ! 892: switch (ap->a_stat) { ! 893: ! 894: case SSTOP: ! 895: stat = 'T'; ! 896: break; ! 897: ! 898: case SSLEEP: ! 899: if (ap->a_pri >= PZERO) ! 900: if (ap->a_slptime >= MAXSLP) ! 901: stat = 'I'; ! 902: else ! 903: stat = 'S'; ! 904: else if (ap->a_flag & SPAGE) ! 905: stat = 'P'; ! 906: else ! 907: stat = 'D'; ! 908: break; ! 909: ! 910: case SWAIT: ! 911: case SRUN: ! 912: case SIDL: ! 913: stat = 'R'; ! 914: break; ! 915: ! 916: case SZOMB: ! 917: stat = 'Z'; ! 918: break; ! 919: ! 920: default: ! 921: stat = '?'; ! 922: } ! 923: load = ap->a_flag & SLOAD ? ' ' : 'W'; ! 924: if (ap->a_nice < NZERO) ! 925: nice = '<'; ! 926: else if (ap->a_nice > NZERO) ! 927: nice = 'N'; ! 928: else ! 929: nice = ' '; ! 930: anom = ap->a_flag & (SANOM|SUANOM) ? 'A' : ' '; ! 931: res[0] = stat; res[1] = load; res[2] = nice; res[3] = anom; ! 932: return (res); ! 933: } ! 934: ! 935: /* ! 936: * Given a base/size pair in virtual swap area, ! 937: * return a physical base/size pair which is the ! 938: * (largest) initial, physically contiguous block. ! 939: */ ! 940: vstodb(vsbase, vssize, dmp, dbp, rev) ! 941: register int vsbase; ! 942: int vssize; ! 943: struct dmap *dmp; ! 944: register struct dblock *dbp; ! 945: { ! 946: register int blk = DMMIN; ! 947: register swblk_t *ip = dmp->dm_map; ! 948: ! 949: if (vsbase < 0 || vsbase + vssize > dmp->dm_size) ! 950: panic("vstodb"); ! 951: while (vsbase >= blk) { ! 952: vsbase -= blk; ! 953: if (blk < DMMAX) ! 954: blk *= 2; ! 955: ip++; ! 956: } ! 957: if (*ip <= 0 || *ip + blk > nswap) ! 958: panic("vstodb *ip"); ! 959: dbp->db_size = min(vssize, blk - vsbase); ! 960: dbp->db_base = *ip + (rev ? blk - (vsbase + dbp->db_size) : vsbase); ! 961: } ! 962: ! 963: /*ARGSUSED*/ ! 964: panic(cp) ! 965: char *cp; ! 966: { ! 967: ! 968: #ifdef DEBUG ! 969: printf("%s\n", cp); ! 970: #endif ! 971: } ! 972: ! 973: min(a, b) ! 974: { ! 975: ! 976: return (a < b ? a : b); ! 977: } ! 978: ! 979: pscomp(s1, s2) ! 980: struct savcom *s1, *s2; ! 981: { ! 982: register int i; ! 983: ! 984: if (uflg) ! 985: return (s2->sun.u_pctcpu > s1->sun.u_pctcpu ? 1 : -1); ! 986: if (vflg) ! 987: return (vsize(s2) - vsize(s1)); ! 988: i = s1->ap->a_ttyd - s2->ap->a_ttyd; ! 989: if (i == 0) ! 990: i = s1->ap->a_pid - s2->ap->a_pid; ! 991: return (i); ! 992: } ! 993: ! 994: vsize(sp) ! 995: struct savcom *sp; ! 996: { ! 997: register struct asav *ap = sp->ap; ! 998: register struct vsav *vp = sp->sun.vp; ! 999: ! 1000: if (ap->a_flag & SLOAD) ! 1001: return (ap->a_rss + ! 1002: ap->a_txtrss / (ap->a_xccount ? ap->a_xccount : 1)); ! 1003: return (vp->v_swrss + (ap->a_xccount ? 0 : vp->v_txtswrss)); ! 1004: } ! 1005: ! 1006: #define NMAX 8 ! 1007: #define NUID 2048 ! 1008: ! 1009: char names[NUID][NMAX+1]; ! 1010: ! 1011: /* ! 1012: * Stolen from ls... ! 1013: */ ! 1014: char * ! 1015: getname(uid) ! 1016: { ! 1017: register struct passwd *pw; ! 1018: static init; ! 1019: struct passwd *getpwent(); ! 1020: ! 1021: if (uid >= 0 && uid < NUID && names[uid][0]) ! 1022: return (&names[uid][0]); ! 1023: if (init == 2) ! 1024: return (0); ! 1025: if (init == 0) ! 1026: setpwent(), init = 1; ! 1027: while (pw = getpwent()) { ! 1028: if (pw->pw_uid >= NUID) ! 1029: continue; ! 1030: if (names[pw->pw_uid][0]) ! 1031: continue; ! 1032: strncpy(names[pw->pw_uid], pw->pw_name, NMAX); ! 1033: if (pw->pw_uid == uid) ! 1034: return (&names[uid][0]); ! 1035: } ! 1036: init = 2; ! 1037: endpwent(); ! 1038: return (0); ! 1039: } ! 1040: ! 1041: char *freebase; ! 1042: int nleft; ! 1043: ! 1044: char * ! 1045: alloc(size) ! 1046: int size; ! 1047: { ! 1048: register char *cp; ! 1049: register int i; ! 1050: ! 1051: if (size > nleft) { ! 1052: freebase = (char *)sbrk(i = size > 2048 ? size : 2048); ! 1053: if (freebase == 0) { ! 1054: fprintf(stderr, "ps: ran out of memory\n"); ! 1055: exit(1); ! 1056: } ! 1057: nleft = i - size; ! 1058: } else ! 1059: nleft -= size; ! 1060: cp = freebase; ! 1061: for (i = size; --i >= 0; ) ! 1062: *cp++ = 0; ! 1063: freebase = cp; ! 1064: return (cp - size); ! 1065: } ! 1066: ! 1067: char * ! 1068: savestr(cp) ! 1069: char *cp; ! 1070: { ! 1071: register int len; ! 1072: register char *dp; ! 1073: ! 1074: len = strlen(cp); ! 1075: dp = (char *)alloc(len+1); ! 1076: strcpy(dp, cp); ! 1077: return (dp); ! 1078: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.