Annotation of 42BSD/bin/tar.c, revision 1.1.1.1

1.1       root        1: #ifndef lint
                      2: static char *sccsid = "@(#)tar.c       4.19 (Berkeley) 9/22/83";
                      3: #endif
                      4: 
                      5: /*
                      6:  * Tape Archival Program
                      7:  */
                      8: #include <stdio.h>
                      9: #include <sys/param.h>
                     10: #include <sys/stat.h>
                     11: #include <sys/dir.h>
                     12: #include <sys/ioctl.h>
                     13: #include <sys/mtio.h>
                     14: #include <sys/time.h>
                     15: #include <signal.h>
                     16: #include <errno.h>
                     17: 
                     18: #define TBLOCK 512
                     19: #define NBLOCK 20
                     20: #define NAMSIZ 100
                     21: 
                     22: union hblock {
                     23:        char dummy[TBLOCK];
                     24:        struct header {
                     25:                char name[NAMSIZ];
                     26:                char mode[8];
                     27:                char uid[8];
                     28:                char gid[8];
                     29:                char size[12];
                     30:                char mtime[12];
                     31:                char chksum[8];
                     32:                char linkflag;
                     33:                char linkname[NAMSIZ];
                     34:        } dbuf;
                     35: };
                     36: 
                     37: struct linkbuf {
                     38:        ino_t   inum;
                     39:        dev_t   devnum;
                     40:        int     count;
                     41:        char    pathname[NAMSIZ];
                     42:        struct  linkbuf *nextp;
                     43: };
                     44: 
                     45: union  hblock dblock;
                     46: union  hblock *tbuf;
                     47: struct linkbuf *ihead;
                     48: struct stat stbuf;
                     49: 
                     50: int    rflag;
                     51: int    xflag;
                     52: int    vflag;
                     53: int    tflag;
                     54: int    cflag;
                     55: int    mflag;
                     56: int    fflag;
                     57: int    iflag;
                     58: int    oflag;
                     59: int    pflag;
                     60: int    wflag;
                     61: int    hflag;
                     62: int    Bflag;
                     63: int    Fflag;
                     64: 
                     65: int    mt;
                     66: int    term;
                     67: int    chksum;
                     68: int    recno;
                     69: int    first;
                     70: int    linkerrok;
                     71: int    freemem = 1;
                     72: int    nblock = NBLOCK;
                     73: int    onintr();
                     74: int    onquit();
                     75: int    onhup();
                     76: int    onterm();
                     77: 
                     78: daddr_t        low;
                     79: daddr_t        high;
                     80: daddr_t        bsrch();
                     81: 
                     82: FILE   *tfile;
                     83: char   tname[] = "/tmp/tarXXXXXX";
                     84: char   *usefile;
                     85: char   magtape[] = "/dev/rmt8";
                     86: char   *malloc();
                     87: char   *sprintf();
                     88: char   *strcat();
                     89: char   *rindex();
                     90: char   *getcwd();
                     91: char   *getwd();
                     92: 
                     93: main(argc, argv)
                     94: int    argc;
                     95: char   *argv[];
                     96: {
                     97:        char *cp;
                     98: 
                     99:        if (argc < 2)
                    100:                usage();
                    101: 
                    102:        tfile = NULL;
                    103:        usefile =  magtape;
                    104:        argv[argc] = 0;
                    105:        argv++;
                    106:        for (cp = *argv++; *cp; cp++) 
                    107:                switch(*cp) {
                    108: 
                    109:                case 'f':
                    110:                        if (*argv == 0) {
                    111:                                fprintf(stderr,
                    112:                        "tar: tapefile must be specified with 'f' option\n");
                    113:                                usage();
                    114:                        }
                    115:                        usefile = *argv++;
                    116:                        fflag++;
                    117:                        break;
                    118: 
                    119:                case 'c':
                    120:                        cflag++;
                    121:                        rflag++;
                    122:                        break;
                    123: 
                    124:                case 'o':
                    125:                        oflag++;
                    126:                        break;
                    127: 
                    128:                case 'p':
                    129:                        pflag++;
                    130:                        break;
                    131:                
                    132:                case 'u':
                    133:                        mktemp(tname);
                    134:                        if ((tfile = fopen(tname, "w")) == NULL) {
                    135:                                fprintf(stderr,
                    136:                                 "Tar: cannot create temporary file (%s)\n",
                    137:                                 tname);
                    138:                                done(1);
                    139:                        }
                    140:                        fprintf(tfile, "!!!!!/!/!/!/!/!/!/! 000\n");
                    141:                        /*FALL THRU*/
                    142: 
                    143:                case 'r':
                    144:                        rflag++;
                    145:                        break;
                    146: 
                    147:                case 'v':
                    148:                        vflag++;
                    149:                        break;
                    150: 
                    151:                case 'w':
                    152:                        wflag++;
                    153:                        break;
                    154: 
                    155:                case 'x':
                    156:                        xflag++;
                    157:                        break;
                    158: 
                    159:                case 't':
                    160:                        tflag++;
                    161:                        break;
                    162: 
                    163:                case 'm':
                    164:                        mflag++;
                    165:                        break;
                    166: 
                    167:                case '-':
                    168:                        break;
                    169: 
                    170:                case '0':
                    171:                case '1':
                    172:                case '4':
                    173:                case '5':
                    174:                case '7':
                    175:                case '8':
                    176:                        magtape[8] = *cp;
                    177:                        usefile = magtape;
                    178:                        break;
                    179: 
                    180:                case 'b':
                    181:                        if (*argv == 0) {
                    182:                                fprintf(stderr,
                    183:                        "tar: blocksize must be specified with 'b' option\n");
                    184:                                usage();
                    185:                        }
                    186:                        nblock = atoi(*argv);
                    187:                        if (nblock <= 0) {
                    188:                                fprintf(stderr,
                    189:                                    "tar: invalid blocksize \"%s\"\n", *argv);
                    190:                                done(1);
                    191:                        }
                    192:                        argv++;
                    193:                        break;
                    194: 
                    195:                case 'l':
                    196:                        linkerrok++;
                    197:                        break;
                    198: 
                    199:                case 'h':
                    200:                        hflag++;
                    201:                        break;
                    202: 
                    203:                case 'i':
                    204:                        iflag++;
                    205:                        break;
                    206: 
                    207:                case 'B':
                    208:                        Bflag++;
                    209:                        break;
                    210: 
                    211:                case 'F':
                    212:                        Fflag++;
                    213:                        break;
                    214: 
                    215:                default:
                    216:                        fprintf(stderr, "tar: %c: unknown option\n", *cp);
                    217:                        usage();
                    218:                }
                    219: 
                    220:        if (!rflag && !xflag && !tflag)
                    221:                usage();
                    222:        tbuf = (union hblock *)malloc(nblock*TBLOCK);
                    223:        if (tbuf == NULL) {
                    224:                fprintf(stderr, "tar: blocksize %d too big, can't get memory\n",
                    225:                    nblock);
                    226:                done(1);
                    227:        }
                    228:        if (rflag) {
                    229:                if (cflag && tfile != NULL)
                    230:                        usage();
                    231:                if (signal(SIGINT, SIG_IGN) != SIG_IGN)
                    232:                        signal(SIGINT, onintr);
                    233:                if (signal(SIGHUP, SIG_IGN) != SIG_IGN)
                    234:                        signal(SIGHUP, onhup);
                    235:                if (signal(SIGQUIT, SIG_IGN) != SIG_IGN)
                    236:                        signal(SIGQUIT, onquit);
                    237: #ifdef notdef
                    238:                if (signal(SIGTERM, SIG_IGN) != SIG_IGN)
                    239:                        signal(SIGTERM, onterm);
                    240: #endif
                    241:                if (strcmp(usefile, "-") == 0) {
                    242:                        if (cflag == 0) {
                    243:                                fprintf(stderr,
                    244:                         "tar: can only create standard output archives\n");
                    245:                                done(1);
                    246:                        }
                    247:                        mt = dup(1);
                    248:                        nblock = 1;
                    249:                } else if ((mt = open(usefile, 2)) < 0) {
                    250:                        if (cflag == 0 || (mt =  creat(usefile, 0666)) < 0) {
                    251:                                fprintf(stderr,
                    252:                                        "tar: cannot open %s\n", usefile);
                    253:                                done(1);
                    254:                        }
                    255:                }
                    256:                dorep(argv);
                    257:                done(0);
                    258:        }
                    259:        if (strcmp(usefile, "-") == 0) {
                    260:                mt = dup(0);
                    261:                nblock = 1;
                    262:        } else if ((mt = open(usefile, 0)) < 0) {
                    263:                fprintf(stderr, "tar: cannot open %s\n", usefile);
                    264:                done(1);
                    265:        }
                    266:        if (xflag)
                    267:                doxtract(argv);
                    268:        else
                    269:                dotable();
                    270:        done(0);
                    271: }
                    272: 
                    273: usage()
                    274: {
                    275:        fprintf(stderr,
                    276: "tar: usage: tar -{txru}[cvfblmhopwBi] [tapefile] [blocksize] file1 file2...\n");
                    277:        done(1);
                    278: }
                    279: 
                    280: dorep(argv)
                    281:        char *argv[];
                    282: {
                    283:        register char *cp, *cp2;
                    284:        char wdir[MAXPATHLEN], tempdir[MAXPATHLEN], *parent;
                    285: 
                    286:        if (!cflag) {
                    287:                getdir();
                    288:                do {
                    289:                        passtape();
                    290:                        if (term)
                    291:                                done(0);
                    292:                        getdir();
                    293:                } while (!endtape());
                    294:                backtape();
                    295:                if (tfile != NULL) {
                    296:                        char buf[200];
                    297: 
                    298:                        sprintf(buf,
                    299: "sort +0 -1 +1nr %s -o %s; awk '$1 != prev {print; prev=$1}' %s >%sX; mv %sX %s",
                    300:                                tname, tname, tname, tname, tname, tname);
                    301:                        fflush(tfile);
                    302:                        system(buf);
                    303:                        freopen(tname, "r", tfile);
                    304:                        fstat(fileno(tfile), &stbuf);
                    305:                        high = stbuf.st_size;
                    306:                }
                    307:        }
                    308: 
                    309:        (void) getcwd(wdir);
                    310:        while (*argv && ! term) {
                    311:                cp2 = *argv;
                    312:                if (!strcmp(cp2, "-C") && argv[1]) {
                    313:                        argv++;
                    314:                        if (chdir(*argv) < 0)
                    315:                                perror(*argv);
                    316:                        else
                    317:                                (void) getcwd(wdir);
                    318:                        argv++;
                    319:                        continue;
                    320:                }
                    321:                parent = wdir;
                    322:                for (cp = *argv; *cp; cp++)
                    323:                        if (*cp == '/')
                    324:                                cp2 = cp;
                    325:                if (cp2 != *argv) {
                    326:                        *cp2 = '\0';
                    327:                        if (chdir(*argv) < 0) {
                    328:                                perror(*argv);
                    329:                                continue;
                    330:                        }
                    331:                        parent = getcwd(tempdir);
                    332:                        *cp2 = '/';
                    333:                        cp2++;
                    334:                }
                    335:                putfile(*argv++, cp2, parent);
                    336:                if (chdir(wdir) < 0) {
                    337:                        fprintf(stderr, "cannot change back?: ");
                    338:                        perror(wdir);
                    339:                }
                    340:        }
                    341:        putempty();
                    342:        putempty();
                    343:        flushtape();
                    344:        if (linkerrok == 0)
                    345:                return;
                    346:        for (; ihead != NULL; ihead = ihead->nextp) {
                    347:                if (ihead->count == 0)
                    348:                        continue;
                    349:                fprintf(stderr, "tar: missing links to %s\n", ihead->pathname);
                    350:        }
                    351: }
                    352: 
                    353: endtape()
                    354: {
                    355:        return (dblock.dbuf.name[0] == '\0');
                    356: }
                    357: 
                    358: getdir()
                    359: {
                    360:        register struct stat *sp;
                    361:        int i;
                    362: 
                    363: top:
                    364:        readtape((char *)&dblock);
                    365:        if (dblock.dbuf.name[0] == '\0')
                    366:                return;
                    367:        sp = &stbuf;
                    368:        sscanf(dblock.dbuf.mode, "%o", &i);
                    369:        sp->st_mode = i;
                    370:        sscanf(dblock.dbuf.uid, "%o", &i);
                    371:        sp->st_uid = i;
                    372:        sscanf(dblock.dbuf.gid, "%o", &i);
                    373:        sp->st_gid = i;
                    374:        sscanf(dblock.dbuf.size, "%lo", &sp->st_size);
                    375:        sscanf(dblock.dbuf.mtime, "%lo", &sp->st_mtime);
                    376:        sscanf(dblock.dbuf.chksum, "%o", &chksum);
                    377:        if (chksum != (i = checksum())) {
                    378:                fprintf(stderr, "tar: directory checksum error (%d != %d)\n",
                    379:                    chksum, i);
                    380:                if (iflag)
                    381:                        goto top;
                    382:                done(2);
                    383:        }
                    384:        if (tfile != NULL)
                    385:                fprintf(tfile, "%s %s\n", dblock.dbuf.name, dblock.dbuf.mtime);
                    386: }
                    387: 
                    388: passtape()
                    389: {
                    390:        long blocks;
                    391:        char buf[TBLOCK];
                    392: 
                    393:        if (dblock.dbuf.linkflag == '1')
                    394:                return;
                    395:        blocks = stbuf.st_size;
                    396:        blocks += TBLOCK-1;
                    397:        blocks /= TBLOCK;
                    398: 
                    399:        while (blocks-- > 0)
                    400:                readtape(buf);
                    401: }
                    402: 
                    403: putfile(longname, shortname, parent)
                    404:        char *longname;
                    405:        char *shortname;
                    406:        char *parent;
                    407: {
                    408:        int infile = 0;
                    409:        long blocks;
                    410:        char buf[TBLOCK];
                    411:        register char *cp, *cp2;
                    412:        struct direct *dp;
                    413:        DIR *dirp;
                    414:        int i, j;
                    415:        char newparent[NAMSIZ+64];
                    416:        extern int errno;
                    417: 
                    418:        if (!hflag)
                    419:                i = lstat(shortname, &stbuf);
                    420:        else
                    421:                i = stat(shortname, &stbuf);
                    422:        if (i < 0) {
                    423:                switch (errno) {
                    424:                case EACCES:
                    425:                        fprintf(stderr, "tar: %s: cannot open file\n", longname);
                    426:                        break;
                    427:                case ENOENT:
                    428:                        fprintf(stderr, "tar: %s: no such file or directory\n",
                    429:                            longname);
                    430:                        break;
                    431:                default:
                    432:                        fprintf(stderr, "tar: %s: cannot stat file\n", longname);
                    433:                        break;
                    434:                }
                    435:                return;
                    436:        }
                    437:        if (tfile != NULL && checkupdate(longname) == 0)
                    438:                return;
                    439:        if (checkw('r', longname) == 0)
                    440:                return;
                    441:        if (Fflag && checkf(shortname, stbuf.st_mode, Fflag) == 0)
                    442:                return;
                    443: 
                    444:        switch (stbuf.st_mode & S_IFMT) {
                    445:        case S_IFDIR:
                    446:                for (i = 0, cp = buf; *cp++ = longname[i++];)
                    447:                        ;
                    448:                *--cp = '/';
                    449:                *++cp = 0  ;
                    450:                if (!oflag) {
                    451:                        if ((cp - buf) >= NAMSIZ) {
                    452:                                fprintf(stderr, "tar: %s: file name too long\n",
                    453:                                    longname);
                    454:                                return;
                    455:                        }
                    456:                        stbuf.st_size = 0;
                    457:                        tomodes(&stbuf);
                    458:                        strcpy(dblock.dbuf.name,buf);
                    459:                        sprintf(dblock.dbuf.chksum, "%6o", checksum());
                    460:                        writetape((char *)&dblock);
                    461:                }
                    462:                sprintf(newparent, "%s/%s", parent, shortname);
                    463:                if (chdir(shortname) < 0) {
                    464:                        perror(shortname);
                    465:                        return;
                    466:                }
                    467:                if ((dirp = opendir(".")) == NULL) {
                    468:                        fprintf(stderr, "tar: %s: directory read error\n",
                    469:                            longname);
                    470:                        if (chdir(parent) < 0) {
                    471:                                fprintf(stderr, "cannot change back?: ");
                    472:                                perror(parent);
                    473:                        }
                    474:                        return;
                    475:                }
                    476:                while ((dp = readdir(dirp)) != NULL && !term) {
                    477:                        if (dp->d_ino == 0)
                    478:                                continue;
                    479:                        if (!strcmp(".", dp->d_name) ||
                    480:                            !strcmp("..", dp->d_name))
                    481:                                continue;
                    482:                        strcpy(cp, dp->d_name);
                    483:                        i = telldir(dirp);
                    484:                        closedir(dirp);
                    485:                        putfile(buf, cp, newparent);
                    486:                        dirp = opendir(".");
                    487:                        seekdir(dirp, i);
                    488:                }
                    489:                closedir(dirp);
                    490:                if (chdir(parent) < 0) {
                    491:                        fprintf(stderr, "cannot change back?: ");
                    492:                        perror(parent);
                    493:                }
                    494:                break;
                    495: 
                    496:        case S_IFLNK:
                    497:                tomodes(&stbuf);
                    498:                if (strlen(longname) >= NAMSIZ) {
                    499:                        fprintf(stderr, "tar: %s: file name too long\n",
                    500:                            longname);
                    501:                        return;
                    502:                }
                    503:                strcpy(dblock.dbuf.name, longname);
                    504:                if (stbuf.st_size + 1 >= NAMSIZ) {
                    505:                        fprintf(stderr, "tar: %s: symbolic link too long\n",
                    506:                            longname);
                    507:                        return;
                    508:                }
                    509:                i = readlink(shortname, dblock.dbuf.linkname, NAMSIZ - 1);
                    510:                if (i < 0) {
                    511:                        perror(longname);
                    512:                        return;
                    513:                }
                    514:                dblock.dbuf.linkname[i] = '\0';
                    515:                dblock.dbuf.linkflag = '2';
                    516:                if (vflag) {
                    517:                        fprintf(stderr, "a %s ", longname);
                    518:                        fprintf(stderr, "symbolic link to %s\n",
                    519:                            dblock.dbuf.linkname);
                    520:                }
                    521:                sprintf(dblock.dbuf.size, "%11lo", 0);
                    522:                sprintf(dblock.dbuf.chksum, "%6o", checksum());
                    523:                writetape((char *)&dblock);
                    524:                break;
                    525: 
                    526:        case S_IFREG:
                    527:                if ((infile = open(shortname, 0)) < 0) {
                    528:                        fprintf(stderr, "tar: %s: cannot open file\n", longname);
                    529:                        return;
                    530:                }
                    531:                tomodes(&stbuf);
                    532:                if (strlen(longname) >= NAMSIZ) {
                    533:                        fprintf(stderr, "tar: %s: file name too long\n",
                    534:                            longname);
                    535:                        return;
                    536:                }
                    537:                strcpy(dblock.dbuf.name, longname);
                    538:                if (stbuf.st_nlink > 1) {
                    539:                        struct linkbuf *lp;
                    540:                        int found = 0;
                    541: 
                    542:                        for (lp = ihead; lp != NULL; lp = lp->nextp)
                    543:                                if (lp->inum == stbuf.st_ino &&
                    544:                                    lp->devnum == stbuf.st_dev) {
                    545:                                        found++;
                    546:                                        break;
                    547:                                }
                    548:                        if (found) {
                    549:                                strcpy(dblock.dbuf.linkname, lp->pathname);
                    550:                                dblock.dbuf.linkflag = '1';
                    551:                                sprintf(dblock.dbuf.chksum, "%6o", checksum());
                    552:                                writetape( (char *) &dblock);
                    553:                                if (vflag) {
                    554:                                        fprintf(stderr, "a %s ", longname);
                    555:                                        fprintf(stderr, "link to %s\n",
                    556:                                            lp->pathname);
                    557:                                }
                    558:                                lp->count--;
                    559:                                close(infile);
                    560:                                return;
                    561:                        }
                    562:                        lp = (struct linkbuf *) malloc(sizeof(*lp));
                    563:                        if (lp == NULL) {
                    564:                                if (freemem) {
                    565:                                        fprintf(stderr,
                    566:                                "tar: out of memory, link information lost\n");
                    567:                                        freemem = 0;
                    568:                                }
                    569:                        } else {
                    570:                                lp->nextp = ihead;
                    571:                                ihead = lp;
                    572:                                lp->inum = stbuf.st_ino;
                    573:                                lp->devnum = stbuf.st_dev;
                    574:                                lp->count = stbuf.st_nlink - 1;
                    575:                                strcpy(lp->pathname, longname);
                    576:                        }
                    577:                }
                    578:                blocks = (stbuf.st_size + (TBLOCK-1)) / TBLOCK;
                    579:                if (vflag) {
                    580:                        fprintf(stderr, "a %s ", longname);
                    581:                        fprintf(stderr, "%ld blocks\n", blocks);
                    582:                }
                    583:                sprintf(dblock.dbuf.chksum, "%6o", checksum());
                    584:                writetape((char *)&dblock);
                    585: 
                    586:                while ((i = read(infile, buf, TBLOCK)) > 0 && blocks > 0) {
                    587:                        writetape(buf);
                    588:                        blocks--;
                    589:                }
                    590:                close(infile);
                    591:                if (blocks != 0 || i != 0)
                    592:                        fprintf(stderr, "tar: %s: file changed size\n",
                    593:                            longname);
                    594:                while (--blocks >=  0)
                    595:                        putempty();
                    596:                break;
                    597: 
                    598:        default:
                    599:                fprintf(stderr, "tar: %s is not a file. Not dumped\n",
                    600:                    longname);
                    601:                break;
                    602:        }
                    603: }
                    604: 
                    605: doxtract(argv)
                    606:        char *argv[];
                    607: {
                    608:        long blocks, bytes;
                    609:        char buf[TBLOCK];
                    610:        char **cp;
                    611:        int ofile;
                    612: 
                    613:        for (;;) {
                    614:                getdir();
                    615:                if (endtape())
                    616:                        break;
                    617:                if (*argv == 0)
                    618:                        goto gotit;
                    619:                for (cp = argv; *cp; cp++)
                    620:                        if (prefix(*cp, dblock.dbuf.name))
                    621:                                goto gotit;
                    622:                passtape();
                    623:                continue;
                    624: 
                    625: gotit:
                    626:                if (checkw('x', dblock.dbuf.name) == 0) {
                    627:                        passtape();
                    628:                        continue;
                    629:                }
                    630:                if (Fflag) {
                    631:                        char *s;
                    632: 
                    633:                        if ((s = rindex(dblock.dbuf.name, '/')) == 0)
                    634:                                s = dblock.dbuf.name;
                    635:                        else
                    636:                                s++;
                    637:                        if (checkf(s, stbuf.st_mode, Fflag) == 0) {
                    638:                                passtape();
                    639:                                continue;
                    640:                        }
                    641:                }
                    642:                if (checkdir(dblock.dbuf.name))
                    643:                        continue;
                    644:                if (dblock.dbuf.linkflag == '2') {
                    645:                        unlink(dblock.dbuf.name);
                    646:                        if (symlink(dblock.dbuf.linkname, dblock.dbuf.name)<0) {
                    647:                                fprintf(stderr, "tar: %s: symbolic link failed\n",
                    648:                                    dblock.dbuf.name);
                    649:                                continue;
                    650:                        }
                    651:                        if (vflag)
                    652:                                fprintf(stderr, "x %s symbolic link to %s\n",
                    653:                                    dblock.dbuf.name, dblock.dbuf.linkname);
                    654: #ifdef notdef
                    655:                        /* ignore alien orders */
                    656:                        chown(dblock.dbuf.name, stbuf.st_uid, stbuf.st_gid);
                    657:                        if (mflag == 0) {
                    658:                                struct timeval tv[2];
                    659: 
                    660:                                tv[0].tv_sec = time(0);
                    661:                                tv[0].tv_usec = 0;
                    662:                                tv[1].tv_sec = stbuf.st_mtime;
                    663:                                tv[1].tv_usec = 0;
                    664:                                utimes(dblock.dbuf.name, tv);
                    665:                        }
                    666:                        if (pflag)
                    667:                                chmod(dblock.dbuf.name, stbuf.st_mode & 07777);
                    668: #endif
                    669:                        continue;
                    670:                }
                    671:                if (dblock.dbuf.linkflag == '1') {
                    672:                        unlink(dblock.dbuf.name);
                    673:                        if (link(dblock.dbuf.linkname, dblock.dbuf.name) < 0) {
                    674:                                fprintf(stderr, "tar: %s: cannot link\n",
                    675:                                    dblock.dbuf.name);
                    676:                                continue;
                    677:                        }
                    678:                        if (vflag)
                    679:                                fprintf(stderr, "%s linked to %s\n",
                    680:                                    dblock.dbuf.name, dblock.dbuf.linkname);
                    681:                        continue;
                    682:                }
                    683:                if ((ofile = creat(dblock.dbuf.name,stbuf.st_mode&0xfff)) < 0) {
                    684:                        fprintf(stderr, "tar: %s - cannot create\n",
                    685:                            dblock.dbuf.name);
                    686:                        passtape();
                    687:                        continue;
                    688:                }
                    689:                chown(dblock.dbuf.name, stbuf.st_uid, stbuf.st_gid);
                    690:                blocks = ((bytes = stbuf.st_size) + TBLOCK-1)/TBLOCK;
                    691:                if (vflag)
                    692:                        fprintf(stderr, "x %s, %ld bytes, %ld tape blocks\n",
                    693:                            dblock.dbuf.name, bytes, blocks);
                    694:                for (; blocks-- > 0; bytes -= TBLOCK) {
                    695:                        readtape(buf);
                    696:                        if (bytes > TBLOCK) {
                    697:                                if (write(ofile, buf, TBLOCK) < 0) {
                    698:                                        fprintf(stderr,
                    699:                                        "tar: %s: HELP - extract write error\n",
                    700:                                            dblock.dbuf.name);
                    701:                                        done(2);
                    702:                                }
                    703:                                continue;
                    704:                        }
                    705:                        if (write(ofile, buf, (int) bytes) < 0) {
                    706:                                fprintf(stderr,
                    707:                                    "tar: %s: HELP - extract write error\n",
                    708:                                    dblock.dbuf.name);
                    709:                                done(2);
                    710:                        }
                    711:                }
                    712:                close(ofile);
                    713:                if (mflag == 0) {
                    714:                        struct timeval tv[2];
                    715: 
                    716:                        tv[0].tv_sec = time(0);
                    717:                        tv[0].tv_usec = 0;
                    718:                        tv[1].tv_sec = stbuf.st_mtime;
                    719:                        tv[1].tv_usec = 0;
                    720:                        utimes(dblock.dbuf.name, tv);
                    721:                }
                    722:                if (pflag)
                    723:                        chmod(dblock.dbuf.name, stbuf.st_mode & 07777);
                    724:        }
                    725: }
                    726: 
                    727: dotable()
                    728: {
                    729:        for (;;) {
                    730:                getdir();
                    731:                if (endtape())
                    732:                        break;
                    733:                if (vflag)
                    734:                        longt(&stbuf);
                    735:                printf("%s", dblock.dbuf.name);
                    736:                if (dblock.dbuf.linkflag == '1')
                    737:                        printf(" linked to %s", dblock.dbuf.linkname);
                    738:                if (dblock.dbuf.linkflag == '2')
                    739:                        printf(" symbolic link to %s", dblock.dbuf.linkname);
                    740:                printf("\n");
                    741:                passtape();
                    742:        }
                    743: }
                    744: 
                    745: putempty()
                    746: {
                    747:        char buf[TBLOCK];
                    748: 
                    749:        bzero(buf, sizeof (buf));
                    750:        writetape(buf);
                    751: }
                    752: 
                    753: longt(st)
                    754:        register struct stat *st;
                    755: {
                    756:        register char *cp;
                    757:        char *ctime();
                    758: 
                    759:        pmode(st);
                    760:        printf("%3d/%1d", st->st_uid, st->st_gid);
                    761:        printf("%7D", st->st_size);
                    762:        cp = ctime(&st->st_mtime);
                    763:        printf(" %-12.12s %-4.4s ", cp+4, cp+20);
                    764: }
                    765: 
                    766: #define        SUID    04000
                    767: #define        SGID    02000
                    768: #define        ROWN    0400
                    769: #define        WOWN    0200
                    770: #define        XOWN    0100
                    771: #define        RGRP    040
                    772: #define        WGRP    020
                    773: #define        XGRP    010
                    774: #define        ROTH    04
                    775: #define        WOTH    02
                    776: #define        XOTH    01
                    777: #define        STXT    01000
                    778: int    m1[] = { 1, ROWN, 'r', '-' };
                    779: int    m2[] = { 1, WOWN, 'w', '-' };
                    780: int    m3[] = { 2, SUID, 's', XOWN, 'x', '-' };
                    781: int    m4[] = { 1, RGRP, 'r', '-' };
                    782: int    m5[] = { 1, WGRP, 'w', '-' };
                    783: int    m6[] = { 2, SGID, 's', XGRP, 'x', '-' };
                    784: int    m7[] = { 1, ROTH, 'r', '-' };
                    785: int    m8[] = { 1, WOTH, 'w', '-' };
                    786: int    m9[] = { 2, STXT, 't', XOTH, 'x', '-' };
                    787: 
                    788: int    *m[] = { m1, m2, m3, m4, m5, m6, m7, m8, m9};
                    789: 
                    790: pmode(st)
                    791:        register struct stat *st;
                    792: {
                    793:        register int **mp;
                    794: 
                    795:        for (mp = &m[0]; mp < &m[9];)
                    796:                select(*mp++, st);
                    797: }
                    798: 
                    799: select(pairp, st)
                    800:        int *pairp;
                    801:        struct stat *st;
                    802: {
                    803:        register int n, *ap;
                    804: 
                    805:        ap = pairp;
                    806:        n = *ap++;
                    807:        while (--n>=0 && (st->st_mode&*ap++)==0)
                    808:                ap++;
                    809:        printf("%c", *ap);
                    810: }
                    811: 
                    812: checkdir(name)
                    813:        register char *name;
                    814: {
                    815:        register char *cp;
                    816: 
                    817:        /*
                    818:         * Quick check for existance of directory.
                    819:         */
                    820:        if ((cp = rindex(name, '/')) == 0)
                    821:                return (0);
                    822:        *cp = '\0';
                    823:        if (access(name, 0) >= 0) {
                    824:                *cp = '/';
                    825:                return (cp[1] == '\0');
                    826:        }
                    827:        *cp = '/';
                    828: 
                    829:        /*
                    830:         * No luck, try to make all directories in path.
                    831:         */
                    832:        for (cp = name; *cp; cp++) {
                    833:                if (*cp != '/')
                    834:                        continue;
                    835:                *cp = '\0';
                    836:                if (access(name, 0) < 0) {
                    837:                        if (mkdir(name, 0777) < 0) {
                    838:                                perror(name);
                    839:                                *cp = '/';
                    840:                                return (0);
                    841:                        }
                    842:                        chown(name, stbuf.st_uid, stbuf.st_gid);
                    843:                        if (pflag && cp[1] == '\0')
                    844:                                chmod(name, stbuf.st_mode & 0777);
                    845:                }
                    846:                *cp = '/';
                    847:        }
                    848:        return (cp[-1]=='/');
                    849: }
                    850: 
                    851: onintr()
                    852: {
                    853:        signal(SIGINT, SIG_IGN);
                    854:        term++;
                    855: }
                    856: 
                    857: onquit()
                    858: {
                    859:        signal(SIGQUIT, SIG_IGN);
                    860:        term++;
                    861: }
                    862: 
                    863: onhup()
                    864: {
                    865:        signal(SIGHUP, SIG_IGN);
                    866:        term++;
                    867: }
                    868: 
                    869: onterm()
                    870: {
                    871:        signal(SIGTERM, SIG_IGN);
                    872:        term++;
                    873: }
                    874: 
                    875: tomodes(sp)
                    876: register struct stat *sp;
                    877: {
                    878:        register char *cp;
                    879: 
                    880:        for (cp = dblock.dummy; cp < &dblock.dummy[TBLOCK]; cp++)
                    881:                *cp = '\0';
                    882:        sprintf(dblock.dbuf.mode, "%6o ", sp->st_mode & 07777);
                    883:        sprintf(dblock.dbuf.uid, "%6o ", sp->st_uid);
                    884:        sprintf(dblock.dbuf.gid, "%6o ", sp->st_gid);
                    885:        sprintf(dblock.dbuf.size, "%11lo ", sp->st_size);
                    886:        sprintf(dblock.dbuf.mtime, "%11lo ", sp->st_mtime);
                    887: }
                    888: 
                    889: checksum()
                    890: {
                    891:        register i;
                    892:        register char *cp;
                    893: 
                    894:        for (cp = dblock.dbuf.chksum;
                    895:             cp < &dblock.dbuf.chksum[sizeof(dblock.dbuf.chksum)]; cp++)
                    896:                *cp = ' ';
                    897:        i = 0;
                    898:        for (cp = dblock.dummy; cp < &dblock.dummy[TBLOCK]; cp++)
                    899:                i += *cp;
                    900:        return (i);
                    901: }
                    902: 
                    903: checkw(c, name)
                    904:        char *name;
                    905: {
                    906:        if (!wflag)
                    907:                return (1);
                    908:        printf("%c ", c);
                    909:        if (vflag)
                    910:                longt(&stbuf);
                    911:        printf("%s: ", name);
                    912:        return (response() == 'y');
                    913: }
                    914: 
                    915: response()
                    916: {
                    917:        char c;
                    918: 
                    919:        c = getchar();
                    920:        if (c != '\n')
                    921:                while (getchar() != '\n')
                    922:                        ;
                    923:        else
                    924:                c = 'n';
                    925:        return (c);
                    926: }
                    927: 
                    928: checkf(name, mode, howmuch)
                    929:        char *name;
                    930:        int mode, howmuch;
                    931: {
                    932:        int l;
                    933: 
                    934:        if ((mode & S_IFMT) == S_IFDIR)
                    935:                return (strcmp(name, "SCCS") != 0);
                    936:        if ((l = strlen(name)) < 3)
                    937:                return (1);
                    938:        if (howmuch > 1 && name[l-2] == '.' && name[l-1] == 'o')
                    939:                return (0);
                    940:        if (strcmp(name, "core") == 0 ||
                    941:            strcmp(name, "errs") == 0 ||
                    942:            (howmuch > 1 && strcmp(name, "a.out") == 0))
                    943:                return (0);
                    944:        /* SHOULD CHECK IF IT IS EXECUTABLE */
                    945:        return (1);
                    946: }
                    947: 
                    948: checkupdate(arg)
                    949:        char *arg;
                    950: {
                    951:        char name[100];
                    952:        long mtime;
                    953:        daddr_t seekp;
                    954:        daddr_t lookup();
                    955: 
                    956:        rewind(tfile);
                    957:        for (;;) {
                    958:                if ((seekp = lookup(arg)) < 0)
                    959:                        return (1);
                    960:                fseek(tfile, seekp, 0);
                    961:                fscanf(tfile, "%s %lo", name, &mtime);
                    962:                return (stbuf.st_mtime > mtime);
                    963:        }
                    964: }
                    965: 
                    966: done(n)
                    967: {
                    968:        unlink(tname);
                    969:        exit(n);
                    970: }
                    971: 
                    972: prefix(s1, s2)
                    973:        register char *s1, *s2;
                    974: {
                    975:        while (*s1)
                    976:                if (*s1++ != *s2++)
                    977:                        return (0);
                    978:        if (*s2)
                    979:                return (*s2 == '/');
                    980:        return (1);
                    981: }
                    982: 
                    983: #define        N       200
                    984: int    njab;
                    985: 
                    986: daddr_t
                    987: lookup(s)
                    988:        char *s;
                    989: {
                    990:        register i;
                    991:        daddr_t a;
                    992: 
                    993:        for(i=0; s[i]; i++)
                    994:                if (s[i] == ' ')
                    995:                        break;
                    996:        a = bsrch(s, i, low, high);
                    997:        return (a);
                    998: }
                    999: 
                   1000: daddr_t
                   1001: bsrch(s, n, l, h)
                   1002:        daddr_t l, h;
                   1003:        char *s;
                   1004: {
                   1005:        register i, j;
                   1006:        char b[N];
                   1007:        daddr_t m, m1;
                   1008: 
                   1009:        njab = 0;
                   1010: 
                   1011: loop:
                   1012:        if (l >= h)
                   1013:                return (-1L);
                   1014:        m = l + (h-l)/2 - N/2;
                   1015:        if (m < l)
                   1016:                m = l;
                   1017:        fseek(tfile, m, 0);
                   1018:        fread(b, 1, N, tfile);
                   1019:        njab++;
                   1020:        for(i=0; i<N; i++) {
                   1021:                if (b[i] == '\n')
                   1022:                        break;
                   1023:                m++;
                   1024:        }
                   1025:        if (m >= h)
                   1026:                return (-1L);
                   1027:        m1 = m;
                   1028:        j = i;
                   1029:        for(i++; i<N; i++) {
                   1030:                m1++;
                   1031:                if (b[i] == '\n')
                   1032:                        break;
                   1033:        }
                   1034:        i = cmp(b+j, s, n);
                   1035:        if (i < 0) {
                   1036:                h = m;
                   1037:                goto loop;
                   1038:        }
                   1039:        if (i > 0) {
                   1040:                l = m1;
                   1041:                goto loop;
                   1042:        }
                   1043:        return (m);
                   1044: }
                   1045: 
                   1046: cmp(b, s, n)
                   1047:        char *b, *s;
                   1048: {
                   1049:        register i;
                   1050: 
                   1051:        if (b[0] != '\n')
                   1052:                exit(2);
                   1053:        for(i=0; i<n; i++) {
                   1054:                if (b[i+1] > s[i])
                   1055:                        return (-1);
                   1056:                if (b[i+1] < s[i])
                   1057:                        return (1);
                   1058:        }
                   1059:        return (b[i+1] == ' '? 0 : -1);
                   1060: }
                   1061: 
                   1062: readtape(buffer)
                   1063:        char *buffer;
                   1064: {
                   1065:        register int i;
                   1066: 
                   1067:        if (recno >= nblock || first == 0) {
                   1068:                if ((i = bread(mt, tbuf, TBLOCK*nblock)) < 0) {
                   1069:                        fprintf(stderr, "tar: tape read error\n");
                   1070:                        done(3);
                   1071:                }
                   1072:                if (first == 0) {
                   1073:                        if ((i % TBLOCK) != 0) {
                   1074:                                fprintf(stderr, "tar: tape blocksize error\n");
                   1075:                                done(3);
                   1076:                        }
                   1077:                        i /= TBLOCK;
                   1078:                        if (i != nblock) {
                   1079:                                fprintf(stderr, "tar: blocksize = %d\n", i);
                   1080:                                nblock = i;
                   1081:                        }
                   1082:                }
                   1083:                recno = 0;
                   1084:        }
                   1085:        first = 1;
                   1086:        bcopy((char *)&tbuf[recno++], buffer, TBLOCK);
                   1087:        return (TBLOCK);
                   1088: }
                   1089: 
                   1090: writetape(buffer)
                   1091:        char *buffer;
                   1092: {
                   1093:        first = 1;
                   1094:        if (recno >= nblock) {
                   1095:                if (write(mt, tbuf, TBLOCK*nblock) < 0) {
                   1096:                        fprintf(stderr, "tar: tape write error\n");
                   1097:                        done(2);
                   1098:                }
                   1099:                recno = 0;
                   1100:        }
                   1101:        bcopy(buffer, (char *)&tbuf[recno++], TBLOCK);
                   1102:        if (recno >= nblock) {
                   1103:                if (write(mt, tbuf, TBLOCK*nblock) < 0) {
                   1104:                        fprintf(stderr, "tar: tape write error\n");
                   1105:                        done(2);
                   1106:                }
                   1107:                recno = 0;
                   1108:        }
                   1109:        return (TBLOCK);
                   1110: }
                   1111: 
                   1112: backtape()
                   1113: {
                   1114:        static int mtdev = 1;
                   1115:        static struct mtop mtop = {MTBSR, 1};
                   1116:        struct mtget mtget;
                   1117: 
                   1118:        if (mtdev == 1)
                   1119:                mtdev = ioctl(mt, MTIOCGET, &mtget);
                   1120:        if (mtdev == 0) {
                   1121:                if (ioctl(mt, MTIOCTOP, &mtop) < 0) {
                   1122:                        fprintf(stderr, "tar: tape backspace error\n");
                   1123:                        done(4);
                   1124:                }
                   1125:        } else
                   1126:                lseek(mt, (long) -TBLOCK*nblock, 1);
                   1127:        recno--;
                   1128: }
                   1129: 
                   1130: flushtape()
                   1131: {
                   1132:        write(mt, tbuf, TBLOCK*nblock);
                   1133: }
                   1134: 
                   1135: bread(fd, buf, size)
                   1136:        int fd;
                   1137:        char *buf;
                   1138:        int size;
                   1139: {
                   1140:        int count;
                   1141:        static int lastread = 0;
                   1142: 
                   1143:        if (!Bflag)
                   1144:                return (read(fd, buf, size));
                   1145:        for (count = 0; count < size; count += lastread) {
                   1146:                if (lastread < 0) {
                   1147:                        if (count > 0)
                   1148:                                return (count);
                   1149:                        return (lastread);
                   1150:                }
                   1151:                lastread = read(fd, buf, size - count);
                   1152:                buf += lastread;
                   1153:        }
                   1154:        return (count);
                   1155: }
                   1156: 
                   1157: char *
                   1158: getcwd(buf)
                   1159:        char *buf;
                   1160: {
                   1161: 
                   1162:        if (getwd(buf) == NULL) {
                   1163:                fprintf(stderr, "tar: %s\n", buf);
                   1164:                exit(1);
                   1165:        }
                   1166:        return (buf);
                   1167: }

unix.superglobalmegacorp.com

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