xref: /freebsd/bin/pax/cpio.c (revision c9a8d1f4ddce0e113129507ab55689b6e922f60d)
14b88c807SRodney W. Grimes /*-
24b88c807SRodney W. Grimes  * Copyright (c) 1992 Keith Muller.
34b88c807SRodney W. Grimes  * Copyright (c) 1992, 1993
44b88c807SRodney W. Grimes  *	The Regents of the University of California.  All rights reserved.
54b88c807SRodney W. Grimes  *
64b88c807SRodney W. Grimes  * This code is derived from software contributed to Berkeley by
74b88c807SRodney W. Grimes  * Keith Muller of the University of California, San Diego.
84b88c807SRodney W. Grimes  *
94b88c807SRodney W. Grimes  * Redistribution and use in source and binary forms, with or without
104b88c807SRodney W. Grimes  * modification, are permitted provided that the following conditions
114b88c807SRodney W. Grimes  * are met:
124b88c807SRodney W. Grimes  * 1. Redistributions of source code must retain the above copyright
134b88c807SRodney W. Grimes  *    notice, this list of conditions and the following disclaimer.
144b88c807SRodney W. Grimes  * 2. Redistributions in binary form must reproduce the above copyright
154b88c807SRodney W. Grimes  *    notice, this list of conditions and the following disclaimer in the
164b88c807SRodney W. Grimes  *    documentation and/or other materials provided with the distribution.
174b88c807SRodney W. Grimes  * 3. All advertising materials mentioning features or use of this software
184b88c807SRodney W. Grimes  *    must display the following acknowledgement:
194b88c807SRodney W. Grimes  *	This product includes software developed by the University of
204b88c807SRodney W. Grimes  *	California, Berkeley and its contributors.
214b88c807SRodney W. Grimes  * 4. Neither the name of the University nor the names of its contributors
224b88c807SRodney W. Grimes  *    may be used to endorse or promote products derived from this software
234b88c807SRodney W. Grimes  *    without specific prior written permission.
244b88c807SRodney W. Grimes  *
254b88c807SRodney W. Grimes  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
264b88c807SRodney W. Grimes  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
274b88c807SRodney W. Grimes  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
284b88c807SRodney W. Grimes  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
294b88c807SRodney W. Grimes  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
304b88c807SRodney W. Grimes  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
314b88c807SRodney W. Grimes  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
324b88c807SRodney W. Grimes  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
334b88c807SRodney W. Grimes  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
344b88c807SRodney W. Grimes  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
354b88c807SRodney W. Grimes  * SUCH DAMAGE.
364b88c807SRodney W. Grimes  */
374b88c807SRodney W. Grimes 
384b88c807SRodney W. Grimes #ifndef lint
39c9a8d1f4SPhilippe Charnier #if 0
40c9a8d1f4SPhilippe Charnier static char sccsid[] = "@(#)cpio.c	8.1 (Berkeley) 5/31/93";
41c9a8d1f4SPhilippe Charnier #endif
42c9a8d1f4SPhilippe Charnier static const char rcsid[] =
43c9a8d1f4SPhilippe Charnier 	"$Id$";
444b88c807SRodney W. Grimes #endif /* not lint */
454b88c807SRodney W. Grimes 
464b88c807SRodney W. Grimes #include <sys/types.h>
474b88c807SRodney W. Grimes #include <sys/time.h>
484b88c807SRodney W. Grimes #include <sys/stat.h>
494b88c807SRodney W. Grimes #include <string.h>
504b88c807SRodney W. Grimes #include <stdio.h>
514b88c807SRodney W. Grimes #include <unistd.h>
524b88c807SRodney W. Grimes #include <stdlib.h>
534b88c807SRodney W. Grimes #include "pax.h"
544b88c807SRodney W. Grimes #include "cpio.h"
554b88c807SRodney W. Grimes #include "extern.h"
564b88c807SRodney W. Grimes 
574b88c807SRodney W. Grimes static int rd_nm __P((register ARCHD *, int));
584b88c807SRodney W. Grimes static int rd_ln_nm __P((register ARCHD *));
594b88c807SRodney W. Grimes static int com_rd __P((register ARCHD *));
604b88c807SRodney W. Grimes 
614b88c807SRodney W. Grimes /*
624b88c807SRodney W. Grimes  * Routines which support the different cpio versions
634b88c807SRodney W. Grimes  */
644b88c807SRodney W. Grimes 
654b88c807SRodney W. Grimes static int swp_head;		/* binary cpio header byte swap */
664b88c807SRodney W. Grimes 
674b88c807SRodney W. Grimes /*
684b88c807SRodney W. Grimes  * Routines common to all versions of cpio
694b88c807SRodney W. Grimes  */
704b88c807SRodney W. Grimes 
714b88c807SRodney W. Grimes /*
724b88c807SRodney W. Grimes  * cpio_strd()
734b88c807SRodney W. Grimes  *	Fire up the hard link detection code
744b88c807SRodney W. Grimes  * Return:
754b88c807SRodney W. Grimes  *      0 if ok -1 otherwise (the return values of lnk_start())
764b88c807SRodney W. Grimes  */
774b88c807SRodney W. Grimes 
784b88c807SRodney W. Grimes #if __STDC__
794b88c807SRodney W. Grimes int
804b88c807SRodney W. Grimes cpio_strd(void)
814b88c807SRodney W. Grimes #else
824b88c807SRodney W. Grimes int
834b88c807SRodney W. Grimes cpio_strd()
844b88c807SRodney W. Grimes #endif
854b88c807SRodney W. Grimes {
864b88c807SRodney W. Grimes 	return(lnk_start());
874b88c807SRodney W. Grimes }
884b88c807SRodney W. Grimes 
894b88c807SRodney W. Grimes /*
904b88c807SRodney W. Grimes  * cpio_trail()
914b88c807SRodney W. Grimes  *	Called to determine if a header block is a valid trailer. We are
924b88c807SRodney W. Grimes  *	passed the block, the in_sync flag (which tells us we are in resync
934b88c807SRodney W. Grimes  *	mode; looking for a valid header), and cnt (which starts at zero)
944b88c807SRodney W. Grimes  *	which is used to count the number of empty blocks we have seen so far.
954b88c807SRodney W. Grimes  * Return:
964b88c807SRodney W. Grimes  *	0 if a valid trailer, -1 if not a valid trailer,
974b88c807SRodney W. Grimes  */
984b88c807SRodney W. Grimes 
994b88c807SRodney W. Grimes #if __STDC__
1004b88c807SRodney W. Grimes int
1014b88c807SRodney W. Grimes cpio_trail(register ARCHD *arcn)
1024b88c807SRodney W. Grimes #else
1034b88c807SRodney W. Grimes int
1044b88c807SRodney W. Grimes cpio_trail(arcn)
1054b88c807SRodney W. Grimes 	register ARCHD *arcn;
1064b88c807SRodney W. Grimes #endif
1074b88c807SRodney W. Grimes {
1084b88c807SRodney W. Grimes 	/*
1094b88c807SRodney W. Grimes 	 * look for trailer id in file we are about to process
1104b88c807SRodney W. Grimes 	 */
1114b88c807SRodney W. Grimes 	if ((strcmp(arcn->name, TRAILER) == 0) && (arcn->sb.st_size == 0))
1124b88c807SRodney W. Grimes 		return(0);
1134b88c807SRodney W. Grimes 	return(-1);
1144b88c807SRodney W. Grimes }
1154b88c807SRodney W. Grimes 
1164b88c807SRodney W. Grimes /*
1174b88c807SRodney W. Grimes  * com_rd()
1184b88c807SRodney W. Grimes  *	operations common to all cpio read functions.
1194b88c807SRodney W. Grimes  * Return:
1204b88c807SRodney W. Grimes  *	0
1214b88c807SRodney W. Grimes  */
1224b88c807SRodney W. Grimes 
1234b88c807SRodney W. Grimes #if __STDC__
1244b88c807SRodney W. Grimes static int
1254b88c807SRodney W. Grimes com_rd(register ARCHD *arcn)
1264b88c807SRodney W. Grimes #else
1274b88c807SRodney W. Grimes static int
1284b88c807SRodney W. Grimes com_rd(arcn)
1294b88c807SRodney W. Grimes 	register ARCHD *arcn;
1304b88c807SRodney W. Grimes #endif
1314b88c807SRodney W. Grimes {
1324b88c807SRodney W. Grimes 	arcn->skip = 0;
1334b88c807SRodney W. Grimes 	arcn->pat = NULL;
1344b88c807SRodney W. Grimes 	arcn->org_name = arcn->name;
1354b88c807SRodney W. Grimes 	switch(arcn->sb.st_mode & C_IFMT) {
1364b88c807SRodney W. Grimes 	case C_ISFIFO:
1374b88c807SRodney W. Grimes 		arcn->type = PAX_FIF;
1384b88c807SRodney W. Grimes 		break;
1394b88c807SRodney W. Grimes 	case C_ISDIR:
1404b88c807SRodney W. Grimes 		arcn->type = PAX_DIR;
1414b88c807SRodney W. Grimes 		break;
1424b88c807SRodney W. Grimes 	case C_ISBLK:
1434b88c807SRodney W. Grimes 		arcn->type = PAX_BLK;
1444b88c807SRodney W. Grimes 		break;
1454b88c807SRodney W. Grimes 	case C_ISCHR:
1464b88c807SRodney W. Grimes 		arcn->type = PAX_CHR;
1474b88c807SRodney W. Grimes 		break;
1484b88c807SRodney W. Grimes 	case C_ISLNK:
1494b88c807SRodney W. Grimes 		arcn->type = PAX_SLK;
1504b88c807SRodney W. Grimes 		break;
1514b88c807SRodney W. Grimes 	case C_ISOCK:
1524b88c807SRodney W. Grimes 		arcn->type = PAX_SCK;
1534b88c807SRodney W. Grimes 		break;
1544b88c807SRodney W. Grimes 	case C_ISCTG:
1554b88c807SRodney W. Grimes 	case C_ISREG:
1564b88c807SRodney W. Grimes 	default:
1574b88c807SRodney W. Grimes 		/*
1584b88c807SRodney W. Grimes 		 * we have file data, set up skip (pad is set in the format
1594b88c807SRodney W. Grimes 		 * specific sections)
1604b88c807SRodney W. Grimes 		 */
1614b88c807SRodney W. Grimes 		arcn->sb.st_mode = (arcn->sb.st_mode & 0xfff) | C_ISREG;
1624b88c807SRodney W. Grimes 		arcn->type = PAX_REG;
1634b88c807SRodney W. Grimes 		arcn->skip = arcn->sb.st_size;
1644b88c807SRodney W. Grimes 		break;
1654b88c807SRodney W. Grimes 	}
1664b88c807SRodney W. Grimes 	if (chk_lnk(arcn) < 0)
1674b88c807SRodney W. Grimes 		return(-1);
1684b88c807SRodney W. Grimes 	return(0);
1694b88c807SRodney W. Grimes }
1704b88c807SRodney W. Grimes 
1714b88c807SRodney W. Grimes /*
1724b88c807SRodney W. Grimes  * cpio_end_wr()
1734b88c807SRodney W. Grimes  *	write the special file with the name trailer in the proper format
1744b88c807SRodney W. Grimes  * Return:
1754b88c807SRodney W. Grimes  *	result of the write of the trailer from the cpio specific write func
1764b88c807SRodney W. Grimes  */
1774b88c807SRodney W. Grimes 
1784b88c807SRodney W. Grimes #if __STDC__
1794b88c807SRodney W. Grimes int
1804b88c807SRodney W. Grimes cpio_endwr(void)
1814b88c807SRodney W. Grimes #else
1824b88c807SRodney W. Grimes int
1834b88c807SRodney W. Grimes cpio_endwr()
1844b88c807SRodney W. Grimes #endif
1854b88c807SRodney W. Grimes {
1864b88c807SRodney W. Grimes 	ARCHD last;
1874b88c807SRodney W. Grimes 
1884b88c807SRodney W. Grimes 	/*
1894b88c807SRodney W. Grimes 	 * create a trailer request and call the proper format write function
1904b88c807SRodney W. Grimes 	 */
1914b88c807SRodney W. Grimes 	bzero((char *)&last, sizeof(last));
1924b88c807SRodney W. Grimes 	last.nlen = sizeof(TRAILER) - 1;
1934b88c807SRodney W. Grimes 	last.type = PAX_REG;
1944b88c807SRodney W. Grimes 	last.sb.st_nlink = 1;
1954b88c807SRodney W. Grimes 	(void)strcpy(last.name, TRAILER);
1964b88c807SRodney W. Grimes 	return((*frmt->wr)(&last));
1974b88c807SRodney W. Grimes }
1984b88c807SRodney W. Grimes 
1994b88c807SRodney W. Grimes /*
2004b88c807SRodney W. Grimes  * rd_nam()
2014b88c807SRodney W. Grimes  *	read in the file name which follows the cpio header
2024b88c807SRodney W. Grimes  * Return:
2034b88c807SRodney W. Grimes  *	0 if ok, -1 otherwise
2044b88c807SRodney W. Grimes  */
2054b88c807SRodney W. Grimes 
2064b88c807SRodney W. Grimes #if __STDC__
2074b88c807SRodney W. Grimes static int
2084b88c807SRodney W. Grimes rd_nm(register ARCHD *arcn, int nsz)
2094b88c807SRodney W. Grimes #else
2104b88c807SRodney W. Grimes static int
2114b88c807SRodney W. Grimes rd_nm(arcn, nsz)
2124b88c807SRodney W. Grimes 	register ARCHD *arcn;
2134b88c807SRodney W. Grimes 	int nsz;
2144b88c807SRodney W. Grimes #endif
2154b88c807SRodney W. Grimes {
2164b88c807SRodney W. Grimes 	/*
2174b88c807SRodney W. Grimes 	 * do not even try bogus values
2184b88c807SRodney W. Grimes 	 */
2194b88c807SRodney W. Grimes 	if ((nsz == 0) || (nsz > sizeof(arcn->name))) {
220a885d9dcSSøren Schmidt 		pax_warn(1, "Cpio file name length %d is out of range", nsz);
2214b88c807SRodney W. Grimes 		return(-1);
2224b88c807SRodney W. Grimes 	}
2234b88c807SRodney W. Grimes 
2244b88c807SRodney W. Grimes 	/*
2254b88c807SRodney W. Grimes 	 * read the name and make sure it is not empty and is \0 terminated
2264b88c807SRodney W. Grimes 	 */
2274b88c807SRodney W. Grimes 	if ((rd_wrbuf(arcn->name,nsz) != nsz) || (arcn->name[nsz-1] != '\0') ||
2284b88c807SRodney W. Grimes 	    (arcn->name[0] == '\0')) {
229a885d9dcSSøren Schmidt 		pax_warn(1, "Cpio file name in header is corrupted");
2304b88c807SRodney W. Grimes 		return(-1);
2314b88c807SRodney W. Grimes 	}
2324b88c807SRodney W. Grimes 	return(0);
2334b88c807SRodney W. Grimes }
2344b88c807SRodney W. Grimes 
2354b88c807SRodney W. Grimes /*
2364b88c807SRodney W. Grimes  * rd_ln_nm()
2374b88c807SRodney W. Grimes  *	read in the link name for a file with links. The link name is stored
2384b88c807SRodney W. Grimes  *	like file data (and is NOT \0 terminated!)
2394b88c807SRodney W. Grimes  * Return:
2404b88c807SRodney W. Grimes  *	0 if ok, -1 otherwise
2414b88c807SRodney W. Grimes  */
2424b88c807SRodney W. Grimes 
2434b88c807SRodney W. Grimes #if __STDC__
2444b88c807SRodney W. Grimes static int
2454b88c807SRodney W. Grimes rd_ln_nm(register ARCHD *arcn)
2464b88c807SRodney W. Grimes #else
2474b88c807SRodney W. Grimes static int
2484b88c807SRodney W. Grimes rd_ln_nm(arcn)
2494b88c807SRodney W. Grimes 	register ARCHD *arcn;
2504b88c807SRodney W. Grimes #endif
2514b88c807SRodney W. Grimes {
2524b88c807SRodney W. Grimes 	/*
2534b88c807SRodney W. Grimes 	 * check the length specified for bogus values
2544b88c807SRodney W. Grimes 	 */
2554b88c807SRodney W. Grimes 	if ((arcn->sb.st_size == 0) ||
2564b88c807SRodney W. Grimes 	    (arcn->sb.st_size >= sizeof(arcn->ln_name))) {
2574b88c807SRodney W. Grimes #		ifdef NET2_STAT
258a885d9dcSSøren Schmidt 		pax_warn(1, "Cpio link name length is invalid: %lu",
2594b88c807SRodney W. Grimes 		    arcn->sb.st_size);
2604b88c807SRodney W. Grimes #		else
261a885d9dcSSøren Schmidt 		pax_warn(1, "Cpio link name length is invalid: %qu",
2624b88c807SRodney W. Grimes 		    arcn->sb.st_size);
2634b88c807SRodney W. Grimes #		endif
2644b88c807SRodney W. Grimes 		return(-1);
2654b88c807SRodney W. Grimes 	}
2664b88c807SRodney W. Grimes 
2674b88c807SRodney W. Grimes 	/*
2684b88c807SRodney W. Grimes 	 * read in the link name and \0 terminate it
2694b88c807SRodney W. Grimes 	 */
2704b88c807SRodney W. Grimes 	if (rd_wrbuf(arcn->ln_name, (int)arcn->sb.st_size) !=
2714b88c807SRodney W. Grimes 	    (int)arcn->sb.st_size) {
272a885d9dcSSøren Schmidt 		pax_warn(1, "Cpio link name read error");
2734b88c807SRodney W. Grimes 		return(-1);
2744b88c807SRodney W. Grimes 	}
2754b88c807SRodney W. Grimes 	arcn->ln_nlen = arcn->sb.st_size;
2764b88c807SRodney W. Grimes 	arcn->ln_name[arcn->ln_nlen] = '\0';
2774b88c807SRodney W. Grimes 
2784b88c807SRodney W. Grimes 	/*
2794b88c807SRodney W. Grimes 	 * watch out for those empty link names
2804b88c807SRodney W. Grimes 	 */
2814b88c807SRodney W. Grimes 	if (arcn->ln_name[0] == '\0') {
282a885d9dcSSøren Schmidt 		pax_warn(1, "Cpio link name is corrupt");
2834b88c807SRodney W. Grimes 		return(-1);
2844b88c807SRodney W. Grimes 	}
2854b88c807SRodney W. Grimes 	return(0);
2864b88c807SRodney W. Grimes }
2874b88c807SRodney W. Grimes 
2884b88c807SRodney W. Grimes /*
2894b88c807SRodney W. Grimes  * Routines common to the extended byte oriented cpio format
2904b88c807SRodney W. Grimes  */
2914b88c807SRodney W. Grimes 
2924b88c807SRodney W. Grimes /*
2934b88c807SRodney W. Grimes  * cpio_id()
2944b88c807SRodney W. Grimes  *      determine if a block given to us is a valid extended byte oriented
2954b88c807SRodney W. Grimes  *	cpio header
2964b88c807SRodney W. Grimes  * Return:
2974b88c807SRodney W. Grimes  *      0 if a valid header, -1 otherwise
2984b88c807SRodney W. Grimes  */
2994b88c807SRodney W. Grimes 
3004b88c807SRodney W. Grimes #if __STDC__
3014b88c807SRodney W. Grimes int
3024b88c807SRodney W. Grimes cpio_id(char *blk, int size)
3034b88c807SRodney W. Grimes #else
3044b88c807SRodney W. Grimes int
3054b88c807SRodney W. Grimes cpio_id(blk, size)
3064b88c807SRodney W. Grimes 	char *blk;
3074b88c807SRodney W. Grimes 	int size;
3084b88c807SRodney W. Grimes #endif
3094b88c807SRodney W. Grimes {
3104b88c807SRodney W. Grimes 	if ((size < sizeof(HD_CPIO)) ||
3114b88c807SRodney W. Grimes 	    (strncmp(blk, AMAGIC, sizeof(AMAGIC) - 1) != 0))
3124b88c807SRodney W. Grimes 		return(-1);
3134b88c807SRodney W. Grimes 	return(0);
3144b88c807SRodney W. Grimes }
3154b88c807SRodney W. Grimes 
3164b88c807SRodney W. Grimes /*
3174b88c807SRodney W. Grimes  * cpio_rd()
3184b88c807SRodney W. Grimes  *	determine if a buffer is a byte oriented extended cpio archive entry.
3194b88c807SRodney W. Grimes  *	convert and store the values in the ARCHD parameter.
3204b88c807SRodney W. Grimes  * Return:
3214b88c807SRodney W. Grimes  *	0 if a valid header, -1 otherwise.
3224b88c807SRodney W. Grimes  */
3234b88c807SRodney W. Grimes 
3244b88c807SRodney W. Grimes #if __STDC__
3254b88c807SRodney W. Grimes int
3264b88c807SRodney W. Grimes cpio_rd(register ARCHD *arcn, register char *buf)
3274b88c807SRodney W. Grimes #else
3284b88c807SRodney W. Grimes int
3294b88c807SRodney W. Grimes cpio_rd(arcn, buf)
3304b88c807SRodney W. Grimes 	register ARCHD *arcn;
3314b88c807SRodney W. Grimes 	register char *buf;
3324b88c807SRodney W. Grimes #endif
3334b88c807SRodney W. Grimes {
3344b88c807SRodney W. Grimes 	register int nsz;
3354b88c807SRodney W. Grimes 	register HD_CPIO *hd;
3364b88c807SRodney W. Grimes 
3374b88c807SRodney W. Grimes 	/*
3384b88c807SRodney W. Grimes 	 * check that this is a valid header, if not return -1
3394b88c807SRodney W. Grimes 	 */
3404b88c807SRodney W. Grimes 	if (cpio_id(buf, sizeof(HD_CPIO)) < 0)
3414b88c807SRodney W. Grimes 		return(-1);
3424b88c807SRodney W. Grimes 	hd = (HD_CPIO *)buf;
3434b88c807SRodney W. Grimes 
3444b88c807SRodney W. Grimes 	/*
3454b88c807SRodney W. Grimes 	 * byte oriented cpio (posix) does not have padding! extract the octal
3464b88c807SRodney W. Grimes 	 * ascii fields from the header
3474b88c807SRodney W. Grimes 	 */
3484b88c807SRodney W. Grimes 	arcn->pad = 0L;
3494b88c807SRodney W. Grimes 	arcn->sb.st_dev = (dev_t)asc_ul(hd->c_dev, sizeof(hd->c_dev), OCT);
3504b88c807SRodney W. Grimes 	arcn->sb.st_ino = (ino_t)asc_ul(hd->c_ino, sizeof(hd->c_ino), OCT);
3514b88c807SRodney W. Grimes 	arcn->sb.st_mode = (mode_t)asc_ul(hd->c_mode, sizeof(hd->c_mode), OCT);
3524b88c807SRodney W. Grimes 	arcn->sb.st_uid = (uid_t)asc_ul(hd->c_uid, sizeof(hd->c_uid), OCT);
3534b88c807SRodney W. Grimes 	arcn->sb.st_gid = (gid_t)asc_ul(hd->c_gid, sizeof(hd->c_gid), OCT);
3544b88c807SRodney W. Grimes 	arcn->sb.st_nlink = (nlink_t)asc_ul(hd->c_nlink, sizeof(hd->c_nlink),
3554b88c807SRodney W. Grimes 	    OCT);
3564b88c807SRodney W. Grimes 	arcn->sb.st_rdev = (dev_t)asc_ul(hd->c_rdev, sizeof(hd->c_rdev), OCT);
3574b88c807SRodney W. Grimes 	arcn->sb.st_mtime = (time_t)asc_ul(hd->c_mtime, sizeof(hd->c_mtime),
3584b88c807SRodney W. Grimes 	    OCT);
3594b88c807SRodney W. Grimes 	arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime;
3604b88c807SRodney W. Grimes #	ifdef NET2_STAT
3614b88c807SRodney W. Grimes 	arcn->sb.st_size = (off_t)asc_ul(hd->c_filesize,sizeof(hd->c_filesize),
3624b88c807SRodney W. Grimes 	    OCT);
3634b88c807SRodney W. Grimes #	else
3644b88c807SRodney W. Grimes 	arcn->sb.st_size = (off_t)asc_uqd(hd->c_filesize,sizeof(hd->c_filesize),
3654b88c807SRodney W. Grimes 	    OCT);
3664b88c807SRodney W. Grimes #	endif
3674b88c807SRodney W. Grimes 
3684b88c807SRodney W. Grimes 	/*
3694b88c807SRodney W. Grimes 	 * check name size and if valid, read in the name of this entry (name
3704b88c807SRodney W. Grimes 	 * follows header in the archive)
3714b88c807SRodney W. Grimes 	 */
3724b88c807SRodney W. Grimes 	if ((nsz = (int)asc_ul(hd->c_namesize,sizeof(hd->c_namesize),OCT)) < 2)
3734b88c807SRodney W. Grimes 		return(-1);
3744b88c807SRodney W. Grimes 	arcn->nlen = nsz - 1;
3754b88c807SRodney W. Grimes 	if (rd_nm(arcn, nsz) < 0)
3764b88c807SRodney W. Grimes 		return(-1);
3774b88c807SRodney W. Grimes 
3784b88c807SRodney W. Grimes 	if (((arcn->sb.st_mode&C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)) {
3794b88c807SRodney W. Grimes 		/*
3804b88c807SRodney W. Grimes 	 	 * no link name to read for this file
3814b88c807SRodney W. Grimes 	 	 */
3824b88c807SRodney W. Grimes 		arcn->ln_nlen = 0;
3834b88c807SRodney W. Grimes 		arcn->ln_name[0] = '\0';
3844b88c807SRodney W. Grimes 		return(com_rd(arcn));
3854b88c807SRodney W. Grimes 	}
3864b88c807SRodney W. Grimes 
3874b88c807SRodney W. Grimes 	/*
3884b88c807SRodney W. Grimes 	 * check link name size and read in the link name. Link names are
3894b88c807SRodney W. Grimes 	 * stored like file data.
3904b88c807SRodney W. Grimes 	 */
3914b88c807SRodney W. Grimes 	if (rd_ln_nm(arcn) < 0)
3924b88c807SRodney W. Grimes 		return(-1);
3934b88c807SRodney W. Grimes 
3944b88c807SRodney W. Grimes 	/*
3954b88c807SRodney W. Grimes 	 * we have a valid header (with a link)
3964b88c807SRodney W. Grimes 	 */
3974b88c807SRodney W. Grimes 	return(com_rd(arcn));
3984b88c807SRodney W. Grimes }
3994b88c807SRodney W. Grimes 
4004b88c807SRodney W. Grimes /*
4014b88c807SRodney W. Grimes  * cpio_endrd()
4024b88c807SRodney W. Grimes  *      no cleanup needed here, just return size of the trailer (for append)
4034b88c807SRodney W. Grimes  * Return:
4044b88c807SRodney W. Grimes  *      size of trailer header in this format
4054b88c807SRodney W. Grimes  */
4064b88c807SRodney W. Grimes 
4074b88c807SRodney W. Grimes #if __STDC__
4084b88c807SRodney W. Grimes off_t
4094b88c807SRodney W. Grimes cpio_endrd(void)
4104b88c807SRodney W. Grimes #else
4114b88c807SRodney W. Grimes off_t
4124b88c807SRodney W. Grimes cpio_endrd()
4134b88c807SRodney W. Grimes #endif
4144b88c807SRodney W. Grimes {
4154b88c807SRodney W. Grimes 	return((off_t)(sizeof(HD_CPIO) + sizeof(TRAILER)));
4164b88c807SRodney W. Grimes }
4174b88c807SRodney W. Grimes 
4184b88c807SRodney W. Grimes /*
4194b88c807SRodney W. Grimes  * cpio_stwr()
4204b88c807SRodney W. Grimes  *	start up the device mapping table
4214b88c807SRodney W. Grimes  * Return:
4224b88c807SRodney W. Grimes  *	0 if ok, -1 otherwise (what dev_start() returns)
4234b88c807SRodney W. Grimes  */
4244b88c807SRodney W. Grimes 
4254b88c807SRodney W. Grimes #if __STDC__
4264b88c807SRodney W. Grimes int
4274b88c807SRodney W. Grimes cpio_stwr(void)
4284b88c807SRodney W. Grimes #else
4294b88c807SRodney W. Grimes int
4304b88c807SRodney W. Grimes cpio_stwr()
4314b88c807SRodney W. Grimes #endif
4324b88c807SRodney W. Grimes {
4334b88c807SRodney W. Grimes 	return(dev_start());
4344b88c807SRodney W. Grimes }
4354b88c807SRodney W. Grimes 
4364b88c807SRodney W. Grimes /*
4374b88c807SRodney W. Grimes  * cpio_wr()
4384b88c807SRodney W. Grimes  *	copy the data in the ARCHD to buffer in extended byte oriented cpio
4394b88c807SRodney W. Grimes  *	format.
4404b88c807SRodney W. Grimes  * Return
4414b88c807SRodney W. Grimes  *      0 if file has data to be written after the header, 1 if file has NO
4424b88c807SRodney W. Grimes  *	data to write after the header, -1 if archive write failed
4434b88c807SRodney W. Grimes  */
4444b88c807SRodney W. Grimes 
4454b88c807SRodney W. Grimes #if __STDC__
4464b88c807SRodney W. Grimes int
4474b88c807SRodney W. Grimes cpio_wr(register ARCHD *arcn)
4484b88c807SRodney W. Grimes #else
4494b88c807SRodney W. Grimes int
4504b88c807SRodney W. Grimes cpio_wr(arcn)
4514b88c807SRodney W. Grimes 	register ARCHD *arcn;
4524b88c807SRodney W. Grimes #endif
4534b88c807SRodney W. Grimes {
4544b88c807SRodney W. Grimes 	register HD_CPIO *hd;
4554b88c807SRodney W. Grimes 	register int nsz;
4564b88c807SRodney W. Grimes 	char hdblk[sizeof(HD_CPIO)];
4574b88c807SRodney W. Grimes 
4584b88c807SRodney W. Grimes 	/*
4594b88c807SRodney W. Grimes 	 * check and repair truncated device and inode fields in the header
4604b88c807SRodney W. Grimes 	 */
4614b88c807SRodney W. Grimes 	if (map_dev(arcn, (u_long)CPIO_MASK, (u_long)CPIO_MASK) < 0)
4624b88c807SRodney W. Grimes 		return(-1);
4634b88c807SRodney W. Grimes 
4644b88c807SRodney W. Grimes 	arcn->pad = 0L;
4654b88c807SRodney W. Grimes 	nsz = arcn->nlen + 1;
4664b88c807SRodney W. Grimes 	hd = (HD_CPIO *)hdblk;
4674b88c807SRodney W. Grimes 	if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR))
4684b88c807SRodney W. Grimes 		arcn->sb.st_rdev = 0;
4694b88c807SRodney W. Grimes 
4704b88c807SRodney W. Grimes 	switch(arcn->type) {
4714b88c807SRodney W. Grimes 	case PAX_CTG:
4724b88c807SRodney W. Grimes 	case PAX_REG:
4734b88c807SRodney W. Grimes 	case PAX_HRG:
4744b88c807SRodney W. Grimes 		/*
4754b88c807SRodney W. Grimes 		 * set data size for file data
4764b88c807SRodney W. Grimes 		 */
4774b88c807SRodney W. Grimes #		ifdef NET2_STAT
4784b88c807SRodney W. Grimes 		if (ul_asc((u_long)arcn->sb.st_size, hd->c_filesize,
4794b88c807SRodney W. Grimes 		    sizeof(hd->c_filesize), OCT)) {
4804b88c807SRodney W. Grimes #		else
4814b88c807SRodney W. Grimes 		if (uqd_asc((u_quad_t)arcn->sb.st_size, hd->c_filesize,
4824b88c807SRodney W. Grimes 		    sizeof(hd->c_filesize), OCT)) {
4834b88c807SRodney W. Grimes #		endif
484a885d9dcSSøren Schmidt 			pax_warn(1,"File is too large for cpio format %s",
4854b88c807SRodney W. Grimes 			    arcn->org_name);
4864b88c807SRodney W. Grimes 			return(1);
4874b88c807SRodney W. Grimes 		}
4884b88c807SRodney W. Grimes 		break;
4894b88c807SRodney W. Grimes 	case PAX_SLK:
4904b88c807SRodney W. Grimes 		/*
4914b88c807SRodney W. Grimes 		 * set data size to hold link name
4924b88c807SRodney W. Grimes 		 */
4934b88c807SRodney W. Grimes 		if (ul_asc((u_long)arcn->ln_nlen, hd->c_filesize,
4944b88c807SRodney W. Grimes 		    sizeof(hd->c_filesize), OCT))
4954b88c807SRodney W. Grimes 			goto out;
4964b88c807SRodney W. Grimes 		break;
4974b88c807SRodney W. Grimes 	default:
4984b88c807SRodney W. Grimes 		/*
4994b88c807SRodney W. Grimes 		 * all other file types have no file data
5004b88c807SRodney W. Grimes 		 */
5014b88c807SRodney W. Grimes 		if (ul_asc((u_long)0, hd->c_filesize, sizeof(hd->c_filesize),
5024b88c807SRodney W. Grimes 		     OCT))
5034b88c807SRodney W. Grimes 			goto out;
5044b88c807SRodney W. Grimes 		break;
5054b88c807SRodney W. Grimes 	}
5064b88c807SRodney W. Grimes 
5074b88c807SRodney W. Grimes 	/*
5084b88c807SRodney W. Grimes 	 * copy the values to the header using octal ascii
5094b88c807SRodney W. Grimes 	 */
5104b88c807SRodney W. Grimes 	if (ul_asc((u_long)MAGIC, hd->c_magic, sizeof(hd->c_magic), OCT) ||
5114b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_dev, hd->c_dev, sizeof(hd->c_dev),
5124b88c807SRodney W. Grimes 	        OCT) ||
5134b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_ino, hd->c_ino, sizeof(hd->c_ino),
5144b88c807SRodney W. Grimes 		OCT) ||
5154b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_mode, hd->c_mode, sizeof(hd->c_mode),
5164b88c807SRodney W. Grimes 		OCT) ||
5174b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_uid, hd->c_uid, sizeof(hd->c_uid),
5184b88c807SRodney W. Grimes 		OCT) ||
5194b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_gid, hd->c_gid, sizeof(hd->c_gid),
5204b88c807SRodney W. Grimes 		OCT) ||
5214b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_nlink, hd->c_nlink, sizeof(hd->c_nlink),
5224b88c807SRodney W. Grimes 		 OCT) ||
5234b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_rdev, hd->c_rdev, sizeof(hd->c_rdev),
5244b88c807SRodney W. Grimes 		OCT) ||
5254b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_mtime,hd->c_mtime,sizeof(hd->c_mtime),
5264b88c807SRodney W. Grimes 		OCT) ||
5274b88c807SRodney W. Grimes 	    ul_asc((u_long)nsz, hd->c_namesize, sizeof(hd->c_namesize), OCT))
5284b88c807SRodney W. Grimes 		goto out;
5294b88c807SRodney W. Grimes 
5304b88c807SRodney W. Grimes 	/*
5314b88c807SRodney W. Grimes 	 * write the file name to the archive
5324b88c807SRodney W. Grimes 	 */
5334b88c807SRodney W. Grimes 	if ((wr_rdbuf(hdblk, (int)sizeof(HD_CPIO)) < 0) ||
5344b88c807SRodney W. Grimes 	    (wr_rdbuf(arcn->name, nsz) < 0)) {
535a885d9dcSSøren Schmidt 		pax_warn(1, "Unable to write cpio header for %s", arcn->org_name);
5364b88c807SRodney W. Grimes 		return(-1);
5374b88c807SRodney W. Grimes 	}
5384b88c807SRodney W. Grimes 
5394b88c807SRodney W. Grimes 	/*
5404b88c807SRodney W. Grimes 	 * if this file has data, we are done. The caller will write the file
5414b88c807SRodney W. Grimes 	 * data, if we are link tell caller we are done, go to next file
5424b88c807SRodney W. Grimes 	 */
5434b88c807SRodney W. Grimes 	if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) ||
5444b88c807SRodney W. Grimes 	    (arcn->type == PAX_HRG))
5454b88c807SRodney W. Grimes 		return(0);
5464b88c807SRodney W. Grimes 	if (arcn->type != PAX_SLK)
5474b88c807SRodney W. Grimes 		return(1);
5484b88c807SRodney W. Grimes 
5494b88c807SRodney W. Grimes 	/*
5504b88c807SRodney W. Grimes 	 * write the link name to the archive, tell the caller to go to the
5514b88c807SRodney W. Grimes 	 * next file as we are done.
5524b88c807SRodney W. Grimes 	 */
5534b88c807SRodney W. Grimes 	if (wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) {
554a885d9dcSSøren Schmidt 		pax_warn(1,"Unable to write cpio link name for %s",arcn->org_name);
5554b88c807SRodney W. Grimes 		return(-1);
5564b88c807SRodney W. Grimes 	}
5574b88c807SRodney W. Grimes 	return(1);
5584b88c807SRodney W. Grimes 
5594b88c807SRodney W. Grimes     out:
5604b88c807SRodney W. Grimes 	/*
5614b88c807SRodney W. Grimes 	 * header field is out of range
5624b88c807SRodney W. Grimes 	 */
563a885d9dcSSøren Schmidt 	pax_warn(1, "Cpio header field is too small to store file %s",
5644b88c807SRodney W. Grimes 	    arcn->org_name);
5654b88c807SRodney W. Grimes 	return(1);
5664b88c807SRodney W. Grimes }
5674b88c807SRodney W. Grimes 
5684b88c807SRodney W. Grimes /*
5694b88c807SRodney W. Grimes  * Routines common to the system VR4 version of cpio (with/without file CRC)
5704b88c807SRodney W. Grimes  */
5714b88c807SRodney W. Grimes 
5724b88c807SRodney W. Grimes /*
5734b88c807SRodney W. Grimes  * vcpio_id()
5744b88c807SRodney W. Grimes  *      determine if a block given to us is a valid system VR4 cpio header
5754b88c807SRodney W. Grimes  *	WITHOUT crc. WATCH it the magic cookies are in OCTAL, the header
5764b88c807SRodney W. Grimes  *	uses HEX
5774b88c807SRodney W. Grimes  * Return:
5784b88c807SRodney W. Grimes  *      0 if a valid header, -1 otherwise
5794b88c807SRodney W. Grimes  */
5804b88c807SRodney W. Grimes 
5814b88c807SRodney W. Grimes #if __STDC__
5824b88c807SRodney W. Grimes int
5834b88c807SRodney W. Grimes vcpio_id(char *blk, int size)
5844b88c807SRodney W. Grimes #else
5854b88c807SRodney W. Grimes int
5864b88c807SRodney W. Grimes vcpio_id(blk, size)
5874b88c807SRodney W. Grimes 	char *blk;
5884b88c807SRodney W. Grimes 	int size;
5894b88c807SRodney W. Grimes #endif
5904b88c807SRodney W. Grimes {
5914b88c807SRodney W. Grimes 	if ((size < sizeof(HD_VCPIO)) ||
5924b88c807SRodney W. Grimes 	    (strncmp(blk, AVMAGIC, sizeof(AVMAGIC) - 1) != 0))
5934b88c807SRodney W. Grimes 		return(-1);
5944b88c807SRodney W. Grimes 	return(0);
5954b88c807SRodney W. Grimes }
5964b88c807SRodney W. Grimes 
5974b88c807SRodney W. Grimes /*
5984b88c807SRodney W. Grimes  * crc_id()
5994b88c807SRodney W. Grimes  *      determine if a block given to us is a valid system VR4 cpio header
6004b88c807SRodney W. Grimes  *	WITH crc. WATCH it the magic cookies are in OCTAL the header uses HEX
6014b88c807SRodney W. Grimes  * Return:
6024b88c807SRodney W. Grimes  *      0 if a valid header, -1 otherwise
6034b88c807SRodney W. Grimes  */
6044b88c807SRodney W. Grimes 
6054b88c807SRodney W. Grimes #if __STDC__
6064b88c807SRodney W. Grimes int
6074b88c807SRodney W. Grimes crc_id(char *blk, int size)
6084b88c807SRodney W. Grimes #else
6094b88c807SRodney W. Grimes int
6104b88c807SRodney W. Grimes crc_id(blk, size)
6114b88c807SRodney W. Grimes 	char *blk;
6124b88c807SRodney W. Grimes 	int size;
6134b88c807SRodney W. Grimes #endif
6144b88c807SRodney W. Grimes {
6154b88c807SRodney W. Grimes 	if ((size < sizeof(HD_VCPIO)) ||
6164b88c807SRodney W. Grimes 	    (strncmp(blk, AVCMAGIC, sizeof(AVCMAGIC) - 1) != 0))
6174b88c807SRodney W. Grimes 		return(-1);
6184b88c807SRodney W. Grimes 	return(0);
6194b88c807SRodney W. Grimes }
6204b88c807SRodney W. Grimes 
6214b88c807SRodney W. Grimes /*
6224b88c807SRodney W. Grimes  * crc_strd()
6234b88c807SRodney W. Grimes  w	set file data CRC calculations. Fire up the hard link detection code
6244b88c807SRodney W. Grimes  * Return:
6254b88c807SRodney W. Grimes  *      0 if ok -1 otherwise (the return values of lnk_start())
6264b88c807SRodney W. Grimes  */
6274b88c807SRodney W. Grimes 
6284b88c807SRodney W. Grimes #if __STDC__
6294b88c807SRodney W. Grimes int
6304b88c807SRodney W. Grimes crc_strd(void)
6314b88c807SRodney W. Grimes #else
6324b88c807SRodney W. Grimes int
6334b88c807SRodney W. Grimes crc_strd()
6344b88c807SRodney W. Grimes #endif
6354b88c807SRodney W. Grimes {
6364b88c807SRodney W. Grimes 	docrc = 1;
6374b88c807SRodney W. Grimes 	return(lnk_start());
6384b88c807SRodney W. Grimes }
6394b88c807SRodney W. Grimes 
6404b88c807SRodney W. Grimes /*
6414b88c807SRodney W. Grimes  * vcpio_rd()
6424b88c807SRodney W. Grimes  *	determine if a buffer is a system VR4 archive entry. (with/without CRC)
6434b88c807SRodney W. Grimes  *	convert and store the values in the ARCHD parameter.
6444b88c807SRodney W. Grimes  * Return:
6454b88c807SRodney W. Grimes  *	0 if a valid header, -1 otherwise.
6464b88c807SRodney W. Grimes  */
6474b88c807SRodney W. Grimes 
6484b88c807SRodney W. Grimes #if __STDC__
6494b88c807SRodney W. Grimes int
6504b88c807SRodney W. Grimes vcpio_rd(register ARCHD *arcn, register char *buf)
6514b88c807SRodney W. Grimes #else
6524b88c807SRodney W. Grimes int
6534b88c807SRodney W. Grimes vcpio_rd(arcn, buf)
6544b88c807SRodney W. Grimes 	register ARCHD *arcn;
6554b88c807SRodney W. Grimes 	register char *buf;
6564b88c807SRodney W. Grimes #endif
6574b88c807SRodney W. Grimes {
6584b88c807SRodney W. Grimes 	register HD_VCPIO *hd;
6594b88c807SRodney W. Grimes 	dev_t devminor;
6604b88c807SRodney W. Grimes 	dev_t devmajor;
6614b88c807SRodney W. Grimes 	register int nsz;
6624b88c807SRodney W. Grimes 
6634b88c807SRodney W. Grimes 	/*
6644b88c807SRodney W. Grimes 	 * during the id phase it was determined if we were using CRC, use the
6654b88c807SRodney W. Grimes 	 * proper id routine.
6664b88c807SRodney W. Grimes 	 */
6674b88c807SRodney W. Grimes 	if (docrc) {
6684b88c807SRodney W. Grimes 		if (crc_id(buf, sizeof(HD_VCPIO)) < 0)
6694b88c807SRodney W. Grimes 			return(-1);
6704b88c807SRodney W. Grimes 	} else {
6714b88c807SRodney W. Grimes 		if (vcpio_id(buf, sizeof(HD_VCPIO)) < 0)
6724b88c807SRodney W. Grimes 			return(-1);
6734b88c807SRodney W. Grimes 	}
6744b88c807SRodney W. Grimes 
6754b88c807SRodney W. Grimes 	hd = (HD_VCPIO *)buf;
6764b88c807SRodney W. Grimes 	arcn->pad = 0L;
6774b88c807SRodney W. Grimes 
6784b88c807SRodney W. Grimes 	/*
6794b88c807SRodney W. Grimes 	 * extract the hex ascii fields from the header
6804b88c807SRodney W. Grimes 	 */
6814b88c807SRodney W. Grimes 	arcn->sb.st_ino = (ino_t)asc_ul(hd->c_ino, sizeof(hd->c_ino), HEX);
6824b88c807SRodney W. Grimes 	arcn->sb.st_mode = (mode_t)asc_ul(hd->c_mode, sizeof(hd->c_mode), HEX);
6834b88c807SRodney W. Grimes 	arcn->sb.st_uid = (uid_t)asc_ul(hd->c_uid, sizeof(hd->c_uid), HEX);
6844b88c807SRodney W. Grimes 	arcn->sb.st_gid = (gid_t)asc_ul(hd->c_gid, sizeof(hd->c_gid), HEX);
6854b88c807SRodney W. Grimes 	arcn->sb.st_mtime = (time_t)asc_ul(hd->c_mtime,sizeof(hd->c_mtime),HEX);
6864b88c807SRodney W. Grimes 	arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime;
6874b88c807SRodney W. Grimes #	ifdef NET2_STAT
6884b88c807SRodney W. Grimes 	arcn->sb.st_size = (off_t)asc_ul(hd->c_filesize,
6894b88c807SRodney W. Grimes 	    sizeof(hd->c_filesize), HEX);
6904b88c807SRodney W. Grimes #	else
6914b88c807SRodney W. Grimes 	arcn->sb.st_size = (off_t)asc_uqd(hd->c_filesize,
6924b88c807SRodney W. Grimes 	    sizeof(hd->c_filesize), HEX);
6934b88c807SRodney W. Grimes #	endif
6944b88c807SRodney W. Grimes 	arcn->sb.st_nlink = (nlink_t)asc_ul(hd->c_nlink, sizeof(hd->c_nlink),
6954b88c807SRodney W. Grimes 	    HEX);
6964b88c807SRodney W. Grimes 	devmajor = (dev_t)asc_ul(hd->c_maj, sizeof(hd->c_maj), HEX);
6974b88c807SRodney W. Grimes 	devminor = (dev_t)asc_ul(hd->c_min, sizeof(hd->c_min), HEX);
6984b88c807SRodney W. Grimes 	arcn->sb.st_dev = TODEV(devmajor, devminor);
6994b88c807SRodney W. Grimes 	devmajor = (dev_t)asc_ul(hd->c_rmaj, sizeof(hd->c_maj), HEX);
7004b88c807SRodney W. Grimes 	devminor = (dev_t)asc_ul(hd->c_rmin, sizeof(hd->c_min), HEX);
7014b88c807SRodney W. Grimes 	arcn->sb.st_rdev = TODEV(devmajor, devminor);
7024b88c807SRodney W. Grimes 	arcn->crc = asc_ul(hd->c_chksum, sizeof(hd->c_chksum), HEX);
7034b88c807SRodney W. Grimes 
7044b88c807SRodney W. Grimes 	/*
7054b88c807SRodney W. Grimes 	 * check the length of the file name, if ok read it in, return -1 if
7064b88c807SRodney W. Grimes 	 * bogus
7074b88c807SRodney W. Grimes 	 */
7084b88c807SRodney W. Grimes 	if ((nsz = (int)asc_ul(hd->c_namesize,sizeof(hd->c_namesize),HEX)) < 2)
7094b88c807SRodney W. Grimes 		return(-1);
7104b88c807SRodney W. Grimes 	arcn->nlen = nsz - 1;
7114b88c807SRodney W. Grimes 	if (rd_nm(arcn, nsz) < 0)
7124b88c807SRodney W. Grimes 		return(-1);
7134b88c807SRodney W. Grimes 
7144b88c807SRodney W. Grimes 	/*
7154b88c807SRodney W. Grimes 	 * skip padding. header + filename is aligned to 4 byte boundries
7164b88c807SRodney W. Grimes 	 */
7174b88c807SRodney W. Grimes 	if (rd_skip((off_t)(VCPIO_PAD(sizeof(HD_VCPIO) + nsz))) < 0)
7184b88c807SRodney W. Grimes 		return(-1);
7194b88c807SRodney W. Grimes 
7204b88c807SRodney W. Grimes 	/*
7214b88c807SRodney W. Grimes 	 * if not a link (or a file with no data), calculate pad size (for
7224b88c807SRodney W. Grimes 	 * padding which follows the file data), clear the link name and return
7234b88c807SRodney W. Grimes 	 */
7244b88c807SRodney W. Grimes 	if (((arcn->sb.st_mode&C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)) {
7254b88c807SRodney W. Grimes 		/*
7264b88c807SRodney W. Grimes 		 * we have a valid header (not a link)
7274b88c807SRodney W. Grimes 		 */
7284b88c807SRodney W. Grimes 		arcn->ln_nlen = 0;
7294b88c807SRodney W. Grimes 		arcn->ln_name[0] = '\0';
7304b88c807SRodney W. Grimes 		arcn->pad = VCPIO_PAD(arcn->sb.st_size);
7314b88c807SRodney W. Grimes 		return(com_rd(arcn));
7324b88c807SRodney W. Grimes 	}
7334b88c807SRodney W. Grimes 
7344b88c807SRodney W. Grimes 	/*
7354b88c807SRodney W. Grimes 	 * read in the link name and skip over the padding
7364b88c807SRodney W. Grimes 	 */
7374b88c807SRodney W. Grimes 	if ((rd_ln_nm(arcn) < 0) ||
7384b88c807SRodney W. Grimes 	    (rd_skip((off_t)(VCPIO_PAD(arcn->sb.st_size))) < 0))
7394b88c807SRodney W. Grimes 		return(-1);
7404b88c807SRodney W. Grimes 
7414b88c807SRodney W. Grimes 	/*
7424b88c807SRodney W. Grimes 	 * we have a valid header (with a link)
7434b88c807SRodney W. Grimes 	 */
7444b88c807SRodney W. Grimes 	return(com_rd(arcn));
7454b88c807SRodney W. Grimes }
7464b88c807SRodney W. Grimes 
7474b88c807SRodney W. Grimes /*
7484b88c807SRodney W. Grimes  * vcpio_endrd()
7494b88c807SRodney W. Grimes  *      no cleanup needed here, just return size of the trailer (for append)
7504b88c807SRodney W. Grimes  * Return:
7514b88c807SRodney W. Grimes  *      size of trailer header in this format
7524b88c807SRodney W. Grimes  */
7534b88c807SRodney W. Grimes 
7544b88c807SRodney W. Grimes #if __STDC__
7554b88c807SRodney W. Grimes off_t
7564b88c807SRodney W. Grimes vcpio_endrd(void)
7574b88c807SRodney W. Grimes #else
7584b88c807SRodney W. Grimes off_t
7594b88c807SRodney W. Grimes vcpio_endrd()
7604b88c807SRodney W. Grimes #endif
7614b88c807SRodney W. Grimes {
7624b88c807SRodney W. Grimes 	return((off_t)(sizeof(HD_VCPIO) + sizeof(TRAILER) +
7634b88c807SRodney W. Grimes 		(VCPIO_PAD(sizeof(HD_VCPIO) + sizeof(TRAILER)))));
7644b88c807SRodney W. Grimes }
7654b88c807SRodney W. Grimes 
7664b88c807SRodney W. Grimes /*
7674b88c807SRodney W. Grimes  * crc_stwr()
7684b88c807SRodney W. Grimes  *	start up the device mapping table, enable crc file calculation
7694b88c807SRodney W. Grimes  * Return:
7704b88c807SRodney W. Grimes  *	0 if ok, -1 otherwise (what dev_start() returns)
7714b88c807SRodney W. Grimes  */
7724b88c807SRodney W. Grimes 
7734b88c807SRodney W. Grimes #if __STDC__
7744b88c807SRodney W. Grimes int
7754b88c807SRodney W. Grimes crc_stwr(void)
7764b88c807SRodney W. Grimes #else
7774b88c807SRodney W. Grimes int
7784b88c807SRodney W. Grimes crc_stwr()
7794b88c807SRodney W. Grimes #endif
7804b88c807SRodney W. Grimes {
7814b88c807SRodney W. Grimes 	docrc = 1;
7824b88c807SRodney W. Grimes 	return(dev_start());
7834b88c807SRodney W. Grimes }
7844b88c807SRodney W. Grimes 
7854b88c807SRodney W. Grimes /*
7864b88c807SRodney W. Grimes  * vcpio_wr()
7874b88c807SRodney W. Grimes  *	copy the data in the ARCHD to buffer in system VR4 cpio
7884b88c807SRodney W. Grimes  *	(with/without crc) format.
7894b88c807SRodney W. Grimes  * Return
7904b88c807SRodney W. Grimes  *	0 if file has data to be written after the header, 1 if file has
7914b88c807SRodney W. Grimes  *	NO data to write after the header, -1 if archive write failed
7924b88c807SRodney W. Grimes  */
7934b88c807SRodney W. Grimes 
7944b88c807SRodney W. Grimes #if __STDC__
7954b88c807SRodney W. Grimes int
7964b88c807SRodney W. Grimes vcpio_wr(register ARCHD *arcn)
7974b88c807SRodney W. Grimes #else
7984b88c807SRodney W. Grimes int
7994b88c807SRodney W. Grimes vcpio_wr(arcn)
8004b88c807SRodney W. Grimes 	register ARCHD *arcn;
8014b88c807SRodney W. Grimes #endif
8024b88c807SRodney W. Grimes {
8034b88c807SRodney W. Grimes 	register HD_VCPIO *hd;
8044b88c807SRodney W. Grimes 	unsigned int nsz;
8054b88c807SRodney W. Grimes 	char hdblk[sizeof(HD_VCPIO)];
8064b88c807SRodney W. Grimes 
8074b88c807SRodney W. Grimes 	/*
8084b88c807SRodney W. Grimes 	 * check and repair truncated device and inode fields in the cpio
8094b88c807SRodney W. Grimes 	 * header
8104b88c807SRodney W. Grimes 	 */
8114b88c807SRodney W. Grimes 	if (map_dev(arcn, (u_long)VCPIO_MASK, (u_long)VCPIO_MASK) < 0)
8124b88c807SRodney W. Grimes 		return(-1);
8134b88c807SRodney W. Grimes 	nsz = arcn->nlen + 1;
8144b88c807SRodney W. Grimes 	hd = (HD_VCPIO *)hdblk;
8154b88c807SRodney W. Grimes 	if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR))
8164b88c807SRodney W. Grimes 		arcn->sb.st_rdev = 0;
8174b88c807SRodney W. Grimes 
8184b88c807SRodney W. Grimes 	/*
8194b88c807SRodney W. Grimes 	 * add the proper magic value depending whether we were asked for
8204b88c807SRodney W. Grimes 	 * file data crc's, and the crc if needed.
8214b88c807SRodney W. Grimes 	 */
8224b88c807SRodney W. Grimes 	if (docrc) {
8234b88c807SRodney W. Grimes 		if (ul_asc((u_long)VCMAGIC, hd->c_magic, sizeof(hd->c_magic),
8244b88c807SRodney W. Grimes 	    		OCT) ||
8254b88c807SRodney W. Grimes 		    ul_asc((u_long)arcn->crc,hd->c_chksum,sizeof(hd->c_chksum),
8264b88c807SRodney W. Grimes 	    		HEX))
8274b88c807SRodney W. Grimes 			goto out;
8284b88c807SRodney W. Grimes 	} else {
8294b88c807SRodney W. Grimes 		if (ul_asc((u_long)VMAGIC, hd->c_magic, sizeof(hd->c_magic),
8304b88c807SRodney W. Grimes 	    		OCT) ||
8314b88c807SRodney W. Grimes 		    ul_asc((u_long)0L, hd->c_chksum, sizeof(hd->c_chksum),HEX))
8324b88c807SRodney W. Grimes 			goto out;
8334b88c807SRodney W. Grimes 	}
8344b88c807SRodney W. Grimes 
8354b88c807SRodney W. Grimes 	switch(arcn->type) {
8364b88c807SRodney W. Grimes 	case PAX_CTG:
8374b88c807SRodney W. Grimes 	case PAX_REG:
8384b88c807SRodney W. Grimes 	case PAX_HRG:
8394b88c807SRodney W. Grimes 		/*
8404b88c807SRodney W. Grimes 		 * caller will copy file data to the archive. tell him how
8414b88c807SRodney W. Grimes 		 * much to pad.
8424b88c807SRodney W. Grimes 		 */
8434b88c807SRodney W. Grimes 		arcn->pad = VCPIO_PAD(arcn->sb.st_size);
8444b88c807SRodney W. Grimes #		ifdef NET2_STAT
8454b88c807SRodney W. Grimes 		if (ul_asc((u_long)arcn->sb.st_size, hd->c_filesize,
8464b88c807SRodney W. Grimes 		    sizeof(hd->c_filesize), HEX)) {
8474b88c807SRodney W. Grimes #		else
8484b88c807SRodney W. Grimes 		if (uqd_asc((u_quad_t)arcn->sb.st_size, hd->c_filesize,
8494b88c807SRodney W. Grimes 		    sizeof(hd->c_filesize), HEX)) {
8504b88c807SRodney W. Grimes #		endif
851a885d9dcSSøren Schmidt 			pax_warn(1,"File is too large for sv4cpio format %s",
8524b88c807SRodney W. Grimes 			    arcn->org_name);
8534b88c807SRodney W. Grimes 			return(1);
8544b88c807SRodney W. Grimes 		}
8554b88c807SRodney W. Grimes 		break;
8564b88c807SRodney W. Grimes 	case PAX_SLK:
8574b88c807SRodney W. Grimes 		/*
8584b88c807SRodney W. Grimes 		 * no file data for the caller to process, the file data has
8594b88c807SRodney W. Grimes 		 * the size of the link
8604b88c807SRodney W. Grimes 		 */
8614b88c807SRodney W. Grimes 		arcn->pad = 0L;
8624b88c807SRodney W. Grimes 		if (ul_asc((u_long)arcn->ln_nlen, hd->c_filesize,
8634b88c807SRodney W. Grimes 		    sizeof(hd->c_filesize), HEX))
8644b88c807SRodney W. Grimes 			goto out;
8654b88c807SRodney W. Grimes 		break;
8664b88c807SRodney W. Grimes 	default:
8674b88c807SRodney W. Grimes 		/*
8684b88c807SRodney W. Grimes 		 * no file data for the caller to process
8694b88c807SRodney W. Grimes 		 */
8704b88c807SRodney W. Grimes 		arcn->pad = 0L;
8714b88c807SRodney W. Grimes 		if (ul_asc((u_long)0L, hd->c_filesize, sizeof(hd->c_filesize),
8724b88c807SRodney W. Grimes 		    HEX))
8734b88c807SRodney W. Grimes 			goto out;
8744b88c807SRodney W. Grimes 		break;
8754b88c807SRodney W. Grimes 	}
8764b88c807SRodney W. Grimes 
8774b88c807SRodney W. Grimes 	/*
8784b88c807SRodney W. Grimes 	 * set the other fields in the header
8794b88c807SRodney W. Grimes 	 */
8804b88c807SRodney W. Grimes 	if (ul_asc((u_long)arcn->sb.st_ino, hd->c_ino, sizeof(hd->c_ino),
8814b88c807SRodney W. Grimes 		HEX) ||
8824b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_mode, hd->c_mode, sizeof(hd->c_mode),
8834b88c807SRodney W. Grimes 		HEX) ||
8844b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_uid, hd->c_uid, sizeof(hd->c_uid),
8854b88c807SRodney W. Grimes 		HEX) ||
8864b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_gid, hd->c_gid, sizeof(hd->c_gid),
8874b88c807SRodney W. Grimes     		HEX) ||
8884b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_mtime, hd->c_mtime, sizeof(hd->c_mtime),
8894b88c807SRodney W. Grimes     		HEX) ||
8904b88c807SRodney W. Grimes 	    ul_asc((u_long)arcn->sb.st_nlink, hd->c_nlink, sizeof(hd->c_nlink),
8914b88c807SRodney W. Grimes     		HEX) ||
8924b88c807SRodney W. Grimes 	    ul_asc((u_long)MAJOR(arcn->sb.st_dev),hd->c_maj, sizeof(hd->c_maj),
8934b88c807SRodney W. Grimes 		HEX) ||
8944b88c807SRodney W. Grimes 	    ul_asc((u_long)MINOR(arcn->sb.st_dev),hd->c_min, sizeof(hd->c_min),
8954b88c807SRodney W. Grimes 		HEX) ||
8964b88c807SRodney W. Grimes 	    ul_asc((u_long)MAJOR(arcn->sb.st_rdev),hd->c_rmaj,sizeof(hd->c_maj),
8974b88c807SRodney W. Grimes 		HEX) ||
8984b88c807SRodney W. Grimes 	    ul_asc((u_long)MINOR(arcn->sb.st_rdev),hd->c_rmin,sizeof(hd->c_min),
8994b88c807SRodney W. Grimes 		HEX) ||
9004b88c807SRodney W. Grimes 	    ul_asc((u_long)nsz, hd->c_namesize, sizeof(hd->c_namesize), HEX))
9014b88c807SRodney W. Grimes 		goto out;
9024b88c807SRodney W. Grimes 
9034b88c807SRodney W. Grimes 	/*
9044b88c807SRodney W. Grimes 	 * write the header, the file name and padding as required.
9054b88c807SRodney W. Grimes 	 */
9064b88c807SRodney W. Grimes 	if ((wr_rdbuf(hdblk, (int)sizeof(HD_VCPIO)) < 0) ||
9074b88c807SRodney W. Grimes 	    (wr_rdbuf(arcn->name, (int)nsz) < 0)  ||
9084b88c807SRodney W. Grimes 	    (wr_skip((off_t)(VCPIO_PAD(sizeof(HD_VCPIO) + nsz))) < 0)) {
909a885d9dcSSøren Schmidt 		pax_warn(1,"Could not write sv4cpio header for %s",arcn->org_name);
9104b88c807SRodney W. Grimes 		return(-1);
9114b88c807SRodney W. Grimes 	}
9124b88c807SRodney W. Grimes 
9134b88c807SRodney W. Grimes 	/*
9144b88c807SRodney W. Grimes 	 * if we have file data, tell the caller we are done, copy the file
9154b88c807SRodney W. Grimes 	 */
9164b88c807SRodney W. Grimes 	if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) ||
9174b88c807SRodney W. Grimes 	    (arcn->type == PAX_HRG))
9184b88c807SRodney W. Grimes 		return(0);
9194b88c807SRodney W. Grimes 
9204b88c807SRodney W. Grimes 	/*
9214b88c807SRodney W. Grimes 	 * if we are not a link, tell the caller we are done, go to next file
9224b88c807SRodney W. Grimes 	 */
9234b88c807SRodney W. Grimes 	if (arcn->type != PAX_SLK)
9244b88c807SRodney W. Grimes 		return(1);
9254b88c807SRodney W. Grimes 
9264b88c807SRodney W. Grimes 	/*
9274b88c807SRodney W. Grimes 	 * write the link name, tell the caller we are done.
9284b88c807SRodney W. Grimes 	 */
9294b88c807SRodney W. Grimes 	if ((wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) ||
9304b88c807SRodney W. Grimes 	    (wr_skip((off_t)(VCPIO_PAD(arcn->ln_nlen))) < 0)) {
931a885d9dcSSøren Schmidt 		pax_warn(1,"Could not write sv4cpio link name for %s",
9324b88c807SRodney W. Grimes 		    arcn->org_name);
9334b88c807SRodney W. Grimes 		return(-1);
9344b88c807SRodney W. Grimes 	}
9354b88c807SRodney W. Grimes 	return(1);
9364b88c807SRodney W. Grimes 
9374b88c807SRodney W. Grimes     out:
9384b88c807SRodney W. Grimes 	/*
9394b88c807SRodney W. Grimes 	 * header field is out of range
9404b88c807SRodney W. Grimes 	 */
941a885d9dcSSøren Schmidt 	pax_warn(1,"Sv4cpio header field is too small for file %s",arcn->org_name);
9424b88c807SRodney W. Grimes 	return(1);
9434b88c807SRodney W. Grimes }
9444b88c807SRodney W. Grimes 
9454b88c807SRodney W. Grimes /*
9464b88c807SRodney W. Grimes  * Routines common to the old binary header cpio
9474b88c807SRodney W. Grimes  */
9484b88c807SRodney W. Grimes 
9494b88c807SRodney W. Grimes /*
9504b88c807SRodney W. Grimes  * bcpio_id()
9514b88c807SRodney W. Grimes  *      determine if a block given to us is a old binary cpio header
9524b88c807SRodney W. Grimes  *	(with/without header byte swapping)
9534b88c807SRodney W. Grimes  * Return:
9544b88c807SRodney W. Grimes  *      0 if a valid header, -1 otherwise
9554b88c807SRodney W. Grimes  */
9564b88c807SRodney W. Grimes 
9574b88c807SRodney W. Grimes #if __STDC__
9584b88c807SRodney W. Grimes int
9594b88c807SRodney W. Grimes bcpio_id(char *blk, int size)
9604b88c807SRodney W. Grimes #else
9614b88c807SRodney W. Grimes int
9624b88c807SRodney W. Grimes bcpio_id(blk, size)
9634b88c807SRodney W. Grimes 	char *blk;
9644b88c807SRodney W. Grimes 	int size;
9654b88c807SRodney W. Grimes #endif
9664b88c807SRodney W. Grimes {
9674b88c807SRodney W. Grimes 	if (size < sizeof(HD_BCPIO))
9684b88c807SRodney W. Grimes 		return(-1);
9694b88c807SRodney W. Grimes 
9704b88c807SRodney W. Grimes 	/*
9714b88c807SRodney W. Grimes 	 * check both normal and byte swapped magic cookies
9724b88c807SRodney W. Grimes 	 */
9734b88c807SRodney W. Grimes 	if (((u_short)SHRT_EXT(blk)) == MAGIC)
9744b88c807SRodney W. Grimes 		return(0);
9754b88c807SRodney W. Grimes 	if (((u_short)RSHRT_EXT(blk)) == MAGIC) {
9764b88c807SRodney W. Grimes 		if (!swp_head)
9774b88c807SRodney W. Grimes 			++swp_head;
9784b88c807SRodney W. Grimes 		return(0);
9794b88c807SRodney W. Grimes 	}
9804b88c807SRodney W. Grimes 	return(-1);
9814b88c807SRodney W. Grimes }
9824b88c807SRodney W. Grimes 
9834b88c807SRodney W. Grimes /*
9844b88c807SRodney W. Grimes  * bcpio_rd()
9854b88c807SRodney W. Grimes  *	determine if a buffer is a old binary archive entry. (it may have byte
9864b88c807SRodney W. Grimes  *	swapped header) convert and store the values in the ARCHD parameter.
9874b88c807SRodney W. Grimes  *	This is a very old header format and should not really be used.
9884b88c807SRodney W. Grimes  * Return:
9894b88c807SRodney W. Grimes  *	0 if a valid header, -1 otherwise.
9904b88c807SRodney W. Grimes  */
9914b88c807SRodney W. Grimes 
9924b88c807SRodney W. Grimes #if __STDC__
9934b88c807SRodney W. Grimes int
9944b88c807SRodney W. Grimes bcpio_rd(register ARCHD *arcn, register char *buf)
9954b88c807SRodney W. Grimes #else
9964b88c807SRodney W. Grimes int
9974b88c807SRodney W. Grimes bcpio_rd(arcn, buf)
9984b88c807SRodney W. Grimes 	register ARCHD *arcn;
9994b88c807SRodney W. Grimes 	register char *buf;
10004b88c807SRodney W. Grimes #endif
10014b88c807SRodney W. Grimes {
10024b88c807SRodney W. Grimes 	register HD_BCPIO *hd;
10034b88c807SRodney W. Grimes 	register int nsz;
10044b88c807SRodney W. Grimes 
10054b88c807SRodney W. Grimes 	/*
10064b88c807SRodney W. Grimes 	 * check the header
10074b88c807SRodney W. Grimes 	 */
10084b88c807SRodney W. Grimes 	if (bcpio_id(buf, sizeof(HD_BCPIO)) < 0)
10094b88c807SRodney W. Grimes 		return(-1);
10104b88c807SRodney W. Grimes 
10114b88c807SRodney W. Grimes 	arcn->pad = 0L;
10124b88c807SRodney W. Grimes 	hd = (HD_BCPIO *)buf;
10134b88c807SRodney W. Grimes 	if (swp_head) {
10144b88c807SRodney W. Grimes 		/*
10154b88c807SRodney W. Grimes 		 * header has swapped bytes on 16 bit boundries
10164b88c807SRodney W. Grimes 		 */
10174b88c807SRodney W. Grimes 		arcn->sb.st_dev = (dev_t)(RSHRT_EXT(hd->h_dev));
10184b88c807SRodney W. Grimes 		arcn->sb.st_ino = (ino_t)(RSHRT_EXT(hd->h_ino));
10194b88c807SRodney W. Grimes 		arcn->sb.st_mode = (mode_t)(RSHRT_EXT(hd->h_mode));
10204b88c807SRodney W. Grimes 		arcn->sb.st_uid = (uid_t)(RSHRT_EXT(hd->h_uid));
10214b88c807SRodney W. Grimes 		arcn->sb.st_gid = (gid_t)(RSHRT_EXT(hd->h_gid));
10224b88c807SRodney W. Grimes 		arcn->sb.st_nlink = (nlink_t)(RSHRT_EXT(hd->h_nlink));
10234b88c807SRodney W. Grimes 		arcn->sb.st_rdev = (dev_t)(RSHRT_EXT(hd->h_rdev));
10244b88c807SRodney W. Grimes 		arcn->sb.st_mtime = (time_t)(RSHRT_EXT(hd->h_mtime_1));
10254b88c807SRodney W. Grimes 		arcn->sb.st_mtime =  (arcn->sb.st_mtime << 16) |
10264b88c807SRodney W. Grimes 			((time_t)(RSHRT_EXT(hd->h_mtime_2)));
10274b88c807SRodney W. Grimes 		arcn->sb.st_size = (off_t)(RSHRT_EXT(hd->h_filesize_1));
10284b88c807SRodney W. Grimes 		arcn->sb.st_size = (arcn->sb.st_size << 16) |
10294b88c807SRodney W. Grimes 			((off_t)(RSHRT_EXT(hd->h_filesize_2)));
10304b88c807SRodney W. Grimes 		nsz = (int)(RSHRT_EXT(hd->h_namesize));
10314b88c807SRodney W. Grimes 	} else {
10324b88c807SRodney W. Grimes 		arcn->sb.st_dev = (dev_t)(SHRT_EXT(hd->h_dev));
10334b88c807SRodney W. Grimes 		arcn->sb.st_ino = (ino_t)(SHRT_EXT(hd->h_ino));
10344b88c807SRodney W. Grimes 		arcn->sb.st_mode = (mode_t)(SHRT_EXT(hd->h_mode));
10354b88c807SRodney W. Grimes 		arcn->sb.st_uid = (uid_t)(SHRT_EXT(hd->h_uid));
10364b88c807SRodney W. Grimes 		arcn->sb.st_gid = (gid_t)(SHRT_EXT(hd->h_gid));
10374b88c807SRodney W. Grimes 		arcn->sb.st_nlink = (nlink_t)(SHRT_EXT(hd->h_nlink));
10384b88c807SRodney W. Grimes 		arcn->sb.st_rdev = (dev_t)(SHRT_EXT(hd->h_rdev));
10394b88c807SRodney W. Grimes 		arcn->sb.st_mtime = (time_t)(SHRT_EXT(hd->h_mtime_1));
10404b88c807SRodney W. Grimes 		arcn->sb.st_mtime =  (arcn->sb.st_mtime << 16) |
10414b88c807SRodney W. Grimes 			((time_t)(SHRT_EXT(hd->h_mtime_2)));
10424b88c807SRodney W. Grimes 		arcn->sb.st_size = (off_t)(SHRT_EXT(hd->h_filesize_1));
10434b88c807SRodney W. Grimes 		arcn->sb.st_size = (arcn->sb.st_size << 16) |
10444b88c807SRodney W. Grimes 			((off_t)(SHRT_EXT(hd->h_filesize_2)));
10454b88c807SRodney W. Grimes 		nsz = (int)(SHRT_EXT(hd->h_namesize));
10464b88c807SRodney W. Grimes 	}
10474b88c807SRodney W. Grimes 	arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime;
10484b88c807SRodney W. Grimes 
10494b88c807SRodney W. Grimes 	/*
10504b88c807SRodney W. Grimes 	 * check the file name size, if bogus give up. otherwise read the file
10514b88c807SRodney W. Grimes 	 * name
10524b88c807SRodney W. Grimes 	 */
10534b88c807SRodney W. Grimes 	if (nsz < 2)
10544b88c807SRodney W. Grimes 		return(-1);
10554b88c807SRodney W. Grimes 	arcn->nlen = nsz - 1;
10564b88c807SRodney W. Grimes 	if (rd_nm(arcn, nsz) < 0)
10574b88c807SRodney W. Grimes 		return(-1);
10584b88c807SRodney W. Grimes 
10594b88c807SRodney W. Grimes 	/*
10604b88c807SRodney W. Grimes 	 * header + file name are aligned to 2 byte boundries, skip if needed
10614b88c807SRodney W. Grimes 	 */
10624b88c807SRodney W. Grimes 	if (rd_skip((off_t)(BCPIO_PAD(sizeof(HD_BCPIO) + nsz))) < 0)
10634b88c807SRodney W. Grimes 		return(-1);
10644b88c807SRodney W. Grimes 
10654b88c807SRodney W. Grimes 	/*
10664b88c807SRodney W. Grimes 	 * if not a link (or a file with no data), calculate pad size (for
10674b88c807SRodney W. Grimes 	 * padding which follows the file data), clear the link name and return
10684b88c807SRodney W. Grimes 	 */
10694b88c807SRodney W. Grimes 	if (((arcn->sb.st_mode & C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)){
10704b88c807SRodney W. Grimes 		/*
10714b88c807SRodney W. Grimes 		 * we have a valid header (not a link)
10724b88c807SRodney W. Grimes 		 */
10734b88c807SRodney W. Grimes 		arcn->ln_nlen = 0;
10744b88c807SRodney W. Grimes 		arcn->ln_name[0] = '\0';
10754b88c807SRodney W. Grimes 		arcn->pad = BCPIO_PAD(arcn->sb.st_size);
10764b88c807SRodney W. Grimes 		return(com_rd(arcn));
10774b88c807SRodney W. Grimes 	}
10784b88c807SRodney W. Grimes 
10794b88c807SRodney W. Grimes 	if ((rd_ln_nm(arcn) < 0) ||
10804b88c807SRodney W. Grimes 	    (rd_skip((off_t)(BCPIO_PAD(arcn->sb.st_size))) < 0))
10814b88c807SRodney W. Grimes 		return(-1);
10824b88c807SRodney W. Grimes 
10834b88c807SRodney W. Grimes 	/*
10844b88c807SRodney W. Grimes 	 * we have a valid header (with a link)
10854b88c807SRodney W. Grimes 	 */
10864b88c807SRodney W. Grimes 	return(com_rd(arcn));
10874b88c807SRodney W. Grimes }
10884b88c807SRodney W. Grimes 
10894b88c807SRodney W. Grimes /*
10904b88c807SRodney W. Grimes  * bcpio_endrd()
10914b88c807SRodney W. Grimes  *      no cleanup needed here, just return size of the trailer (for append)
10924b88c807SRodney W. Grimes  * Return:
10934b88c807SRodney W. Grimes  *      size of trailer header in this format
10944b88c807SRodney W. Grimes  */
10954b88c807SRodney W. Grimes 
10964b88c807SRodney W. Grimes #if __STDC__
10974b88c807SRodney W. Grimes off_t
10984b88c807SRodney W. Grimes bcpio_endrd(void)
10994b88c807SRodney W. Grimes #else
11004b88c807SRodney W. Grimes off_t
11014b88c807SRodney W. Grimes bcpio_endrd()
11024b88c807SRodney W. Grimes #endif
11034b88c807SRodney W. Grimes {
11044b88c807SRodney W. Grimes 	return((off_t)(sizeof(HD_BCPIO) + sizeof(TRAILER) +
11054b88c807SRodney W. Grimes 		(BCPIO_PAD(sizeof(HD_BCPIO) + sizeof(TRAILER)))));
11064b88c807SRodney W. Grimes }
11074b88c807SRodney W. Grimes 
11084b88c807SRodney W. Grimes /*
11094b88c807SRodney W. Grimes  * bcpio_wr()
11104b88c807SRodney W. Grimes  *	copy the data in the ARCHD to buffer in old binary cpio format
11114b88c807SRodney W. Grimes  *	There is a real chance of field overflow with this critter. So we
11124b88c807SRodney W. Grimes  *	always check the conversion is ok. nobody in his their right mind
11134b88c807SRodney W. Grimes  *	should write an achive in this format...
11144b88c807SRodney W. Grimes  * Return
11154b88c807SRodney W. Grimes  *      0 if file has data to be written after the header, 1 if file has NO
11164b88c807SRodney W. Grimes  *	data to write after the header, -1 if archive write failed
11174b88c807SRodney W. Grimes  */
11184b88c807SRodney W. Grimes 
11194b88c807SRodney W. Grimes #if __STDC__
11204b88c807SRodney W. Grimes int
11214b88c807SRodney W. Grimes bcpio_wr(register ARCHD *arcn)
11224b88c807SRodney W. Grimes #else
11234b88c807SRodney W. Grimes int
11244b88c807SRodney W. Grimes bcpio_wr(arcn)
11254b88c807SRodney W. Grimes 	register ARCHD *arcn;
11264b88c807SRodney W. Grimes #endif
11274b88c807SRodney W. Grimes {
11284b88c807SRodney W. Grimes 	register HD_BCPIO *hd;
11294b88c807SRodney W. Grimes 	register int nsz;
11304b88c807SRodney W. Grimes 	char hdblk[sizeof(HD_BCPIO)];
11314b88c807SRodney W. Grimes 	off_t t_offt;
11324b88c807SRodney W. Grimes 	int t_int;
11334b88c807SRodney W. Grimes 	time_t t_timet;
11344b88c807SRodney W. Grimes 
11354b88c807SRodney W. Grimes 	/*
11364b88c807SRodney W. Grimes 	 * check and repair truncated device and inode fields in the cpio
11374b88c807SRodney W. Grimes 	 * header
11384b88c807SRodney W. Grimes 	 */
11394b88c807SRodney W. Grimes 	if (map_dev(arcn, (u_long)BCPIO_MASK, (u_long)BCPIO_MASK) < 0)
11404b88c807SRodney W. Grimes 		return(-1);
11414b88c807SRodney W. Grimes 
11424b88c807SRodney W. Grimes 	if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR))
11434b88c807SRodney W. Grimes 		arcn->sb.st_rdev = 0;
11444b88c807SRodney W. Grimes 	hd = (HD_BCPIO *)hdblk;
11454b88c807SRodney W. Grimes 
11464b88c807SRodney W. Grimes 	switch(arcn->type) {
11474b88c807SRodney W. Grimes 	case PAX_CTG:
11484b88c807SRodney W. Grimes 	case PAX_REG:
11494b88c807SRodney W. Grimes 	case PAX_HRG:
11504b88c807SRodney W. Grimes 		/*
11514b88c807SRodney W. Grimes 		 * caller will copy file data to the archive. tell him how
11524b88c807SRodney W. Grimes 		 * much to pad.
11534b88c807SRodney W. Grimes 		 */
11544b88c807SRodney W. Grimes 		arcn->pad = BCPIO_PAD(arcn->sb.st_size);
11554b88c807SRodney W. Grimes 		hd->h_filesize_1[0] = CHR_WR_0(arcn->sb.st_size);
11564b88c807SRodney W. Grimes 		hd->h_filesize_1[1] = CHR_WR_1(arcn->sb.st_size);
11574b88c807SRodney W. Grimes 		hd->h_filesize_2[0] = CHR_WR_2(arcn->sb.st_size);
11584b88c807SRodney W. Grimes 		hd->h_filesize_2[1] = CHR_WR_3(arcn->sb.st_size);
11594b88c807SRodney W. Grimes 		t_offt = (off_t)(SHRT_EXT(hd->h_filesize_1));
11604b88c807SRodney W. Grimes 		t_offt = (t_offt<<16) | ((off_t)(SHRT_EXT(hd->h_filesize_2)));
11614b88c807SRodney W. Grimes 		if (arcn->sb.st_size != t_offt) {
1162a885d9dcSSøren Schmidt 			pax_warn(1,"File is too large for bcpio format %s",
11634b88c807SRodney W. Grimes 			    arcn->org_name);
11644b88c807SRodney W. Grimes 			return(1);
11654b88c807SRodney W. Grimes 		}
11664b88c807SRodney W. Grimes 		break;
11674b88c807SRodney W. Grimes 	case PAX_SLK:
11684b88c807SRodney W. Grimes 		/*
11694b88c807SRodney W. Grimes 		 * no file data for the caller to process, the file data has
11704b88c807SRodney W. Grimes 		 * the size of the link
11714b88c807SRodney W. Grimes 		 */
11724b88c807SRodney W. Grimes 		arcn->pad = 0L;
11734b88c807SRodney W. Grimes 		hd->h_filesize_1[0] = CHR_WR_0(arcn->ln_nlen);
11744b88c807SRodney W. Grimes 		hd->h_filesize_1[1] = CHR_WR_1(arcn->ln_nlen);
11754b88c807SRodney W. Grimes 		hd->h_filesize_2[0] = CHR_WR_2(arcn->ln_nlen);
11764b88c807SRodney W. Grimes 		hd->h_filesize_2[1] = CHR_WR_3(arcn->ln_nlen);
11774b88c807SRodney W. Grimes 		t_int = (int)(SHRT_EXT(hd->h_filesize_1));
11784b88c807SRodney W. Grimes 		t_int = (t_int << 16) | ((int)(SHRT_EXT(hd->h_filesize_2)));
11794b88c807SRodney W. Grimes 		if (arcn->ln_nlen != t_int)
11804b88c807SRodney W. Grimes 			goto out;
11814b88c807SRodney W. Grimes 		break;
11824b88c807SRodney W. Grimes 	default:
11834b88c807SRodney W. Grimes 		/*
11844b88c807SRodney W. Grimes 		 * no file data for the caller to process
11854b88c807SRodney W. Grimes 		 */
11864b88c807SRodney W. Grimes 		arcn->pad = 0L;
11874b88c807SRodney W. Grimes 		hd->h_filesize_1[0] = (char)0;
11884b88c807SRodney W. Grimes 		hd->h_filesize_1[1] = (char)0;
11894b88c807SRodney W. Grimes 		hd->h_filesize_2[0] = (char)0;
11904b88c807SRodney W. Grimes 		hd->h_filesize_2[1] = (char)0;
11914b88c807SRodney W. Grimes 		break;
11924b88c807SRodney W. Grimes 	}
11934b88c807SRodney W. Grimes 
11944b88c807SRodney W. Grimes 	/*
11954b88c807SRodney W. Grimes 	 * build up the rest of the fields
11964b88c807SRodney W. Grimes 	 */
11974b88c807SRodney W. Grimes 	hd->h_magic[0] = CHR_WR_2(MAGIC);
11984b88c807SRodney W. Grimes 	hd->h_magic[1] = CHR_WR_3(MAGIC);
11994b88c807SRodney W. Grimes 	hd->h_dev[0] = CHR_WR_2(arcn->sb.st_dev);
12004b88c807SRodney W. Grimes 	hd->h_dev[1] = CHR_WR_3(arcn->sb.st_dev);
12014b88c807SRodney W. Grimes 	if (arcn->sb.st_dev != (dev_t)(SHRT_EXT(hd->h_dev)))
12024b88c807SRodney W. Grimes 		goto out;
12034b88c807SRodney W. Grimes 	hd->h_ino[0] = CHR_WR_2(arcn->sb.st_ino);
12044b88c807SRodney W. Grimes 	hd->h_ino[1] = CHR_WR_3(arcn->sb.st_ino);
12054b88c807SRodney W. Grimes 	if (arcn->sb.st_ino != (ino_t)(SHRT_EXT(hd->h_ino)))
12064b88c807SRodney W. Grimes 		goto out;
12074b88c807SRodney W. Grimes 	hd->h_mode[0] = CHR_WR_2(arcn->sb.st_mode);
12084b88c807SRodney W. Grimes 	hd->h_mode[1] = CHR_WR_3(arcn->sb.st_mode);
12094b88c807SRodney W. Grimes 	if (arcn->sb.st_mode != (mode_t)(SHRT_EXT(hd->h_mode)))
12104b88c807SRodney W. Grimes 		goto out;
12114b88c807SRodney W. Grimes 	hd->h_uid[0] = CHR_WR_2(arcn->sb.st_uid);
12124b88c807SRodney W. Grimes 	hd->h_uid[1] = CHR_WR_3(arcn->sb.st_uid);
12134b88c807SRodney W. Grimes 	if (arcn->sb.st_uid != (uid_t)(SHRT_EXT(hd->h_uid)))
12144b88c807SRodney W. Grimes 		goto out;
12154b88c807SRodney W. Grimes 	hd->h_gid[0] = CHR_WR_2(arcn->sb.st_gid);
12164b88c807SRodney W. Grimes 	hd->h_gid[1] = CHR_WR_3(arcn->sb.st_gid);
12174b88c807SRodney W. Grimes 	if (arcn->sb.st_gid != (gid_t)(SHRT_EXT(hd->h_gid)))
12184b88c807SRodney W. Grimes 		goto out;
12194b88c807SRodney W. Grimes 	hd->h_nlink[0] = CHR_WR_2(arcn->sb.st_nlink);
12204b88c807SRodney W. Grimes 	hd->h_nlink[1] = CHR_WR_3(arcn->sb.st_nlink);
12214b88c807SRodney W. Grimes 	if (arcn->sb.st_nlink != (nlink_t)(SHRT_EXT(hd->h_nlink)))
12224b88c807SRodney W. Grimes 		goto out;
12234b88c807SRodney W. Grimes 	hd->h_rdev[0] = CHR_WR_2(arcn->sb.st_rdev);
12244b88c807SRodney W. Grimes 	hd->h_rdev[1] = CHR_WR_3(arcn->sb.st_rdev);
12254b88c807SRodney W. Grimes 	if (arcn->sb.st_rdev != (dev_t)(SHRT_EXT(hd->h_rdev)))
12264b88c807SRodney W. Grimes 		goto out;
12274b88c807SRodney W. Grimes 	hd->h_mtime_1[0] = CHR_WR_0(arcn->sb.st_mtime);
12284b88c807SRodney W. Grimes 	hd->h_mtime_1[1] = CHR_WR_1(arcn->sb.st_mtime);
12294b88c807SRodney W. Grimes 	hd->h_mtime_2[0] = CHR_WR_2(arcn->sb.st_mtime);
12304b88c807SRodney W. Grimes 	hd->h_mtime_2[1] = CHR_WR_3(arcn->sb.st_mtime);
12314b88c807SRodney W. Grimes 	t_timet = (time_t)(SHRT_EXT(hd->h_mtime_1));
12324b88c807SRodney W. Grimes 	t_timet =  (t_timet << 16) | ((time_t)(SHRT_EXT(hd->h_mtime_2)));
12334b88c807SRodney W. Grimes 	if (arcn->sb.st_mtime != t_timet)
12344b88c807SRodney W. Grimes 		goto out;
12354b88c807SRodney W. Grimes 	nsz = arcn->nlen + 1;
12364b88c807SRodney W. Grimes 	hd->h_namesize[0] = CHR_WR_2(nsz);
12374b88c807SRodney W. Grimes 	hd->h_namesize[1] = CHR_WR_3(nsz);
12384b88c807SRodney W. Grimes 	if (nsz != (int)(SHRT_EXT(hd->h_namesize)))
12394b88c807SRodney W. Grimes 		goto out;
12404b88c807SRodney W. Grimes 
12414b88c807SRodney W. Grimes 	/*
12424b88c807SRodney W. Grimes 	 * write the header, the file name and padding as required.
12434b88c807SRodney W. Grimes 	 */
12444b88c807SRodney W. Grimes 	if ((wr_rdbuf(hdblk, (int)sizeof(HD_BCPIO)) < 0) ||
12454b88c807SRodney W. Grimes 	    (wr_rdbuf(arcn->name, nsz) < 0) ||
12464b88c807SRodney W. Grimes 	    (wr_skip((off_t)(BCPIO_PAD(sizeof(HD_BCPIO) + nsz))) < 0)) {
1247a885d9dcSSøren Schmidt 		pax_warn(1, "Could not write bcpio header for %s", arcn->org_name);
12484b88c807SRodney W. Grimes 		return(-1);
12494b88c807SRodney W. Grimes 	}
12504b88c807SRodney W. Grimes 
12514b88c807SRodney W. Grimes 	/*
12524b88c807SRodney W. Grimes 	 * if we have file data, tell the caller we are done
12534b88c807SRodney W. Grimes 	 */
12544b88c807SRodney W. Grimes 	if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) ||
12554b88c807SRodney W. Grimes 	    (arcn->type == PAX_HRG))
12564b88c807SRodney W. Grimes 		return(0);
12574b88c807SRodney W. Grimes 
12584b88c807SRodney W. Grimes 	/*
12594b88c807SRodney W. Grimes 	 * if we are not a link, tell the caller we are done, go to next file
12604b88c807SRodney W. Grimes 	 */
12614b88c807SRodney W. Grimes 	if (arcn->type != PAX_SLK)
12624b88c807SRodney W. Grimes 		return(1);
12634b88c807SRodney W. Grimes 
12644b88c807SRodney W. Grimes 	/*
12654b88c807SRodney W. Grimes 	 * write the link name, tell the caller we are done.
12664b88c807SRodney W. Grimes 	 */
12674b88c807SRodney W. Grimes 	if ((wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) ||
12684b88c807SRodney W. Grimes 	    (wr_skip((off_t)(BCPIO_PAD(arcn->ln_nlen))) < 0)) {
1269a885d9dcSSøren Schmidt 		pax_warn(1,"Could not write bcpio link name for %s",arcn->org_name);
12704b88c807SRodney W. Grimes 		return(-1);
12714b88c807SRodney W. Grimes 	}
12724b88c807SRodney W. Grimes 	return(1);
12734b88c807SRodney W. Grimes 
12744b88c807SRodney W. Grimes     out:
12754b88c807SRodney W. Grimes 	/*
12764b88c807SRodney W. Grimes 	 * header field is out of range
12774b88c807SRodney W. Grimes 	 */
1278a885d9dcSSøren Schmidt 	pax_warn(1,"Bcpio header field is too small for file %s", arcn->org_name);
12794b88c807SRodney W. Grimes 	return(1);
12804b88c807SRodney W. Grimes }
1281