|
|
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: static char sccsid[] = "@(#)ar.c 5.3 (Berkeley) 10/22/87";
9: #endif not lint
10:
11: #ifndef lint
12: char copyright[] =
13: "@(#) Copyright (c) 1980 Regents of the University of California.\n\
14: All rights reserved.\n";
15: #endif not lint
16:
17: /*
18: * ar - portable (ascii) format version
19: */
20: #include <sys/param.h>
21: #include <sys/stat.h>
22: #include <sys/time.h>
23:
24: #include <stdio.h>
25: #include <ar.h>
26:
27: struct stat stbuf;
28: struct ar_hdr arbuf;
29: struct lar_hdr {
30: char lar_name[16];
31: long lar_date;
32: u_short lar_uid;
33: u_short lar_gid;
34: u_short lar_mode;
35: long lar_size;
36: } larbuf;
37:
38: #define SKIP 1
39: #define IODD 2
40: #define OODD 4
41: #define HEAD 8
42:
43: char *man = { "mrxtdpq" };
44: char *opt = { "uvnbailo" };
45:
46: int signum[] = {SIGHUP, SIGINT, SIGQUIT, 0};
47: int sigdone();
48: long lseek();
49: int rcmd();
50: int dcmd();
51: int xcmd();
52: int tcmd();
53: int pcmd();
54: int mcmd();
55: int qcmd();
56: int (*comfun)();
57: char flg[26];
58: char **namv;
59: int namc;
60: char *arnam;
61: char *ponam;
62: char *tmpnam = { "/tmp/vXXXXX" };
63: char *tmp1nam = { "/tmp/v1XXXXX" };
64: char *tmp2nam = { "/tmp/v2XXXXX" };
65: char *tfnam;
66: char *tf1nam;
67: char *tf2nam;
68: char *file;
69: char name[16];
70: int af;
71: int tf;
72: int tf1;
73: int tf2;
74: int qf;
75: int bastate;
76: char buf[MAXBSIZE];
77: int truncate; /* ok to truncate argument filenames */
78:
79: char *trim();
80: char *mktemp();
81: char *ctime();
82:
83: main(argc, argv)
84: char *argv[];
85: {
86: register i;
87: register char *cp;
88:
89: for(i=0; signum[i]; i++)
90: if(signal(signum[i], SIG_IGN) != SIG_IGN)
91: signal(signum[i], sigdone);
92: if(argc < 3)
93: usage();
94: cp = argv[1];
95: for(cp = argv[1]; *cp; cp++)
96: switch(*cp) {
97: case 'o':
98: case 'l':
99: case 'v':
100: case 'u':
101: case 'n':
102: case 'a':
103: case 'b':
104: case 'c':
105: case 'i':
106: flg[*cp - 'a']++;
107: continue;
108:
109: case 'r':
110: setcom(rcmd);
111: continue;
112:
113: case 'd':
114: setcom(dcmd);
115: continue;
116:
117: case 'x':
118: setcom(xcmd);
119: continue;
120:
121: case 't':
122: setcom(tcmd);
123: continue;
124:
125: case 'p':
126: setcom(pcmd);
127: continue;
128:
129: case 'm':
130: setcom(mcmd);
131: continue;
132:
133: case 'q':
134: setcom(qcmd);
135: continue;
136:
137: default:
138: fprintf(stderr, "ar: bad option `%c'\n", *cp);
139: done(1);
140: }
141: if(flg['l'-'a']) {
142: tmpnam = "vXXXXX";
143: tmp1nam = "v1XXXXX";
144: tmp2nam = "v2XXXXX";
145: }
146: if(flg['i'-'a'])
147: flg['b'-'a']++;
148: if(flg['a'-'a'] || flg['b'-'a']) {
149: bastate = 1;
150: ponam = trim(argv[2]);
151: argv++;
152: argc--;
153: if(argc < 3)
154: usage();
155: }
156: arnam = argv[2];
157: namv = argv+3;
158: namc = argc-3;
159: if(comfun == 0) {
160: if(flg['u'-'a'] == 0) {
161: fprintf(stderr, "ar: one of [%s] must be specified\n", man);
162: done(1);
163: }
164: setcom(rcmd);
165: }
166: (*comfun)();
167: done(notfound());
168: }
169:
170: setcom(fun)
171: int (*fun)();
172: {
173:
174: if(comfun != 0) {
175: fprintf(stderr, "ar: only one of [%s] allowed\n", man);
176: done(1);
177: }
178: comfun = fun;
179: }
180:
181: rcmd()
182: {
183: register f;
184:
185: init();
186: getaf();
187: while(!getdir()) {
188: bamatch();
189: if(namc == 0 || match()) {
190: f = stats();
191: if(f < 0) {
192: if(namc)
193: fprintf(stderr, "ar: cannot open %s\n", file);
194: goto cp;
195: }
196: if(flg['u'-'a'])
197: if(stbuf.st_mtime <= larbuf.lar_date) {
198: close(f);
199: goto cp;
200: }
201: mesg('r');
202: copyfil(af, -1, IODD+SKIP);
203: movefil(f);
204: continue;
205: }
206: cp:
207: mesg('c');
208: copyfil(af, tf, IODD+OODD+HEAD);
209: }
210: cleanup();
211: }
212:
213: dcmd()
214: {
215:
216: init();
217: if(getaf())
218: noar();
219: while(!getdir()) {
220: if(match()) {
221: mesg('d');
222: copyfil(af, -1, IODD+SKIP);
223: continue;
224: }
225: mesg('c');
226: copyfil(af, tf, IODD+OODD+HEAD);
227: }
228: install();
229: }
230:
231: xcmd()
232: {
233: register f;
234: struct timeval tv[2];
235:
236: if(getaf())
237: noar();
238: while(!getdir()) {
239: if(namc == 0 || match()) {
240: f = creat(file, larbuf.lar_mode & 0777);
241: if(f < 0) {
242: fprintf(stderr, "ar: %s cannot create\n", file);
243: goto sk;
244: }
245: mesg('x');
246: copyfil(af, f, IODD);
247: close(f);
248: if (flg['o'-'a']) {
249: tv[0].tv_sec = tv[1].tv_sec = larbuf.lar_date;
250: tv[0].tv_usec = tv[1].tv_usec = 0;
251: utimes(file, tv);
252: }
253: continue;
254: }
255: sk:
256: mesg('c');
257: copyfil(af, -1, IODD+SKIP);
258: if (namc > 0 && !morefil())
259: done(0);
260: }
261: }
262:
263: pcmd()
264: {
265:
266: if(getaf())
267: noar();
268: while(!getdir()) {
269: if(namc == 0 || match()) {
270: if(flg['v'-'a']) {
271: printf("\n<%s>\n\n", file);
272: fflush(stdout);
273: }
274: copyfil(af, 1, IODD);
275: continue;
276: }
277: copyfil(af, -1, IODD+SKIP);
278: }
279: }
280:
281: mcmd()
282: {
283:
284: init();
285: if(getaf())
286: noar();
287: tf2nam = mktemp(tmp2nam);
288: close(creat(tf2nam, 0600));
289: tf2 = open(tf2nam, 2);
290: if(tf2 < 0) {
291: fprintf(stderr, "ar: cannot create third temp\n");
292: done(1);
293: }
294: while(!getdir()) {
295: bamatch();
296: if(match()) {
297: mesg('m');
298: copyfil(af, tf2, IODD+OODD+HEAD);
299: continue;
300: }
301: mesg('c');
302: copyfil(af, tf, IODD+OODD+HEAD);
303: }
304: install();
305: }
306:
307: tcmd()
308: {
309:
310: if(getaf())
311: noar();
312: while(!getdir()) {
313: if(namc == 0 || match()) {
314: if(flg['v'-'a'])
315: longt();
316: printf("%s\n", trim(file));
317: }
318: copyfil(af, -1, IODD+SKIP);
319: }
320: }
321:
322: qcmd()
323: {
324: register i, f;
325:
326: if (flg['a'-'a'] || flg['b'-'a']) {
327: fprintf(stderr, "ar: abi not allowed with q\n");
328: done(1);
329: }
330: truncate++;
331: getqf();
332: for(i=0; signum[i]; i++)
333: signal(signum[i], SIG_IGN);
334: lseek(qf, 0l, 2);
335: for(i=0; i<namc; i++) {
336: file = namv[i];
337: if(file == 0)
338: continue;
339: namv[i] = 0;
340: mesg('q');
341: f = stats();
342: if(f < 0) {
343: fprintf(stderr, "ar: %s cannot open\n", file);
344: continue;
345: }
346: tf = qf;
347: movefil(f);
348: qf = tf;
349: }
350: }
351:
352: init()
353: {
354:
355: tfnam = mktemp(tmpnam);
356: close(creat(tfnam, 0600));
357: tf = open(tfnam, 2);
358: if(tf < 0) {
359: fprintf(stderr, "ar: cannot create temp file\n");
360: done(1);
361: }
362: if (write(tf, ARMAG, SARMAG) != SARMAG)
363: wrerr();
364: }
365:
366: getaf()
367: {
368: char mbuf[SARMAG];
369:
370: af = open(arnam, 0);
371: if(af < 0)
372: return(1);
373: if (read(af, mbuf, SARMAG) != SARMAG || strncmp(mbuf, ARMAG, SARMAG)) {
374: fprintf(stderr, "ar: %s not in archive format\n", arnam);
375: done(1);
376: }
377: return(0);
378: }
379:
380: getqf()
381: {
382: char mbuf[SARMAG];
383:
384: if ((qf = open(arnam, 2)) < 0) {
385: if(!flg['c'-'a'])
386: fprintf(stderr, "ar: creating %s\n", arnam);
387: if ((qf = creat(arnam, 0666)) < 0) {
388: fprintf(stderr, "ar: cannot create %s\n", arnam);
389: done(1);
390: }
391: if (write(qf, ARMAG, SARMAG) != SARMAG)
392: wrerr();
393: } else if (read(qf, mbuf, SARMAG) != SARMAG
394: || strncmp(mbuf, ARMAG, SARMAG)) {
395: fprintf(stderr, "ar: %s not in archive format\n", arnam);
396: done(1);
397: }
398: }
399:
400: usage()
401: {
402: printf("usage: ar [%s][%s] archive files ...\n", man, opt);
403: done(1);
404: }
405:
406: noar()
407: {
408:
409: fprintf(stderr, "ar: %s does not exist\n", arnam);
410: done(1);
411: }
412:
413: sigdone()
414: {
415: done(100);
416: }
417:
418: done(c)
419: {
420:
421: if(tfnam)
422: unlink(tfnam);
423: if(tf1nam)
424: unlink(tf1nam);
425: if(tf2nam)
426: unlink(tf2nam);
427: exit(c);
428: }
429:
430: notfound()
431: {
432: register i, n;
433:
434: n = 0;
435: for(i=0; i<namc; i++)
436: if(namv[i]) {
437: fprintf(stderr, "ar: %s not found\n", namv[i]);
438: n++;
439: }
440: return(n);
441: }
442:
443: morefil()
444: {
445: register i, n;
446:
447: n = 0;
448: for(i=0; i<namc; i++)
449: if(namv[i])
450: n++;
451: return(n);
452: }
453:
454: cleanup()
455: {
456: register i, f;
457:
458: truncate++;
459: for(i=0; i<namc; i++) {
460: file = namv[i];
461: if(file == 0)
462: continue;
463: namv[i] = 0;
464: mesg('a');
465: f = stats();
466: if(f < 0) {
467: fprintf(stderr, "ar: %s cannot open\n", file);
468: continue;
469: }
470: movefil(f);
471: }
472: install();
473: }
474:
475: install()
476: {
477: register i;
478:
479: for(i=0; signum[i]; i++)
480: signal(signum[i], SIG_IGN);
481: if(af < 0)
482: if(!flg['c'-'a'])
483: fprintf(stderr, "ar: creating %s\n", arnam);
484: close(af);
485: af = creat(arnam, 0666);
486: if(af < 0) {
487: fprintf(stderr, "ar: cannot create %s\n", arnam);
488: done(1);
489: }
490: if(tfnam) {
491: lseek(tf, 0l, 0);
492: while((i = read(tf, buf, MAXBSIZE)) > 0)
493: if (write(af, buf, i) != i)
494: wrerr();
495: }
496: if(tf2nam) {
497: lseek(tf2, 0l, 0);
498: while((i = read(tf2, buf, MAXBSIZE)) > 0)
499: if (write(af, buf, i) != i)
500: wrerr();
501: }
502: if(tf1nam) {
503: lseek(tf1, 0l, 0);
504: while((i = read(tf1, buf, MAXBSIZE)) > 0)
505: if (write(af, buf, i) != i)
506: wrerr();
507: }
508: }
509:
510: /*
511: * insert the file 'file'
512: * into the temporary file
513: */
514: movefil(f)
515: {
516: char buf[sizeof(arbuf)+1];
517:
518: (void)sprintf(buf, "%-16s%-12ld%-6u%-6u%-8o%-10ld%-2s",
519: trim(file),
520: stbuf.st_mtime,
521: (u_short)stbuf.st_uid,
522: (u_short)stbuf.st_gid,
523: stbuf.st_mode,
524: stbuf.st_size,
525: ARFMAG);
526: strncpy((char *)&arbuf, buf, sizeof(arbuf));
527: larbuf.lar_size = stbuf.st_size;
528: copyfil(f, tf, OODD+HEAD);
529: close(f);
530: }
531:
532: stats()
533: {
534: register f;
535:
536: f = open(file, 0);
537: if(f < 0)
538: return(f);
539: if(fstat(f, &stbuf) < 0) {
540: close(f);
541: return(-1);
542: }
543: return(f);
544: }
545:
546: /*
547: * copy next file
548: * size given in arbuf
549: */
550: copyfil(fi, fo, flag)
551: {
552: register i, o;
553: int pe;
554:
555: if(flag & HEAD) {
556: for (i=sizeof(arbuf.ar_name)-1; i>=0; i--) {
557: if (arbuf.ar_name[i]==' ')
558: continue;
559: else if (arbuf.ar_name[i]=='\0')
560: arbuf.ar_name[i] = ' ';
561: else
562: break;
563: }
564: if (write(fo, (char *)&arbuf, sizeof arbuf) != sizeof arbuf)
565: wrerr();
566: }
567: pe = 0;
568: while(larbuf.lar_size > 0) {
569: i = o = MAXBSIZE;
570: if(larbuf.lar_size < i) {
571: i = o = larbuf.lar_size;
572: if(i&1) {
573: buf[i] = '\n';
574: if(flag & IODD)
575: i++;
576: if(flag & OODD)
577: o++;
578: }
579: }
580: if(read(fi, buf, i) != i)
581: pe++;
582: if((flag & SKIP) == 0)
583: if (write(fo, buf, o) != o)
584: wrerr();
585: larbuf.lar_size -= MAXBSIZE;
586: }
587: if(pe)
588: phserr();
589: }
590:
591: getdir()
592: {
593: register char *cp;
594: register i;
595:
596: i = read(af, (char *)&arbuf, sizeof arbuf);
597: if(i != sizeof arbuf) {
598: if(tf1nam) {
599: i = tf;
600: tf = tf1;
601: tf1 = i;
602: }
603: return(1);
604: }
605: if (strncmp(arbuf.ar_fmag, ARFMAG, sizeof(arbuf.ar_fmag))) {
606: fprintf(stderr, "ar: malformed archive (at %ld)\n", lseek(af, 0L, 1));
607: done(1);
608: }
609: cp = arbuf.ar_name + sizeof(arbuf.ar_name);
610: while (*--cp==' ')
611: ;
612: *++cp = '\0';
613: strncpy(name, arbuf.ar_name, sizeof(arbuf.ar_name));
614: file = name;
615: strncpy(larbuf.lar_name, name, sizeof(larbuf.lar_name));
616: sscanf(arbuf.ar_date, "%ld", &larbuf.lar_date);
617: sscanf(arbuf.ar_uid, "%hd", &larbuf.lar_uid);
618: sscanf(arbuf.ar_gid, "%hd", &larbuf.lar_gid);
619: sscanf(arbuf.ar_mode, "%ho", &larbuf.lar_mode);
620: sscanf(arbuf.ar_size, "%ld", &larbuf.lar_size);
621: return(0);
622: }
623:
624: match()
625: {
626: register i;
627:
628: for(i=0; i<namc; i++) {
629: if(namv[i] == 0)
630: continue;
631: if(strcmp(trim(namv[i]), file) == 0) {
632: file = namv[i];
633: namv[i] = 0;
634: return(1);
635: }
636: }
637: return(0);
638: }
639:
640: bamatch()
641: {
642: register f;
643:
644: switch(bastate) {
645:
646: case 1:
647: if(strcmp(file, ponam) != 0)
648: return;
649: bastate = 2;
650: if(flg['a'-'a'])
651: return;
652:
653: case 2:
654: bastate = 0;
655: tf1nam = mktemp(tmp1nam);
656: close(creat(tf1nam, 0600));
657: f = open(tf1nam, 2);
658: if(f < 0) {
659: fprintf(stderr, "ar: cannot create second temp\n");
660: return;
661: }
662: tf1 = tf;
663: tf = f;
664: }
665: }
666:
667: phserr()
668: {
669:
670: fprintf(stderr, "ar: phase error on %s\n", file);
671: }
672:
673: mesg(c)
674: {
675:
676: if(flg['v'-'a'])
677: if(c != 'c' || flg['v'-'a'] > 1)
678: printf("%c - %s\n", c, file);
679: }
680:
681: char *
682: trim(s)
683: char *s;
684: {
685: register char *p1, *p2;
686:
687: /* Strip trailing slashes */
688: for(p1 = s; *p1; p1++)
689: ;
690: while(p1 > s) {
691: if(*--p1 != '/')
692: break;
693: *p1 = 0;
694: }
695:
696: /* Find last component of path; do not zap the path */
697: p2 = s;
698: for(p1 = s; *p1; p1++)
699: if(*p1 == '/')
700: p2 = p1+1;
701:
702: /*
703: * Truncate name if too long, only if we are doing an 'add'
704: * type operation. We only allow 15 cause rest of ar
705: * isn't smart enough to deal with non-null terminated
706: * names. Need an exit status convention...
707: * Need yet another new archive format...
708: */
709: if (truncate && strlen(p2) > sizeof(arbuf.ar_name) - 1) {
710: fprintf(stderr, "ar: filename %s truncated to ", p2);
711: *(p2 + sizeof(arbuf.ar_name) - 1) = '\0';
712: fprintf(stderr, "%s\n", p2);
713: }
714: return(p2);
715: }
716:
717: #define IFMT 060000
718: #define ISARG 01000
719: #define LARGE 010000
720: #define SUID 04000
721: #define SGID 02000
722: #define ROWN 0400
723: #define WOWN 0200
724: #define XOWN 0100
725: #define RGRP 040
726: #define WGRP 020
727: #define XGRP 010
728: #define ROTH 04
729: #define WOTH 02
730: #define XOTH 01
731: #define STXT 01000
732:
733: longt()
734: {
735: register char *cp;
736:
737: pmode();
738: printf("%3d/%1d", larbuf.lar_uid, larbuf.lar_gid);
739: printf("%7ld", larbuf.lar_size);
740: cp = ctime(&larbuf.lar_date);
741: printf(" %-12.12s %-4.4s ", cp+4, cp+20);
742: }
743:
744: int m1[] = { 1, ROWN, 'r', '-' };
745: int m2[] = { 1, WOWN, 'w', '-' };
746: int m3[] = { 2, SUID, 's', XOWN, 'x', '-' };
747: int m4[] = { 1, RGRP, 'r', '-' };
748: int m5[] = { 1, WGRP, 'w', '-' };
749: int m6[] = { 2, SGID, 's', XGRP, 'x', '-' };
750: int m7[] = { 1, ROTH, 'r', '-' };
751: int m8[] = { 1, WOTH, 'w', '-' };
752: int m9[] = { 2, STXT, 't', XOTH, 'x', '-' };
753:
754: int *m[] = { m1, m2, m3, m4, m5, m6, m7, m8, m9};
755:
756: pmode()
757: {
758: register int **mp;
759:
760: for (mp = &m[0]; mp < &m[9];)
761: select(*mp++);
762: }
763:
764: select(pairp)
765: int *pairp;
766: {
767: register int n, *ap;
768:
769: ap = pairp;
770: n = *ap++;
771: while (--n>=0 && (larbuf.lar_mode&*ap++)==0)
772: ap++;
773: putchar(*ap);
774: }
775:
776: wrerr()
777: {
778: perror("ar write error");
779: done(1);
780: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.