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