Annotation of 43BSD/bin/tar.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1980 Regents of the University of California.
        !             3:  * All rights reserved.  The Berkeley software License Agreement
        !             4:  * specifies the terms and conditions for redistribution.
        !             5:  */
        !             6: 
        !             7: #ifndef lint
        !             8: char copyright[] =
        !             9: "@(#) Copyright (c) 1980 Regents of the University of California.\n\
        !            10:  All rights reserved.\n";
        !            11: #endif not lint
        !            12: 
        !            13: #ifndef lint
        !            14: static char sccsid[] = "@(#)tar.c      5.7 (Berkeley) 4/26/86";
        !            15: #endif not lint
        !            16: 
        !            17: /*
        !            18:  * Tape Archival Program
        !            19:  */
        !            20: #include <stdio.h>
        !            21: #include <sys/param.h>
        !            22: #include <sys/stat.h>
        !            23: #include <sys/dir.h>
        !            24: #include <sys/ioctl.h>
        !            25: #include <sys/mtio.h>
        !            26: #include <sys/time.h>
        !            27: #include <signal.h>
        !            28: #include <errno.h>
        !            29: #include <fcntl.h>
        !            30: 
        !            31: #define TBLOCK 512
        !            32: #define NBLOCK 20
        !            33: #define NAMSIZ 100
        !            34: 
        !            35: #define        writetape(b)    writetbuf(b, 1)
        !            36: #define        min(a,b)  ((a) < (b) ? (a) : (b))
        !            37: #define        max(a,b)  ((a) > (b) ? (a) : (b))
        !            38: 
        !            39: union hblock {
        !            40:        char dummy[TBLOCK];
        !            41:        struct header {
        !            42:                char name[NAMSIZ];
        !            43:                char mode[8];
        !            44:                char uid[8];
        !            45:                char gid[8];
        !            46:                char size[12];
        !            47:                char mtime[12];
        !            48:                char chksum[8];
        !            49:                char linkflag;
        !            50:                char linkname[NAMSIZ];
        !            51:        } dbuf;
        !            52: };
        !            53: 
        !            54: struct linkbuf {
        !            55:        ino_t   inum;
        !            56:        dev_t   devnum;
        !            57:        int     count;
        !            58:        char    pathname[NAMSIZ];
        !            59:        struct  linkbuf *nextp;
        !            60: };
        !            61: 
        !            62: union  hblock dblock;
        !            63: union  hblock *tbuf;
        !            64: struct linkbuf *ihead;
        !            65: struct stat stbuf;
        !            66: 
        !            67: int    rflag;
        !            68: int    xflag;
        !            69: int    vflag;
        !            70: int    tflag;
        !            71: int    cflag;
        !            72: int    mflag;
        !            73: int    fflag;
        !            74: int    iflag;
        !            75: int    oflag;
        !            76: int    pflag;
        !            77: int    wflag;
        !            78: int    hflag;
        !            79: int    Bflag;
        !            80: int    Fflag;
        !            81: 
        !            82: int    mt;
        !            83: int    term;
        !            84: int    chksum;
        !            85: int    recno;
        !            86: int    first;
        !            87: int    prtlinkerr;
        !            88: int    freemem = 1;
        !            89: int    nblock = 0;
        !            90: int    onintr();
        !            91: int    onquit();
        !            92: int    onhup();
        !            93: #ifdef notdef
        !            94: int    onterm();
        !            95: #endif
        !            96: 
        !            97: daddr_t        low;
        !            98: daddr_t        high;
        !            99: daddr_t        bsrch();
        !           100: 
        !           101: FILE   *vfile = stdout;
        !           102: FILE   *tfile;
        !           103: char   tname[] = "/tmp/tarXXXXXX";
        !           104: char   *usefile;
        !           105: char   magtape[] = "/dev/rmt8";
        !           106: char   *malloc();
        !           107: long   time();
        !           108: off_t  lseek();
        !           109: char   *mktemp();
        !           110: char   *sprintf();
        !           111: char   *strcat();
        !           112: char   *strcpy();
        !           113: char   *rindex();
        !           114: char   *getcwd();
        !           115: char   *getwd();
        !           116: char   *getmem();
        !           117: 
        !           118: main(argc, argv)
        !           119: int    argc;
        !           120: char   *argv[];
        !           121: {
        !           122:        char *cp;
        !           123: 
        !           124:        if (argc < 2)
        !           125:                usage();
        !           126: 
        !           127:        tfile = NULL;
        !           128:        usefile =  magtape;
        !           129:        argv[argc] = 0;
        !           130:        argv++;
        !           131:        for (cp = *argv++; *cp; cp++) 
        !           132:                switch(*cp) {
        !           133: 
        !           134:                case 'f':
        !           135:                        if (*argv == 0) {
        !           136:                                fprintf(stderr,
        !           137:                        "tar: tapefile must be specified with 'f' option\n");
        !           138:                                usage();
        !           139:                        }
        !           140:                        usefile = *argv++;
        !           141:                        fflag++;
        !           142:                        break;
        !           143: 
        !           144:                case 'c':
        !           145:                        cflag++;
        !           146:                        rflag++;
        !           147:                        break;
        !           148: 
        !           149:                case 'o':
        !           150:                        oflag++;
        !           151:                        break;
        !           152: 
        !           153:                case 'p':
        !           154:                        pflag++;
        !           155:                        break;
        !           156:                
        !           157:                case 'u':
        !           158:                        mktemp(tname);
        !           159:                        if ((tfile = fopen(tname, "w")) == NULL) {
        !           160:                                fprintf(stderr,
        !           161:                                 "tar: cannot create temporary file (%s)\n",
        !           162:                                 tname);
        !           163:                                done(1);
        !           164:                        }
        !           165:                        fprintf(tfile, "!!!!!/!/!/!/!/!/!/! 000\n");
        !           166:                        /*FALL THRU*/
        !           167: 
        !           168:                case 'r':
        !           169:                        rflag++;
        !           170:                        break;
        !           171: 
        !           172:                case 'v':
        !           173:                        vflag++;
        !           174:                        break;
        !           175: 
        !           176:                case 'w':
        !           177:                        wflag++;
        !           178:                        break;
        !           179: 
        !           180:                case 'x':
        !           181:                        xflag++;
        !           182:                        break;
        !           183: 
        !           184:                case 't':
        !           185:                        tflag++;
        !           186:                        break;
        !           187: 
        !           188:                case 'm':
        !           189:                        mflag++;
        !           190:                        break;
        !           191: 
        !           192:                case '-':
        !           193:                        break;
        !           194: 
        !           195:                case '0':
        !           196:                case '1':
        !           197:                case '4':
        !           198:                case '5':
        !           199:                case '7':
        !           200:                case '8':
        !           201:                        magtape[8] = *cp;
        !           202:                        usefile = magtape;
        !           203:                        break;
        !           204: 
        !           205:                case 'b':
        !           206:                        if (*argv == 0) {
        !           207:                                fprintf(stderr,
        !           208:                        "tar: blocksize must be specified with 'b' option\n");
        !           209:                                usage();
        !           210:                        }
        !           211:                        nblock = atoi(*argv);
        !           212:                        if (nblock <= 0) {
        !           213:                                fprintf(stderr,
        !           214:                                    "tar: invalid blocksize \"%s\"\n", *argv);
        !           215:                                done(1);
        !           216:                        }
        !           217:                        argv++;
        !           218:                        break;
        !           219: 
        !           220:                case 'l':
        !           221:                        prtlinkerr++;
        !           222:                        break;
        !           223: 
        !           224:                case 'h':
        !           225:                        hflag++;
        !           226:                        break;
        !           227: 
        !           228:                case 'i':
        !           229:                        iflag++;
        !           230:                        break;
        !           231: 
        !           232:                case 'B':
        !           233:                        Bflag++;
        !           234:                        break;
        !           235: 
        !           236:                case 'F':
        !           237:                        Fflag++;
        !           238:                        break;
        !           239: 
        !           240:                default:
        !           241:                        fprintf(stderr, "tar: %c: unknown option\n", *cp);
        !           242:                        usage();
        !           243:                }
        !           244: 
        !           245:        if (!rflag && !xflag && !tflag)
        !           246:                usage();
        !           247:        if (rflag) {
        !           248:                if (cflag && tfile != NULL)
        !           249:                        usage();
        !           250:                if (signal(SIGINT, SIG_IGN) != SIG_IGN)
        !           251:                        (void) signal(SIGINT, onintr);
        !           252:                if (signal(SIGHUP, SIG_IGN) != SIG_IGN)
        !           253:                        (void) signal(SIGHUP, onhup);
        !           254:                if (signal(SIGQUIT, SIG_IGN) != SIG_IGN)
        !           255:                        (void) signal(SIGQUIT, onquit);
        !           256: #ifdef notdef
        !           257:                if (signal(SIGTERM, SIG_IGN) != SIG_IGN)
        !           258:                        (void) signal(SIGTERM, onterm);
        !           259: #endif
        !           260:                mt = openmt(usefile, 1);
        !           261:                dorep(argv);
        !           262:                done(0);
        !           263:        }
        !           264:        mt = openmt(usefile, 0);
        !           265:        if (xflag)
        !           266:                doxtract(argv);
        !           267:        else
        !           268:                dotable(argv);
        !           269:        done(0);
        !           270: }
        !           271: 
        !           272: usage()
        !           273: {
        !           274:        fprintf(stderr,
        !           275: "tar: usage: tar -{txru}[cvfblmhopwBi] [tapefile] [blocksize] file1 file2...\n");
        !           276:        done(1);
        !           277: }
        !           278: 
        !           279: int
        !           280: openmt(tape, writing)
        !           281:        char *tape;
        !           282:        int writing;
        !           283: {
        !           284: 
        !           285:        if (strcmp(tape, "-") == 0) {
        !           286:                /*
        !           287:                 * Read from standard input or write to standard output.
        !           288:                 */
        !           289:                if (writing) {
        !           290:                        if (cflag == 0) {
        !           291:                                fprintf(stderr,
        !           292:                         "tar: can only create standard output archives\n");
        !           293:                                done(1);
        !           294:                        }
        !           295:                        vfile = stderr;
        !           296:                        setlinebuf(vfile);
        !           297:                        mt = dup(1);
        !           298:                } else {
        !           299:                        mt = dup(0);
        !           300:                        Bflag++;
        !           301:                }
        !           302:        } else {
        !           303:                /*
        !           304:                 * Use file or tape on local machine.
        !           305:                 */
        !           306:                if (writing) {
        !           307:                        if (cflag)
        !           308:                                mt = open(tape, O_RDWR|O_CREAT|O_TRUNC, 0666);
        !           309:                        else
        !           310:                                mt = open(tape, O_RDWR);
        !           311:                } else
        !           312:                        mt = open(tape, O_RDONLY);
        !           313:                if (mt < 0) {
        !           314:                        fprintf(stderr, "tar: ");
        !           315:                        perror(tape);
        !           316:                        done(1);
        !           317:                }
        !           318:        }
        !           319:        return(mt);
        !           320: }
        !           321: 
        !           322: dorep(argv)
        !           323:        char *argv[];
        !           324: {
        !           325:        register char *cp, *cp2;
        !           326:        char wdir[MAXPATHLEN], tempdir[MAXPATHLEN], *parent;
        !           327: 
        !           328:        if (!cflag) {
        !           329:                getdir();
        !           330:                do {
        !           331:                        passtape();
        !           332:                        if (term)
        !           333:                                done(0);
        !           334:                        getdir();
        !           335:                } while (!endtape());
        !           336:                backtape();
        !           337:                if (tfile != NULL) {
        !           338:                        char buf[200];
        !           339: 
        !           340:                        sprintf(buf,
        !           341: "sort +0 -1 +1nr %s -o %s; awk '$1 != prev {print; prev=$1}' %s >%sX; mv %sX %s",
        !           342:                                tname, tname, tname, tname, tname, tname);
        !           343:                        fflush(tfile);
        !           344:                        system(buf);
        !           345:                        freopen(tname, "r", tfile);
        !           346:                        fstat(fileno(tfile), &stbuf);
        !           347:                        high = stbuf.st_size;
        !           348:                }
        !           349:        }
        !           350: 
        !           351:        (void) getcwd(wdir);
        !           352:        while (*argv && ! term) {
        !           353:                cp2 = *argv;
        !           354:                if (!strcmp(cp2, "-C") && argv[1]) {
        !           355:                        argv++;
        !           356:                        if (chdir(*argv) < 0) {
        !           357:                                fprintf(stderr, "tar: can't change directories to ");
        !           358:                                perror(*argv);
        !           359:                        } else
        !           360:                                (void) getcwd(wdir);
        !           361:                        argv++;
        !           362:                        continue;
        !           363:                }
        !           364:                parent = wdir;
        !           365:                for (cp = *argv; *cp; cp++)
        !           366:                        if (*cp == '/')
        !           367:                                cp2 = cp;
        !           368:                if (cp2 != *argv) {
        !           369:                        *cp2 = '\0';
        !           370:                        if (chdir(*argv) < 0) {
        !           371:                                fprintf(stderr, "tar: can't change directories to ");
        !           372:                                perror(*argv);
        !           373:                                continue;
        !           374:                        }
        !           375:                        parent = getcwd(tempdir);
        !           376:                        *cp2 = '/';
        !           377:                        cp2++;
        !           378:                }
        !           379:                putfile(*argv++, cp2, parent);
        !           380:                if (chdir(wdir) < 0) {
        !           381:                        fprintf(stderr, "tar: cannot change back?: ");
        !           382:                        perror(wdir);
        !           383:                }
        !           384:        }
        !           385:        putempty();
        !           386:        putempty();
        !           387:        flushtape();
        !           388:        if (prtlinkerr == 0)
        !           389:                return;
        !           390:        for (; ihead != NULL; ihead = ihead->nextp) {
        !           391:                if (ihead->count == 0)
        !           392:                        continue;
        !           393:                fprintf(stderr, "tar: missing links to %s\n", ihead->pathname);
        !           394:        }
        !           395: }
        !           396: 
        !           397: endtape()
        !           398: {
        !           399:        return (dblock.dbuf.name[0] == '\0');
        !           400: }
        !           401: 
        !           402: getdir()
        !           403: {
        !           404:        register struct stat *sp;
        !           405:        int i;
        !           406: 
        !           407: top:
        !           408:        readtape((char *)&dblock);
        !           409:        if (dblock.dbuf.name[0] == '\0')
        !           410:                return;
        !           411:        sp = &stbuf;
        !           412:        sscanf(dblock.dbuf.mode, "%o", &i);
        !           413:        sp->st_mode = i;
        !           414:        sscanf(dblock.dbuf.uid, "%o", &i);
        !           415:        sp->st_uid = i;
        !           416:        sscanf(dblock.dbuf.gid, "%o", &i);
        !           417:        sp->st_gid = i;
        !           418:        sscanf(dblock.dbuf.size, "%lo", &sp->st_size);
        !           419:        sscanf(dblock.dbuf.mtime, "%lo", &sp->st_mtime);
        !           420:        sscanf(dblock.dbuf.chksum, "%o", &chksum);
        !           421:        if (chksum != (i = checksum())) {
        !           422:                fprintf(stderr, "tar: directory checksum error (%d != %d)\n",
        !           423:                    chksum, i);
        !           424:                if (iflag)
        !           425:                        goto top;
        !           426:                done(2);
        !           427:        }
        !           428:        if (tfile != NULL)
        !           429:                fprintf(tfile, "%s %s\n", dblock.dbuf.name, dblock.dbuf.mtime);
        !           430: }
        !           431: 
        !           432: passtape()
        !           433: {
        !           434:        long blocks;
        !           435:        char *bufp;
        !           436: 
        !           437:        if (dblock.dbuf.linkflag == '1')
        !           438:                return;
        !           439:        blocks = stbuf.st_size;
        !           440:        blocks += TBLOCK-1;
        !           441:        blocks /= TBLOCK;
        !           442: 
        !           443:        while (blocks-- > 0)
        !           444:                (void) readtbuf(&bufp, TBLOCK);
        !           445: }
        !           446: 
        !           447: putfile(longname, shortname, parent)
        !           448:        char *longname;
        !           449:        char *shortname;
        !           450:        char *parent;
        !           451: {
        !           452:        int infile = 0;
        !           453:        long blocks;
        !           454:        char buf[TBLOCK];
        !           455:        char *bigbuf;
        !           456:        register char *cp;
        !           457:        struct direct *dp;
        !           458:        DIR *dirp;
        !           459:        register int i;
        !           460:        long l;
        !           461:        char newparent[NAMSIZ+64];
        !           462:        extern int errno;
        !           463:        int     maxread;
        !           464:        int     hint;           /* amount to write to get "in sync" */
        !           465: 
        !           466:        if (!hflag)
        !           467:                i = lstat(shortname, &stbuf);
        !           468:        else
        !           469:                i = stat(shortname, &stbuf);
        !           470:        if (i < 0) {
        !           471:                fprintf(stderr, "tar: ");
        !           472:                perror(longname);
        !           473:                return;
        !           474:        }
        !           475:        if (tfile != NULL && checkupdate(longname) == 0)
        !           476:                return;
        !           477:        if (checkw('r', longname) == 0)
        !           478:                return;
        !           479:        if (Fflag && checkf(shortname, stbuf.st_mode, Fflag) == 0)
        !           480:                return;
        !           481: 
        !           482:        switch (stbuf.st_mode & S_IFMT) {
        !           483:        case S_IFDIR:
        !           484:                for (i = 0, cp = buf; *cp++ = longname[i++];)
        !           485:                        ;
        !           486:                *--cp = '/';
        !           487:                *++cp = 0  ;
        !           488:                if (!oflag) {
        !           489:                        if ((cp - buf) >= NAMSIZ) {
        !           490:                                fprintf(stderr, "tar: %s: file name too long\n",
        !           491:                                    longname);
        !           492:                                return;
        !           493:                        }
        !           494:                        stbuf.st_size = 0;
        !           495:                        tomodes(&stbuf);
        !           496:                        strcpy(dblock.dbuf.name,buf);
        !           497:                        sprintf(dblock.dbuf.chksum, "%6o", checksum());
        !           498:                        (void) writetape((char *)&dblock);
        !           499:                }
        !           500:                sprintf(newparent, "%s/%s", parent, shortname);
        !           501:                if (chdir(shortname) < 0) {
        !           502:                        perror(shortname);
        !           503:                        return;
        !           504:                }
        !           505:                if ((dirp = opendir(".")) == NULL) {
        !           506:                        fprintf(stderr, "tar: %s: directory read error\n",
        !           507:                            longname);
        !           508:                        if (chdir(parent) < 0) {
        !           509:                                fprintf(stderr, "tar: cannot change back?: ");
        !           510:                                perror(parent);
        !           511:                        }
        !           512:                        return;
        !           513:                }
        !           514:                while ((dp = readdir(dirp)) != NULL && !term) {
        !           515:                        if (dp->d_ino == 0)
        !           516:                                continue;
        !           517:                        if (!strcmp(".", dp->d_name) ||
        !           518:                            !strcmp("..", dp->d_name))
        !           519:                                continue;
        !           520:                        strcpy(cp, dp->d_name);
        !           521:                        l = telldir(dirp);
        !           522:                        closedir(dirp);
        !           523:                        putfile(buf, cp, newparent);
        !           524:                        dirp = opendir(".");
        !           525:                        seekdir(dirp, l);
        !           526:                }
        !           527:                closedir(dirp);
        !           528:                if (chdir(parent) < 0) {
        !           529:                        fprintf(stderr, "tar: cannot change back?: ");
        !           530:                        perror(parent);
        !           531:                }
        !           532:                break;
        !           533: 
        !           534:        case S_IFLNK:
        !           535:                tomodes(&stbuf);
        !           536:                if (strlen(longname) >= NAMSIZ) {
        !           537:                        fprintf(stderr, "tar: %s: file name too long\n",
        !           538:                            longname);
        !           539:                        return;
        !           540:                }
        !           541:                strcpy(dblock.dbuf.name, longname);
        !           542:                if (stbuf.st_size + 1 >= NAMSIZ) {
        !           543:                        fprintf(stderr, "tar: %s: symbolic link too long\n",
        !           544:                            longname);
        !           545:                        return;
        !           546:                }
        !           547:                i = readlink(shortname, dblock.dbuf.linkname, NAMSIZ - 1);
        !           548:                if (i < 0) {
        !           549:                        fprintf(stderr, "tar: can't read symbolic link ");
        !           550:                        perror(longname);
        !           551:                        return;
        !           552:                }
        !           553:                dblock.dbuf.linkname[i] = '\0';
        !           554:                dblock.dbuf.linkflag = '2';
        !           555:                if (vflag)
        !           556:                        fprintf(vfile, "a %s symbolic link to %s\n",
        !           557:                            longname, dblock.dbuf.linkname);
        !           558:                sprintf(dblock.dbuf.size, "%11lo", 0);
        !           559:                sprintf(dblock.dbuf.chksum, "%6o", checksum());
        !           560:                (void) writetape((char *)&dblock);
        !           561:                break;
        !           562: 
        !           563:        case S_IFREG:
        !           564:                if ((infile = open(shortname, 0)) < 0) {
        !           565:                        fprintf(stderr, "tar: ");
        !           566:                        perror(longname);
        !           567:                        return;
        !           568:                }
        !           569:                tomodes(&stbuf);
        !           570:                if (strlen(longname) >= NAMSIZ) {
        !           571:                        fprintf(stderr, "tar: %s: file name too long\n",
        !           572:                            longname);
        !           573:                        close(infile);
        !           574:                        return;
        !           575:                }
        !           576:                strcpy(dblock.dbuf.name, longname);
        !           577:                if (stbuf.st_nlink > 1) {
        !           578:                        struct linkbuf *lp;
        !           579:                        int found = 0;
        !           580: 
        !           581:                        for (lp = ihead; lp != NULL; lp = lp->nextp)
        !           582:                                if (lp->inum == stbuf.st_ino &&
        !           583:                                    lp->devnum == stbuf.st_dev) {
        !           584:                                        found++;
        !           585:                                        break;
        !           586:                                }
        !           587:                        if (found) {
        !           588:                                strcpy(dblock.dbuf.linkname, lp->pathname);
        !           589:                                dblock.dbuf.linkflag = '1';
        !           590:                                sprintf(dblock.dbuf.chksum, "%6o", checksum());
        !           591:                                (void) writetape( (char *) &dblock);
        !           592:                                if (vflag)
        !           593:                                        fprintf(vfile, "a %s link to %s\n",
        !           594:                                            longname, lp->pathname);
        !           595:                                lp->count--;
        !           596:                                close(infile);
        !           597:                                return;
        !           598:                        }
        !           599:                        lp = (struct linkbuf *) getmem(sizeof(*lp));
        !           600:                        if (lp != NULL) {
        !           601:                                lp->nextp = ihead;
        !           602:                                ihead = lp;
        !           603:                                lp->inum = stbuf.st_ino;
        !           604:                                lp->devnum = stbuf.st_dev;
        !           605:                                lp->count = stbuf.st_nlink - 1;
        !           606:                                strcpy(lp->pathname, longname);
        !           607:                        }
        !           608:                }
        !           609:                blocks = (stbuf.st_size + (TBLOCK-1)) / TBLOCK;
        !           610:                if (vflag)
        !           611:                        fprintf(vfile, "a %s %ld blocks\n", longname, blocks);
        !           612:                sprintf(dblock.dbuf.chksum, "%6o", checksum());
        !           613:                hint = writetape((char *)&dblock);
        !           614:                maxread = max(stbuf.st_blksize, (nblock * TBLOCK));
        !           615:                if ((bigbuf = malloc((unsigned)maxread)) == 0) {
        !           616:                        maxread = TBLOCK;
        !           617:                        bigbuf = buf;
        !           618:                }
        !           619: 
        !           620:                while ((i = read(infile, bigbuf, min((hint*TBLOCK), maxread))) > 0
        !           621:                  && blocks > 0) {
        !           622:                        register int nblks;
        !           623: 
        !           624:                        nblks = ((i-1)/TBLOCK)+1;
        !           625:                        if (nblks > blocks)
        !           626:                                nblks = blocks;
        !           627:                        hint = writetbuf(bigbuf, nblks);
        !           628:                        blocks -= nblks;
        !           629:                }
        !           630:                close(infile);
        !           631:                if (bigbuf != buf)
        !           632:                        free(bigbuf);
        !           633:                if (i < 0) {
        !           634:                        fprintf(stderr, "tar: Read error on ");
        !           635:                        perror(longname);
        !           636:                } else if (blocks != 0 || i != 0)
        !           637:                        fprintf(stderr, "tar: %s: file changed size\n",
        !           638:                            longname);
        !           639:                while (--blocks >=  0)
        !           640:                        putempty();
        !           641:                break;
        !           642: 
        !           643:        default:
        !           644:                fprintf(stderr, "tar: %s is not a file. Not dumped\n",
        !           645:                    longname);
        !           646:                break;
        !           647:        }
        !           648: }
        !           649: 
        !           650: doxtract(argv)
        !           651:        char *argv[];
        !           652: {
        !           653:        long blocks, bytes;
        !           654:        int ofile, i;
        !           655: 
        !           656:        for (;;) {
        !           657:                if ((i = wantit(argv)) == 0)
        !           658:                        continue;
        !           659:                if (i == -1)
        !           660:                        break;          /* end of tape */
        !           661:                if (checkw('x', dblock.dbuf.name) == 0) {
        !           662:                        passtape();
        !           663:                        continue;
        !           664:                }
        !           665:                if (Fflag) {
        !           666:                        char *s;
        !           667: 
        !           668:                        if ((s = rindex(dblock.dbuf.name, '/')) == 0)
        !           669:                                s = dblock.dbuf.name;
        !           670:                        else
        !           671:                                s++;
        !           672:                        if (checkf(s, stbuf.st_mode, Fflag) == 0) {
        !           673:                                passtape();
        !           674:                                continue;
        !           675:                        }
        !           676:                }
        !           677:                if (checkdir(dblock.dbuf.name)) {       /* have a directory */
        !           678:                        if (mflag == 0)
        !           679:                                dodirtimes(&dblock);
        !           680:                        continue;
        !           681:                }
        !           682:                if (dblock.dbuf.linkflag == '2') {      /* symlink */
        !           683:                        /*
        !           684:                         * only unlink non directories or empty
        !           685:                         * directories
        !           686:                         */
        !           687:                        if (rmdir(dblock.dbuf.name) < 0) {
        !           688:                                if (errno == ENOTDIR)
        !           689:                                        unlink(dblock.dbuf.name);
        !           690:                        }
        !           691:                        if (symlink(dblock.dbuf.linkname, dblock.dbuf.name)<0) {
        !           692:                                fprintf(stderr, "tar: %s: symbolic link failed: ",
        !           693:                                    dblock.dbuf.name);
        !           694:                                perror("");
        !           695:                                continue;
        !           696:                        }
        !           697:                        if (vflag)
        !           698:                                fprintf(vfile, "x %s symbolic link to %s\n",
        !           699:                                    dblock.dbuf.name, dblock.dbuf.linkname);
        !           700: #ifdef notdef
        !           701:                        /* ignore alien orders */
        !           702:                        chown(dblock.dbuf.name, stbuf.st_uid, stbuf.st_gid);
        !           703:                        if (mflag == 0)
        !           704:                                setimes(dblock.dbuf.name, stbuf.st_mtime);
        !           705:                        if (pflag)
        !           706:                                chmod(dblock.dbuf.name, stbuf.st_mode & 07777);
        !           707: #endif
        !           708:                        continue;
        !           709:                }
        !           710:                if (dblock.dbuf.linkflag == '1') {      /* regular link */
        !           711:                        /*
        !           712:                         * only unlink non directories or empty
        !           713:                         * directories
        !           714:                         */
        !           715:                        if (rmdir(dblock.dbuf.name) < 0) {
        !           716:                                if (errno == ENOTDIR)
        !           717:                                        unlink(dblock.dbuf.name);
        !           718:                        }
        !           719:                        if (link(dblock.dbuf.linkname, dblock.dbuf.name) < 0) {
        !           720:                                fprintf(stderr, "tar: can't link %s to %s: ",
        !           721:                                    dblock.dbuf.name, dblock.dbuf.linkname);
        !           722:                                perror("");
        !           723:                                continue;
        !           724:                        }
        !           725:                        if (vflag)
        !           726:                                fprintf(vfile, "%s linked to %s\n",
        !           727:                                    dblock.dbuf.name, dblock.dbuf.linkname);
        !           728:                        continue;
        !           729:                }
        !           730:                if ((ofile = creat(dblock.dbuf.name,stbuf.st_mode&0xfff)) < 0) {
        !           731:                        fprintf(stderr, "tar: can't create %s: ",
        !           732:                            dblock.dbuf.name);
        !           733:                        perror("");
        !           734:                        passtape();
        !           735:                        continue;
        !           736:                }
        !           737:                chown(dblock.dbuf.name, stbuf.st_uid, stbuf.st_gid);
        !           738:                blocks = ((bytes = stbuf.st_size) + TBLOCK-1)/TBLOCK;
        !           739:                if (vflag)
        !           740:                        fprintf(vfile, "x %s, %ld bytes, %ld tape blocks\n",
        !           741:                            dblock.dbuf.name, bytes, blocks);
        !           742:                for (; blocks > 0;) {
        !           743:                        register int nread;
        !           744:                        char    *bufp;
        !           745:                        register int nwant;
        !           746:                        
        !           747:                        nwant = NBLOCK*TBLOCK;
        !           748:                        if (nwant > (blocks*TBLOCK))
        !           749:                                nwant = (blocks*TBLOCK);
        !           750:                        nread = readtbuf(&bufp, nwant);
        !           751:                        if (write(ofile, bufp, (int)min(nread, bytes)) < 0) {
        !           752:                                fprintf(stderr,
        !           753:                                    "tar: %s: HELP - extract write error",
        !           754:                                    dblock.dbuf.name);
        !           755:                                perror("");
        !           756:                                done(2);
        !           757:                        }
        !           758:                        bytes -= nread;
        !           759:                        blocks -= (((nread-1)/TBLOCK)+1);
        !           760:                }
        !           761:                close(ofile);
        !           762:                if (mflag == 0)
        !           763:                        setimes(dblock.dbuf.name, stbuf.st_mtime);
        !           764:                if (pflag)
        !           765:                        chmod(dblock.dbuf.name, stbuf.st_mode & 07777);
        !           766:        }
        !           767:        if (mflag == 0) {
        !           768:                dblock.dbuf.name[0] = '\0';     /* process the whole stack */
        !           769:                dodirtimes(&dblock);
        !           770:        }
        !           771: }
        !           772: 
        !           773: dotable(argv)
        !           774:        char *argv[];
        !           775: {
        !           776:        register int i;
        !           777: 
        !           778:        for (;;) {
        !           779:                if ((i = wantit(argv)) == 0)
        !           780:                        continue;
        !           781:                if (i == -1)
        !           782:                        break;          /* end of tape */
        !           783:                if (vflag)
        !           784:                        longt(&stbuf);
        !           785:                printf("%s", dblock.dbuf.name);
        !           786:                if (dblock.dbuf.linkflag == '1')
        !           787:                        printf(" linked to %s", dblock.dbuf.linkname);
        !           788:                if (dblock.dbuf.linkflag == '2')
        !           789:                        printf(" symbolic link to %s", dblock.dbuf.linkname);
        !           790:                printf("\n");
        !           791:                passtape();
        !           792:        }
        !           793: }
        !           794: 
        !           795: putempty()
        !           796: {
        !           797:        char buf[TBLOCK];
        !           798: 
        !           799:        bzero(buf, sizeof (buf));
        !           800:        (void) writetape(buf);
        !           801: }
        !           802: 
        !           803: longt(st)
        !           804:        register struct stat *st;
        !           805: {
        !           806:        register char *cp;
        !           807:        char *ctime();
        !           808: 
        !           809:        pmode(st);
        !           810:        printf("%3d/%1d", st->st_uid, st->st_gid);
        !           811:        printf("%7ld", st->st_size);
        !           812:        cp = ctime(&st->st_mtime);
        !           813:        printf(" %-12.12s %-4.4s ", cp+4, cp+20);
        !           814: }
        !           815: 
        !           816: #define        SUID    04000
        !           817: #define        SGID    02000
        !           818: #define        ROWN    0400
        !           819: #define        WOWN    0200
        !           820: #define        XOWN    0100
        !           821: #define        RGRP    040
        !           822: #define        WGRP    020
        !           823: #define        XGRP    010
        !           824: #define        ROTH    04
        !           825: #define        WOTH    02
        !           826: #define        XOTH    01
        !           827: #define        STXT    01000
        !           828: int    m1[] = { 1, ROWN, 'r', '-' };
        !           829: int    m2[] = { 1, WOWN, 'w', '-' };
        !           830: int    m3[] = { 2, SUID, 's', XOWN, 'x', '-' };
        !           831: int    m4[] = { 1, RGRP, 'r', '-' };
        !           832: int    m5[] = { 1, WGRP, 'w', '-' };
        !           833: int    m6[] = { 2, SGID, 's', XGRP, 'x', '-' };
        !           834: int    m7[] = { 1, ROTH, 'r', '-' };
        !           835: int    m8[] = { 1, WOTH, 'w', '-' };
        !           836: int    m9[] = { 2, STXT, 't', XOTH, 'x', '-' };
        !           837: 
        !           838: int    *m[] = { m1, m2, m3, m4, m5, m6, m7, m8, m9};
        !           839: 
        !           840: pmode(st)
        !           841:        register struct stat *st;
        !           842: {
        !           843:        register int **mp;
        !           844: 
        !           845:        for (mp = &m[0]; mp < &m[9];)
        !           846:                selectbits(*mp++, st);
        !           847: }
        !           848: 
        !           849: selectbits(pairp, st)
        !           850:        int *pairp;
        !           851:        struct stat *st;
        !           852: {
        !           853:        register int n, *ap;
        !           854: 
        !           855:        ap = pairp;
        !           856:        n = *ap++;
        !           857:        while (--n>=0 && (st->st_mode&*ap++)==0)
        !           858:                ap++;
        !           859:        putchar(*ap);
        !           860: }
        !           861: 
        !           862: /*
        !           863:  * Make all directories needed by `name'.  If `name' is itself
        !           864:  * a directory on the tar tape (indicated by a trailing '/'),
        !           865:  * return 1; else 0.
        !           866:  */
        !           867: checkdir(name)
        !           868:        register char *name;
        !           869: {
        !           870:        register char *cp;
        !           871: 
        !           872:        /*
        !           873:         * Quick check for existence of directory.
        !           874:         */
        !           875:        if ((cp = rindex(name, '/')) == 0)
        !           876:                return (0);
        !           877:        *cp = '\0';
        !           878:        if (access(name, 0) == 0) {     /* already exists */
        !           879:                *cp = '/';
        !           880:                return (cp[1] == '\0'); /* return (lastchar == '/') */
        !           881:        }
        !           882:        *cp = '/';
        !           883: 
        !           884:        /*
        !           885:         * No luck, try to make all directories in path.
        !           886:         */
        !           887:        for (cp = name; *cp; cp++) {
        !           888:                if (*cp != '/')
        !           889:                        continue;
        !           890:                *cp = '\0';
        !           891:                if (access(name, 0) < 0) {
        !           892:                        if (mkdir(name, 0777) < 0) {
        !           893:                                perror(name);
        !           894:                                *cp = '/';
        !           895:                                return (0);
        !           896:                        }
        !           897:                        chown(name, stbuf.st_uid, stbuf.st_gid);
        !           898:                        if (pflag && cp[1] == '\0')     /* dir on the tape */
        !           899:                                chmod(name, stbuf.st_mode & 07777);
        !           900:                }
        !           901:                *cp = '/';
        !           902:        }
        !           903:        return (cp[-1]=='/');
        !           904: }
        !           905: 
        !           906: onintr()
        !           907: {
        !           908:        (void) signal(SIGINT, SIG_IGN);
        !           909:        term++;
        !           910: }
        !           911: 
        !           912: onquit()
        !           913: {
        !           914:        (void) signal(SIGQUIT, SIG_IGN);
        !           915:        term++;
        !           916: }
        !           917: 
        !           918: onhup()
        !           919: {
        !           920:        (void) signal(SIGHUP, SIG_IGN);
        !           921:        term++;
        !           922: }
        !           923: 
        !           924: #ifdef notdef
        !           925: onterm()
        !           926: {
        !           927:        (void) signal(SIGTERM, SIG_IGN);
        !           928:        term++;
        !           929: }
        !           930: #endif
        !           931: 
        !           932: tomodes(sp)
        !           933: register struct stat *sp;
        !           934: {
        !           935:        register char *cp;
        !           936: 
        !           937:        for (cp = dblock.dummy; cp < &dblock.dummy[TBLOCK]; cp++)
        !           938:                *cp = '\0';
        !           939:        sprintf(dblock.dbuf.mode, "%6o ", sp->st_mode & 07777);
        !           940:        sprintf(dblock.dbuf.uid, "%6o ", sp->st_uid);
        !           941:        sprintf(dblock.dbuf.gid, "%6o ", sp->st_gid);
        !           942:        sprintf(dblock.dbuf.size, "%11lo ", sp->st_size);
        !           943:        sprintf(dblock.dbuf.mtime, "%11lo ", sp->st_mtime);
        !           944: }
        !           945: 
        !           946: checksum()
        !           947: {
        !           948:        register i;
        !           949:        register char *cp;
        !           950: 
        !           951:        for (cp = dblock.dbuf.chksum;
        !           952:             cp < &dblock.dbuf.chksum[sizeof(dblock.dbuf.chksum)]; cp++)
        !           953:                *cp = ' ';
        !           954:        i = 0;
        !           955:        for (cp = dblock.dummy; cp < &dblock.dummy[TBLOCK]; cp++)
        !           956:                i += *cp;
        !           957:        return (i);
        !           958: }
        !           959: 
        !           960: checkw(c, name)
        !           961:        char *name;
        !           962: {
        !           963:        if (!wflag)
        !           964:                return (1);
        !           965:        printf("%c ", c);
        !           966:        if (vflag)
        !           967:                longt(&stbuf);
        !           968:        printf("%s: ", name);
        !           969:        return (response() == 'y');
        !           970: }
        !           971: 
        !           972: response()
        !           973: {
        !           974:        char c;
        !           975: 
        !           976:        c = getchar();
        !           977:        if (c != '\n')
        !           978:                while (getchar() != '\n')
        !           979:                        ;
        !           980:        else
        !           981:                c = 'n';
        !           982:        return (c);
        !           983: }
        !           984: 
        !           985: checkf(name, mode, howmuch)
        !           986:        char *name;
        !           987:        int mode, howmuch;
        !           988: {
        !           989:        int l;
        !           990: 
        !           991:        if ((mode & S_IFMT) == S_IFDIR){
        !           992:                if ((strcmp(name, "SCCS")==0) || (strcmp(name, "RCS")==0)) 
        !           993:                        return(0)        !           994:                return(1);
        !           995:        }
        !           996:        if ((l = strlen(name)) < 3)
        !           997:                return (1);
        !           998:        if (howmuch > 1 && name[l-2] == '.' && name[l-1] == 'o')
        !           999:                return (0);
        !          1000:        if (strcmp(name, "core") == 0 ||
        !          1001:            strcmp(name, "errs") == 0 ||
        !          1002:            (howmuch > 1 && strcmp(name, "a.out") == 0))
        !          1003:                return (0);
        !          1004:        /* SHOULD CHECK IF IT IS EXECUTABLE */
        !          1005:        return (1);
        !          1006: }
        !          1007: 
        !          1008: /* Is the current file a new file, or the newest one of the same name? */
        !          1009: checkupdate(arg)
        !          1010:        char *arg;
        !          1011: {
        !          1012:        char name[100];
        !          1013:        long mtime;
        !          1014:        daddr_t seekp;
        !          1015:        daddr_t lookup();
        !          1016: 
        !          1017:        rewind(tfile);
        !          1018:        for (;;) {
        !          1019:                if ((seekp = lookup(arg)) < 0)
        !          1020:                        return (1);
        !          1021:                fseek(tfile, seekp, 0);
        !          1022:                fscanf(tfile, "%s %lo", name, &mtime);
        !          1023:                return (stbuf.st_mtime > mtime);
        !          1024:        }
        !          1025: }
        !          1026: 
        !          1027: done(n)
        !          1028: {
        !          1029:        unlink(tname);
        !          1030:        exit(n);
        !          1031: }
        !          1032: 
        !          1033: /* 
        !          1034:  * Do we want the next entry on the tape, i.e. is it selected?  If
        !          1035:  * not, skip over the entire entry.  Return -1 if reached end of tape.
        !          1036:  */
        !          1037: wantit(argv)
        !          1038:        char *argv[];
        !          1039: {
        !          1040:        register char **cp;
        !          1041: 
        !          1042:        getdir();
        !          1043:        if (endtape())
        !          1044:                return (-1);
        !          1045:        if (*argv == 0)
        !          1046:                return (1);
        !          1047:        for (cp = argv; *cp; cp++)
        !          1048:                if (prefix(*cp, dblock.dbuf.name))
        !          1049:                        return (1);
        !          1050:        passtape();
        !          1051:        return (0);
        !          1052: }
        !          1053: 
        !          1054: /*
        !          1055:  * Does s2 begin with the string s1, on a directory boundary?
        !          1056:  */
        !          1057: prefix(s1, s2)
        !          1058:        register char *s1, *s2;
        !          1059: {
        !          1060:        while (*s1)
        !          1061:                if (*s1++ != *s2++)
        !          1062:                        return (0);
        !          1063:        if (*s2)
        !          1064:                return (*s2 == '/');
        !          1065:        return (1);
        !          1066: }
        !          1067: 
        !          1068: #define        N       200
        !          1069: int    njab;
        !          1070: 
        !          1071: daddr_t
        !          1072: lookup(s)
        !          1073:        char *s;
        !          1074: {
        !          1075:        register i;
        !          1076:        daddr_t a;
        !          1077: 
        !          1078:        for(i=0; s[i]; i++)
        !          1079:                if (s[i] == ' ')
        !          1080:                        break;
        !          1081:        a = bsrch(s, i, low, high);
        !          1082:        return (a);
        !          1083: }
        !          1084: 
        !          1085: daddr_t
        !          1086: bsrch(s, n, l, h)
        !          1087:        daddr_t l, h;
        !          1088:        char *s;
        !          1089: {
        !          1090:        register i, j;
        !          1091:        char b[N];
        !          1092:        daddr_t m, m1;
        !          1093: 
        !          1094:        njab = 0;
        !          1095: 
        !          1096: loop:
        !          1097:        if (l >= h)
        !          1098:                return ((daddr_t) -1);
        !          1099:        m = l + (h-l)/2 - N/2;
        !          1100:        if (m < l)
        !          1101:                m = l;
        !          1102:        fseek(tfile, m, 0);
        !          1103:        fread(b, 1, N, tfile);
        !          1104:        njab++;
        !          1105:        for(i=0; i<N; i++) {
        !          1106:                if (b[i] == '\n')
        !          1107:                        break;
        !          1108:                m++;
        !          1109:        }
        !          1110:        if (m >= h)
        !          1111:                return ((daddr_t) -1);
        !          1112:        m1 = m;
        !          1113:        j = i;
        !          1114:        for(i++; i<N; i++) {
        !          1115:                m1++;
        !          1116:                if (b[i] == '\n')
        !          1117:                        break;
        !          1118:        }
        !          1119:        i = cmp(b+j, s, n);
        !          1120:        if (i < 0) {
        !          1121:                h = m;
        !          1122:                goto loop;
        !          1123:        }
        !          1124:        if (i > 0) {
        !          1125:                l = m1;
        !          1126:                goto loop;
        !          1127:        }
        !          1128:        return (m);
        !          1129: }
        !          1130: 
        !          1131: cmp(b, s, n)
        !          1132:        char *b, *s;
        !          1133: {
        !          1134:        register i;
        !          1135: 
        !          1136:        if (b[0] != '\n')
        !          1137:                exit(2);
        !          1138:        for(i=0; i<n; i++) {
        !          1139:                if (b[i+1] > s[i])
        !          1140:                        return (-1);
        !          1141:                if (b[i+1] < s[i])
        !          1142:                        return (1);
        !          1143:        }
        !          1144:        return (b[i+1] == ' '? 0 : -1);
        !          1145: }
        !          1146: 
        !          1147: readtape(buffer)
        !          1148:        char *buffer;
        !          1149: {
        !          1150:        char *bufp;
        !          1151: 
        !          1152:        if (first == 0)
        !          1153:                getbuf();
        !          1154:        (void) readtbuf(&bufp, TBLOCK);
        !          1155:        bcopy(bufp, buffer, TBLOCK);
        !          1156:        return(TBLOCK);
        !          1157: }
        !          1158: 
        !          1159: readtbuf(bufpp, size)
        !          1160:        char **bufpp;
        !          1161:        int size;
        !          1162: {
        !          1163:        register int i;
        !          1164: 
        !          1165:        if (recno >= nblock || first == 0) {
        !          1166:                if ((i = bread(mt, (char *)tbuf, TBLOCK*nblock)) < 0)
        !          1167:                        mterr("read", i, 3);
        !          1168:                if (first == 0) {
        !          1169:                        if ((i % TBLOCK) != 0) {
        !          1170:                                fprintf(stderr, "tar: tape blocksize error\n");
        !          1171:                                done(3);
        !          1172:                        }
        !          1173:                        i /= TBLOCK;
        !          1174:                        if (i != nblock) {
        !          1175:                                fprintf(stderr, "tar: blocksize = %d\n", i);
        !          1176:                                nblock = i;
        !          1177:                        }
        !          1178:                        first = 1;
        !          1179:                }
        !          1180:                recno = 0;
        !          1181:        }
        !          1182:        if (size > ((nblock-recno)*TBLOCK))
        !          1183:                size = (nblock-recno)*TBLOCK;
        !          1184:        *bufpp = (char *)&tbuf[recno];
        !          1185:        recno += (size/TBLOCK);
        !          1186:        return (size);
        !          1187: }
        !          1188: 
        !          1189: writetbuf(buffer, n)
        !          1190:        register char *buffer;
        !          1191:        register int n;
        !          1192: {
        !          1193:        int i;
        !          1194: 
        !          1195:        if (first == 0) {
        !          1196:                getbuf();
        !          1197:                first = 1;
        !          1198:        }
        !          1199:        if (recno >= nblock) {
        !          1200:                i = write(mt, (char *)tbuf, TBLOCK*nblock);
        !          1201:                if (i != TBLOCK*nblock)
        !          1202:                        mterr("write", i, 2);
        !          1203:                recno = 0;
        !          1204:        }
        !          1205: 
        !          1206:        /*
        !          1207:         *  Special case:  We have an empty tape buffer, and the
        !          1208:         *  users data size is >= the tape block size:  Avoid
        !          1209:         *  the bcopy and dma direct to tape.  BIG WIN.  Add the
        !          1210:         *  residual to the tape buffer.
        !          1211:         */
        !          1212:        while (recno == 0 && n >= nblock) {
        !          1213:                i = write(mt, buffer, TBLOCK*nblock);
        !          1214:                if (i != TBLOCK*nblock)
        !          1215:                        mterr("write", i, 2);
        !          1216:                n -= nblock;
        !          1217:                buffer += (nblock * TBLOCK);
        !          1218:        }
        !          1219:                
        !          1220:        while (n-- > 0) {
        !          1221:                bcopy(buffer, (char *)&tbuf[recno++], TBLOCK);
        !          1222:                buffer += TBLOCK;
        !          1223:                if (recno >= nblock) {
        !          1224:                        i = write(mt, (char *)tbuf, TBLOCK*nblock);
        !          1225:                        if (i != TBLOCK*nblock)
        !          1226:                                mterr("write", i, 2);
        !          1227:                        recno = 0;
        !          1228:                }
        !          1229:        }
        !          1230: 
        !          1231:        /* Tell the user how much to write to get in sync */
        !          1232:        return (nblock - recno);
        !          1233: }
        !          1234: 
        !          1235: backtape()
        !          1236: {
        !          1237:        static int mtdev = 1;
        !          1238:        static struct mtop mtop = {MTBSR, 1};
        !          1239:        struct mtget mtget;
        !          1240:        
        !          1241:        if (mtdev == 1)
        !          1242:                mtdev = ioctl(mt, MTIOCGET, (char *)&mtget);
        !          1243:        if (mtdev == 0) {
        !          1244:                if (ioctl(mt, MTIOCTOP, (char *)&mtop) < 0) {
        !          1245:                        fprintf(stderr, "tar: tape backspace error: ");
        !          1246:                        perror("");
        !          1247:                        done(4);
        !          1248:                }
        !          1249:        } else
        !          1250:                lseek(mt, (daddr_t) -TBLOCK*nblock, 1);
        !          1251:        recno--;
        !          1252: }
        !          1253: 
        !          1254: flushtape()
        !          1255: {
        !          1256:        int i;
        !          1257: 
        !          1258:        i = write(mt, (char *)tbuf, TBLOCK*nblock);
        !          1259:        if (i != TBLOCK*nblock)
        !          1260:                mterr("write", i, 2);
        !          1261: }
        !          1262: 
        !          1263: mterr(operation, i, exitcode)
        !          1264:        char *operation;
        !          1265:        int i;
        !          1266: {
        !          1267:        fprintf(stderr, "tar: tape %s error: ", operation);
        !          1268:        if (i < 0)
        !          1269:                perror("");
        !          1270:        else
        !          1271:                fprintf(stderr, "unexpected EOF\n");
        !          1272:        done(exitcode);
        !          1273: }
        !          1274: 
        !          1275: bread(fd, buf, size)
        !          1276:        int fd;
        !          1277:        char *buf;
        !          1278:        int size;
        !          1279: {
        !          1280:        int count;
        !          1281:        static int lastread = 0;
        !          1282: 
        !          1283:        if (!Bflag)
        !          1284:                return (read(fd, buf, size)); 
        !          1285: 
        !          1286:        for (count = 0; count < size; count += lastread) {
        !          1287:                lastread = read(fd, buf, size - count);
        !          1288:                if (lastread <= 0) {
        !          1289:                        if (count > 0)
        !          1290:                                return (count);
        !          1291:                        return (lastread);
        !          1292:                }
        !          1293:                buf += lastread;
        !          1294:        }
        !          1295:        return (count);
        !          1296: }
        !          1297: 
        !          1298: char *
        !          1299: getcwd(buf)
        !          1300:        char *buf;
        !          1301: {
        !          1302:        if (getwd(buf) == NULL) {
        !          1303:                fprintf(stderr, "tar: %s\n", buf);
        !          1304:                exit(1);
        !          1305:        }
        !          1306:        return (buf);
        !          1307: }
        !          1308: 
        !          1309: getbuf()
        !          1310: {
        !          1311:        
        !          1312:        if (nblock == 0) {
        !          1313:                fstat(mt, &stbuf);
        !          1314:                if ((stbuf.st_mode & S_IFMT) == S_IFCHR)
        !          1315:                        nblock = NBLOCK;
        !          1316:                else {
        !          1317:                        nblock = stbuf.st_blksize / TBLOCK;
        !          1318:                        if (nblock == 0)
        !          1319:                                nblock = NBLOCK;
        !          1320:                }
        !          1321:        }
        !          1322:        tbuf = (union hblock *)malloc((unsigned)nblock*TBLOCK);
        !          1323:        if (tbuf == NULL) {
        !          1324:                fprintf(stderr, "tar: blocksize %d too big, can't get memory\n",
        !          1325:                    nblock);
        !          1326:                done(1);
        !          1327:        }
        !          1328: }
        !          1329: 
        !          1330: /*
        !          1331:  * Save this directory and its mtime on the stack, popping and setting
        !          1332:  * the mtimes of any stacked dirs which aren't parents of this one.
        !          1333:  * A null directory causes the entire stack to be unwound and set.
        !          1334:  *
        !          1335:  * Since all the elements of the directory "stack" share a common
        !          1336:  * prefix, we can make do with one string.  We keep only the current
        !          1337:  * directory path, with an associated array of mtime's, one for each
        !          1338:  * '/' in the path.  A negative mtime means no mtime.  The mtime's are
        !          1339:  * offset by one (first index 1, not 0) because calling this with a null
        !          1340:  * directory causes mtime[0] to be set.
        !          1341:  * 
        !          1342:  * This stack algorithm is not guaranteed to work for tapes created
        !          1343:  * with the 'r' option, but the vast majority of tapes with
        !          1344:  * directories are not.  This avoids saving every directory record on
        !          1345:  * the tape and setting all the times at the end.
        !          1346:  */
        !          1347: char dirstack[NAMSIZ];
        !          1348: #define NTIM (NAMSIZ/2+1)              /* a/b/c/d/... */
        !          1349: time_t mtime[NTIM];
        !          1350: 
        !          1351: dodirtimes(hp)
        !          1352:        union hblock *hp;
        !          1353: {
        !          1354:        register char *p = dirstack;
        !          1355:        register char *q = hp->dbuf.name;
        !          1356:        register int ndir = 0;
        !          1357:        char *savp;
        !          1358:        int savndir;
        !          1359: 
        !          1360:        /* Find common prefix */
        !          1361:        while (*p == *q) {
        !          1362:                if (*p++ == '/')
        !          1363:                        ++ndir;
        !          1364:                q++;
        !          1365:        }
        !          1366: 
        !          1367:        savp = p;
        !          1368:        savndir = ndir;
        !          1369:        while (*p) {
        !          1370:                /*
        !          1371:                 * Not a child: unwind the stack, setting the times.
        !          1372:                 * The order we do this doesn't matter, so we go "forward."
        !          1373:                 */
        !          1374:                if (*p++ == '/')
        !          1375:                        if (mtime[++ndir] >= 0) {
        !          1376:                                *--p = '\0';    /* zap the slash */
        !          1377:                                setimes(dirstack, mtime[ndir]);
        !          1378:                                *p++ = '/';
        !          1379:                        }
        !          1380:        }
        !          1381:        p = savp;
        !          1382:        ndir = savndir;
        !          1383: 
        !          1384:        /* Push this one on the "stack" */
        !          1385:        while (*p = *q++)       /* append the rest of the new dir */
        !          1386:                if (*p++ == '/')
        !          1387:                        mtime[++ndir] = -1;
        !          1388:        mtime[ndir] = stbuf.st_mtime;   /* overwrite the last one */
        !          1389: }
        !          1390: 
        !          1391: setimes(path, mt)
        !          1392:        char *path;
        !          1393:        time_t mt;
        !          1394: {
        !          1395:        struct timeval tv[2];
        !          1396: 
        !          1397:        tv[0].tv_sec = time((time_t *) 0);
        !          1398:        tv[1].tv_sec = mt;
        !          1399:        tv[0].tv_usec = tv[1].tv_usec = 0;
        !          1400:        if (utimes(path, tv) < 0) {
        !          1401:                fprintf(stderr, "tar: can't set time on %s: ", path);
        !          1402:                perror("");
        !          1403:        }
        !          1404: }
        !          1405: 
        !          1406: char *
        !          1407: getmem(size)
        !          1408: {
        !          1409:        char *p = malloc((unsigned) size);
        !          1410: 
        !          1411:        if (p == NULL && freemem) {
        !          1412:                fprintf(stderr,
        !          1413:                    "tar: out of memory, link and directory modtime info lost\n");
        !          1414:                freemem = 0;
        !          1415:        }
        !          1416:        return (p);
        !          1417: }

unix.superglobalmegacorp.com

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