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 424b88c807SRodney W. Grimes #endif /* not lint */ 432749b141SDavid E. O'Brien #include <sys/cdefs.h> 442749b141SDavid E. O'Brien __FBSDID("$FreeBSD$"); 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> 505b94264cSTim J. Robbins #include <stdint.h> 514b88c807SRodney W. Grimes #include <stdio.h> 524b88c807SRodney W. Grimes #include <unistd.h> 534b88c807SRodney W. Grimes #include <stdlib.h> 544b88c807SRodney W. Grimes #include "pax.h" 554b88c807SRodney W. Grimes #include "cpio.h" 564b88c807SRodney W. Grimes #include "extern.h" 574b88c807SRodney W. Grimes 58f789b261SWarner Losh static int rd_nm(ARCHD *, int); 59f789b261SWarner Losh static int rd_ln_nm(ARCHD *); 60f789b261SWarner Losh static int com_rd(ARCHD *); 614b88c807SRodney W. Grimes 624b88c807SRodney W. Grimes /* 634b88c807SRodney W. Grimes * Routines which support the different cpio versions 644b88c807SRodney W. Grimes */ 654b88c807SRodney W. Grimes 664b88c807SRodney W. Grimes static int swp_head; /* binary cpio header byte swap */ 674b88c807SRodney W. Grimes 684b88c807SRodney W. Grimes /* 694b88c807SRodney W. Grimes * Routines common to all versions of cpio 704b88c807SRodney W. Grimes */ 714b88c807SRodney W. Grimes 724b88c807SRodney W. Grimes /* 734b88c807SRodney W. Grimes * cpio_strd() 744b88c807SRodney W. Grimes * Fire up the hard link detection code 754b88c807SRodney W. Grimes * Return: 764b88c807SRodney W. Grimes * 0 if ok -1 otherwise (the return values of lnk_start()) 774b88c807SRodney W. Grimes */ 784b88c807SRodney W. Grimes 794b88c807SRodney W. Grimes int 804b88c807SRodney W. Grimes cpio_strd(void) 814b88c807SRodney W. Grimes { 824b88c807SRodney W. Grimes return(lnk_start()); 834b88c807SRodney W. Grimes } 844b88c807SRodney W. Grimes 854b88c807SRodney W. Grimes /* 864b88c807SRodney W. Grimes * cpio_trail() 874b88c807SRodney W. Grimes * Called to determine if a header block is a valid trailer. We are 884b88c807SRodney W. Grimes * passed the block, the in_sync flag (which tells us we are in resync 894b88c807SRodney W. Grimes * mode; looking for a valid header), and cnt (which starts at zero) 904b88c807SRodney W. Grimes * which is used to count the number of empty blocks we have seen so far. 914b88c807SRodney W. Grimes * Return: 924b88c807SRodney W. Grimes * 0 if a valid trailer, -1 if not a valid trailer, 934b88c807SRodney W. Grimes */ 944b88c807SRodney W. Grimes 954b88c807SRodney W. Grimes int 96f789b261SWarner Losh cpio_trail(ARCHD *arcn) 974b88c807SRodney W. Grimes { 984b88c807SRodney W. Grimes /* 994b88c807SRodney W. Grimes * look for trailer id in file we are about to process 1004b88c807SRodney W. Grimes */ 1014b88c807SRodney W. Grimes if ((strcmp(arcn->name, TRAILER) == 0) && (arcn->sb.st_size == 0)) 1024b88c807SRodney W. Grimes return(0); 1034b88c807SRodney W. Grimes return(-1); 1044b88c807SRodney W. Grimes } 1054b88c807SRodney W. Grimes 1064b88c807SRodney W. Grimes /* 1074b88c807SRodney W. Grimes * com_rd() 1084b88c807SRodney W. Grimes * operations common to all cpio read functions. 1094b88c807SRodney W. Grimes * Return: 1104b88c807SRodney W. Grimes * 0 1114b88c807SRodney W. Grimes */ 1124b88c807SRodney W. Grimes 1134b88c807SRodney W. Grimes static int 114f789b261SWarner Losh com_rd(ARCHD *arcn) 1154b88c807SRodney W. Grimes { 1164b88c807SRodney W. Grimes arcn->skip = 0; 1174b88c807SRodney W. Grimes arcn->pat = NULL; 1184b88c807SRodney W. Grimes arcn->org_name = arcn->name; 1194b88c807SRodney W. Grimes switch(arcn->sb.st_mode & C_IFMT) { 1204b88c807SRodney W. Grimes case C_ISFIFO: 1214b88c807SRodney W. Grimes arcn->type = PAX_FIF; 1224b88c807SRodney W. Grimes break; 1234b88c807SRodney W. Grimes case C_ISDIR: 1244b88c807SRodney W. Grimes arcn->type = PAX_DIR; 1254b88c807SRodney W. Grimes break; 1264b88c807SRodney W. Grimes case C_ISBLK: 1274b88c807SRodney W. Grimes arcn->type = PAX_BLK; 1284b88c807SRodney W. Grimes break; 1294b88c807SRodney W. Grimes case C_ISCHR: 1304b88c807SRodney W. Grimes arcn->type = PAX_CHR; 1314b88c807SRodney W. Grimes break; 1324b88c807SRodney W. Grimes case C_ISLNK: 1334b88c807SRodney W. Grimes arcn->type = PAX_SLK; 1344b88c807SRodney W. Grimes break; 1354b88c807SRodney W. Grimes case C_ISOCK: 1364b88c807SRodney W. Grimes arcn->type = PAX_SCK; 1374b88c807SRodney W. Grimes break; 1384b88c807SRodney W. Grimes case C_ISCTG: 1394b88c807SRodney W. Grimes case C_ISREG: 1404b88c807SRodney W. Grimes default: 1414b88c807SRodney W. Grimes /* 1424b88c807SRodney W. Grimes * we have file data, set up skip (pad is set in the format 1434b88c807SRodney W. Grimes * specific sections) 1444b88c807SRodney W. Grimes */ 1454b88c807SRodney W. Grimes arcn->sb.st_mode = (arcn->sb.st_mode & 0xfff) | C_ISREG; 1464b88c807SRodney W. Grimes arcn->type = PAX_REG; 1474b88c807SRodney W. Grimes arcn->skip = arcn->sb.st_size; 1484b88c807SRodney W. Grimes break; 1494b88c807SRodney W. Grimes } 1504b88c807SRodney W. Grimes if (chk_lnk(arcn) < 0) 1514b88c807SRodney W. Grimes return(-1); 1524b88c807SRodney W. Grimes return(0); 1534b88c807SRodney W. Grimes } 1544b88c807SRodney W. Grimes 1554b88c807SRodney W. Grimes /* 1564b88c807SRodney W. Grimes * cpio_end_wr() 1574b88c807SRodney W. Grimes * write the special file with the name trailer in the proper format 1584b88c807SRodney W. Grimes * Return: 1594b88c807SRodney W. Grimes * result of the write of the trailer from the cpio specific write func 1604b88c807SRodney W. Grimes */ 1614b88c807SRodney W. Grimes 1624b88c807SRodney W. Grimes int 1634b88c807SRodney W. Grimes cpio_endwr(void) 1644b88c807SRodney W. Grimes { 1654b88c807SRodney W. Grimes ARCHD last; 1664b88c807SRodney W. Grimes 1674b88c807SRodney W. Grimes /* 1684b88c807SRodney W. Grimes * create a trailer request and call the proper format write function 1694b88c807SRodney W. Grimes */ 170778766feSKris Kennaway memset(&last, 0, sizeof(last)); 1714b88c807SRodney W. Grimes last.nlen = sizeof(TRAILER) - 1; 1724b88c807SRodney W. Grimes last.type = PAX_REG; 1734b88c807SRodney W. Grimes last.sb.st_nlink = 1; 1744b88c807SRodney W. Grimes (void)strcpy(last.name, TRAILER); 1754b88c807SRodney W. Grimes return((*frmt->wr)(&last)); 1764b88c807SRodney W. Grimes } 1774b88c807SRodney W. Grimes 1784b88c807SRodney W. Grimes /* 1794b88c807SRodney W. Grimes * rd_nam() 1804b88c807SRodney W. Grimes * read in the file name which follows the cpio header 1814b88c807SRodney W. Grimes * Return: 1824b88c807SRodney W. Grimes * 0 if ok, -1 otherwise 1834b88c807SRodney W. Grimes */ 1844b88c807SRodney W. Grimes 1854b88c807SRodney W. Grimes static int 186f789b261SWarner Losh rd_nm(ARCHD *arcn, int nsz) 1874b88c807SRodney W. Grimes { 1884b88c807SRodney W. Grimes /* 1894b88c807SRodney W. Grimes * do not even try bogus values 1904b88c807SRodney W. Grimes */ 191cee22cbdSDavid E. O'Brien if ((nsz == 0) || (nsz > (int)sizeof(arcn->name))) { 192778766feSKris Kennaway paxwarn(1, "Cpio file name length %d is out of range", nsz); 1934b88c807SRodney W. Grimes return(-1); 1944b88c807SRodney W. Grimes } 1954b88c807SRodney W. Grimes 1964b88c807SRodney W. Grimes /* 1974b88c807SRodney W. Grimes * read the name and make sure it is not empty and is \0 terminated 1984b88c807SRodney W. Grimes */ 1994b88c807SRodney W. Grimes if ((rd_wrbuf(arcn->name,nsz) != nsz) || (arcn->name[nsz-1] != '\0') || 2004b88c807SRodney W. Grimes (arcn->name[0] == '\0')) { 201778766feSKris Kennaway paxwarn(1, "Cpio file name in header is corrupted"); 2024b88c807SRodney W. Grimes return(-1); 2034b88c807SRodney W. Grimes } 2044b88c807SRodney W. Grimes return(0); 2054b88c807SRodney W. Grimes } 2064b88c807SRodney W. Grimes 2074b88c807SRodney W. Grimes /* 2084b88c807SRodney W. Grimes * rd_ln_nm() 2094b88c807SRodney W. Grimes * read in the link name for a file with links. The link name is stored 2104b88c807SRodney W. Grimes * like file data (and is NOT \0 terminated!) 2114b88c807SRodney W. Grimes * Return: 2124b88c807SRodney W. Grimes * 0 if ok, -1 otherwise 2134b88c807SRodney W. Grimes */ 2144b88c807SRodney W. Grimes 2154b88c807SRodney W. Grimes static int 216f789b261SWarner Losh rd_ln_nm(ARCHD *arcn) 2174b88c807SRodney W. Grimes { 2184b88c807SRodney W. Grimes /* 2194b88c807SRodney W. Grimes * check the length specified for bogus values 2204b88c807SRodney W. Grimes */ 2214b88c807SRodney W. Grimes if ((arcn->sb.st_size == 0) || 2224b88c807SRodney W. Grimes (arcn->sb.st_size >= sizeof(arcn->ln_name))) { 2234b88c807SRodney W. Grimes # ifdef NET2_STAT 224778766feSKris Kennaway paxwarn(1, "Cpio link name length is invalid: %lu", 2254b88c807SRodney W. Grimes arcn->sb.st_size); 2264b88c807SRodney W. Grimes # else 2275b94264cSTim J. Robbins paxwarn(1, "Cpio link name length is invalid: %ju", 2285b94264cSTim J. Robbins (uintmax_t)arcn->sb.st_size); 2294b88c807SRodney W. Grimes # endif 2304b88c807SRodney W. Grimes return(-1); 2314b88c807SRodney W. Grimes } 2324b88c807SRodney W. Grimes 2334b88c807SRodney W. Grimes /* 2344b88c807SRodney W. Grimes * read in the link name and \0 terminate it 2354b88c807SRodney W. Grimes */ 2364b88c807SRodney W. Grimes if (rd_wrbuf(arcn->ln_name, (int)arcn->sb.st_size) != 2374b88c807SRodney W. Grimes (int)arcn->sb.st_size) { 238778766feSKris Kennaway paxwarn(1, "Cpio link name read error"); 2394b88c807SRodney W. Grimes return(-1); 2404b88c807SRodney W. Grimes } 2414b88c807SRodney W. Grimes arcn->ln_nlen = arcn->sb.st_size; 2424b88c807SRodney W. Grimes arcn->ln_name[arcn->ln_nlen] = '\0'; 2434b88c807SRodney W. Grimes 2444b88c807SRodney W. Grimes /* 2454b88c807SRodney W. Grimes * watch out for those empty link names 2464b88c807SRodney W. Grimes */ 2474b88c807SRodney W. Grimes if (arcn->ln_name[0] == '\0') { 248778766feSKris Kennaway paxwarn(1, "Cpio link name is corrupt"); 2494b88c807SRodney W. Grimes return(-1); 2504b88c807SRodney W. Grimes } 2514b88c807SRodney W. Grimes return(0); 2524b88c807SRodney W. Grimes } 2534b88c807SRodney W. Grimes 2544b88c807SRodney W. Grimes /* 2554b88c807SRodney W. Grimes * Routines common to the extended byte oriented cpio format 2564b88c807SRodney W. Grimes */ 2574b88c807SRodney W. Grimes 2584b88c807SRodney W. Grimes /* 2594b88c807SRodney W. Grimes * cpio_id() 2604b88c807SRodney W. Grimes * determine if a block given to us is a valid extended byte oriented 2614b88c807SRodney W. Grimes * cpio header 2624b88c807SRodney W. Grimes * Return: 2634b88c807SRodney W. Grimes * 0 if a valid header, -1 otherwise 2644b88c807SRodney W. Grimes */ 2654b88c807SRodney W. Grimes 2664b88c807SRodney W. Grimes int 2674b88c807SRodney W. Grimes cpio_id(char *blk, int size) 2684b88c807SRodney W. Grimes { 269cee22cbdSDavid E. O'Brien if ((size < (int)sizeof(HD_CPIO)) || 2704b88c807SRodney W. Grimes (strncmp(blk, AMAGIC, sizeof(AMAGIC) - 1) != 0)) 2714b88c807SRodney W. Grimes return(-1); 2724b88c807SRodney W. Grimes return(0); 2734b88c807SRodney W. Grimes } 2744b88c807SRodney W. Grimes 2754b88c807SRodney W. Grimes /* 2764b88c807SRodney W. Grimes * cpio_rd() 2774b88c807SRodney W. Grimes * determine if a buffer is a byte oriented extended cpio archive entry. 2784b88c807SRodney W. Grimes * convert and store the values in the ARCHD parameter. 2794b88c807SRodney W. Grimes * Return: 2804b88c807SRodney W. Grimes * 0 if a valid header, -1 otherwise. 2814b88c807SRodney W. Grimes */ 2824b88c807SRodney W. Grimes 2834b88c807SRodney W. Grimes int 284f789b261SWarner Losh cpio_rd(ARCHD *arcn, char *buf) 2854b88c807SRodney W. Grimes { 286f789b261SWarner Losh int nsz; 287f789b261SWarner Losh HD_CPIO *hd; 2884b88c807SRodney W. Grimes 2894b88c807SRodney W. Grimes /* 2904b88c807SRodney W. Grimes * check that this is a valid header, if not return -1 2914b88c807SRodney W. Grimes */ 2924b88c807SRodney W. Grimes if (cpio_id(buf, sizeof(HD_CPIO)) < 0) 2934b88c807SRodney W. Grimes return(-1); 2944b88c807SRodney W. Grimes hd = (HD_CPIO *)buf; 2954b88c807SRodney W. Grimes 2964b88c807SRodney W. Grimes /* 2974b88c807SRodney W. Grimes * byte oriented cpio (posix) does not have padding! extract the octal 2984b88c807SRodney W. Grimes * ascii fields from the header 2994b88c807SRodney W. Grimes */ 3004b88c807SRodney W. Grimes arcn->pad = 0L; 3014b88c807SRodney W. Grimes arcn->sb.st_dev = (dev_t)asc_ul(hd->c_dev, sizeof(hd->c_dev), OCT); 3024b88c807SRodney W. Grimes arcn->sb.st_ino = (ino_t)asc_ul(hd->c_ino, sizeof(hd->c_ino), OCT); 3034b88c807SRodney W. Grimes arcn->sb.st_mode = (mode_t)asc_ul(hd->c_mode, sizeof(hd->c_mode), OCT); 3044b88c807SRodney W. Grimes arcn->sb.st_uid = (uid_t)asc_ul(hd->c_uid, sizeof(hd->c_uid), OCT); 3054b88c807SRodney W. Grimes arcn->sb.st_gid = (gid_t)asc_ul(hd->c_gid, sizeof(hd->c_gid), OCT); 3064b88c807SRodney W. Grimes arcn->sb.st_nlink = (nlink_t)asc_ul(hd->c_nlink, sizeof(hd->c_nlink), 3074b88c807SRodney W. Grimes OCT); 3084b88c807SRodney W. Grimes arcn->sb.st_rdev = (dev_t)asc_ul(hd->c_rdev, sizeof(hd->c_rdev), OCT); 3092ad45bbeSMatthew Dillon #ifdef NET2_STAT 3104b88c807SRodney W. Grimes arcn->sb.st_mtime = (time_t)asc_ul(hd->c_mtime, sizeof(hd->c_mtime), 3114b88c807SRodney W. Grimes OCT); 3122ad45bbeSMatthew Dillon #else 3132ad45bbeSMatthew Dillon arcn->sb.st_mtime = (time_t)asc_uqd(hd->c_mtime, sizeof(hd->c_mtime), 3142ad45bbeSMatthew Dillon OCT); 3152ad45bbeSMatthew Dillon #endif 3164b88c807SRodney W. Grimes arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime; 3174b88c807SRodney W. Grimes #ifdef NET2_STAT 3184b88c807SRodney W. Grimes arcn->sb.st_size = (off_t)asc_ul(hd->c_filesize,sizeof(hd->c_filesize), 3194b88c807SRodney W. Grimes OCT); 3204b88c807SRodney W. Grimes #else 3214b88c807SRodney W. Grimes arcn->sb.st_size = (off_t)asc_uqd(hd->c_filesize,sizeof(hd->c_filesize), 3224b88c807SRodney W. Grimes OCT); 3234b88c807SRodney W. Grimes #endif 3244b88c807SRodney W. Grimes 3254b88c807SRodney W. Grimes /* 3264b88c807SRodney W. Grimes * check name size and if valid, read in the name of this entry (name 3274b88c807SRodney W. Grimes * follows header in the archive) 3284b88c807SRodney W. Grimes */ 3294b88c807SRodney W. Grimes if ((nsz = (int)asc_ul(hd->c_namesize,sizeof(hd->c_namesize),OCT)) < 2) 3304b88c807SRodney W. Grimes return(-1); 3314b88c807SRodney W. Grimes arcn->nlen = nsz - 1; 3324b88c807SRodney W. Grimes if (rd_nm(arcn, nsz) < 0) 3334b88c807SRodney W. Grimes return(-1); 3344b88c807SRodney W. Grimes 3354b88c807SRodney W. Grimes if (((arcn->sb.st_mode&C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)) { 3364b88c807SRodney W. Grimes /* 3374b88c807SRodney W. Grimes * no link name to read for this file 3384b88c807SRodney W. Grimes */ 3394b88c807SRodney W. Grimes arcn->ln_nlen = 0; 3404b88c807SRodney W. Grimes arcn->ln_name[0] = '\0'; 3414b88c807SRodney W. Grimes return(com_rd(arcn)); 3424b88c807SRodney W. Grimes } 3434b88c807SRodney W. Grimes 3444b88c807SRodney W. Grimes /* 3454b88c807SRodney W. Grimes * check link name size and read in the link name. Link names are 3464b88c807SRodney W. Grimes * stored like file data. 3474b88c807SRodney W. Grimes */ 3484b88c807SRodney W. Grimes if (rd_ln_nm(arcn) < 0) 3494b88c807SRodney W. Grimes return(-1); 3504b88c807SRodney W. Grimes 3514b88c807SRodney W. Grimes /* 3524b88c807SRodney W. Grimes * we have a valid header (with a link) 3534b88c807SRodney W. Grimes */ 3544b88c807SRodney W. Grimes return(com_rd(arcn)); 3554b88c807SRodney W. Grimes } 3564b88c807SRodney W. Grimes 3574b88c807SRodney W. Grimes /* 3584b88c807SRodney W. Grimes * cpio_endrd() 3594b88c807SRodney W. Grimes * no cleanup needed here, just return size of the trailer (for append) 3604b88c807SRodney W. Grimes * Return: 3614b88c807SRodney W. Grimes * size of trailer header in this format 3624b88c807SRodney W. Grimes */ 3634b88c807SRodney W. Grimes 3644b88c807SRodney W. Grimes off_t 3654b88c807SRodney W. Grimes cpio_endrd(void) 3664b88c807SRodney W. Grimes { 3674b88c807SRodney W. Grimes return((off_t)(sizeof(HD_CPIO) + sizeof(TRAILER))); 3684b88c807SRodney W. Grimes } 3694b88c807SRodney W. Grimes 3704b88c807SRodney W. Grimes /* 3714b88c807SRodney W. Grimes * cpio_stwr() 3724b88c807SRodney W. Grimes * start up the device mapping table 3734b88c807SRodney W. Grimes * Return: 3744b88c807SRodney W. Grimes * 0 if ok, -1 otherwise (what dev_start() returns) 3754b88c807SRodney W. Grimes */ 3764b88c807SRodney W. Grimes 3774b88c807SRodney W. Grimes int 3784b88c807SRodney W. Grimes cpio_stwr(void) 3794b88c807SRodney W. Grimes { 3804b88c807SRodney W. Grimes return(dev_start()); 3814b88c807SRodney W. Grimes } 3824b88c807SRodney W. Grimes 3834b88c807SRodney W. Grimes /* 3844b88c807SRodney W. Grimes * cpio_wr() 3854b88c807SRodney W. Grimes * copy the data in the ARCHD to buffer in extended byte oriented cpio 3864b88c807SRodney W. Grimes * format. 3874b88c807SRodney W. Grimes * Return 3884b88c807SRodney W. Grimes * 0 if file has data to be written after the header, 1 if file has NO 3894b88c807SRodney W. Grimes * data to write after the header, -1 if archive write failed 3904b88c807SRodney W. Grimes */ 3914b88c807SRodney W. Grimes 3924b88c807SRodney W. Grimes int 393f789b261SWarner Losh cpio_wr(ARCHD *arcn) 3944b88c807SRodney W. Grimes { 395f789b261SWarner Losh HD_CPIO *hd; 396f789b261SWarner Losh int nsz; 3974b88c807SRodney W. Grimes char hdblk[sizeof(HD_CPIO)]; 3984b88c807SRodney W. Grimes 3994b88c807SRodney W. Grimes /* 4004b88c807SRodney W. Grimes * check and repair truncated device and inode fields in the header 4014b88c807SRodney W. Grimes */ 4024b88c807SRodney W. Grimes if (map_dev(arcn, (u_long)CPIO_MASK, (u_long)CPIO_MASK) < 0) 4034b88c807SRodney W. Grimes return(-1); 4044b88c807SRodney W. Grimes 4054b88c807SRodney W. Grimes arcn->pad = 0L; 4064b88c807SRodney W. Grimes nsz = arcn->nlen + 1; 4074b88c807SRodney W. Grimes hd = (HD_CPIO *)hdblk; 4084b88c807SRodney W. Grimes if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR)) 4094b88c807SRodney W. Grimes arcn->sb.st_rdev = 0; 4104b88c807SRodney W. Grimes 4114b88c807SRodney W. Grimes switch(arcn->type) { 4124b88c807SRodney W. Grimes case PAX_CTG: 4134b88c807SRodney W. Grimes case PAX_REG: 4144b88c807SRodney W. Grimes case PAX_HRG: 4154b88c807SRodney W. Grimes /* 4164b88c807SRodney W. Grimes * set data size for file data 4174b88c807SRodney W. Grimes */ 4184b88c807SRodney W. Grimes # ifdef NET2_STAT 4194b88c807SRodney W. Grimes if (ul_asc((u_long)arcn->sb.st_size, hd->c_filesize, 4204b88c807SRodney W. Grimes sizeof(hd->c_filesize), OCT)) { 4214b88c807SRodney W. Grimes # else 4224b88c807SRodney W. Grimes if (uqd_asc((u_quad_t)arcn->sb.st_size, hd->c_filesize, 4234b88c807SRodney W. Grimes sizeof(hd->c_filesize), OCT)) { 4244b88c807SRodney W. Grimes # endif 425778766feSKris Kennaway paxwarn(1,"File is too large for cpio format %s", 4264b88c807SRodney W. Grimes arcn->org_name); 4274b88c807SRodney W. Grimes return(1); 4284b88c807SRodney W. Grimes } 4294b88c807SRodney W. Grimes break; 4304b88c807SRodney W. Grimes case PAX_SLK: 4314b88c807SRodney W. Grimes /* 4324b88c807SRodney W. Grimes * set data size to hold link name 4334b88c807SRodney W. Grimes */ 4344b88c807SRodney W. Grimes if (ul_asc((u_long)arcn->ln_nlen, hd->c_filesize, 4354b88c807SRodney W. Grimes sizeof(hd->c_filesize), OCT)) 4364b88c807SRodney W. Grimes goto out; 4374b88c807SRodney W. Grimes break; 4384b88c807SRodney W. Grimes default: 4394b88c807SRodney W. Grimes /* 4404b88c807SRodney W. Grimes * all other file types have no file data 4414b88c807SRodney W. Grimes */ 4424b88c807SRodney W. Grimes if (ul_asc((u_long)0, hd->c_filesize, sizeof(hd->c_filesize), 4434b88c807SRodney W. Grimes OCT)) 4444b88c807SRodney W. Grimes goto out; 4454b88c807SRodney W. Grimes break; 4464b88c807SRodney W. Grimes } 4474b88c807SRodney W. Grimes 4484b88c807SRodney W. Grimes /* 4494b88c807SRodney W. Grimes * copy the values to the header using octal ascii 4504b88c807SRodney W. Grimes */ 4514b88c807SRodney W. Grimes if (ul_asc((u_long)MAGIC, hd->c_magic, sizeof(hd->c_magic), OCT) || 4524b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_dev, hd->c_dev, sizeof(hd->c_dev), 4534b88c807SRodney W. Grimes OCT) || 4544b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_ino, hd->c_ino, sizeof(hd->c_ino), 4554b88c807SRodney W. Grimes OCT) || 4564b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_mode, hd->c_mode, sizeof(hd->c_mode), 4574b88c807SRodney W. Grimes OCT) || 4584b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_uid, hd->c_uid, sizeof(hd->c_uid), 4594b88c807SRodney W. Grimes OCT) || 4604b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_gid, hd->c_gid, sizeof(hd->c_gid), 4614b88c807SRodney W. Grimes OCT) || 4624b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_nlink, hd->c_nlink, sizeof(hd->c_nlink), 4634b88c807SRodney W. Grimes OCT) || 4644b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_rdev, hd->c_rdev, sizeof(hd->c_rdev), 4654b88c807SRodney W. Grimes OCT) || 4664b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_mtime,hd->c_mtime,sizeof(hd->c_mtime), 4674b88c807SRodney W. Grimes OCT) || 4684b88c807SRodney W. Grimes ul_asc((u_long)nsz, hd->c_namesize, sizeof(hd->c_namesize), OCT)) 4694b88c807SRodney W. Grimes goto out; 4704b88c807SRodney W. Grimes 4714b88c807SRodney W. Grimes /* 4724b88c807SRodney W. Grimes * write the file name to the archive 4734b88c807SRodney W. Grimes */ 4744b88c807SRodney W. Grimes if ((wr_rdbuf(hdblk, (int)sizeof(HD_CPIO)) < 0) || 4754b88c807SRodney W. Grimes (wr_rdbuf(arcn->name, nsz) < 0)) { 476778766feSKris Kennaway paxwarn(1, "Unable to write cpio header for %s", arcn->org_name); 4774b88c807SRodney W. Grimes return(-1); 4784b88c807SRodney W. Grimes } 4794b88c807SRodney W. Grimes 4804b88c807SRodney W. Grimes /* 4814b88c807SRodney W. Grimes * if this file has data, we are done. The caller will write the file 4824b88c807SRodney W. Grimes * data, if we are link tell caller we are done, go to next file 4834b88c807SRodney W. Grimes */ 4844b88c807SRodney W. Grimes if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) || 4854b88c807SRodney W. Grimes (arcn->type == PAX_HRG)) 4864b88c807SRodney W. Grimes return(0); 4874b88c807SRodney W. Grimes if (arcn->type != PAX_SLK) 4884b88c807SRodney W. Grimes return(1); 4894b88c807SRodney W. Grimes 4904b88c807SRodney W. Grimes /* 4914b88c807SRodney W. Grimes * write the link name to the archive, tell the caller to go to the 4924b88c807SRodney W. Grimes * next file as we are done. 4934b88c807SRodney W. Grimes */ 4944b88c807SRodney W. Grimes if (wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) { 495778766feSKris Kennaway paxwarn(1,"Unable to write cpio link name for %s",arcn->org_name); 4964b88c807SRodney W. Grimes return(-1); 4974b88c807SRodney W. Grimes } 4984b88c807SRodney W. Grimes return(1); 4994b88c807SRodney W. Grimes 5004b88c807SRodney W. Grimes out: 5014b88c807SRodney W. Grimes /* 5024b88c807SRodney W. Grimes * header field is out of range 5034b88c807SRodney W. Grimes */ 504778766feSKris Kennaway paxwarn(1, "Cpio header field is too small to store file %s", 5054b88c807SRodney W. Grimes arcn->org_name); 5064b88c807SRodney W. Grimes return(1); 5074b88c807SRodney W. Grimes } 5084b88c807SRodney W. Grimes 5094b88c807SRodney W. Grimes /* 5104b88c807SRodney W. Grimes * Routines common to the system VR4 version of cpio (with/without file CRC) 5114b88c807SRodney W. Grimes */ 5124b88c807SRodney W. Grimes 5134b88c807SRodney W. Grimes /* 5144b88c807SRodney W. Grimes * vcpio_id() 5154b88c807SRodney W. Grimes * determine if a block given to us is a valid system VR4 cpio header 5164b88c807SRodney W. Grimes * WITHOUT crc. WATCH it the magic cookies are in OCTAL, the header 5174b88c807SRodney W. Grimes * uses HEX 5184b88c807SRodney W. Grimes * Return: 5194b88c807SRodney W. Grimes * 0 if a valid header, -1 otherwise 5204b88c807SRodney W. Grimes */ 5214b88c807SRodney W. Grimes 5224b88c807SRodney W. Grimes int 5234b88c807SRodney W. Grimes vcpio_id(char *blk, int size) 5244b88c807SRodney W. Grimes { 525cee22cbdSDavid E. O'Brien if ((size < (int)sizeof(HD_VCPIO)) || 5264b88c807SRodney W. Grimes (strncmp(blk, AVMAGIC, sizeof(AVMAGIC) - 1) != 0)) 5274b88c807SRodney W. Grimes return(-1); 5284b88c807SRodney W. Grimes return(0); 5294b88c807SRodney W. Grimes } 5304b88c807SRodney W. Grimes 5314b88c807SRodney W. Grimes /* 5324b88c807SRodney W. Grimes * crc_id() 5334b88c807SRodney W. Grimes * determine if a block given to us is a valid system VR4 cpio header 5344b88c807SRodney W. Grimes * WITH crc. WATCH it the magic cookies are in OCTAL the header uses HEX 5354b88c807SRodney W. Grimes * Return: 5364b88c807SRodney W. Grimes * 0 if a valid header, -1 otherwise 5374b88c807SRodney W. Grimes */ 5384b88c807SRodney W. Grimes 5394b88c807SRodney W. Grimes int 5404b88c807SRodney W. Grimes crc_id(char *blk, int size) 5414b88c807SRodney W. Grimes { 542cee22cbdSDavid E. O'Brien if ((size < (int)sizeof(HD_VCPIO)) || 543cee22cbdSDavid E. O'Brien (strncmp(blk, AVCMAGIC, (int)sizeof(AVCMAGIC) - 1) != 0)) 5444b88c807SRodney W. Grimes return(-1); 5454b88c807SRodney W. Grimes return(0); 5464b88c807SRodney W. Grimes } 5474b88c807SRodney W. Grimes 5484b88c807SRodney W. Grimes /* 5494b88c807SRodney W. Grimes * crc_strd() 5504b88c807SRodney W. Grimes w set file data CRC calculations. Fire up the hard link detection code 5514b88c807SRodney W. Grimes * Return: 5524b88c807SRodney W. Grimes * 0 if ok -1 otherwise (the return values of lnk_start()) 5534b88c807SRodney W. Grimes */ 5544b88c807SRodney W. Grimes 5554b88c807SRodney W. Grimes int 5564b88c807SRodney W. Grimes crc_strd(void) 5574b88c807SRodney W. Grimes { 5584b88c807SRodney W. Grimes docrc = 1; 5594b88c807SRodney W. Grimes return(lnk_start()); 5604b88c807SRodney W. Grimes } 5614b88c807SRodney W. Grimes 5624b88c807SRodney W. Grimes /* 5634b88c807SRodney W. Grimes * vcpio_rd() 5644b88c807SRodney W. Grimes * determine if a buffer is a system VR4 archive entry. (with/without CRC) 5654b88c807SRodney W. Grimes * convert and store the values in the ARCHD parameter. 5664b88c807SRodney W. Grimes * Return: 5674b88c807SRodney W. Grimes * 0 if a valid header, -1 otherwise. 5684b88c807SRodney W. Grimes */ 5694b88c807SRodney W. Grimes 5704b88c807SRodney W. Grimes int 571f789b261SWarner Losh vcpio_rd(ARCHD *arcn, char *buf) 5724b88c807SRodney W. Grimes { 573f789b261SWarner Losh HD_VCPIO *hd; 5744b88c807SRodney W. Grimes dev_t devminor; 5754b88c807SRodney W. Grimes dev_t devmajor; 576f789b261SWarner Losh int nsz; 5774b88c807SRodney W. Grimes 5784b88c807SRodney W. Grimes /* 5794b88c807SRodney W. Grimes * during the id phase it was determined if we were using CRC, use the 5804b88c807SRodney W. Grimes * proper id routine. 5814b88c807SRodney W. Grimes */ 5824b88c807SRodney W. Grimes if (docrc) { 5834b88c807SRodney W. Grimes if (crc_id(buf, sizeof(HD_VCPIO)) < 0) 5844b88c807SRodney W. Grimes return(-1); 5854b88c807SRodney W. Grimes } else { 5864b88c807SRodney W. Grimes if (vcpio_id(buf, sizeof(HD_VCPIO)) < 0) 5874b88c807SRodney W. Grimes return(-1); 5884b88c807SRodney W. Grimes } 5894b88c807SRodney W. Grimes 5904b88c807SRodney W. Grimes hd = (HD_VCPIO *)buf; 5914b88c807SRodney W. Grimes arcn->pad = 0L; 5924b88c807SRodney W. Grimes 5934b88c807SRodney W. Grimes /* 5944b88c807SRodney W. Grimes * extract the hex ascii fields from the header 5954b88c807SRodney W. Grimes */ 5964b88c807SRodney W. Grimes arcn->sb.st_ino = (ino_t)asc_ul(hd->c_ino, sizeof(hd->c_ino), HEX); 5974b88c807SRodney W. Grimes arcn->sb.st_mode = (mode_t)asc_ul(hd->c_mode, sizeof(hd->c_mode), HEX); 5984b88c807SRodney W. Grimes arcn->sb.st_uid = (uid_t)asc_ul(hd->c_uid, sizeof(hd->c_uid), HEX); 5994b88c807SRodney W. Grimes arcn->sb.st_gid = (gid_t)asc_ul(hd->c_gid, sizeof(hd->c_gid), HEX); 6002ad45bbeSMatthew Dillon #ifdef NET2_STAT 6014b88c807SRodney W. Grimes arcn->sb.st_mtime = (time_t)asc_ul(hd->c_mtime,sizeof(hd->c_mtime),HEX); 6022ad45bbeSMatthew Dillon #else 6032ad45bbeSMatthew Dillon arcn->sb.st_mtime = (time_t)asc_uqd(hd->c_mtime,sizeof(hd->c_mtime),HEX); 6042ad45bbeSMatthew Dillon #endif 6054b88c807SRodney W. Grimes arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime; 6064b88c807SRodney W. Grimes #ifdef NET2_STAT 6074b88c807SRodney W. Grimes arcn->sb.st_size = (off_t)asc_ul(hd->c_filesize, 6084b88c807SRodney W. Grimes sizeof(hd->c_filesize), HEX); 6094b88c807SRodney W. Grimes #else 6104b88c807SRodney W. Grimes arcn->sb.st_size = (off_t)asc_uqd(hd->c_filesize, 6114b88c807SRodney W. Grimes sizeof(hd->c_filesize), HEX); 6124b88c807SRodney W. Grimes #endif 6134b88c807SRodney W. Grimes arcn->sb.st_nlink = (nlink_t)asc_ul(hd->c_nlink, sizeof(hd->c_nlink), 6144b88c807SRodney W. Grimes HEX); 6154b88c807SRodney W. Grimes devmajor = (dev_t)asc_ul(hd->c_maj, sizeof(hd->c_maj), HEX); 6164b88c807SRodney W. Grimes devminor = (dev_t)asc_ul(hd->c_min, sizeof(hd->c_min), HEX); 6174b88c807SRodney W. Grimes arcn->sb.st_dev = TODEV(devmajor, devminor); 6184b88c807SRodney W. Grimes devmajor = (dev_t)asc_ul(hd->c_rmaj, sizeof(hd->c_maj), HEX); 6194b88c807SRodney W. Grimes devminor = (dev_t)asc_ul(hd->c_rmin, sizeof(hd->c_min), HEX); 6204b88c807SRodney W. Grimes arcn->sb.st_rdev = TODEV(devmajor, devminor); 6214b88c807SRodney W. Grimes arcn->crc = asc_ul(hd->c_chksum, sizeof(hd->c_chksum), HEX); 6224b88c807SRodney W. Grimes 6234b88c807SRodney W. Grimes /* 6244b88c807SRodney W. Grimes * check the length of the file name, if ok read it in, return -1 if 6254b88c807SRodney W. Grimes * bogus 6264b88c807SRodney W. Grimes */ 6274b88c807SRodney W. Grimes if ((nsz = (int)asc_ul(hd->c_namesize,sizeof(hd->c_namesize),HEX)) < 2) 6284b88c807SRodney W. Grimes return(-1); 6294b88c807SRodney W. Grimes arcn->nlen = nsz - 1; 6304b88c807SRodney W. Grimes if (rd_nm(arcn, nsz) < 0) 6314b88c807SRodney W. Grimes return(-1); 6324b88c807SRodney W. Grimes 6334b88c807SRodney W. Grimes /* 6344b88c807SRodney W. Grimes * skip padding. header + filename is aligned to 4 byte boundries 6354b88c807SRodney W. Grimes */ 6364b88c807SRodney W. Grimes if (rd_skip((off_t)(VCPIO_PAD(sizeof(HD_VCPIO) + nsz))) < 0) 6374b88c807SRodney W. Grimes return(-1); 6384b88c807SRodney W. Grimes 6394b88c807SRodney W. Grimes /* 6404b88c807SRodney W. Grimes * if not a link (or a file with no data), calculate pad size (for 6414b88c807SRodney W. Grimes * padding which follows the file data), clear the link name and return 6424b88c807SRodney W. Grimes */ 6434b88c807SRodney W. Grimes if (((arcn->sb.st_mode&C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)) { 6444b88c807SRodney W. Grimes /* 6454b88c807SRodney W. Grimes * we have a valid header (not a link) 6464b88c807SRodney W. Grimes */ 6474b88c807SRodney W. Grimes arcn->ln_nlen = 0; 6484b88c807SRodney W. Grimes arcn->ln_name[0] = '\0'; 6494b88c807SRodney W. Grimes arcn->pad = VCPIO_PAD(arcn->sb.st_size); 6504b88c807SRodney W. Grimes return(com_rd(arcn)); 6514b88c807SRodney W. Grimes } 6524b88c807SRodney W. Grimes 6534b88c807SRodney W. Grimes /* 6544b88c807SRodney W. Grimes * read in the link name and skip over the padding 6554b88c807SRodney W. Grimes */ 6564b88c807SRodney W. Grimes if ((rd_ln_nm(arcn) < 0) || 6574b88c807SRodney W. Grimes (rd_skip((off_t)(VCPIO_PAD(arcn->sb.st_size))) < 0)) 6584b88c807SRodney W. Grimes return(-1); 6594b88c807SRodney W. Grimes 6604b88c807SRodney W. Grimes /* 6614b88c807SRodney W. Grimes * we have a valid header (with a link) 6624b88c807SRodney W. Grimes */ 6634b88c807SRodney W. Grimes return(com_rd(arcn)); 6644b88c807SRodney W. Grimes } 6654b88c807SRodney W. Grimes 6664b88c807SRodney W. Grimes /* 6674b88c807SRodney W. Grimes * vcpio_endrd() 6684b88c807SRodney W. Grimes * no cleanup needed here, just return size of the trailer (for append) 6694b88c807SRodney W. Grimes * Return: 6704b88c807SRodney W. Grimes * size of trailer header in this format 6714b88c807SRodney W. Grimes */ 6724b88c807SRodney W. Grimes 6734b88c807SRodney W. Grimes off_t 6744b88c807SRodney W. Grimes vcpio_endrd(void) 6754b88c807SRodney W. Grimes { 6764b88c807SRodney W. Grimes return((off_t)(sizeof(HD_VCPIO) + sizeof(TRAILER) + 6774b88c807SRodney W. Grimes (VCPIO_PAD(sizeof(HD_VCPIO) + sizeof(TRAILER))))); 6784b88c807SRodney W. Grimes } 6794b88c807SRodney W. Grimes 6804b88c807SRodney W. Grimes /* 6814b88c807SRodney W. Grimes * crc_stwr() 6824b88c807SRodney W. Grimes * start up the device mapping table, enable crc file calculation 6834b88c807SRodney W. Grimes * Return: 6844b88c807SRodney W. Grimes * 0 if ok, -1 otherwise (what dev_start() returns) 6854b88c807SRodney W. Grimes */ 6864b88c807SRodney W. Grimes 6874b88c807SRodney W. Grimes int 6884b88c807SRodney W. Grimes crc_stwr(void) 6894b88c807SRodney W. Grimes { 6904b88c807SRodney W. Grimes docrc = 1; 6914b88c807SRodney W. Grimes return(dev_start()); 6924b88c807SRodney W. Grimes } 6934b88c807SRodney W. Grimes 6944b88c807SRodney W. Grimes /* 6954b88c807SRodney W. Grimes * vcpio_wr() 6964b88c807SRodney W. Grimes * copy the data in the ARCHD to buffer in system VR4 cpio 6974b88c807SRodney W. Grimes * (with/without crc) format. 6984b88c807SRodney W. Grimes * Return 6994b88c807SRodney W. Grimes * 0 if file has data to be written after the header, 1 if file has 7004b88c807SRodney W. Grimes * NO data to write after the header, -1 if archive write failed 7014b88c807SRodney W. Grimes */ 7024b88c807SRodney W. Grimes 7034b88c807SRodney W. Grimes int 704f789b261SWarner Losh vcpio_wr(ARCHD *arcn) 7054b88c807SRodney W. Grimes { 706f789b261SWarner Losh HD_VCPIO *hd; 7074b88c807SRodney W. Grimes unsigned int nsz; 7084b88c807SRodney W. Grimes char hdblk[sizeof(HD_VCPIO)]; 7094b88c807SRodney W. Grimes 7104b88c807SRodney W. Grimes /* 7114b88c807SRodney W. Grimes * check and repair truncated device and inode fields in the cpio 7124b88c807SRodney W. Grimes * header 7134b88c807SRodney W. Grimes */ 7144b88c807SRodney W. Grimes if (map_dev(arcn, (u_long)VCPIO_MASK, (u_long)VCPIO_MASK) < 0) 7154b88c807SRodney W. Grimes return(-1); 7164b88c807SRodney W. Grimes nsz = arcn->nlen + 1; 7174b88c807SRodney W. Grimes hd = (HD_VCPIO *)hdblk; 7184b88c807SRodney W. Grimes if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR)) 7194b88c807SRodney W. Grimes arcn->sb.st_rdev = 0; 7204b88c807SRodney W. Grimes 7214b88c807SRodney W. Grimes /* 7224b88c807SRodney W. Grimes * add the proper magic value depending whether we were asked for 7234b88c807SRodney W. Grimes * file data crc's, and the crc if needed. 7244b88c807SRodney W. Grimes */ 7254b88c807SRodney W. Grimes if (docrc) { 7264b88c807SRodney W. Grimes if (ul_asc((u_long)VCMAGIC, hd->c_magic, sizeof(hd->c_magic), 7274b88c807SRodney W. Grimes OCT) || 7284b88c807SRodney W. Grimes ul_asc((u_long)arcn->crc,hd->c_chksum,sizeof(hd->c_chksum), 7294b88c807SRodney W. Grimes HEX)) 7304b88c807SRodney W. Grimes goto out; 7314b88c807SRodney W. Grimes } else { 7324b88c807SRodney W. Grimes if (ul_asc((u_long)VMAGIC, hd->c_magic, sizeof(hd->c_magic), 7334b88c807SRodney W. Grimes OCT) || 7344b88c807SRodney W. Grimes ul_asc((u_long)0L, hd->c_chksum, sizeof(hd->c_chksum),HEX)) 7354b88c807SRodney W. Grimes goto out; 7364b88c807SRodney W. Grimes } 7374b88c807SRodney W. Grimes 7384b88c807SRodney W. Grimes switch(arcn->type) { 7394b88c807SRodney W. Grimes case PAX_CTG: 7404b88c807SRodney W. Grimes case PAX_REG: 7414b88c807SRodney W. Grimes case PAX_HRG: 7424b88c807SRodney W. Grimes /* 7434b88c807SRodney W. Grimes * caller will copy file data to the archive. tell him how 7444b88c807SRodney W. Grimes * much to pad. 7454b88c807SRodney W. Grimes */ 7464b88c807SRodney W. Grimes arcn->pad = VCPIO_PAD(arcn->sb.st_size); 7474b88c807SRodney W. Grimes # ifdef NET2_STAT 7484b88c807SRodney W. Grimes if (ul_asc((u_long)arcn->sb.st_size, hd->c_filesize, 7494b88c807SRodney W. Grimes sizeof(hd->c_filesize), HEX)) { 7504b88c807SRodney W. Grimes # else 7514b88c807SRodney W. Grimes if (uqd_asc((u_quad_t)arcn->sb.st_size, hd->c_filesize, 7524b88c807SRodney W. Grimes sizeof(hd->c_filesize), HEX)) { 7534b88c807SRodney W. Grimes # endif 754778766feSKris Kennaway paxwarn(1,"File is too large for sv4cpio format %s", 7554b88c807SRodney W. Grimes arcn->org_name); 7564b88c807SRodney W. Grimes return(1); 7574b88c807SRodney W. Grimes } 7584b88c807SRodney W. Grimes break; 7594b88c807SRodney W. Grimes case PAX_SLK: 7604b88c807SRodney W. Grimes /* 7614b88c807SRodney W. Grimes * no file data for the caller to process, the file data has 7624b88c807SRodney W. Grimes * the size of the link 7634b88c807SRodney W. Grimes */ 7644b88c807SRodney W. Grimes arcn->pad = 0L; 7654b88c807SRodney W. Grimes if (ul_asc((u_long)arcn->ln_nlen, hd->c_filesize, 7664b88c807SRodney W. Grimes sizeof(hd->c_filesize), HEX)) 7674b88c807SRodney W. Grimes goto out; 7684b88c807SRodney W. Grimes break; 7694b88c807SRodney W. Grimes default: 7704b88c807SRodney W. Grimes /* 7714b88c807SRodney W. Grimes * no file data for the caller to process 7724b88c807SRodney W. Grimes */ 7734b88c807SRodney W. Grimes arcn->pad = 0L; 7744b88c807SRodney W. Grimes if (ul_asc((u_long)0L, hd->c_filesize, sizeof(hd->c_filesize), 7754b88c807SRodney W. Grimes HEX)) 7764b88c807SRodney W. Grimes goto out; 7774b88c807SRodney W. Grimes break; 7784b88c807SRodney W. Grimes } 7794b88c807SRodney W. Grimes 7804b88c807SRodney W. Grimes /* 7814b88c807SRodney W. Grimes * set the other fields in the header 7824b88c807SRodney W. Grimes */ 7834b88c807SRodney W. Grimes if (ul_asc((u_long)arcn->sb.st_ino, hd->c_ino, sizeof(hd->c_ino), 7844b88c807SRodney W. Grimes HEX) || 7854b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_mode, hd->c_mode, sizeof(hd->c_mode), 7864b88c807SRodney W. Grimes HEX) || 7874b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_uid, hd->c_uid, sizeof(hd->c_uid), 7884b88c807SRodney W. Grimes HEX) || 7894b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_gid, hd->c_gid, sizeof(hd->c_gid), 7904b88c807SRodney W. Grimes HEX) || 7914b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_mtime, hd->c_mtime, sizeof(hd->c_mtime), 7924b88c807SRodney W. Grimes HEX) || 7934b88c807SRodney W. Grimes ul_asc((u_long)arcn->sb.st_nlink, hd->c_nlink, sizeof(hd->c_nlink), 7944b88c807SRodney W. Grimes HEX) || 7954b88c807SRodney W. Grimes ul_asc((u_long)MAJOR(arcn->sb.st_dev),hd->c_maj, sizeof(hd->c_maj), 7964b88c807SRodney W. Grimes HEX) || 7974b88c807SRodney W. Grimes ul_asc((u_long)MINOR(arcn->sb.st_dev),hd->c_min, sizeof(hd->c_min), 7984b88c807SRodney W. Grimes HEX) || 7994b88c807SRodney W. Grimes ul_asc((u_long)MAJOR(arcn->sb.st_rdev),hd->c_rmaj,sizeof(hd->c_maj), 8004b88c807SRodney W. Grimes HEX) || 8014b88c807SRodney W. Grimes ul_asc((u_long)MINOR(arcn->sb.st_rdev),hd->c_rmin,sizeof(hd->c_min), 8024b88c807SRodney W. Grimes HEX) || 8034b88c807SRodney W. Grimes ul_asc((u_long)nsz, hd->c_namesize, sizeof(hd->c_namesize), HEX)) 8044b88c807SRodney W. Grimes goto out; 8054b88c807SRodney W. Grimes 8064b88c807SRodney W. Grimes /* 8074b88c807SRodney W. Grimes * write the header, the file name and padding as required. 8084b88c807SRodney W. Grimes */ 8094b88c807SRodney W. Grimes if ((wr_rdbuf(hdblk, (int)sizeof(HD_VCPIO)) < 0) || 8104b88c807SRodney W. Grimes (wr_rdbuf(arcn->name, (int)nsz) < 0) || 8114b88c807SRodney W. Grimes (wr_skip((off_t)(VCPIO_PAD(sizeof(HD_VCPIO) + nsz))) < 0)) { 812778766feSKris Kennaway paxwarn(1,"Could not write sv4cpio header for %s",arcn->org_name); 8134b88c807SRodney W. Grimes return(-1); 8144b88c807SRodney W. Grimes } 8154b88c807SRodney W. Grimes 8164b88c807SRodney W. Grimes /* 8174b88c807SRodney W. Grimes * if we have file data, tell the caller we are done, copy the file 8184b88c807SRodney W. Grimes */ 8194b88c807SRodney W. Grimes if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) || 8204b88c807SRodney W. Grimes (arcn->type == PAX_HRG)) 8214b88c807SRodney W. Grimes return(0); 8224b88c807SRodney W. Grimes 8234b88c807SRodney W. Grimes /* 8244b88c807SRodney W. Grimes * if we are not a link, tell the caller we are done, go to next file 8254b88c807SRodney W. Grimes */ 8264b88c807SRodney W. Grimes if (arcn->type != PAX_SLK) 8274b88c807SRodney W. Grimes return(1); 8284b88c807SRodney W. Grimes 8294b88c807SRodney W. Grimes /* 8304b88c807SRodney W. Grimes * write the link name, tell the caller we are done. 8314b88c807SRodney W. Grimes */ 8324b88c807SRodney W. Grimes if ((wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) || 8334b88c807SRodney W. Grimes (wr_skip((off_t)(VCPIO_PAD(arcn->ln_nlen))) < 0)) { 834778766feSKris Kennaway paxwarn(1,"Could not write sv4cpio link name for %s", 8354b88c807SRodney W. Grimes arcn->org_name); 8364b88c807SRodney W. Grimes return(-1); 8374b88c807SRodney W. Grimes } 8384b88c807SRodney W. Grimes return(1); 8394b88c807SRodney W. Grimes 8404b88c807SRodney W. Grimes out: 8414b88c807SRodney W. Grimes /* 8424b88c807SRodney W. Grimes * header field is out of range 8434b88c807SRodney W. Grimes */ 844778766feSKris Kennaway paxwarn(1,"Sv4cpio header field is too small for file %s",arcn->org_name); 8454b88c807SRodney W. Grimes return(1); 8464b88c807SRodney W. Grimes } 8474b88c807SRodney W. Grimes 8484b88c807SRodney W. Grimes /* 8494b88c807SRodney W. Grimes * Routines common to the old binary header cpio 8504b88c807SRodney W. Grimes */ 8514b88c807SRodney W. Grimes 8524b88c807SRodney W. Grimes /* 8534b88c807SRodney W. Grimes * bcpio_id() 8549d5abbddSJens Schweikhardt * determine if a block given to us is an old binary cpio header 8554b88c807SRodney W. Grimes * (with/without header byte swapping) 8564b88c807SRodney W. Grimes * Return: 8574b88c807SRodney W. Grimes * 0 if a valid header, -1 otherwise 8584b88c807SRodney W. Grimes */ 8594b88c807SRodney W. Grimes 8604b88c807SRodney W. Grimes int 8614b88c807SRodney W. Grimes bcpio_id(char *blk, int size) 8624b88c807SRodney W. Grimes { 863cee22cbdSDavid E. O'Brien if (size < (int)sizeof(HD_BCPIO)) 8644b88c807SRodney W. Grimes return(-1); 8654b88c807SRodney W. Grimes 8664b88c807SRodney W. Grimes /* 8674b88c807SRodney W. Grimes * check both normal and byte swapped magic cookies 8684b88c807SRodney W. Grimes */ 8694b88c807SRodney W. Grimes if (((u_short)SHRT_EXT(blk)) == MAGIC) 8704b88c807SRodney W. Grimes return(0); 8714b88c807SRodney W. Grimes if (((u_short)RSHRT_EXT(blk)) == MAGIC) { 8724b88c807SRodney W. Grimes if (!swp_head) 8734b88c807SRodney W. Grimes ++swp_head; 8744b88c807SRodney W. Grimes return(0); 8754b88c807SRodney W. Grimes } 8764b88c807SRodney W. Grimes return(-1); 8774b88c807SRodney W. Grimes } 8784b88c807SRodney W. Grimes 8794b88c807SRodney W. Grimes /* 8804b88c807SRodney W. Grimes * bcpio_rd() 8819d5abbddSJens Schweikhardt * determine if a buffer is an old binary archive entry. (It may have byte 8824b88c807SRodney W. Grimes * swapped header) convert and store the values in the ARCHD parameter. 8834b88c807SRodney W. Grimes * This is a very old header format and should not really be used. 8844b88c807SRodney W. Grimes * Return: 8854b88c807SRodney W. Grimes * 0 if a valid header, -1 otherwise. 8864b88c807SRodney W. Grimes */ 8874b88c807SRodney W. Grimes 8884b88c807SRodney W. Grimes int 889f789b261SWarner Losh bcpio_rd(ARCHD *arcn, char *buf) 8904b88c807SRodney W. Grimes { 891f789b261SWarner Losh HD_BCPIO *hd; 892f789b261SWarner Losh int nsz; 8934b88c807SRodney W. Grimes 8944b88c807SRodney W. Grimes /* 8954b88c807SRodney W. Grimes * check the header 8964b88c807SRodney W. Grimes */ 8974b88c807SRodney W. Grimes if (bcpio_id(buf, sizeof(HD_BCPIO)) < 0) 8984b88c807SRodney W. Grimes return(-1); 8994b88c807SRodney W. Grimes 9004b88c807SRodney W. Grimes arcn->pad = 0L; 9014b88c807SRodney W. Grimes hd = (HD_BCPIO *)buf; 9024b88c807SRodney W. Grimes if (swp_head) { 9034b88c807SRodney W. Grimes /* 90446be34b9SKris Kennaway * header has swapped bytes on 16 bit boundaries 9054b88c807SRodney W. Grimes */ 9064b88c807SRodney W. Grimes arcn->sb.st_dev = (dev_t)(RSHRT_EXT(hd->h_dev)); 9074b88c807SRodney W. Grimes arcn->sb.st_ino = (ino_t)(RSHRT_EXT(hd->h_ino)); 9084b88c807SRodney W. Grimes arcn->sb.st_mode = (mode_t)(RSHRT_EXT(hd->h_mode)); 9094b88c807SRodney W. Grimes arcn->sb.st_uid = (uid_t)(RSHRT_EXT(hd->h_uid)); 9104b88c807SRodney W. Grimes arcn->sb.st_gid = (gid_t)(RSHRT_EXT(hd->h_gid)); 9114b88c807SRodney W. Grimes arcn->sb.st_nlink = (nlink_t)(RSHRT_EXT(hd->h_nlink)); 9124b88c807SRodney W. Grimes arcn->sb.st_rdev = (dev_t)(RSHRT_EXT(hd->h_rdev)); 9134b88c807SRodney W. Grimes arcn->sb.st_mtime = (time_t)(RSHRT_EXT(hd->h_mtime_1)); 9144b88c807SRodney W. Grimes arcn->sb.st_mtime = (arcn->sb.st_mtime << 16) | 9154b88c807SRodney W. Grimes ((time_t)(RSHRT_EXT(hd->h_mtime_2))); 9164b88c807SRodney W. Grimes arcn->sb.st_size = (off_t)(RSHRT_EXT(hd->h_filesize_1)); 9174b88c807SRodney W. Grimes arcn->sb.st_size = (arcn->sb.st_size << 16) | 9184b88c807SRodney W. Grimes ((off_t)(RSHRT_EXT(hd->h_filesize_2))); 9194b88c807SRodney W. Grimes nsz = (int)(RSHRT_EXT(hd->h_namesize)); 9204b88c807SRodney W. Grimes } else { 9214b88c807SRodney W. Grimes arcn->sb.st_dev = (dev_t)(SHRT_EXT(hd->h_dev)); 9224b88c807SRodney W. Grimes arcn->sb.st_ino = (ino_t)(SHRT_EXT(hd->h_ino)); 9234b88c807SRodney W. Grimes arcn->sb.st_mode = (mode_t)(SHRT_EXT(hd->h_mode)); 9244b88c807SRodney W. Grimes arcn->sb.st_uid = (uid_t)(SHRT_EXT(hd->h_uid)); 9254b88c807SRodney W. Grimes arcn->sb.st_gid = (gid_t)(SHRT_EXT(hd->h_gid)); 9264b88c807SRodney W. Grimes arcn->sb.st_nlink = (nlink_t)(SHRT_EXT(hd->h_nlink)); 9274b88c807SRodney W. Grimes arcn->sb.st_rdev = (dev_t)(SHRT_EXT(hd->h_rdev)); 9284b88c807SRodney W. Grimes arcn->sb.st_mtime = (time_t)(SHRT_EXT(hd->h_mtime_1)); 9294b88c807SRodney W. Grimes arcn->sb.st_mtime = (arcn->sb.st_mtime << 16) | 9304b88c807SRodney W. Grimes ((time_t)(SHRT_EXT(hd->h_mtime_2))); 9314b88c807SRodney W. Grimes arcn->sb.st_size = (off_t)(SHRT_EXT(hd->h_filesize_1)); 9324b88c807SRodney W. Grimes arcn->sb.st_size = (arcn->sb.st_size << 16) | 9334b88c807SRodney W. Grimes ((off_t)(SHRT_EXT(hd->h_filesize_2))); 9344b88c807SRodney W. Grimes nsz = (int)(SHRT_EXT(hd->h_namesize)); 9354b88c807SRodney W. Grimes } 9364b88c807SRodney W. Grimes arcn->sb.st_ctime = arcn->sb.st_atime = arcn->sb.st_mtime; 9374b88c807SRodney W. Grimes 9384b88c807SRodney W. Grimes /* 9394b88c807SRodney W. Grimes * check the file name size, if bogus give up. otherwise read the file 9404b88c807SRodney W. Grimes * name 9414b88c807SRodney W. Grimes */ 9424b88c807SRodney W. Grimes if (nsz < 2) 9434b88c807SRodney W. Grimes return(-1); 9444b88c807SRodney W. Grimes arcn->nlen = nsz - 1; 9454b88c807SRodney W. Grimes if (rd_nm(arcn, nsz) < 0) 9464b88c807SRodney W. Grimes return(-1); 9474b88c807SRodney W. Grimes 9484b88c807SRodney W. Grimes /* 9494b88c807SRodney W. Grimes * header + file name are aligned to 2 byte boundries, skip if needed 9504b88c807SRodney W. Grimes */ 9514b88c807SRodney W. Grimes if (rd_skip((off_t)(BCPIO_PAD(sizeof(HD_BCPIO) + nsz))) < 0) 9524b88c807SRodney W. Grimes return(-1); 9534b88c807SRodney W. Grimes 9544b88c807SRodney W. Grimes /* 9554b88c807SRodney W. Grimes * if not a link (or a file with no data), calculate pad size (for 9564b88c807SRodney W. Grimes * padding which follows the file data), clear the link name and return 9574b88c807SRodney W. Grimes */ 9584b88c807SRodney W. Grimes if (((arcn->sb.st_mode & C_IFMT) != C_ISLNK)||(arcn->sb.st_size == 0)){ 9594b88c807SRodney W. Grimes /* 9604b88c807SRodney W. Grimes * we have a valid header (not a link) 9614b88c807SRodney W. Grimes */ 9624b88c807SRodney W. Grimes arcn->ln_nlen = 0; 9634b88c807SRodney W. Grimes arcn->ln_name[0] = '\0'; 9644b88c807SRodney W. Grimes arcn->pad = BCPIO_PAD(arcn->sb.st_size); 9654b88c807SRodney W. Grimes return(com_rd(arcn)); 9664b88c807SRodney W. Grimes } 9674b88c807SRodney W. Grimes 9684b88c807SRodney W. Grimes if ((rd_ln_nm(arcn) < 0) || 9694b88c807SRodney W. Grimes (rd_skip((off_t)(BCPIO_PAD(arcn->sb.st_size))) < 0)) 9704b88c807SRodney W. Grimes return(-1); 9714b88c807SRodney W. Grimes 9724b88c807SRodney W. Grimes /* 9734b88c807SRodney W. Grimes * we have a valid header (with a link) 9744b88c807SRodney W. Grimes */ 9754b88c807SRodney W. Grimes return(com_rd(arcn)); 9764b88c807SRodney W. Grimes } 9774b88c807SRodney W. Grimes 9784b88c807SRodney W. Grimes /* 9794b88c807SRodney W. Grimes * bcpio_endrd() 9804b88c807SRodney W. Grimes * no cleanup needed here, just return size of the trailer (for append) 9814b88c807SRodney W. Grimes * Return: 9824b88c807SRodney W. Grimes * size of trailer header in this format 9834b88c807SRodney W. Grimes */ 9844b88c807SRodney W. Grimes 9854b88c807SRodney W. Grimes off_t 9864b88c807SRodney W. Grimes bcpio_endrd(void) 9874b88c807SRodney W. Grimes { 9884b88c807SRodney W. Grimes return((off_t)(sizeof(HD_BCPIO) + sizeof(TRAILER) + 9894b88c807SRodney W. Grimes (BCPIO_PAD(sizeof(HD_BCPIO) + sizeof(TRAILER))))); 9904b88c807SRodney W. Grimes } 9914b88c807SRodney W. Grimes 9924b88c807SRodney W. Grimes /* 9934b88c807SRodney W. Grimes * bcpio_wr() 9944b88c807SRodney W. Grimes * copy the data in the ARCHD to buffer in old binary cpio format 9954b88c807SRodney W. Grimes * There is a real chance of field overflow with this critter. So we 9964b88c807SRodney W. Grimes * always check the conversion is ok. nobody in his their right mind 9974b88c807SRodney W. Grimes * should write an achive in this format... 9984b88c807SRodney W. Grimes * Return 9994b88c807SRodney W. Grimes * 0 if file has data to be written after the header, 1 if file has NO 10004b88c807SRodney W. Grimes * data to write after the header, -1 if archive write failed 10014b88c807SRodney W. Grimes */ 10024b88c807SRodney W. Grimes 10034b88c807SRodney W. Grimes int 1004f789b261SWarner Losh bcpio_wr(ARCHD *arcn) 10054b88c807SRodney W. Grimes { 1006f789b261SWarner Losh HD_BCPIO *hd; 1007f789b261SWarner Losh int nsz; 10084b88c807SRodney W. Grimes char hdblk[sizeof(HD_BCPIO)]; 10094b88c807SRodney W. Grimes off_t t_offt; 10104b88c807SRodney W. Grimes int t_int; 10114b88c807SRodney W. Grimes time_t t_timet; 10124b88c807SRodney W. Grimes 10134b88c807SRodney W. Grimes /* 10144b88c807SRodney W. Grimes * check and repair truncated device and inode fields in the cpio 10154b88c807SRodney W. Grimes * header 10164b88c807SRodney W. Grimes */ 10174b88c807SRodney W. Grimes if (map_dev(arcn, (u_long)BCPIO_MASK, (u_long)BCPIO_MASK) < 0) 10184b88c807SRodney W. Grimes return(-1); 10194b88c807SRodney W. Grimes 10204b88c807SRodney W. Grimes if ((arcn->type != PAX_BLK) && (arcn->type != PAX_CHR)) 10214b88c807SRodney W. Grimes arcn->sb.st_rdev = 0; 10224b88c807SRodney W. Grimes hd = (HD_BCPIO *)hdblk; 10234b88c807SRodney W. Grimes 10244b88c807SRodney W. Grimes switch(arcn->type) { 10254b88c807SRodney W. Grimes case PAX_CTG: 10264b88c807SRodney W. Grimes case PAX_REG: 10274b88c807SRodney W. Grimes case PAX_HRG: 10284b88c807SRodney W. Grimes /* 10294b88c807SRodney W. Grimes * caller will copy file data to the archive. tell him how 10304b88c807SRodney W. Grimes * much to pad. 10314b88c807SRodney W. Grimes */ 10324b88c807SRodney W. Grimes arcn->pad = BCPIO_PAD(arcn->sb.st_size); 10334b88c807SRodney W. Grimes hd->h_filesize_1[0] = CHR_WR_0(arcn->sb.st_size); 10344b88c807SRodney W. Grimes hd->h_filesize_1[1] = CHR_WR_1(arcn->sb.st_size); 10354b88c807SRodney W. Grimes hd->h_filesize_2[0] = CHR_WR_2(arcn->sb.st_size); 10364b88c807SRodney W. Grimes hd->h_filesize_2[1] = CHR_WR_3(arcn->sb.st_size); 10374b88c807SRodney W. Grimes t_offt = (off_t)(SHRT_EXT(hd->h_filesize_1)); 10384b88c807SRodney W. Grimes t_offt = (t_offt<<16) | ((off_t)(SHRT_EXT(hd->h_filesize_2))); 10394b88c807SRodney W. Grimes if (arcn->sb.st_size != t_offt) { 1040778766feSKris Kennaway paxwarn(1,"File is too large for bcpio format %s", 10414b88c807SRodney W. Grimes arcn->org_name); 10424b88c807SRodney W. Grimes return(1); 10434b88c807SRodney W. Grimes } 10444b88c807SRodney W. Grimes break; 10454b88c807SRodney W. Grimes case PAX_SLK: 10464b88c807SRodney W. Grimes /* 10474b88c807SRodney W. Grimes * no file data for the caller to process, the file data has 10484b88c807SRodney W. Grimes * the size of the link 10494b88c807SRodney W. Grimes */ 10504b88c807SRodney W. Grimes arcn->pad = 0L; 10514b88c807SRodney W. Grimes hd->h_filesize_1[0] = CHR_WR_0(arcn->ln_nlen); 10524b88c807SRodney W. Grimes hd->h_filesize_1[1] = CHR_WR_1(arcn->ln_nlen); 10534b88c807SRodney W. Grimes hd->h_filesize_2[0] = CHR_WR_2(arcn->ln_nlen); 10544b88c807SRodney W. Grimes hd->h_filesize_2[1] = CHR_WR_3(arcn->ln_nlen); 10554b88c807SRodney W. Grimes t_int = (int)(SHRT_EXT(hd->h_filesize_1)); 10564b88c807SRodney W. Grimes t_int = (t_int << 16) | ((int)(SHRT_EXT(hd->h_filesize_2))); 10574b88c807SRodney W. Grimes if (arcn->ln_nlen != t_int) 10584b88c807SRodney W. Grimes goto out; 10594b88c807SRodney W. Grimes break; 10604b88c807SRodney W. Grimes default: 10614b88c807SRodney W. Grimes /* 10624b88c807SRodney W. Grimes * no file data for the caller to process 10634b88c807SRodney W. Grimes */ 10644b88c807SRodney W. Grimes arcn->pad = 0L; 10654b88c807SRodney W. Grimes hd->h_filesize_1[0] = (char)0; 10664b88c807SRodney W. Grimes hd->h_filesize_1[1] = (char)0; 10674b88c807SRodney W. Grimes hd->h_filesize_2[0] = (char)0; 10684b88c807SRodney W. Grimes hd->h_filesize_2[1] = (char)0; 10694b88c807SRodney W. Grimes break; 10704b88c807SRodney W. Grimes } 10714b88c807SRodney W. Grimes 10724b88c807SRodney W. Grimes /* 10734b88c807SRodney W. Grimes * build up the rest of the fields 10744b88c807SRodney W. Grimes */ 10754b88c807SRodney W. Grimes hd->h_magic[0] = CHR_WR_2(MAGIC); 10764b88c807SRodney W. Grimes hd->h_magic[1] = CHR_WR_3(MAGIC); 10774b88c807SRodney W. Grimes hd->h_dev[0] = CHR_WR_2(arcn->sb.st_dev); 10784b88c807SRodney W. Grimes hd->h_dev[1] = CHR_WR_3(arcn->sb.st_dev); 10794b88c807SRodney W. Grimes if (arcn->sb.st_dev != (dev_t)(SHRT_EXT(hd->h_dev))) 10804b88c807SRodney W. Grimes goto out; 10814b88c807SRodney W. Grimes hd->h_ino[0] = CHR_WR_2(arcn->sb.st_ino); 10824b88c807SRodney W. Grimes hd->h_ino[1] = CHR_WR_3(arcn->sb.st_ino); 10834b88c807SRodney W. Grimes if (arcn->sb.st_ino != (ino_t)(SHRT_EXT(hd->h_ino))) 10844b88c807SRodney W. Grimes goto out; 10854b88c807SRodney W. Grimes hd->h_mode[0] = CHR_WR_2(arcn->sb.st_mode); 10864b88c807SRodney W. Grimes hd->h_mode[1] = CHR_WR_3(arcn->sb.st_mode); 10874b88c807SRodney W. Grimes if (arcn->sb.st_mode != (mode_t)(SHRT_EXT(hd->h_mode))) 10884b88c807SRodney W. Grimes goto out; 10894b88c807SRodney W. Grimes hd->h_uid[0] = CHR_WR_2(arcn->sb.st_uid); 10904b88c807SRodney W. Grimes hd->h_uid[1] = CHR_WR_3(arcn->sb.st_uid); 10914b88c807SRodney W. Grimes if (arcn->sb.st_uid != (uid_t)(SHRT_EXT(hd->h_uid))) 10924b88c807SRodney W. Grimes goto out; 10934b88c807SRodney W. Grimes hd->h_gid[0] = CHR_WR_2(arcn->sb.st_gid); 10944b88c807SRodney W. Grimes hd->h_gid[1] = CHR_WR_3(arcn->sb.st_gid); 10954b88c807SRodney W. Grimes if (arcn->sb.st_gid != (gid_t)(SHRT_EXT(hd->h_gid))) 10964b88c807SRodney W. Grimes goto out; 10974b88c807SRodney W. Grimes hd->h_nlink[0] = CHR_WR_2(arcn->sb.st_nlink); 10984b88c807SRodney W. Grimes hd->h_nlink[1] = CHR_WR_3(arcn->sb.st_nlink); 10994b88c807SRodney W. Grimes if (arcn->sb.st_nlink != (nlink_t)(SHRT_EXT(hd->h_nlink))) 11004b88c807SRodney W. Grimes goto out; 11014b88c807SRodney W. Grimes hd->h_rdev[0] = CHR_WR_2(arcn->sb.st_rdev); 11024b88c807SRodney W. Grimes hd->h_rdev[1] = CHR_WR_3(arcn->sb.st_rdev); 11034b88c807SRodney W. Grimes if (arcn->sb.st_rdev != (dev_t)(SHRT_EXT(hd->h_rdev))) 11044b88c807SRodney W. Grimes goto out; 11054b88c807SRodney W. Grimes hd->h_mtime_1[0] = CHR_WR_0(arcn->sb.st_mtime); 11064b88c807SRodney W. Grimes hd->h_mtime_1[1] = CHR_WR_1(arcn->sb.st_mtime); 11074b88c807SRodney W. Grimes hd->h_mtime_2[0] = CHR_WR_2(arcn->sb.st_mtime); 11084b88c807SRodney W. Grimes hd->h_mtime_2[1] = CHR_WR_3(arcn->sb.st_mtime); 11094b88c807SRodney W. Grimes t_timet = (time_t)(SHRT_EXT(hd->h_mtime_1)); 11104b88c807SRodney W. Grimes t_timet = (t_timet << 16) | ((time_t)(SHRT_EXT(hd->h_mtime_2))); 11114b88c807SRodney W. Grimes if (arcn->sb.st_mtime != t_timet) 11124b88c807SRodney W. Grimes goto out; 11134b88c807SRodney W. Grimes nsz = arcn->nlen + 1; 11144b88c807SRodney W. Grimes hd->h_namesize[0] = CHR_WR_2(nsz); 11154b88c807SRodney W. Grimes hd->h_namesize[1] = CHR_WR_3(nsz); 11164b88c807SRodney W. Grimes if (nsz != (int)(SHRT_EXT(hd->h_namesize))) 11174b88c807SRodney W. Grimes goto out; 11184b88c807SRodney W. Grimes 11194b88c807SRodney W. Grimes /* 11204b88c807SRodney W. Grimes * write the header, the file name and padding as required. 11214b88c807SRodney W. Grimes */ 11224b88c807SRodney W. Grimes if ((wr_rdbuf(hdblk, (int)sizeof(HD_BCPIO)) < 0) || 11234b88c807SRodney W. Grimes (wr_rdbuf(arcn->name, nsz) < 0) || 11244b88c807SRodney W. Grimes (wr_skip((off_t)(BCPIO_PAD(sizeof(HD_BCPIO) + nsz))) < 0)) { 1125778766feSKris Kennaway paxwarn(1, "Could not write bcpio header for %s", arcn->org_name); 11264b88c807SRodney W. Grimes return(-1); 11274b88c807SRodney W. Grimes } 11284b88c807SRodney W. Grimes 11294b88c807SRodney W. Grimes /* 11304b88c807SRodney W. Grimes * if we have file data, tell the caller we are done 11314b88c807SRodney W. Grimes */ 11324b88c807SRodney W. Grimes if ((arcn->type == PAX_CTG) || (arcn->type == PAX_REG) || 11334b88c807SRodney W. Grimes (arcn->type == PAX_HRG)) 11344b88c807SRodney W. Grimes return(0); 11354b88c807SRodney W. Grimes 11364b88c807SRodney W. Grimes /* 11374b88c807SRodney W. Grimes * if we are not a link, tell the caller we are done, go to next file 11384b88c807SRodney W. Grimes */ 11394b88c807SRodney W. Grimes if (arcn->type != PAX_SLK) 11404b88c807SRodney W. Grimes return(1); 11414b88c807SRodney W. Grimes 11424b88c807SRodney W. Grimes /* 11434b88c807SRodney W. Grimes * write the link name, tell the caller we are done. 11444b88c807SRodney W. Grimes */ 11454b88c807SRodney W. Grimes if ((wr_rdbuf(arcn->ln_name, arcn->ln_nlen) < 0) || 11464b88c807SRodney W. Grimes (wr_skip((off_t)(BCPIO_PAD(arcn->ln_nlen))) < 0)) { 1147778766feSKris Kennaway paxwarn(1,"Could not write bcpio link name for %s",arcn->org_name); 11484b88c807SRodney W. Grimes return(-1); 11494b88c807SRodney W. Grimes } 11504b88c807SRodney W. Grimes return(1); 11514b88c807SRodney W. Grimes 11524b88c807SRodney W. Grimes out: 11534b88c807SRodney W. Grimes /* 11544b88c807SRodney W. Grimes * header field is out of range 11554b88c807SRodney W. Grimes */ 1156778766feSKris Kennaway paxwarn(1,"Bcpio header field is too small for file %s", arcn->org_name); 11574b88c807SRodney W. Grimes return(1); 11584b88c807SRodney W. Grimes } 1159