xref: /freebsd/bin/pax/cpio.c (revision 952d112864d8008aa87278a30a539d888a8493cd)
1 /*-
2  * Copyright (c) 1992 Keith Muller.
3  * Copyright (c) 1992, 1993
4  *	The Regents of the University of California.  All rights reserved.
5  *
6  * This code is derived from software contributed to Berkeley by
7  * Keith Muller of the University of California, San Diego.
8  *
9  * Redistribution and use in source and binary forms, with or without
10  * modification, are permitted provided that the following conditions
11  * are met:
12  * 1. Redistributions of source code must retain the above copyright
13  *    notice, this list of conditions and the following disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  * 3. All advertising materials mentioning features or use of this software
18  *    must display the following acknowledgement:
19  *	This product includes software developed by the University of
20  *	California, Berkeley and its contributors.
21  * 4. Neither the name of the University nor the names of its contributors
22  *    may be used to endorse or promote products derived from this software
23  *    without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
26  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
27  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
28  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
29  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
30  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
31  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
32  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
33  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
34  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35  * SUCH DAMAGE.
36  *
37  *	$Id$
38  */
39 
40 #ifndef lint
41 static char const sccsid[] = "@(#)cpio.c	8.1 (Berkeley) 5/31/93";
42 #endif /* not lint */
43 
44 #include <sys/types.h>
45 #include <sys/time.h>
46 #include <sys/stat.h>
47 #include <sys/param.h>
48 #include <string.h>
49 #include <stdio.h>
50 #include <unistd.h>
51 #include <stdlib.h>
52 #include "pax.h"
53 #include "cpio.h"
54 #include "extern.h"
55 
56 static int rd_nm __P((register ARCHD *, int));
57 static int rd_ln_nm __P((register ARCHD *));
58 static int com_rd __P((register ARCHD *));
59 
60 /*
61  * Routines which support the different cpio versions
62  */
63 
64 static int swp_head;		/* binary cpio header byte swap */
65 
66 /*
67  * Routines common to all versions of cpio
68  */
69 
70 /*
71  * cpio_strd()
72  *	Fire up the hard link detection code
73  * Return:
74  *      0 if ok -1 otherwise (the return values of lnk_start())
75  */
76 
77 #if __STDC__
78 int
79 cpio_strd(void)
80 #else
81 int
82 cpio_strd()
83 #endif
84 {
85 	return(lnk_start());
86 }
87 
88 /*
89  * cpio_trail()
90  *	Called to determine if a header block is a valid trailer. We are
91  *	passed the block, the in_sync flag (which tells us we are in resync
92  *	mode; looking for a valid header), and cnt (which starts at zero)
93  *	which is used to count the number of empty blocks we have seen so far.
94  * Return:
95  *	0 if a valid trailer, -1 if not a valid trailer,
96  */
97 
98 #if __STDC__
99 int
100 cpio_trail(register ARCHD *arcn)
101 #else
102 int
103 cpio_trail(arcn)
104 	register ARCHD *arcn;
105 #endif
106 {
107 	/*
108 	 * look for trailer id in file we are about to process
109 	 */
110 	if ((strcmp(arcn->name, TRAILER) == 0) && (arcn->sb.st_size == 0))
111 		return(0);
112 	return(-1);
113 }
114 
115 /*
116  * com_rd()
117  *	operations common to all cpio read functions.
118  * Return:
119  *	0
120  */
121 
122 #if __STDC__
123 static int
124 com_rd(register ARCHD *arcn)
125 #else
126 static int
127 com_rd(arcn)
128 	register ARCHD *arcn;
129 #endif
130 {
131 	arcn->skip = 0;
132 	arcn->pat = NULL;
133 	arcn->org_name = arcn->name;
134 	switch(arcn->sb.st_mode & C_IFMT) {
135 	case C_ISFIFO:
136 		arcn->type = PAX_FIF;
137 		break;
138 	case C_ISDIR:
139 		arcn->type = PAX_DIR;
140 		break;
141 	case C_ISBLK:
142 		arcn->type = PAX_BLK;
143 		break;
144 	case C_ISCHR:
145 		arcn->type = PAX_CHR;
146 		break;
147 	case C_ISLNK:
148 		arcn->type = PAX_SLK;
149 		break;
150 	case C_ISOCK:
151 		arcn->type = PAX_SCK;
152 		break;
153 	case C_ISCTG:
154 	case C_ISREG:
155 	default:
156 		/*
157 		 * we have file data, set up skip (pad is set in the format
158 		 * specific sections)
159 		 */
160 		arcn->sb.st_mode = (arcn->sb.st_mode & 0xfff) | C_ISREG;
161 		arcn->type = PAX_REG;
162 		arcn->skip = arcn->sb.st_size;
163 		break;
164 	}
165 	if (chk_lnk(arcn) < 0)
166 		return(-1);
167 	return(0);
168 }
169 
170 /*
171  * cpio_end_wr()
172  *	write the special file with the name trailer in the proper format
173  * Return:
174  *	result of the write of the trailer from the cpio specific write func
175  */
176 
177 #if __STDC__
178 int
179 cpio_endwr(void)
180 #else
181 int
182 cpio_endwr()
183 #endif
184 {
185 	ARCHD last;
186 
187 	/*
188 	 * create a trailer request and call the proper format write function
189 	 */
190 	bzero((char *)&last, sizeof(last));
191 	last.nlen = sizeof(TRAILER) - 1;
192 	last.type = PAX_REG;
193 	last.sb.st_nlink = 1;
194 	(void)strcpy(last.name, TRAILER);
195 	return((*frmt->wr)(&last));
196 }
197 
198 /*
199  * rd_nam()
200  *	read in the file name which follows the cpio header
201  * Return:
202  *	0 if ok, -1 otherwise
203  */
204 
205 #if __STDC__
206 static int
207 rd_nm(register ARCHD *arcn, int nsz)
208 #else
209 static int
210 rd_nm(arcn, nsz)
211 	register ARCHD *arcn;
212 	int nsz;
213 #endif
214 {
215 	/*
216 	 * do not even try bogus values
217 	 */
218 	if ((nsz == 0) || (nsz > sizeof(arcn->name))) {
219 		warn(1, "Cpio file name length %d is out of range", nsz);
220 		return(-1);
221 	}
222 
223 	/*
224 	 * read the name and make sure it is not empty and is \0 terminated
225 	 */
226 	if ((rd_wrbuf(arcn->name,nsz) != nsz) || (arcn->name[nsz-1] != '\0') ||
227 	    (arcn->name[0] == '\0')) {
228 		warn(1, "Cpio file name in header is corrupted");
229 		return(-1);
230 	}
231 	return(0);
232 }
233 
234 /*
235  * rd_ln_nm()
236  *	read in the link name for a file with links. The link name is stored
237  *	like file data (and is NOT \0 terminated!)
238  * Return:
239  *	0 if ok, -1 otherwise
240  */
241 
242 #if __STDC__
243 static int
244 rd_ln_nm(register ARCHD *arcn)
245 #else
246 static int
247 rd_ln_nm(arcn)
248 	register ARCHD *arcn;
249 #endif
250 {
251 	/*
252 	 * check the length specified for bogus values
253 	 */
254 	if ((arcn->sb.st_size == 0) ||
255 	    (arcn->sb.st_size >= sizeof(arcn->ln_name))) {
256 #		ifdef NET2_STAT
257 		warn(1, "Cpio link name length is invalid: %lu",
258 		    arcn->sb.st_size);
259 #		else
260 		warn(1, "Cpio link name length is invalid: %qu",
261 		    arcn->sb.st_size);
262 #		endif
263 		return(-1);
264 	}
265 
266 	/*
267 	 * read in the link name and \0 terminate it
268 	 */
269 	if (rd_wrbuf(arcn->ln_name, (int)arcn->sb.st_size) !=
270 	    (int)arcn->sb.st_size) {
271 		warn(1, "Cpio link name read error");
272 		return(-1);
273 	}
274 	arcn->ln_nlen = arcn->sb.st_size;
275 	arcn->ln_name[arcn->ln_nlen] = '\0';
276 
277 	/*
278 	 * watch out for those empty link names
279 	 */
280 	if (arcn->ln_name[0] == '\0') {
281 		warn(1, "Cpio link name is corrupt");
282 		return(-1);
283 	}
284 	return(0);
285 }
286 
287 /*
288  * Routines common to the extended byte oriented cpio format
289  */
290 
291 /*
292  * cpio_id()
293  *      determine if a block given to us is a valid extended byte oriented
294  *	cpio header
295  * Return:
296  *      0 if a valid header, -1 otherwise
297  */
298 
299 #if __STDC__
300 int
301 cpio_id(char *blk, int size)
302 #else
303 int
304 cpio_id(blk, size)
305 	char *blk;
306 	int size;
307 #endif
308 {
309 	if ((size < sizeof(HD_CPIO)) ||
310 	    (strncmp(blk, AMAGIC, sizeof(AMAGIC) - 1) != 0))
311 		return(-1);
312 	return(0);
313 }
314 
315 /*
316  * cpio_rd()
317  *	determine if a buffer is a byte oriented extended cpio archive entry.
318  *	convert and store the values in the ARCHD parameter.
319  * Return:
320  *	0 if a valid header, -1 otherwise.
321  */
322 
323 #if __STDC__
324 int
325 cpio_rd(register ARCHD *arcn, register char *buf)
326 #else
327 int
328 cpio_rd(arcn, buf)
329 	register ARCHD *arcn;
330 	register char *buf;
331 #endif
332 {
333 	register int nsz;
334 	register HD_CPIO *hd;
335 
336 	/*
337 	 * check that this is a valid header, if not return -1
338 	 */
339 	if (cpio_id(buf, sizeof(HD_CPIO)) < 0)
340 		return(-1);
341 	hd = (HD_CPIO *)buf;
342 
343 	/*
344 	 * byte oriented cpio (posix) does not have padding! extract the octal
345 	 * ascii fields from the header
346 	 */
347 	arcn->pad = 0L;
348 	arcn->sb.st_dev = (dev_t)asc_ul(hd->c_dev, sizeof(hd->c_dev), OCT);
349 	arcn->sb.st_ino = (ino_t)asc_ul(hd->c_ino, sizeof(hd->c_ino), OCT);
350 	arcn->sb.st_mode = (mode_t)asc_ul(hd->c_mode, sizeof(hd->c_mode), OCT);
351 	arcn->sb.st_uid = (uid_t)asc_ul(hd->c_uid, sizeof(hd->c_uid), OCT);
352 	arcn->sb.st_gid = (gid_t)asc_ul(hd->c_gid, sizeof(hd->c_gid), OCT);
353 	arcn->sb.st_nlink = (nlink_t)asc_ul(hd->c_nlink, sizeof(hd->c_nlink),
354 	    OCT);
355 	arcn->sb.st_rdev = (dev_t)asc_ul(hd->c_rdev, sizeof(hd->c_rdev), OCT);
356 	arcn->sb.st_mtime = (time_t)asc_ul(hd->c_mtime, sizeof(hd->c_mtime),
357 	    OCT);
358 	arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime;
359 #	ifdef NET2_STAT
360 	arcn->sb.st_size = (off_t)asc_ul(hd->c_filesize,sizeof(hd->c_filesize),
361 	    OCT);
362 #	else
363 	arcn->sb.st_size = (off_t)asc_uqd(hd->c_filesize,sizeof(hd->c_filesize),
364 	    OCT);
365 #	endif
366 
367 	/*
368 	 * check name size and if valid, read in the name of this entry (name
369 	 * follows header in the archive)
370 	 */
371 	if ((nsz = (int)asc_ul(hd->c_namesize,sizeof(hd->c_namesize),OCT)) < 2)
372 		return(-1);
373 	arcn->nlen = nsz - 1;
374 	if (rd_nm(arcn, nsz) < 0)
375 		return(-1);
376 
377 	if (((arcn->sb.st_mode&C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)) {
378 		/*
379 	 	 * no link name to read for this file
380 	 	 */
381 		arcn->ln_nlen = 0;
382 		arcn->ln_name[0] = '\0';
383 		return(com_rd(arcn));
384 	}
385 
386 	/*
387 	 * check link name size and read in the link name. Link names are
388 	 * stored like file data.
389 	 */
390 	if (rd_ln_nm(arcn) < 0)
391 		return(-1);
392 
393 	/*
394 	 * we have a valid header (with a link)
395 	 */
396 	return(com_rd(arcn));
397 }
398 
399 /*
400  * cpio_endrd()
401  *      no cleanup needed here, just return size of the trailer (for append)
402  * Return:
403  *      size of trailer header in this format
404  */
405 
406 #if __STDC__
407 off_t
408 cpio_endrd(void)
409 #else
410 off_t
411 cpio_endrd()
412 #endif
413 {
414 	return((off_t)(sizeof(HD_CPIO) + sizeof(TRAILER)));
415 }
416 
417 /*
418  * cpio_stwr()
419  *	start up the device mapping table
420  * Return:
421  *	0 if ok, -1 otherwise (what dev_start() returns)
422  */
423 
424 #if __STDC__
425 int
426 cpio_stwr(void)
427 #else
428 int
429 cpio_stwr()
430 #endif
431 {
432 	return(dev_start());
433 }
434 
435 /*
436  * cpio_wr()
437  *	copy the data in the ARCHD to buffer in extended byte oriented cpio
438  *	format.
439  * Return
440  *      0 if file has data to be written after the header, 1 if file has NO
441  *	data to write after the header, -1 if archive write failed
442  */
443 
444 #if __STDC__
445 int
446 cpio_wr(register ARCHD *arcn)
447 #else
448 int
449 cpio_wr(arcn)
450 	register ARCHD *arcn;
451 #endif
452 {
453 	register HD_CPIO *hd;
454 	register int nsz;
455 	char hdblk[sizeof(HD_CPIO)];
456 
457 	/*
458 	 * check and repair truncated device and inode fields in the header
459 	 */
460 	if (map_dev(arcn, (u_long)CPIO_MASK, (u_long)CPIO_MASK) < 0)
461 		return(-1);
462 
463 	arcn->pad = 0L;
464 	nsz = arcn->nlen + 1;
465 	hd = (HD_CPIO *)hdblk;
466 	if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR))
467 		arcn->sb.st_rdev = 0;
468 
469 	switch(arcn->type) {
470 	case PAX_CTG:
471 	case PAX_REG:
472 	case PAX_HRG:
473 		/*
474 		 * set data size for file data
475 		 */
476 #		ifdef NET2_STAT
477 		if (ul_asc((u_long)arcn->sb.st_size, hd->c_filesize,
478 		    sizeof(hd->c_filesize), OCT)) {
479 #		else
480 		if (uqd_asc((u_quad_t)arcn->sb.st_size, hd->c_filesize,
481 		    sizeof(hd->c_filesize), OCT)) {
482 #		endif
483 			warn(1,"File is too large for cpio format %s",
484 			    arcn->org_name);
485 			return(1);
486 		}
487 		break;
488 	case PAX_SLK:
489 		/*
490 		 * set data size to hold link name
491 		 */
492 		if (ul_asc((u_long)arcn->ln_nlen, hd->c_filesize,
493 		    sizeof(hd->c_filesize), OCT))
494 			goto out;
495 		break;
496 	default:
497 		/*
498 		 * all other file types have no file data
499 		 */
500 		if (ul_asc((u_long)0, hd->c_filesize, sizeof(hd->c_filesize),
501 		     OCT))
502 			goto out;
503 		break;
504 	}
505 
506 	/*
507 	 * copy the values to the header using octal ascii
508 	 */
509 	if (ul_asc((u_long)MAGIC, hd->c_magic, sizeof(hd->c_magic), OCT) ||
510 	    ul_asc((u_long)arcn->sb.st_dev, hd->c_dev, sizeof(hd->c_dev),
511 	        OCT) ||
512 	    ul_asc((u_long)arcn->sb.st_ino, hd->c_ino, sizeof(hd->c_ino),
513 		OCT) ||
514 	    ul_asc((u_long)arcn->sb.st_mode, hd->c_mode, sizeof(hd->c_mode),
515 		OCT) ||
516 	    ul_asc((u_long)arcn->sb.st_uid, hd->c_uid, sizeof(hd->c_uid),
517 		OCT) ||
518 	    ul_asc((u_long)arcn->sb.st_gid, hd->c_gid, sizeof(hd->c_gid),
519 		OCT) ||
520 	    ul_asc((u_long)arcn->sb.st_nlink, hd->c_nlink, sizeof(hd->c_nlink),
521 		 OCT) ||
522 	    ul_asc((u_long)arcn->sb.st_rdev, hd->c_rdev, sizeof(hd->c_rdev),
523 		OCT) ||
524 	    ul_asc((u_long)arcn->sb.st_mtime,hd->c_mtime,sizeof(hd->c_mtime),
525 		OCT) ||
526 	    ul_asc((u_long)nsz, hd->c_namesize, sizeof(hd->c_namesize), OCT))
527 		goto out;
528 
529 	/*
530 	 * write the file name to the archive
531 	 */
532 	if ((wr_rdbuf(hdblk, (int)sizeof(HD_CPIO)) < 0) ||
533 	    (wr_rdbuf(arcn->name, nsz) < 0)) {
534 		warn(1, "Unable to write cpio header for %s", arcn->org_name);
535 		return(-1);
536 	}
537 
538 	/*
539 	 * if this file has data, we are done. The caller will write the file
540 	 * data, if we are link tell caller we are done, go to next file
541 	 */
542 	if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) ||
543 	    (arcn->type == PAX_HRG))
544 		return(0);
545 	if (arcn->type != PAX_SLK)
546 		return(1);
547 
548 	/*
549 	 * write the link name to the archive, tell the caller to go to the
550 	 * next file as we are done.
551 	 */
552 	if (wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) {
553 		warn(1,"Unable to write cpio link name for %s",arcn->org_name);
554 		return(-1);
555 	}
556 	return(1);
557 
558     out:
559 	/*
560 	 * header field is out of range
561 	 */
562 	warn(1, "Cpio header field is too small to store file %s",
563 	    arcn->org_name);
564 	return(1);
565 }
566 
567 /*
568  * Routines common to the system VR4 version of cpio (with/without file CRC)
569  */
570 
571 /*
572  * vcpio_id()
573  *      determine if a block given to us is a valid system VR4 cpio header
574  *	WITHOUT crc. WATCH it the magic cookies are in OCTAL, the header
575  *	uses HEX
576  * Return:
577  *      0 if a valid header, -1 otherwise
578  */
579 
580 #if __STDC__
581 int
582 vcpio_id(char *blk, int size)
583 #else
584 int
585 vcpio_id(blk, size)
586 	char *blk;
587 	int size;
588 #endif
589 {
590 	if ((size < sizeof(HD_VCPIO)) ||
591 	    (strncmp(blk, AVMAGIC, sizeof(AVMAGIC) - 1) != 0))
592 		return(-1);
593 	return(0);
594 }
595 
596 /*
597  * crc_id()
598  *      determine if a block given to us is a valid system VR4 cpio header
599  *	WITH crc. WATCH it the magic cookies are in OCTAL the header uses HEX
600  * Return:
601  *      0 if a valid header, -1 otherwise
602  */
603 
604 #if __STDC__
605 int
606 crc_id(char *blk, int size)
607 #else
608 int
609 crc_id(blk, size)
610 	char *blk;
611 	int size;
612 #endif
613 {
614 	if ((size < sizeof(HD_VCPIO)) ||
615 	    (strncmp(blk, AVCMAGIC, sizeof(AVCMAGIC) - 1) != 0))
616 		return(-1);
617 	return(0);
618 }
619 
620 /*
621  * crc_strd()
622  w	set file data CRC calculations. Fire up the hard link detection code
623  * Return:
624  *      0 if ok -1 otherwise (the return values of lnk_start())
625  */
626 
627 #if __STDC__
628 int
629 crc_strd(void)
630 #else
631 int
632 crc_strd()
633 #endif
634 {
635 	docrc = 1;
636 	return(lnk_start());
637 }
638 
639 /*
640  * vcpio_rd()
641  *	determine if a buffer is a system VR4 archive entry. (with/without CRC)
642  *	convert and store the values in the ARCHD parameter.
643  * Return:
644  *	0 if a valid header, -1 otherwise.
645  */
646 
647 #if __STDC__
648 int
649 vcpio_rd(register ARCHD *arcn, register char *buf)
650 #else
651 int
652 vcpio_rd(arcn, buf)
653 	register ARCHD *arcn;
654 	register char *buf;
655 #endif
656 {
657 	register HD_VCPIO *hd;
658 	dev_t devminor;
659 	dev_t devmajor;
660 	register int nsz;
661 
662 	/*
663 	 * during the id phase it was determined if we were using CRC, use the
664 	 * proper id routine.
665 	 */
666 	if (docrc) {
667 		if (crc_id(buf, sizeof(HD_VCPIO)) < 0)
668 			return(-1);
669 	} else {
670 		if (vcpio_id(buf, sizeof(HD_VCPIO)) < 0)
671 			return(-1);
672 	}
673 
674 	hd = (HD_VCPIO *)buf;
675 	arcn->pad = 0L;
676 
677 	/*
678 	 * extract the hex ascii fields from the header
679 	 */
680 	arcn->sb.st_ino = (ino_t)asc_ul(hd->c_ino, sizeof(hd->c_ino), HEX);
681 	arcn->sb.st_mode = (mode_t)asc_ul(hd->c_mode, sizeof(hd->c_mode), HEX);
682 	arcn->sb.st_uid = (uid_t)asc_ul(hd->c_uid, sizeof(hd->c_uid), HEX);
683 	arcn->sb.st_gid = (gid_t)asc_ul(hd->c_gid, sizeof(hd->c_gid), HEX);
684 	arcn->sb.st_mtime = (time_t)asc_ul(hd->c_mtime,sizeof(hd->c_mtime),HEX);
685 	arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime;
686 #	ifdef NET2_STAT
687 	arcn->sb.st_size = (off_t)asc_ul(hd->c_filesize,
688 	    sizeof(hd->c_filesize), HEX);
689 #	else
690 	arcn->sb.st_size = (off_t)asc_uqd(hd->c_filesize,
691 	    sizeof(hd->c_filesize), HEX);
692 #	endif
693 	arcn->sb.st_nlink = (nlink_t)asc_ul(hd->c_nlink, sizeof(hd->c_nlink),
694 	    HEX);
695 	devmajor = (dev_t)asc_ul(hd->c_maj, sizeof(hd->c_maj), HEX);
696 	devminor = (dev_t)asc_ul(hd->c_min, sizeof(hd->c_min), HEX);
697 	arcn->sb.st_dev = TODEV(devmajor, devminor);
698 	devmajor = (dev_t)asc_ul(hd->c_rmaj, sizeof(hd->c_maj), HEX);
699 	devminor = (dev_t)asc_ul(hd->c_rmin, sizeof(hd->c_min), HEX);
700 	arcn->sb.st_rdev = TODEV(devmajor, devminor);
701 	arcn->crc = asc_ul(hd->c_chksum, sizeof(hd->c_chksum), HEX);
702 
703 	/*
704 	 * check the length of the file name, if ok read it in, return -1 if
705 	 * bogus
706 	 */
707 	if ((nsz = (int)asc_ul(hd->c_namesize,sizeof(hd->c_namesize),HEX)) < 2)
708 		return(-1);
709 	arcn->nlen = nsz - 1;
710 	if (rd_nm(arcn, nsz) < 0)
711 		return(-1);
712 
713 	/*
714 	 * skip padding. header + filename is aligned to 4 byte boundries
715 	 */
716 	if (rd_skip((off_t)(VCPIO_PAD(sizeof(HD_VCPIO) + nsz))) < 0)
717 		return(-1);
718 
719 	/*
720 	 * if not a link (or a file with no data), calculate pad size (for
721 	 * padding which follows the file data), clear the link name and return
722 	 */
723 	if (((arcn->sb.st_mode&C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)) {
724 		/*
725 		 * we have a valid header (not a link)
726 		 */
727 		arcn->ln_nlen = 0;
728 		arcn->ln_name[0] = '\0';
729 		arcn->pad = VCPIO_PAD(arcn->sb.st_size);
730 		return(com_rd(arcn));
731 	}
732 
733 	/*
734 	 * read in the link name and skip over the padding
735 	 */
736 	if ((rd_ln_nm(arcn) < 0) ||
737 	    (rd_skip((off_t)(VCPIO_PAD(arcn->sb.st_size))) < 0))
738 		return(-1);
739 
740 	/*
741 	 * we have a valid header (with a link)
742 	 */
743 	return(com_rd(arcn));
744 }
745 
746 /*
747  * vcpio_endrd()
748  *      no cleanup needed here, just return size of the trailer (for append)
749  * Return:
750  *      size of trailer header in this format
751  */
752 
753 #if __STDC__
754 off_t
755 vcpio_endrd(void)
756 #else
757 off_t
758 vcpio_endrd()
759 #endif
760 {
761 	return((off_t)(sizeof(HD_VCPIO) + sizeof(TRAILER) +
762 		(VCPIO_PAD(sizeof(HD_VCPIO) + sizeof(TRAILER)))));
763 }
764 
765 /*
766  * crc_stwr()
767  *	start up the device mapping table, enable crc file calculation
768  * Return:
769  *	0 if ok, -1 otherwise (what dev_start() returns)
770  */
771 
772 #if __STDC__
773 int
774 crc_stwr(void)
775 #else
776 int
777 crc_stwr()
778 #endif
779 {
780 	docrc = 1;
781 	return(dev_start());
782 }
783 
784 /*
785  * vcpio_wr()
786  *	copy the data in the ARCHD to buffer in system VR4 cpio
787  *	(with/without crc) format.
788  * Return
789  *	0 if file has data to be written after the header, 1 if file has
790  *	NO data to write after the header, -1 if archive write failed
791  */
792 
793 #if __STDC__
794 int
795 vcpio_wr(register ARCHD *arcn)
796 #else
797 int
798 vcpio_wr(arcn)
799 	register ARCHD *arcn;
800 #endif
801 {
802 	register HD_VCPIO *hd;
803 	unsigned int nsz;
804 	char hdblk[sizeof(HD_VCPIO)];
805 
806 	/*
807 	 * check and repair truncated device and inode fields in the cpio
808 	 * header
809 	 */
810 	if (map_dev(arcn, (u_long)VCPIO_MASK, (u_long)VCPIO_MASK) < 0)
811 		return(-1);
812 	nsz = arcn->nlen + 1;
813 	hd = (HD_VCPIO *)hdblk;
814 	if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR))
815 		arcn->sb.st_rdev = 0;
816 
817 	/*
818 	 * add the proper magic value depending whether we were asked for
819 	 * file data crc's, and the crc if needed.
820 	 */
821 	if (docrc) {
822 		if (ul_asc((u_long)VCMAGIC, hd->c_magic, sizeof(hd->c_magic),
823 	    		OCT) ||
824 		    ul_asc((u_long)arcn->crc,hd->c_chksum,sizeof(hd->c_chksum),
825 	    		HEX))
826 			goto out;
827 	} else {
828 		if (ul_asc((u_long)VMAGIC, hd->c_magic, sizeof(hd->c_magic),
829 	    		OCT) ||
830 		    ul_asc((u_long)0L, hd->c_chksum, sizeof(hd->c_chksum),HEX))
831 			goto out;
832 	}
833 
834 	switch(arcn->type) {
835 	case PAX_CTG:
836 	case PAX_REG:
837 	case PAX_HRG:
838 		/*
839 		 * caller will copy file data to the archive. tell him how
840 		 * much to pad.
841 		 */
842 		arcn->pad = VCPIO_PAD(arcn->sb.st_size);
843 #		ifdef NET2_STAT
844 		if (ul_asc((u_long)arcn->sb.st_size, hd->c_filesize,
845 		    sizeof(hd->c_filesize), HEX)) {
846 #		else
847 		if (uqd_asc((u_quad_t)arcn->sb.st_size, hd->c_filesize,
848 		    sizeof(hd->c_filesize), HEX)) {
849 #		endif
850 			warn(1,"File is too large for sv4cpio format %s",
851 			    arcn->org_name);
852 			return(1);
853 		}
854 		break;
855 	case PAX_SLK:
856 		/*
857 		 * no file data for the caller to process, the file data has
858 		 * the size of the link
859 		 */
860 		arcn->pad = 0L;
861 		if (ul_asc((u_long)arcn->ln_nlen, hd->c_filesize,
862 		    sizeof(hd->c_filesize), HEX))
863 			goto out;
864 		break;
865 	default:
866 		/*
867 		 * no file data for the caller to process
868 		 */
869 		arcn->pad = 0L;
870 		if (ul_asc((u_long)0L, hd->c_filesize, sizeof(hd->c_filesize),
871 		    HEX))
872 			goto out;
873 		break;
874 	}
875 
876 	/*
877 	 * set the other fields in the header
878 	 */
879 	if (ul_asc((u_long)arcn->sb.st_ino, hd->c_ino, sizeof(hd->c_ino),
880 		HEX) ||
881 	    ul_asc((u_long)arcn->sb.st_mode, hd->c_mode, sizeof(hd->c_mode),
882 		HEX) ||
883 	    ul_asc((u_long)arcn->sb.st_uid, hd->c_uid, sizeof(hd->c_uid),
884 		HEX) ||
885 	    ul_asc((u_long)arcn->sb.st_gid, hd->c_gid, sizeof(hd->c_gid),
886     		HEX) ||
887 	    ul_asc((u_long)arcn->sb.st_mtime, hd->c_mtime, sizeof(hd->c_mtime),
888     		HEX) ||
889 	    ul_asc((u_long)arcn->sb.st_nlink, hd->c_nlink, sizeof(hd->c_nlink),
890     		HEX) ||
891 	    ul_asc((u_long)MAJOR(arcn->sb.st_dev),hd->c_maj, sizeof(hd->c_maj),
892 		HEX) ||
893 	    ul_asc((u_long)MINOR(arcn->sb.st_dev),hd->c_min, sizeof(hd->c_min),
894 		HEX) ||
895 	    ul_asc((u_long)MAJOR(arcn->sb.st_rdev),hd->c_rmaj,sizeof(hd->c_maj),
896 		HEX) ||
897 	    ul_asc((u_long)MINOR(arcn->sb.st_rdev),hd->c_rmin,sizeof(hd->c_min),
898 		HEX) ||
899 	    ul_asc((u_long)nsz, hd->c_namesize, sizeof(hd->c_namesize), HEX))
900 		goto out;
901 
902 	/*
903 	 * write the header, the file name and padding as required.
904 	 */
905 	if ((wr_rdbuf(hdblk, (int)sizeof(HD_VCPIO)) < 0) ||
906 	    (wr_rdbuf(arcn->name, (int)nsz) < 0)  ||
907 	    (wr_skip((off_t)(VCPIO_PAD(sizeof(HD_VCPIO) + nsz))) < 0)) {
908 		warn(1,"Could not write sv4cpio header for %s",arcn->org_name);
909 		return(-1);
910 	}
911 
912 	/*
913 	 * if we have file data, tell the caller we are done, copy the file
914 	 */
915 	if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) ||
916 	    (arcn->type == PAX_HRG))
917 		return(0);
918 
919 	/*
920 	 * if we are not a link, tell the caller we are done, go to next file
921 	 */
922 	if (arcn->type != PAX_SLK)
923 		return(1);
924 
925 	/*
926 	 * write the link name, tell the caller we are done.
927 	 */
928 	if ((wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) ||
929 	    (wr_skip((off_t)(VCPIO_PAD(arcn->ln_nlen))) < 0)) {
930 		warn(1,"Could not write sv4cpio link name for %s",
931 		    arcn->org_name);
932 		return(-1);
933 	}
934 	return(1);
935 
936     out:
937 	/*
938 	 * header field is out of range
939 	 */
940 	warn(1,"Sv4cpio header field is too small for file %s",arcn->org_name);
941 	return(1);
942 }
943 
944 /*
945  * Routines common to the old binary header cpio
946  */
947 
948 /*
949  * bcpio_id()
950  *      determine if a block given to us is a old binary cpio header
951  *	(with/without header byte swapping)
952  * Return:
953  *      0 if a valid header, -1 otherwise
954  */
955 
956 #if __STDC__
957 int
958 bcpio_id(char *blk, int size)
959 #else
960 int
961 bcpio_id(blk, size)
962 	char *blk;
963 	int size;
964 #endif
965 {
966 	if (size < sizeof(HD_BCPIO))
967 		return(-1);
968 
969 	/*
970 	 * check both normal and byte swapped magic cookies
971 	 */
972 	if (((u_short)SHRT_EXT(blk)) == MAGIC)
973 		return(0);
974 	if (((u_short)RSHRT_EXT(blk)) == MAGIC) {
975 		if (!swp_head)
976 			++swp_head;
977 		return(0);
978 	}
979 	return(-1);
980 }
981 
982 /*
983  * bcpio_rd()
984  *	determine if a buffer is a old binary archive entry. (it may have byte
985  *	swapped header) convert and store the values in the ARCHD parameter.
986  *	This is a very old header format and should not really be used.
987  * Return:
988  *	0 if a valid header, -1 otherwise.
989  */
990 
991 #if __STDC__
992 int
993 bcpio_rd(register ARCHD *arcn, register char *buf)
994 #else
995 int
996 bcpio_rd(arcn, buf)
997 	register ARCHD *arcn;
998 	register char *buf;
999 #endif
1000 {
1001 	register HD_BCPIO *hd;
1002 	register int nsz;
1003 
1004 	/*
1005 	 * check the header
1006 	 */
1007 	if (bcpio_id(buf, sizeof(HD_BCPIO)) < 0)
1008 		return(-1);
1009 
1010 	arcn->pad = 0L;
1011 	hd = (HD_BCPIO *)buf;
1012 	if (swp_head) {
1013 		/*
1014 		 * header has swapped bytes on 16 bit boundries
1015 		 */
1016 		arcn->sb.st_dev = (dev_t)(RSHRT_EXT(hd->h_dev));
1017 		arcn->sb.st_ino = (ino_t)(RSHRT_EXT(hd->h_ino));
1018 		arcn->sb.st_mode = (mode_t)(RSHRT_EXT(hd->h_mode));
1019 		arcn->sb.st_uid = (uid_t)(RSHRT_EXT(hd->h_uid));
1020 		arcn->sb.st_gid = (gid_t)(RSHRT_EXT(hd->h_gid));
1021 		arcn->sb.st_nlink = (nlink_t)(RSHRT_EXT(hd->h_nlink));
1022 		arcn->sb.st_rdev = (dev_t)(RSHRT_EXT(hd->h_rdev));
1023 		arcn->sb.st_mtime = (time_t)(RSHRT_EXT(hd->h_mtime_1));
1024 		arcn->sb.st_mtime =  (arcn->sb.st_mtime << 16) |
1025 			((time_t)(RSHRT_EXT(hd->h_mtime_2)));
1026 		arcn->sb.st_size = (off_t)(RSHRT_EXT(hd->h_filesize_1));
1027 		arcn->sb.st_size = (arcn->sb.st_size << 16) |
1028 			((off_t)(RSHRT_EXT(hd->h_filesize_2)));
1029 		nsz = (int)(RSHRT_EXT(hd->h_namesize));
1030 	} else {
1031 		arcn->sb.st_dev = (dev_t)(SHRT_EXT(hd->h_dev));
1032 		arcn->sb.st_ino = (ino_t)(SHRT_EXT(hd->h_ino));
1033 		arcn->sb.st_mode = (mode_t)(SHRT_EXT(hd->h_mode));
1034 		arcn->sb.st_uid = (uid_t)(SHRT_EXT(hd->h_uid));
1035 		arcn->sb.st_gid = (gid_t)(SHRT_EXT(hd->h_gid));
1036 		arcn->sb.st_nlink = (nlink_t)(SHRT_EXT(hd->h_nlink));
1037 		arcn->sb.st_rdev = (dev_t)(SHRT_EXT(hd->h_rdev));
1038 		arcn->sb.st_mtime = (time_t)(SHRT_EXT(hd->h_mtime_1));
1039 		arcn->sb.st_mtime =  (arcn->sb.st_mtime << 16) |
1040 			((time_t)(SHRT_EXT(hd->h_mtime_2)));
1041 		arcn->sb.st_size = (off_t)(SHRT_EXT(hd->h_filesize_1));
1042 		arcn->sb.st_size = (arcn->sb.st_size << 16) |
1043 			((off_t)(SHRT_EXT(hd->h_filesize_2)));
1044 		nsz = (int)(SHRT_EXT(hd->h_namesize));
1045 	}
1046 	arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime;
1047 
1048 	/*
1049 	 * check the file name size, if bogus give up. otherwise read the file
1050 	 * name
1051 	 */
1052 	if (nsz < 2)
1053 		return(-1);
1054 	arcn->nlen = nsz - 1;
1055 	if (rd_nm(arcn, nsz) < 0)
1056 		return(-1);
1057 
1058 	/*
1059 	 * header + file name are aligned to 2 byte boundries, skip if needed
1060 	 */
1061 	if (rd_skip((off_t)(BCPIO_PAD(sizeof(HD_BCPIO) + nsz))) < 0)
1062 		return(-1);
1063 
1064 	/*
1065 	 * if not a link (or a file with no data), calculate pad size (for
1066 	 * padding which follows the file data), clear the link name and return
1067 	 */
1068 	if (((arcn->sb.st_mode & C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)){
1069 		/*
1070 		 * we have a valid header (not a link)
1071 		 */
1072 		arcn->ln_nlen = 0;
1073 		arcn->ln_name[0] = '\0';
1074 		arcn->pad = BCPIO_PAD(arcn->sb.st_size);
1075 		return(com_rd(arcn));
1076 	}
1077 
1078 	if ((rd_ln_nm(arcn) < 0) ||
1079 	    (rd_skip((off_t)(BCPIO_PAD(arcn->sb.st_size))) < 0))
1080 		return(-1);
1081 
1082 	/*
1083 	 * we have a valid header (with a link)
1084 	 */
1085 	return(com_rd(arcn));
1086 }
1087 
1088 /*
1089  * bcpio_endrd()
1090  *      no cleanup needed here, just return size of the trailer (for append)
1091  * Return:
1092  *      size of trailer header in this format
1093  */
1094 
1095 #if __STDC__
1096 off_t
1097 bcpio_endrd(void)
1098 #else
1099 off_t
1100 bcpio_endrd()
1101 #endif
1102 {
1103 	return((off_t)(sizeof(HD_BCPIO) + sizeof(TRAILER) +
1104 		(BCPIO_PAD(sizeof(HD_BCPIO) + sizeof(TRAILER)))));
1105 }
1106 
1107 /*
1108  * bcpio_wr()
1109  *	copy the data in the ARCHD to buffer in old binary cpio format
1110  *	There is a real chance of field overflow with this critter. So we
1111  *	always check the conversion is ok. nobody in his their right mind
1112  *	should write an achive in this format...
1113  * Return
1114  *      0 if file has data to be written after the header, 1 if file has NO
1115  *	data to write after the header, -1 if archive write failed
1116  */
1117 
1118 #if __STDC__
1119 int
1120 bcpio_wr(register ARCHD *arcn)
1121 #else
1122 int
1123 bcpio_wr(arcn)
1124 	register ARCHD *arcn;
1125 #endif
1126 {
1127 	register HD_BCPIO *hd;
1128 	register int nsz;
1129 	char hdblk[sizeof(HD_BCPIO)];
1130 	off_t t_offt;
1131 	int t_int;
1132 	time_t t_timet;
1133 
1134 	/*
1135 	 * check and repair truncated device and inode fields in the cpio
1136 	 * header
1137 	 */
1138 	if (map_dev(arcn, (u_long)BCPIO_MASK, (u_long)BCPIO_MASK) < 0)
1139 		return(-1);
1140 
1141 	if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR))
1142 		arcn->sb.st_rdev = 0;
1143 	hd = (HD_BCPIO *)hdblk;
1144 
1145 	switch(arcn->type) {
1146 	case PAX_CTG:
1147 	case PAX_REG:
1148 	case PAX_HRG:
1149 		/*
1150 		 * caller will copy file data to the archive. tell him how
1151 		 * much to pad.
1152 		 */
1153 		arcn->pad = BCPIO_PAD(arcn->sb.st_size);
1154 		hd->h_filesize_1[0] = CHR_WR_0(arcn->sb.st_size);
1155 		hd->h_filesize_1[1] = CHR_WR_1(arcn->sb.st_size);
1156 		hd->h_filesize_2[0] = CHR_WR_2(arcn->sb.st_size);
1157 		hd->h_filesize_2[1] = CHR_WR_3(arcn->sb.st_size);
1158 		t_offt = (off_t)(SHRT_EXT(hd->h_filesize_1));
1159 		t_offt = (t_offt<<16) | ((off_t)(SHRT_EXT(hd->h_filesize_2)));
1160 		if (arcn->sb.st_size != t_offt) {
1161 			warn(1,"File is too large for bcpio format %s",
1162 			    arcn->org_name);
1163 			return(1);
1164 		}
1165 		break;
1166 	case PAX_SLK:
1167 		/*
1168 		 * no file data for the caller to process, the file data has
1169 		 * the size of the link
1170 		 */
1171 		arcn->pad = 0L;
1172 		hd->h_filesize_1[0] = CHR_WR_0(arcn->ln_nlen);
1173 		hd->h_filesize_1[1] = CHR_WR_1(arcn->ln_nlen);
1174 		hd->h_filesize_2[0] = CHR_WR_2(arcn->ln_nlen);
1175 		hd->h_filesize_2[1] = CHR_WR_3(arcn->ln_nlen);
1176 		t_int = (int)(SHRT_EXT(hd->h_filesize_1));
1177 		t_int = (t_int << 16) | ((int)(SHRT_EXT(hd->h_filesize_2)));
1178 		if (arcn->ln_nlen != t_int)
1179 			goto out;
1180 		break;
1181 	default:
1182 		/*
1183 		 * no file data for the caller to process
1184 		 */
1185 		arcn->pad = 0L;
1186 		hd->h_filesize_1[0] = (char)0;
1187 		hd->h_filesize_1[1] = (char)0;
1188 		hd->h_filesize_2[0] = (char)0;
1189 		hd->h_filesize_2[1] = (char)0;
1190 		break;
1191 	}
1192 
1193 	/*
1194 	 * build up the rest of the fields
1195 	 */
1196 	hd->h_magic[0] = CHR_WR_2(MAGIC);
1197 	hd->h_magic[1] = CHR_WR_3(MAGIC);
1198 	hd->h_dev[0] = CHR_WR_2(arcn->sb.st_dev);
1199 	hd->h_dev[1] = CHR_WR_3(arcn->sb.st_dev);
1200 	if (arcn->sb.st_dev != (dev_t)(SHRT_EXT(hd->h_dev)))
1201 		goto out;
1202 	hd->h_ino[0] = CHR_WR_2(arcn->sb.st_ino);
1203 	hd->h_ino[1] = CHR_WR_3(arcn->sb.st_ino);
1204 	if (arcn->sb.st_ino != (ino_t)(SHRT_EXT(hd->h_ino)))
1205 		goto out;
1206 	hd->h_mode[0] = CHR_WR_2(arcn->sb.st_mode);
1207 	hd->h_mode[1] = CHR_WR_3(arcn->sb.st_mode);
1208 	if (arcn->sb.st_mode != (mode_t)(SHRT_EXT(hd->h_mode)))
1209 		goto out;
1210 	hd->h_uid[0] = CHR_WR_2(arcn->sb.st_uid);
1211 	hd->h_uid[1] = CHR_WR_3(arcn->sb.st_uid);
1212 	if (arcn->sb.st_uid != (uid_t)(SHRT_EXT(hd->h_uid)))
1213 		goto out;
1214 	hd->h_gid[0] = CHR_WR_2(arcn->sb.st_gid);
1215 	hd->h_gid[1] = CHR_WR_3(arcn->sb.st_gid);
1216 	if (arcn->sb.st_gid != (gid_t)(SHRT_EXT(hd->h_gid)))
1217 		goto out;
1218 	hd->h_nlink[0] = CHR_WR_2(arcn->sb.st_nlink);
1219 	hd->h_nlink[1] = CHR_WR_3(arcn->sb.st_nlink);
1220 	if (arcn->sb.st_nlink != (nlink_t)(SHRT_EXT(hd->h_nlink)))
1221 		goto out;
1222 	hd->h_rdev[0] = CHR_WR_2(arcn->sb.st_rdev);
1223 	hd->h_rdev[1] = CHR_WR_3(arcn->sb.st_rdev);
1224 	if (arcn->sb.st_rdev != (dev_t)(SHRT_EXT(hd->h_rdev)))
1225 		goto out;
1226 	hd->h_mtime_1[0] = CHR_WR_0(arcn->sb.st_mtime);
1227 	hd->h_mtime_1[1] = CHR_WR_1(arcn->sb.st_mtime);
1228 	hd->h_mtime_2[0] = CHR_WR_2(arcn->sb.st_mtime);
1229 	hd->h_mtime_2[1] = CHR_WR_3(arcn->sb.st_mtime);
1230 	t_timet = (time_t)(SHRT_EXT(hd->h_mtime_1));
1231 	t_timet =  (t_timet << 16) | ((time_t)(SHRT_EXT(hd->h_mtime_2)));
1232 	if (arcn->sb.st_mtime != t_timet)
1233 		goto out;
1234 	nsz = arcn->nlen + 1;
1235 	hd->h_namesize[0] = CHR_WR_2(nsz);
1236 	hd->h_namesize[1] = CHR_WR_3(nsz);
1237 	if (nsz != (int)(SHRT_EXT(hd->h_namesize)))
1238 		goto out;
1239 
1240 	/*
1241 	 * write the header, the file name and padding as required.
1242 	 */
1243 	if ((wr_rdbuf(hdblk, (int)sizeof(HD_BCPIO)) < 0) ||
1244 	    (wr_rdbuf(arcn->name, nsz) < 0) ||
1245 	    (wr_skip((off_t)(BCPIO_PAD(sizeof(HD_BCPIO) + nsz))) < 0)) {
1246 		warn(1, "Could not write bcpio header for %s", arcn->org_name);
1247 		return(-1);
1248 	}
1249 
1250 	/*
1251 	 * if we have file data, tell the caller we are done
1252 	 */
1253 	if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) ||
1254 	    (arcn->type == PAX_HRG))
1255 		return(0);
1256 
1257 	/*
1258 	 * if we are not a link, tell the caller we are done, go to next file
1259 	 */
1260 	if (arcn->type != PAX_SLK)
1261 		return(1);
1262 
1263 	/*
1264 	 * write the link name, tell the caller we are done.
1265 	 */
1266 	if ((wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) ||
1267 	    (wr_skip((off_t)(BCPIO_PAD(arcn->ln_nlen))) < 0)) {
1268 		warn(1,"Could not write bcpio link name for %s",arcn->org_name);
1269 		return(-1);
1270 	}
1271 	return(1);
1272 
1273     out:
1274 	/*
1275 	 * header field is out of range
1276 	 */
1277 	warn(1,"Bcpio header field is too small for file %s", arcn->org_name);
1278 	return(1);
1279 }
1280