Annotation of 42BSD/etc/telnetd.c, revision 1.1

1.1     ! root        1: #ifndef lint
        !             2: static char sccsid[] = "@(#)telnetd.c  4.26 (Berkeley) 83/08/06";
        !             3: #endif
        !             4: 
        !             5: /*
        !             6:  * Stripped-down telnet server.
        !             7:  */
        !             8: #include <sys/types.h>
        !             9: #include <sys/socket.h>
        !            10: #include <sys/wait.h>
        !            11: 
        !            12: #include <netinet/in.h>
        !            13: 
        !            14: #include <arpa/telnet.h>
        !            15: 
        !            16: #include <stdio.h>
        !            17: #include <signal.h>
        !            18: #include <errno.h>
        !            19: #include <sgtty.h>
        !            20: #include <netdb.h>
        !            21: 
        !            22: #define        BELL    '\07'
        !            23: #define BANNER "\r\n\r\n4.2 BSD UNIX (%s)\r\n\r\r\n\r%s"
        !            24: 
        !            25: char   hisopts[256];
        !            26: char   myopts[256];
        !            27: 
        !            28: char   doopt[] = { IAC, DO, '%', 'c', 0 };
        !            29: char   dont[] = { IAC, DONT, '%', 'c', 0 };
        !            30: char   will[] = { IAC, WILL, '%', 'c', 0 };
        !            31: char   wont[] = { IAC, WONT, '%', 'c', 0 };
        !            32: 
        !            33: /*
        !            34:  * I/O data buffers, pointers, and counters.
        !            35:  */
        !            36: char   ptyibuf[BUFSIZ], *ptyip = ptyibuf;
        !            37: char   ptyobuf[BUFSIZ], *pfrontp = ptyobuf, *pbackp = ptyobuf;
        !            38: char   netibuf[BUFSIZ], *netip = netibuf;
        !            39: char   netobuf[BUFSIZ], *nfrontp = netobuf, *nbackp = netobuf;
        !            40: int    pcc, ncc;
        !            41: 
        !            42: int    pty, net;
        !            43: int    inter;
        !            44: int    reapchild();
        !            45: extern char **environ;
        !            46: extern int errno;
        !            47: char   line[] = "/dev/ptyp0";
        !            48: 
        !            49: struct sockaddr_in sin = { AF_INET };
        !            50: 
        !            51: main(argc, argv)
        !            52:        char *argv[];
        !            53: {
        !            54:        int s, pid, options;
        !            55:        struct servent *sp;
        !            56: 
        !            57:        sp = getservbyname("telnet", "tcp");
        !            58:        if (sp == 0) {
        !            59:                fprintf(stderr, "telnetd: tcp/telnet: unknown service\n");
        !            60:                exit(1);
        !            61:        }
        !            62:        sin.sin_port = sp->s_port;
        !            63:        argc--, argv++;
        !            64:        if (argc > 0 && !strcmp(*argv, "-d")) {
        !            65:                options |= SO_DEBUG;
        !            66:                argc--, argv++;
        !            67:        }
        !            68:        if (argc > 0) {
        !            69:                sin.sin_port = atoi(*argv);
        !            70:                if (sin.sin_port <= 0) {
        !            71:                        fprintf(stderr, "telnetd: %s: bad port #\n", *argv);
        !            72:                        exit(1);
        !            73:                }
        !            74:                sin.sin_port = htons((u_short)sin.sin_port);
        !            75:        }
        !            76: #ifndef DEBUG
        !            77:        if (fork())
        !            78:                exit(0);
        !            79:        for (s = 0; s < 10; s++)
        !            80:                (void) close(s);
        !            81:        (void) open("/", 0);
        !            82:        (void) dup2(0, 1);
        !            83:        (void) dup2(0, 2);
        !            84:        { int tt = open("/dev/tty", 2);
        !            85:          if (tt > 0) {
        !            86:                ioctl(tt, TIOCNOTTY, 0);
        !            87:                close(tt);
        !            88:          }
        !            89:        }
        !            90: #endif
        !            91: again:
        !            92:        s = socket(AF_INET, SOCK_STREAM, 0, 0);
        !            93:        if (s < 0) {
        !            94:                perror("telnetd: socket");;
        !            95:                sleep(5);
        !            96:                goto again;
        !            97:        }
        !            98:        if (options & SO_DEBUG)
        !            99:                if (setsockopt(s, SOL_SOCKET, SO_DEBUG, 0, 0) < 0)
        !           100:                        perror("telnetd: setsockopt (SO_DEBUG)");
        !           101:        if (setsockopt(s, SOL_SOCKET, SO_KEEPALIVE, 0, 0) < 0)
        !           102:                perror("telnetd: setsockopt (SO_KEEPALIVE)");
        !           103:        while (bind(s, (caddr_t)&sin, sizeof (sin), 0) < 0) {
        !           104:                perror("telnetd: bind");
        !           105:                sleep(5);
        !           106:        }
        !           107:        signal(SIGCHLD, reapchild);
        !           108:        listen(s, 10);
        !           109:        for (;;) {
        !           110:                struct sockaddr_in from;
        !           111:                int s2, fromlen = sizeof (from);
        !           112: 
        !           113:                s2 = accept(s, (caddr_t)&from, &fromlen);
        !           114:                if (s2 < 0) {
        !           115:                        if (errno == EINTR)
        !           116:                                continue;
        !           117:                        perror("telnetd: accept");
        !           118:                        sleep(1);
        !           119:                        continue;
        !           120:                }
        !           121:                if ((pid = fork()) < 0)
        !           122:                        printf("Out of processes\n");
        !           123:                else if (pid == 0) {
        !           124:                        signal(SIGCHLD, SIG_DFL);
        !           125:                        doit(s2, &from);
        !           126:                }
        !           127:                close(s2);
        !           128:        }
        !           129:        /*NOTREACHED*/
        !           130: }
        !           131: 
        !           132: reapchild()
        !           133: {
        !           134:        union wait status;
        !           135: 
        !           136:        while (wait3(&status, WNOHANG, 0) > 0)
        !           137:                ;
        !           138: }
        !           139: 
        !           140: char   *envinit[] = { "TERM=network", 0 };
        !           141: int    cleanup();
        !           142: 
        !           143: /*
        !           144:  * Get a pty, scan input lines.
        !           145:  */
        !           146: doit(f, who)
        !           147:        int f;
        !           148:        struct sockaddr_in *who;
        !           149: {
        !           150:        char *cp = line, *host, *ntoa();
        !           151:        int i, p, cc, t;
        !           152:        struct sgttyb b;
        !           153:        struct hostent *hp;
        !           154: 
        !           155:        for (i = 0; i < 16; i++) {
        !           156:                cp[strlen("/dev/ptyp")] = "0123456789abcdef"[i];
        !           157:                p = open(cp, 2);
        !           158:                if (p > 0)
        !           159:                        goto gotpty;
        !           160:        }
        !           161:        fatal(f, "All network ports in use");
        !           162:        /*NOTREACHED*/
        !           163: gotpty:
        !           164:        dup2(f, 0);
        !           165:        cp[strlen("/dev/")] = 't';
        !           166:        t = open("/dev/tty", 2);
        !           167:        if (t >= 0) {
        !           168:                ioctl(t, TIOCNOTTY, 0);
        !           169:                close(t);
        !           170:        }
        !           171:        t = open(cp, 2);
        !           172:        if (t < 0)
        !           173:                fatalperror(f, cp, errno);
        !           174:        ioctl(t, TIOCGETP, &b);
        !           175:        b.sg_flags = CRMOD|XTABS|ANYP;
        !           176:        ioctl(t, TIOCSETP, &b);
        !           177:        ioctl(p, TIOCGETP, &b);
        !           178:        b.sg_flags &= ~ECHO;
        !           179:        ioctl(p, TIOCSETP, &b);
        !           180:        hp = gethostbyaddr(&who->sin_addr, sizeof (struct in_addr),
        !           181:                who->sin_family);
        !           182:        if (hp)
        !           183:                host = hp->h_name;
        !           184:        else
        !           185:                host = ntoa(who->sin_addr);
        !           186:        if ((i = fork()) < 0)
        !           187:                fatalperror(f, "fork", errno);
        !           188:        if (i)
        !           189:                telnet(f, p);
        !           190:        close(f);
        !           191:        close(p);
        !           192:        dup2(t, 0);
        !           193:        dup2(t, 1);
        !           194:        dup2(t, 2);
        !           195:        close(t);
        !           196:        environ = envinit;
        !           197:        execl("/bin/login", "login", "-h", host, 0);
        !           198:        fatalperror(f, "/bin/login", errno);
        !           199:        /*NOTREACHED*/
        !           200: }
        !           201: 
        !           202: fatal(f, msg)
        !           203:        int f;
        !           204:        char *msg;
        !           205: {
        !           206:        char buf[BUFSIZ];
        !           207: 
        !           208:        (void) sprintf(buf, "telnetd: %s.\n", msg);
        !           209:        (void) write(f, buf, strlen(buf));
        !           210:        exit(1);
        !           211: }
        !           212: 
        !           213: fatalperror(f, msg, errno)
        !           214:        int f;
        !           215:        char *msg;
        !           216:        int errno;
        !           217: {
        !           218:        char buf[BUFSIZ];
        !           219:        extern char *sys_errlist[];
        !           220: 
        !           221:        (void) sprintf(buf, "%s: %s", msg, sys_errlist[errno]);
        !           222:        fatal(f, buf);
        !           223: }
        !           224: 
        !           225: /*
        !           226:  * Main loop.  Select from pty and network, and
        !           227:  * hand data to telnet receiver finite state machine.
        !           228:  */
        !           229: telnet(f, p)
        !           230: {
        !           231:        int on = 1;
        !           232:        char hostname[32];
        !           233: 
        !           234:        net = f, pty = p;
        !           235:        ioctl(f, FIONBIO, &on);
        !           236:        ioctl(p, FIONBIO, &on);
        !           237:        signal(SIGTSTP, SIG_IGN);
        !           238:        signal(SIGCHLD, cleanup);
        !           239: 
        !           240:        /*
        !           241:         * Request to do remote echo.
        !           242:         */
        !           243:        dooption(TELOPT_ECHO);
        !           244:        myopts[TELOPT_ECHO] = 1;
        !           245:        /*
        !           246:         * Show banner that getty never gave.
        !           247:         */
        !           248:        gethostname(hostname, sizeof (hostname));
        !           249:        sprintf(nfrontp, BANNER, hostname, "");
        !           250:        nfrontp += strlen(nfrontp);
        !           251:        for (;;) {
        !           252:                int ibits = 0, obits = 0;
        !           253:                register int c;
        !           254: 
        !           255:                /*
        !           256:                 * Never look for input if there's still
        !           257:                 * stuff in the corresponding output buffer
        !           258:                 */
        !           259:                if (nfrontp - nbackp)
        !           260:                        obits |= (1 << f);
        !           261:                else
        !           262:                        ibits |= (1 << p);
        !           263:                if (pfrontp - pbackp)
        !           264:                        obits |= (1 << p);
        !           265:                else
        !           266:                        ibits |= (1 << f);
        !           267:                if (ncc < 0 && pcc < 0)
        !           268:                        break;
        !           269:                select(16, &ibits, &obits, 0, 0);
        !           270:                if (ibits == 0 && obits == 0) {
        !           271:                        sleep(5);
        !           272:                        continue;
        !           273:                }
        !           274: 
        !           275:                /*
        !           276:                 * Something to read from the network...
        !           277:                 */
        !           278:                if (ibits & (1 << f)) {
        !           279:                        ncc = read(f, netibuf, BUFSIZ);
        !           280:                        if (ncc < 0 && errno == EWOULDBLOCK)
        !           281:                                ncc = 0;
        !           282:                        else {
        !           283:                                if (ncc <= 0)
        !           284:                                        break;
        !           285:                                netip = netibuf;
        !           286:                        }
        !           287:                }
        !           288: 
        !           289:                /*
        !           290:                 * Something to read from the pty...
        !           291:                 */
        !           292:                if (ibits & (1 << p)) {
        !           293:                        pcc = read(p, ptyibuf, BUFSIZ);
        !           294:                        if (pcc < 0 && errno == EWOULDBLOCK)
        !           295:                                pcc = 0;
        !           296:                        else {
        !           297:                                if (pcc <= 0)
        !           298:                                        break;
        !           299:                                ptyip = ptyibuf;
        !           300:                        }
        !           301:                }
        !           302: 
        !           303:                while (pcc > 0) {
        !           304:                        if ((&netobuf[BUFSIZ] - nfrontp) < 2)
        !           305:                                break;
        !           306:                        c = *ptyip++ & 0377, pcc--;
        !           307:                        if (c == IAC)
        !           308:                                *nfrontp++ = c;
        !           309:                        *nfrontp++ = c;
        !           310:                }
        !           311:                if ((obits & (1 << f)) && (nfrontp - nbackp) > 0)
        !           312:                        netflush();
        !           313:                if (ncc > 0)
        !           314:                        telrcv();
        !           315:                if ((obits & (1 << p)) && (pfrontp - pbackp) > 0)
        !           316:                        ptyflush();
        !           317:        }
        !           318:        cleanup();
        !           319: }
        !           320:        
        !           321: /*
        !           322:  * State for recv fsm
        !           323:  */
        !           324: #define        TS_DATA         0       /* base state */
        !           325: #define        TS_IAC          1       /* look for double IAC's */
        !           326: #define        TS_CR           2       /* CR-LF ->'s CR */
        !           327: #define        TS_BEGINNEG     3       /* throw away begin's... */
        !           328: #define        TS_ENDNEG       4       /* ...end's (suboption negotiation) */
        !           329: #define        TS_WILL         5       /* will option negotiation */
        !           330: #define        TS_WONT         6       /* wont " */
        !           331: #define        TS_DO           7       /* do " */
        !           332: #define        TS_DONT         8       /* dont " */
        !           333: 
        !           334: telrcv()
        !           335: {
        !           336:        register int c;
        !           337:        static int state = TS_DATA;
        !           338:        struct sgttyb b;
        !           339: 
        !           340:        while (ncc > 0) {
        !           341:                if ((&ptyobuf[BUFSIZ] - pfrontp) < 2)
        !           342:                        return;
        !           343:                c = *netip++ & 0377, ncc--;
        !           344:                switch (state) {
        !           345: 
        !           346:                case TS_DATA:
        !           347:                        if (c == IAC) {
        !           348:                                state = TS_IAC;
        !           349:                                break;
        !           350:                        }
        !           351:                        if (inter > 0)
        !           352:                                break;
        !           353:                        *pfrontp++ = c;
        !           354:                        if (!myopts[TELOPT_BINARY] && c == '\r')
        !           355:                                state = TS_CR;
        !           356:                        break;
        !           357: 
        !           358:                case TS_CR:
        !           359:                        if (c && c != '\n')
        !           360:                                *pfrontp++ = c;
        !           361:                        state = TS_DATA;
        !           362:                        break;
        !           363: 
        !           364:                case TS_IAC:
        !           365:                        switch (c) {
        !           366: 
        !           367:                        /*
        !           368:                         * Send the process on the pty side an
        !           369:                         * interrupt.  Do this with a NULL or
        !           370:                         * interrupt char; depending on the tty mode.
        !           371:                         */
        !           372:                        case BREAK:
        !           373:                        case IP:
        !           374:                                interrupt();
        !           375:                                break;
        !           376: 
        !           377:                        /*
        !           378:                         * Are You There?
        !           379:                         */
        !           380:                        case AYT:
        !           381:                                *pfrontp++ = BELL;
        !           382:                                break;
        !           383: 
        !           384:                        /*
        !           385:                         * Erase Character and
        !           386:                         * Erase Line
        !           387:                         */
        !           388:                        case EC:
        !           389:                        case EL:
        !           390:                                ptyflush();     /* half-hearted */
        !           391:                                ioctl(pty, TIOCGETP, &b);
        !           392:                                *pfrontp++ = (c == EC) ?
        !           393:                                        b.sg_erase : b.sg_kill;
        !           394:                                break;
        !           395: 
        !           396:                        /*
        !           397:                         * Check for urgent data...
        !           398:                         */
        !           399:                        case DM:
        !           400:                                break;
        !           401: 
        !           402:                        /*
        !           403:                         * Begin option subnegotiation...
        !           404:                         */
        !           405:                        case SB:
        !           406:                                state = TS_BEGINNEG;
        !           407:                                continue;
        !           408: 
        !           409:                        case WILL:
        !           410:                        case WONT:
        !           411:                        case DO:
        !           412:                        case DONT:
        !           413:                                state = TS_WILL + (c - WILL);
        !           414:                                continue;
        !           415: 
        !           416:                        case IAC:
        !           417:                                *pfrontp++ = c;
        !           418:                                break;
        !           419:                        }
        !           420:                        state = TS_DATA;
        !           421:                        break;
        !           422: 
        !           423:                case TS_BEGINNEG:
        !           424:                        if (c == IAC)
        !           425:                                state = TS_ENDNEG;
        !           426:                        break;
        !           427: 
        !           428:                case TS_ENDNEG:
        !           429:                        state = c == SE ? TS_DATA : TS_BEGINNEG;
        !           430:                        break;
        !           431: 
        !           432:                case TS_WILL:
        !           433:                        if (!hisopts[c])
        !           434:                                willoption(c);
        !           435:                        state = TS_DATA;
        !           436:                        continue;
        !           437: 
        !           438:                case TS_WONT:
        !           439:                        if (hisopts[c])
        !           440:                                wontoption(c);
        !           441:                        state = TS_DATA;
        !           442:                        continue;
        !           443: 
        !           444:                case TS_DO:
        !           445:                        if (!myopts[c])
        !           446:                                dooption(c);
        !           447:                        state = TS_DATA;
        !           448:                        continue;
        !           449: 
        !           450:                case TS_DONT:
        !           451:                        if (myopts[c]) {
        !           452:                                myopts[c] = 0;
        !           453:                                sprintf(nfrontp, wont, c);
        !           454:                                nfrontp += sizeof (wont) - 2;
        !           455:                        }
        !           456:                        state = TS_DATA;
        !           457:                        continue;
        !           458: 
        !           459:                default:
        !           460:                        printf("telnetd: panic state=%d\n", state);
        !           461:                        exit(1);
        !           462:                }
        !           463:        }
        !           464: }
        !           465: 
        !           466: willoption(option)
        !           467:        int option;
        !           468: {
        !           469:        char *fmt;
        !           470: 
        !           471:        switch (option) {
        !           472: 
        !           473:        case TELOPT_BINARY:
        !           474:                mode(RAW, 0);
        !           475:                goto common;
        !           476: 
        !           477:        case TELOPT_ECHO:
        !           478:                mode(0, ECHO|CRMOD);
        !           479:                /*FALL THRU*/
        !           480: 
        !           481:        case TELOPT_SGA:
        !           482:        common:
        !           483:                hisopts[option] = 1;
        !           484:                fmt = doopt;
        !           485:                break;
        !           486: 
        !           487:        case TELOPT_TM:
        !           488:                fmt = dont;
        !           489:                break;
        !           490: 
        !           491:        default:
        !           492:                fmt = dont;
        !           493:                break;
        !           494:        }
        !           495:        sprintf(nfrontp, fmt, option);
        !           496:        nfrontp += sizeof (dont) - 2;
        !           497: }
        !           498: 
        !           499: wontoption(option)
        !           500:        int option;
        !           501: {
        !           502:        char *fmt;
        !           503: 
        !           504:        switch (option) {
        !           505: 
        !           506:        case TELOPT_ECHO:
        !           507:                mode(ECHO|CRMOD, 0);
        !           508:                goto common;
        !           509: 
        !           510:        case TELOPT_BINARY:
        !           511:                mode(0, RAW);
        !           512:                /*FALL THRU*/
        !           513: 
        !           514:        case TELOPT_SGA:
        !           515:        common:
        !           516:                hisopts[option] = 0;
        !           517:                fmt = dont;
        !           518:                break;
        !           519: 
        !           520:        default:
        !           521:                fmt = dont;
        !           522:        }
        !           523:        sprintf(nfrontp, fmt, option);
        !           524:        nfrontp += sizeof (doopt) - 2;
        !           525: }
        !           526: 
        !           527: dooption(option)
        !           528:        int option;
        !           529: {
        !           530:        char *fmt;
        !           531: 
        !           532:        switch (option) {
        !           533: 
        !           534:        case TELOPT_TM:
        !           535:                fmt = wont;
        !           536:                break;
        !           537: 
        !           538:        case TELOPT_ECHO:
        !           539:                mode(ECHO|CRMOD, 0);
        !           540:                goto common;
        !           541: 
        !           542:        case TELOPT_BINARY:
        !           543:                mode(RAW, 0);
        !           544:                /*FALL THRU*/
        !           545: 
        !           546:        case TELOPT_SGA:
        !           547:        common:
        !           548:                fmt = will;
        !           549:                break;
        !           550: 
        !           551:        default:
        !           552:                fmt = wont;
        !           553:                break;
        !           554:        }
        !           555:        sprintf(nfrontp, fmt, option);
        !           556:        nfrontp += sizeof (doopt) - 2;
        !           557: }
        !           558: 
        !           559: mode(on, off)
        !           560:        int on, off;
        !           561: {
        !           562:        struct sgttyb b;
        !           563: 
        !           564:        ptyflush();
        !           565:        ioctl(pty, TIOCGETP, &b);
        !           566:        b.sg_flags |= on;
        !           567:        b.sg_flags &= ~off;
        !           568:        ioctl(pty, TIOCSETP, &b);
        !           569: }
        !           570: 
        !           571: /*
        !           572:  * Send interrupt to process on other side of pty.
        !           573:  * If it is in raw mode, just write NULL;
        !           574:  * otherwise, write intr char.
        !           575:  */
        !           576: interrupt()
        !           577: {
        !           578:        struct sgttyb b;
        !           579:        struct tchars tchars;
        !           580: 
        !           581:        ptyflush();     /* half-hearted */
        !           582:        ioctl(pty, TIOCGETP, &b);
        !           583:        if (b.sg_flags & RAW) {
        !           584:                *pfrontp++ = '\0';
        !           585:                return;
        !           586:        }
        !           587:        *pfrontp++ = ioctl(pty, TIOCGETC, &tchars) < 0 ?
        !           588:                '\177' : tchars.t_intrc;
        !           589: }
        !           590: 
        !           591: ptyflush()
        !           592: {
        !           593:        int n;
        !           594: 
        !           595:        if ((n = pfrontp - pbackp) > 0)
        !           596:                n = write(pty, pbackp, n);
        !           597:        if (n < 0)
        !           598:                return;
        !           599:        pbackp += n;
        !           600:        if (pbackp == pfrontp)
        !           601:                pbackp = pfrontp = ptyobuf;
        !           602: }
        !           603: 
        !           604: netflush()
        !           605: {
        !           606:        int n;
        !           607: 
        !           608:        if ((n = nfrontp - nbackp) > 0)
        !           609:                n = write(net, nbackp, n);
        !           610:        if (n < 0) {
        !           611:                if (errno == EWOULDBLOCK)
        !           612:                        return;
        !           613:                /* should blow this guy away... */
        !           614:                return;
        !           615:        }
        !           616:        nbackp += n;
        !           617:        if (nbackp == nfrontp)
        !           618:                nbackp = nfrontp = netobuf;
        !           619: }
        !           620: 
        !           621: cleanup()
        !           622: {
        !           623: 
        !           624:        rmut();
        !           625:        vhangup();      /* XXX */
        !           626:        shutdown(net, 2);
        !           627:        kill(0, SIGKILL);
        !           628:        exit(1);
        !           629: }
        !           630: 
        !           631: #include <utmp.h>
        !           632: 
        !           633: struct utmp wtmp;
        !           634: char   wtmpf[] = "/usr/adm/wtmp";
        !           635: char   utmp[] = "/etc/utmp";
        !           636: #define SCPYN(a, b)    strncpy(a, b, sizeof (a))
        !           637: #define SCMPN(a, b)    strncmp(a, b, sizeof (a))
        !           638: 
        !           639: rmut()
        !           640: {
        !           641:        register f;
        !           642:        int found = 0;
        !           643: 
        !           644:        f = open(utmp, 2);
        !           645:        if (f >= 0) {
        !           646:                while(read(f, (char *)&wtmp, sizeof (wtmp)) == sizeof (wtmp)) {
        !           647:                        if (SCMPN(wtmp.ut_line, line+5) || wtmp.ut_name[0]==0)
        !           648:                                continue;
        !           649:                        lseek(f, -(long)sizeof (wtmp), 1);
        !           650:                        SCPYN(wtmp.ut_name, "");
        !           651:                        SCPYN(wtmp.ut_host, "");
        !           652:                        time(&wtmp.ut_time);
        !           653:                        write(f, (char *)&wtmp, sizeof (wtmp));
        !           654:                        found++;
        !           655:                }
        !           656:                close(f);
        !           657:        }
        !           658:        if (found) {
        !           659:                f = open(wtmpf, 1);
        !           660:                if (f >= 0) {
        !           661:                        SCPYN(wtmp.ut_line, line+5);
        !           662:                        SCPYN(wtmp.ut_name, "");
        !           663:                        SCPYN(wtmp.ut_host, "");
        !           664:                        time(&wtmp.ut_time);
        !           665:                        lseek(f, (long)0, 2);
        !           666:                        write(f, (char *)&wtmp, sizeof (wtmp));
        !           667:                        close(f);
        !           668:                }
        !           669:        }
        !           670:        chmod(line, 0666);
        !           671:        chown(line, 0, 0);
        !           672:        line[strlen("/dev/")] = 'p';
        !           673:        chmod(line, 0666);
        !           674:        chown(line, 0, 0);
        !           675: }
        !           676: 
        !           677: /*
        !           678:  * Convert network-format internet address
        !           679:  * to base 256 d.d.d.d representation.
        !           680:  */
        !           681: char *
        !           682: ntoa(in)
        !           683:        struct in_addr in;
        !           684: {
        !           685:        static char b[18];
        !           686:        register char *p;
        !           687: 
        !           688:        p = (char *)&in;
        !           689: #define        UC(b)   (((int)b)&0xff)
        !           690:        sprintf(b, "%d.%d.%d.%d", UC(p[0]), UC(p[1]), UC(p[2]), UC(p[3]));
        !           691:        return (b);
        !           692: }

unix.superglobalmegacorp.com

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