Annotation of 42BSD/bin/ps.c, revision 1.1

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: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.