1 /*
2 * CDDL HEADER START
3 *
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
7 *
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
12 *
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
18 *
19 * CDDL HEADER END
20 */
21 /*
22 * Copyright (c) 1988, 2010, Oracle and/or its affiliates. All rights reserved.
23 * Copyright 2012 Milan Jurik. All rights reserved.
24 * Copyright (c) 2012 Gary Mills
25 */
26
27 /* Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T */
28 /* All Rights Reserved */
29
30 /*
31 * Copyright 2022 OmniOS Community Edition (OmniOSce) Association.
32 */
33
34 /*
35 * Portions of this source code were derived from Berkeley 4.3 BSD
36 * under license from the Regents of the University of California.
37 */
38
39 #include <stdio.h>
40 #include <sys/types.h>
41 #include <errno.h>
42 #include <unistd.h>
43 #include <stdlib.h>
44 #include <fcntl.h>
45 #include <memory.h>
46 #include <string.h>
47 #include <stdarg.h>
48 #include <sys/stat.h>
49 #include <sys/statvfs.h>
50 #include <sys/mkdev.h>
51 #include <sys/param.h>
52 #include <utime.h>
53 #include <pwd.h>
54 #include <grp.h>
55 #include <signal.h>
56 #include <ctype.h>
57 #include <locale.h>
58 #include <sys/ioctl.h>
59 #include <sys/mtio.h>
60 #include <sys/fdio.h>
61 #include "cpio.h"
62 #include <sys/acl.h>
63 #include <sys/time.h>
64 #include <sys/resource.h>
65 #include <fnmatch.h>
66 #include <libgen.h>
67 #include <libintl.h>
68 #include <dirent.h>
69 #include <limits.h>
70 #include <aclutils.h>
71 #if defined(_PC_SATTR_ENABLED)
72 #include <libnvpair.h>
73 #include <attr.h>
74 #include <libcmdutils.h>
75 #endif /* _PC_SATTR_ENABLED */
76 #ifdef SOLARIS_PRIVS
77 #include <priv.h>
78 #endif /* SOLARIS_PRIVS */
79
80 /*
81 * Special kludge for off_t being a signed quantity.
82 */
83 typedef ulong_t u_off_t;
84
85 #define SECMODE 0xe080
86
87 #define DEVNULL "/dev/null"
88 #define XATTRHDR ".hdr"
89
90 #define NAMELEN 32
91 #define TYPELEN 16
92 #define PERMLEN 4
93
94 #define FILE_COPIED 1
95 #define FILE_LINKED 2
96 #define FILE_PASS_ERR -1
97
98 #define ARCHIVE_NORMAL 0
99 #define ARCHIVE_ACL 1
100 #define ARCHIVE_XATTR 2
101 #define ARCHIVE_SPARSE 3
102
103 #ifndef VIEW_READONLY
104 #define VIEW_READONLY "SUNWattr_ro"
105 #endif
106
107 #ifndef VIEW_READWRITE
108 #define VIEW_READWRITE "SUNWattr_rw"
109 #endif
110
111
112 #define LSTAT(dir, path, statbuf) fstatat(dir, \
113 get_component((Gen.g_attrnam_p == NULL) ? \
114 path : Gen.g_attrnam_p), statbuf, AT_SYMLINK_NOFOLLOW)
115 #define STAT(dir, path, statbuf) fstatat(dir, \
116 get_component((Gen.g_attrnam_p == NULL) ? \
117 path : Gen.g_attrnam_p), statbuf, 0)
118
119 /*
120 * Convert from and to old dev_t formats.
121 */
122 #define SVR3_MAJOR(x) ((major_t)((dev_t)(x) >> ONBITSMINOR) & OMAXMAJ)
123 #define SVR3_MINOR(x) ((minor_t)((dev_t)(x) & OMAXMIN))
124 #define TO_SVR3(maj, min) \
125 ((((ushort_t)(maj) & OMAXMAJ) << ONBITSMINOR) | \
126 ((ushort_t)(min) & OMAXMIN))
127
128 /*
129 * These limits reflect the maximum size regular file that
130 * can be archived, depending on the archive type. For archives
131 * with character-format headers (odc, tar, ustar) we use
132 * CHAR_OFFSET_MAX. For archives with SVR4 ASCII headers (-c, -H crc)
133 * we store filesize in an 8-char hexadecimal string and use
134 * ASC_OFFSET_MAX. Otherwise, we are limited to the size that will
135 * fit in a signed int value.
136 */
137 #define CHAR_OFFSET_MAX 077777777777ULL /* 11 octal digits */
138 #define ASC_OFFSET_MAX 0XFFFFFFFF /* 8 hexadecimal digits */
139 #define BIN_OFFSET_MAX INT_MAX /* signed int max value */
140
141 #define POSIXMODES 07777
142
143 static char aclchar = ' ';
144
145 static struct Lnk *add_lnk(struct Lnk **);
146 static int bfill(void);
147 static void bflush(void);
148 static int chgreel(int dir);
149 static int ckname(int);
150 static void ckopts(long mask);
151 static uint_t cksum(char hdr, int byt_cnt, int *err);
152 static int creat_hdr(void);
153 static int creat_lnk(int dirfd, char *name1_p, char *name2_p);
154 static int creat_spec(int dirfd);
155 static int creat_tmp(char *nam_p);
156 static void data_in(int proc_mode);
157 static void data_out(void);
158 static void data_pass(void);
159 static void file_in(void);
160 static int file_out(void);
161 static int file_pass(void);
162 static void flush_lnks(void);
163 static int gethdr(void);
164 static int getname(void);
165 static void getpats(int largc, char **largv);
166 static void ioerror(int dir);
167 static int matched(void);
168 static int missdir(char *nam_p);
169 static long mklong(short v[]);
170 static void mkshort(short sval[], long v);
171 static int openout(int dirfd);
172 static int read_hdr(int hdr);
173 static void reclaim(struct Lnk *l_p);
174 static void rstbuf(void);
175 static void setpasswd(char *nam);
176 static void rstfiles(int over, int dirfd);
177 static void scan4trail(void);
178 static void setup(int largc, char **largv);
179 static void set_tym(int dirfd, char *nam_p, time_t atime, time_t mtime);
180 static void sigint(int sig);
181 static void swap(char *buf_p, int cnt);
182 static void usage(void);
183 static void verbose(char *nam_p);
184 static void write_hdr(int arcflag, off_t len);
185 static void write_trail(void);
186 static int ustar_dir(void);
187 static int ustar_spec(void);
188 static struct stat *convert_to_old_stat(struct stat *, char *, char *);
189 static void read_bar_vol_hdr(void);
190 static void read_bar_file_hdr(void);
191 static void setup_uncompress(FILE **);
192 static void skip_bar_volhdr(void);
193 static void bar_file_in(void);
194 static int g_init(int *devtype, int *fdes);
195 static int g_read(int, int, char *, unsigned);
196 static int g_write(int, int, char *, unsigned);
197 static int is_floppy(int);
198 static int is_tape(int);
199 static void write_ancillary(char *buf, size_t len, boolean_t padding);
200 static int remove_dir(char *);
201 static int save_cwd(void);
202 static void rest_cwd(int cwd);
203
204 static void xattrs_out(int (*func)());
205 static void get_parent(char *path, char *dir);
206 static void prepare_xattr_hdr(char **attrbuf, char *filename,
207 char *attrname, char typeflag, struct Lnk *linkinfo, int *rlen);
208 static char tartype(int type);
209 static int openfile(int omode);
210 static mode_t attrmode(char type);
211 static char *get_component(char *path);
212 static int open_dir(char *name);
213 static int open_dirfd();
214 static void close_dirfd();
215 static void write_xattr_hdr();
216 static char *skipslashes(char *string, char *start);
217 static int read_xattr_hdr();
218 static void chop_endslashes(char *path);
219
220
221 /* helpful types */
222
223 static
224 struct passwd *Curpw_p, /* Current password entry for -t option */
225 *Rpw_p, /* Password entry for -R option */
226 *dpasswd;
227
228 static
229 struct group *Curgr_p, /* Current group entry for -t option */
230 *dgroup;
231
232 /* Data structure for buffered I/O. */
233
234 static
235 struct buf_info {
236 char *b_base_p, /* Pointer to base of buffer */
237 *b_out_p, /* Position to take bytes from buffer at */
238 *b_in_p, /* Position to put bytes into buffer at */
239 *b_end_p; /* Pointer to end of buffer */
240 int b_cnt, /* Count of unprocessed bytes */
241 b_size; /* Size of buffer in bytes */
242 } Buffr;
243
244 /* Generic header format */
245
246 static
247 struct gen_hdr {
248 uint_t g_magic, /* Magic number field */
249 g_ino, /* Inode number of file */
250 g_mode, /* Mode of file */
251 g_uid, /* Uid of file */
252 g_gid, /* Gid of file */
253 g_nlink, /* Number of links */
254 g_mtime; /* Modification time */
255 off_t g_filesz; /* Length of file */
256 dev_t g_dev, /* File system of file */
257 g_rdev; /* Major/minor numbers of special files */
258 uint_t g_namesz, /* Length of filename */
259 g_cksum; /* Checksum of file */
260 char g_gname[32],
261 g_uname[32],
262 g_version[2],
263 g_tmagic[6],
264 g_typeflag;
265 char *g_tname,
266 *g_prefix,
267 *g_nam_p, /* Filename */
268 *g_attrparent_p, /* attribute parent */
269 *g_attrpath_p, /* attribute path */
270 *g_attrnam_p, /* attribute */
271 *g_attrfnam_p, /* Real file name attr belongs to */
272 *g_linktoattrfnam_p, /* file linked attribute belongs to */
273 *g_linktoattrnam_p, /* attribute g_attrnam_p is linked to */
274 *g_dirpath; /* dirname currently opened */
275 int g_dirfd; /* directory file descriptor */
276 int g_passdirfd; /* directory fd to pass to */
277 int g_rw_sysattr; /* read-write system attribute */
278 int g_baseparent_fd; /* base file's parent fd */
279 holes_info_t *g_holes; /* sparse file information */
280
281 } Gen, *G_p;
282
283 /* Data structure for handling multiply-linked files */
284 static
285 char prebuf[PRESIZ+1],
286 nambuf[NAMSIZ+1],
287 fullnam[MAXNAM+1];
288
289
290 static
291 struct Lnk {
292 short L_cnt, /* Number of links encountered */
293 L_data; /* Data has been encountered if 1 */
294 struct gen_hdr L_gen; /* gen_hdr information for this file */
295 struct Lnk *L_nxt_p, /* Next file in list */
296 *L_bck_p, /* Previous file in list */
297 *L_lnk_p; /* Next link for this file */
298 } Lnk_hd;
299
300 static
301 struct hdr_cpio Hdr;
302
303 /*
304 * -------------------------------------------------------------------------
305 * Stuff needed to pre-view the name stream
306 *
307 * issymlink is used to remember that the current file is a symlink between
308 * getname() and file_pass(); the former trashes this information immediately
309 * when -L is specified.
310 */
311
312 static
313 int issymlink = 0;
314
315 static
316 FILE *In_p = stdin; /* Where the input comes from */
317
318 typedef struct sl_info
319 {
320 struct sl_info *llink; /* Left subtree ptr (tree depth in *sl_head) */
321 struct sl_info *rlink; /* Right subtree ptr */
322 int bal; /* Subtree balance factor */
323 uint_t sl_count; /* Number of symlinks */
324 int sl_ftype; /* file type of inode */
325 ino_t sl_ino; /* Inode of file */
326 ino_t sl_ino2; /* alternate inode for -Hodc */
327 } sl_info_t;
328
329 typedef struct data_in
330 {
331 int data_in_errno;
332 char data_in_swapfile;
333 char data_in_proc_mode;
334 char data_in_rd_eof;
335 char data_in_wr_part;
336 char data_in_compress_flag;
337 long data_in_cksumval;
338 FILE *data_in_pipef;
339 } data_in_t;
340
341 /*
342 * The following structure maintains a hash entry for the
343 * balancing trees which are allocated for each device nodes.
344 */
345 typedef struct sl_info_link
346 {
347 dev_t dev;
348 sl_info_t *head;
349 struct sl_info_link *next;
350 } sl_info_link_t;
351
352 #define SL_INFO_ALLOC_CHUNK 1024
353 #define NDEVHENTRY 0x40
354 #define DEV_HASHKEY(x) ((x) & (NDEVHENTRY -1))
355
356 /*
357 * For remapping dev,inode for -Hodc archives.
358 */
359
360 typedef struct sl_remap
361 {
362 dev_t dev; /* device */
363 int inode_count; /* # inodes seen on dev */
364 struct sl_remap *next; /* next in the chain */
365 } sl_remap_t;
366
367 /* forward declarations */
368
369 static sl_info_t *sl_info_alloc(void);
370 static sl_info_t *sl_insert(dev_t, ino_t, int);
371 static uint_t sl_numlinks(dev_t, ino_t, int);
372 static void sl_preview_synonyms(void);
373 static void sl_remember_tgt(const struct stat *, int, int);
374 static sl_info_t *sl_search(dev_t, ino_t, int);
375 static sl_info_t *sl_devhash_lookup(dev_t);
376 static void sl_devhash_insert(dev_t, sl_info_t *);
377
378 extern int sl_compare(ino_t, int, ino_t, int);
379 #define sl_compare(lino, lftype, rino, rftype) (lino < rino ? -1 : \
380 (lino > rino ? 1 : (lftype < rftype ? -1 : \
381 (lftype > rftype ? 1 : 0))))
382
383 /* global storage */
384
385 static sl_remap_t *sl_remap_head = NULL; /* head of the inode-remap list */
386 static sl_info_link_t *sl_devhash[NDEVHENTRY]; /* hash table */
387
388 /*
389 * -------------------------------------------------------------------------
390 */
391
392 static
393 struct stat ArchSt, /* stat(2) information of the archive */
394 SrcSt, /* stat(2) information of source file */
395 DesSt, /* stat(2) of destination file */
396 *OldSt = NULL; /* stat info converted to svr32 format */
397
398 /*
399 * bin_mag: Used to validate a binary magic number,
400 * by combining to bytes into an unsigned short.
401 */
402
403 static
404 union bin_mag {
405 unsigned char b_byte[2];
406 ushort_t b_half;
407 } Binmag;
408
409 static
410 union tblock *Thdr_p; /* TAR header pointer */
411
412 static union b_block *bar_Vhdr;
413 static struct gen_hdr Gen_bar_vol;
414
415 /*
416 * swpbuf: Used in swap() to swap bytes within a halfword,
417 * halfwords within a word, or to reverse the order of the
418 * bytes within a word. Also used in mklong() and mkshort().
419 */
420
421 static
422 union swpbuf {
423 unsigned char s_byte[4];
424 ushort_t s_half[2];
425 uint_t s_word;
426 } *Swp_p;
427
428 static
429 char *myname, /* program name */
430 Adir, /* Flags object as a directory */
431 Hiddendir, /* Processing hidden attribute directory */
432 Aspec, /* Flags object as a special file */
433 Do_rename, /* Indicates rename() is to be used */
434 Time[50], /* Array to hold date and time */
435 Ttyname[] = "/dev/tty", /* Controlling console */
436 T_lname[MAXPATHLEN], /* Array to hold links name for tar */
437 *Buf_p, /* Buffer for file system I/O */
438 *Full_p, /* Pointer to full pathname */
439 *Efil_p, /* -E pattern file string */
440 *Eom_p = "Change to part %d and press RETURN key. [q] ",
441 *Fullnam_p, /* Full pathname */
442 *Attrfile_p, /* attribute file */
443 *Hdr_p, /* -H header type string */
444 *IOfil_p, /* -I/-O input/output archive string */
445 *Lnkend_p, /* Pointer to end of Lnknam_p */
446 *Lnknam_p, /* Buffer for linking files with -p option */
447 *Nam_p, /* Array to hold filename */
448 *Savenam_p, /* copy of filename xattr belongs to */
449 *Own_p, /* New owner login id string */
450 *Renam_p, /* Buffer for renaming files */
451 *Renam_attr_p, /* Buffer for renaming attr with sys attrs */
452 *Renametmp_p, /* Tmp Buffer for renaming files */
453 *Symlnk_p, /* Buffer for holding symbolic link name */
454 *Over_p, /* Holds temporary filename when overwriting */
455 **Pat_pp = 0, /* Pattern strings */
456 bar_linkflag, /* flag to indicate if the file is a link */
457 bar_linkname[MAXPATHLEN]; /* store the name of the link */
458
459 static
460 int Append = 0, /* Flag set while searching to end of archive */
461 Archive, /* File descriptor of the archive */
462 Buf_error = 0, /* I/O error occurred during buffer fill */
463 Compress_sparse = 0, /* Compress sparse files */
464 Def_mode = 0777, /* Default file/directory protection modes */
465 Device, /* Device type being accessed (used with libgenIO) */
466 Error_cnt = 0, /* Cumulative count of I/O errors */
467 Finished = 1, /* Indicates that a file transfer has completed */
468 Hdrsz = ASCSZ, /* Fixed length portion of the header */
469 Hdr_type, /* Flag to indicate type of header selected */
470 Ifile, /* File des. of file being archived */
471 Ofile, /* File des. of file being extracted from archive */
472 Use_old_stat = 0, /* Create an old style -Hodc hdr (small dev's) */
473 Onecopy = 0, /* Flags old vs. new link handling */
474 Pad_val = 0, /* Indicates the number of bytes to pad (if any) */
475 PageSize = 0, /* The native page size, used for figuring block size */
476 Volcnt = 1, /* Number of archive volumes processed */
477 Verbcnt = 0, /* Count of number of dots '.' output */
478 Eomflag = 0,
479 Dflag = 0,
480 Atflag = 0, /* Archive/restore extended attributes */
481 SysAtflag = 0, /* Archive/restore extended system attributes */
482 Compressed, /* Flag to indicate if the bar archive is compressed */
483 Bar_vol_num = 0, /* Volume number count for bar archive */
484 privileged = 0, /* Flag set if running with higher privileges */
485 attr_baseparent_fd = -1; /* attribute's base file descriptor */
486
487
488 static
489 gid_t Lastgid = (gid_t)-1; /* Used with -t & -v to record current gid */
490
491 static
492 uid_t Lastuid = (uid_t)-1; /* Used with -t & -v to record current uid */
493
494 static
495 long Args, /* Mask of selected options */
496 Max_namesz = CPATH; /* Maximum size of pathnames/filenames */
497
498 static
499 int Bufsize = BUFSZ; /* Default block size */
500
501
502 static ulong_t Blocks; /* full blocks transferred */
503 static ulong_t SBlocks; /* cumulative char count from short reads */
504
505
506 static off_t Max_offset = BIN_OFFSET_MAX; /* largest file size */
507 static off_t Max_filesz; /* from getrlimit */
508
509 static uint_t Savedev;
510
511 static
512 FILE *Ef_p, /* File pointer of pattern input file */
513 *Err_p = stderr, /* File pointer for error reporting */
514 *Out_p = stdout, /* File pointer for non-archive output */
515 *Rtty_p, /* Input file pointer for interactive rename */
516 *Wtty_p; /* Output file ptr for interactive rename */
517
518 static
519 ushort_t Ftype = S_IFMT; /* File type mask */
520
521 /* ACL support */
522 static struct sec_attr {
523 char attr_type;
524 char attr_len[7];
525 char attr_info[1];
526 } *attr;
527
528 static int Pflag = 0; /* flag indicates that acl is preserved */
529 static int acl_is_set = 0; /* True if an acl was set on the file */
530
531 acl_t *aclp;
532
533 #if defined(O_XATTR)
534 typedef enum {
535 ATTR_OK,
536 ATTR_SKIP,
537 ATTR_CHDIR_ERR,
538 ATTR_OPEN_ERR,
539 ATTR_XATTR_ERR,
540 ATTR_SATTR_ERR
541 } attr_status_t;
542 #endif
543
544 #if defined(O_XATTR)
545 typedef enum {
546 ARC_CREATE,
547 ARC_RESTORE
548 } arc_action_t;
549 #endif
550
551
552 /*
553 *
554 * cpio has been changed to support extended attributes.
555 *
556 * As part of this change cpio has been changed to use the new *at() syscalls
557 * such as openat, fchownat(), unlinkat()...
558 *
559 * This was done so that attributes can be handled with as few code changes
560 * as possible.
561 *
562 * What this means is that cpio now opens the directory that a file or directory
563 * resides in and then performs *at() functions to manipulate the entry.
564 *
565 * For example a new file is now created like this:
566 *
567 * dfd = open(<some dir path>)
568 * fd = openat(dfd, <name>,....);
569 *
570 * or in the case of an extended attribute
571 *
572 * dfd = attropen(<pathname>, ".", ....)
573 *
574 * Once we have a directory file descriptor all of the *at() functions can
575 * be applied to it.
576 *
577 * unlinkat(dfd, <component name>,...)
578 * fchownat(dfd, <component name>,..)
579 *
580 * This works for both normal namespace files and extended attribute file
581 *
582 */
583
584 /*
585 * Extended attribute layout
586 *
587 * Extended attributes are stored in two pieces.
588 * 1. An attribute header which has information about
589 * what file the attribute is for and what the attribute
590 * is named.
591 * 2. The attribute record itself. Stored as a normal file type
592 * of entry.
593 * Both the header and attribute record have special modes/typeflags
594 * associated with them.
595 *
596 * The names of the header in the archive look like:
597 * /dev/null/attr.hdr
598 *
599 * The name of the attribute looks like:
600 * /dev/null/attr.
601 *
602 * This is done so that an archiver that doesn't understand these formats
603 * can just dispose of the attribute records unless the user chooses to
604 * rename them via cpio -r or pax -i
605 *
606 * The format is composed of a fixed size header followed
607 * by a variable sized xattr_buf. If the attribute is a hard link
608 * to another attribute, then another xattr_buf section is included
609 * for the link.
610 *
611 * The xattr_buf is used to define the necessary "pathing" steps
612 * to get to the extended attribute. This is necessary to support
613 * a fully recursive attribute model where an attribute may itself
614 * have an attribute.
615 *
616 * The basic layout looks like this.
617 *
618 * --------------------------------
619 * | |
620 * | xattr_hdr |
621 * | |
622 * --------------------------------
623 * --------------------------------
624 * | |
625 * | xattr_buf |
626 * | |
627 * --------------------------------
628 * --------------------------------
629 * | |
630 * | (optional link info) |
631 * | |
632 * --------------------------------
633 * --------------------------------
634 * | |
635 * | attribute itself |
636 * | stored as normal tar |
637 * | or cpio data with |
638 * | special mode or |
639 * | typeflag |
640 * | |
641 * --------------------------------
642 *
643 */
644
645 /*
646 * Extended attributes structures
647 *
648 * xattrhead is the complete extended attribute header, as read of off
649 * disk/tape. It includes the variable xattr_buf portion.
650 *
651 * xattrp is basically an offset into xattrhead that points to the
652 * "pathing" section which defines how to get to the attribute.
653 *
654 * xattr_linkp is identical to xattrp except that it is used for linked
655 * attributes. It provides the pathing steps to get to the linked
656 * attribute.
657 *
658 * These structures are updated when an extended attribute header is read off
659 * of disk/tape.
660 */
661 static struct xattr_hdr *xattrhead;
662 static struct xattr_buf *xattrp;
663 static struct xattr_buf *xattr_linkp;
664 static int xattrbadhead; /* is extended attribute header bad? */
665
666 static int append_secattr(char **, int *, acl_t *);
667
668 /*
669 * Note regarding cpio and changes to ensure cpio doesn't try to second
670 * guess whether it runs with sufficient privileges or not:
671 *
672 * cpio has been changed so that it doesn't carry a second implementation of
673 * the kernel's policy with respect to privileges. Instead of attempting
674 * to restore uid and gid from an archive only if cpio is run as uid 0,
675 * cpio now *always* tries to restore the uid and gid from the archive
676 * except when the -R option is specified. When the -R is specified,
677 * the uid and gid of the restored file will be changed to those of the
678 * login id specified. In addition, chown(), set_tym(), and chmod() should
679 * only be executed once during archive extraction, and to ensure
680 * setuid/setgid bits are restored properly, chown() should always be
681 * executed before chmod().
682 *
683 * Note regarding debugging mechanism for cpio:
684 *
685 * The following mechanism is provided to allow us to debug cpio in complicated
686 * situations, like when it is part of a pipe. The idea is that you compile
687 * with -DWAITAROUND defined, and then add the "-z" command line option to the
688 * target cpio invocation. If stderr is available, it will tell you to which
689 * pid to attach the debugger; otherwise, use ps to find it. Attach to the
690 * process from the debugger, and, *PRESTO*, you are there!
691 *
692 * Simply assign "waitaround = 0" once you attach to the process, and then
693 * proceed from there as usual.
694 */
695
696 #ifdef WAITAROUND
697 int waitaround = 0; /* wait for rendezvous with the debugger */
698 #endif
699
700 #define EXIT_CODE (Error_cnt > 255 ? 255 : Error_cnt)
701
702 /*
703 * main: Call setup() to process options and perform initializations,
704 * and then select either copy in (-i), copy out (-o), or pass (-p) action.
705 */
706
707 int
main(int argc,char ** argv)708 main(int argc, char **argv)
709 {
710 int i;
711 int passret;
712
713 (void) setlocale(LC_ALL, "");
714 #if !defined(TEXT_DOMAIN) /* Should be defined by cc -D */
715 #define TEXT_DOMAIN "SYS_TEST" /* Use this only if it weren't */
716 #endif
717 (void) textdomain(TEXT_DOMAIN);
718
719 (void) memset(&Gen, 0, sizeof (Gen));
720 myname = e_strdup(E_EXIT, basename(argv[0]));
721 setup(argc, argv);
722
723 if (signal(SIGINT, sigint) == SIG_IGN)
724 (void) signal(SIGINT, SIG_IGN);
725 switch (Args & (OCi | OCo | OCp)) {
726 case OCi: /* COPY IN */
727 Hdr_type = NONE;
728 if (Atflag || SysAtflag) {
729 /*
730 * Save the current working directory, so
731 * we can change back here after cd'ing into
732 * the attribute directory when processing
733 * attributes.
734 */
735 if ((attr_baseparent_fd = save_cwd()) < 0) {
736 msg(EXT, "Unable to open current directory.");
737 }
738 }
739 while ((i = gethdr()) != 0) {
740 Gen.g_dirfd = -1;
741 if (i == 1) {
742 file_in();
743 /*
744 * Any ACL info for this file would or should
745 * have been used after file_in(); clear out
746 * aclp so it is is not erroneously used on
747 * the next file.
748 */
749 if (aclp != NULL) {
750 acl_free(aclp);
751 aclp = NULL;
752 }
753 acl_is_set = 0;
754 }
755 (void) memset(&Gen, 0, sizeof (Gen));
756 }
757 /* Do not count "extra" "read-ahead" buffered data */
758 if (Buffr.b_cnt > Bufsize)
759 Blocks -= (Buffr.b_cnt / Bufsize);
760 break;
761 case OCo: /* COPY OUT */
762 if (Args & OCA) {
763 scan4trail();
764 }
765
766 Gen.g_dirfd = -1;
767 Gen.g_dirpath = NULL;
768 sl_preview_synonyms();
769
770 while ((i = getname()) != 0) {
771 if (i == 1) {
772 (void) file_out();
773 if (Atflag || SysAtflag) {
774 if (Gen.g_dirfd != -1) {
775 (void) close(Gen.g_dirfd);
776 }
777 Gen.g_dirfd = -1;
778 xattrs_out(file_out);
779 }
780 }
781 if (aclp != NULL) {
782 acl_free(aclp);
783 aclp = NULL;
784 acl_is_set = 0;
785 }
786 }
787 write_trail();
788 break;
789 case OCp: /* PASS */
790 sl_preview_synonyms();
791
792 Gen.g_dirfd = -1;
793 Gen.g_passdirfd = -1;
794 Gen.g_dirpath = NULL;
795 Compress_sparse = 1;
796 while (getname()) {
797 /*
798 * If file is a fully qualified path then
799 * file_pass will strip off the leading '/'
800 * and we need to save off the unstripped
801 * name for attribute traversal.
802 */
803 if (Atflag || SysAtflag) {
804 (void) strcpy(Savenam_p, Gen.g_nam_p);
805 }
806 passret = file_pass();
807 if (aclp != NULL) {
808 acl_free(aclp);
809 aclp = NULL;
810 acl_is_set = 0;
811 }
812 if (Gen.g_passdirfd != -1)
813 (void) close(Gen.g_passdirfd);
814 Gen.g_passdirfd = -1;
815 if (Atflag || SysAtflag) {
816 if (Gen.g_dirfd != -1) {
817 (void) close(Gen.g_dirfd);
818 }
819 Gen.g_dirfd = -1;
820 if (passret != FILE_LINKED) {
821 Gen.g_nam_p = Savenam_p;
822 xattrs_out(file_pass);
823 }
824 }
825 }
826 break;
827 default:
828 msg(EXT, "Impossible action.");
829 }
830 if (Ofile > 0) {
831 if (close(Ofile) != 0)
832 msg(EXTN, "close error");
833 }
834 if (Archive > 0) {
835 if (close(Archive) != 0)
836 msg(EXTN, "close error");
837 }
838 if ((Args & OCq) == 0) {
839 Blocks = (ulong_t)(Blocks * Bufsize + SBlocks +
840 0x1FF) >> 9;
841 msg(EPOST, "%ld blocks", Blocks);
842 }
843 if (Error_cnt)
844 msg(EPOST, "%d error(s)", Error_cnt);
845 return (EXIT_CODE);
846 }
847
848 /*
849 * add_lnk: Add a linked file's header to the linked file data structure, by
850 * either adding it to the end of an existing sub-list or starting
851 * a new sub-list. Each sub-list saves the links to a given file.
852 *
853 * Directly returns a pointer to the new entry; returns a pointer to the head
854 * of the sub-list in which that entry is located through the argument.
855 */
856
857 static struct Lnk *
add_lnk(struct Lnk ** sublist_return)858 add_lnk(struct Lnk **sublist_return)
859 {
860 struct Lnk *new_entry, *sublist;
861
862 for (sublist = Lnk_hd.L_nxt_p;
863 sublist != &Lnk_hd;
864 sublist = sublist->L_nxt_p) {
865 if (sublist->L_gen.g_ino == G_p->g_ino &&
866 sublist->L_gen.g_dev == G_p->g_dev) {
867 /* found */
868 break;
869 }
870 }
871
872 new_entry = e_zalloc(E_EXIT, sizeof (struct Lnk));
873
874 new_entry->L_lnk_p = NULL;
875 new_entry->L_gen = *G_p; /* structure copy */
876
877 new_entry->L_gen.g_nam_p = e_zalloc(E_EXIT, (size_t)G_p->g_namesz);
878
879 (void) strcpy(new_entry->L_gen.g_nam_p, G_p->g_nam_p);
880
881 if (sublist == &Lnk_hd) {
882 /* start new sub-list */
883 new_entry->L_nxt_p = &Lnk_hd;
884 new_entry->L_bck_p = Lnk_hd.L_bck_p;
885 Lnk_hd.L_bck_p = new_entry->L_bck_p->L_nxt_p = new_entry;
886 new_entry->L_lnk_p = NULL;
887 new_entry->L_cnt = 1;
888 new_entry->L_data = Onecopy ? 0 : 1;
889 sublist = new_entry;
890 } else {
891 /* add to existing sub-list */
892 struct Lnk *ptr;
893
894 sublist->L_cnt++;
895
896 for (ptr = sublist;
897 ptr->L_lnk_p != NULL;
898 ptr = ptr->L_lnk_p) {
899 ptr->L_gen.g_filesz = G_p->g_filesz;
900 }
901
902 ptr->L_gen.g_filesz = G_p->g_filesz;
903 ptr->L_lnk_p = new_entry;
904 }
905
906 *sublist_return = sublist;
907 return (new_entry);
908 }
909
910 /*
911 * bfill: Read req_cnt bytes (out of filelen bytes) from the I/O buffer,
912 * moving them to rd_buf_p. When there are no bytes left in the I/O buffer,
913 * Fillbuf is set and the I/O buffer is filled. The variable dist is the
914 * distance to lseek if an I/O error is encountered with the -k option set
915 * (converted to a multiple of Bufsize).
916 */
917
918 static int
bfill(void)919 bfill(void)
920 {
921 int i = 0, rv = 0;
922 static int eof = 0;
923
924 if (!Dflag) {
925 while ((Buffr.b_end_p - Buffr.b_in_p) >= Bufsize) {
926 errno = 0;
927 if ((rv = g_read(Device, Archive, Buffr.b_in_p, Bufsize)) < 0) {
928 if (((Buffr.b_end_p - Buffr.b_in_p) >= Bufsize) &&
929 (Eomflag == 0)) {
930 Eomflag = 1;
931 return (1);
932 }
933 if (errno == ENOSPC) {
934 (void) chgreel(INPUT);
935 if (Hdr_type == BAR) {
936 skip_bar_volhdr();
937 }
938 continue;
939 } else if (Args & OCk) {
940 if (i++ > MX_SEEKS)
941 msg(EXT, "Cannot recover.");
942 if (lseek(Archive, Bufsize, SEEK_REL) < 0)
943 msg(EXTN, "Cannot lseek()");
944 Error_cnt++;
945 Buf_error++;
946 rv = 0;
947 continue;
948 } else
949 ioerror(INPUT);
950 } /* (rv = g_read(Device, Archive ... */
951 if (Hdr_type != BAR || rv == Bufsize) {
952 Buffr.b_in_p += rv;
953 Buffr.b_cnt += rv;
954 }
955 if (rv == Bufsize) {
956 eof = 0;
957 Blocks++;
958 } else if (rv == 0) {
959 if (!eof) {
960 eof = 1;
961 break;
962 }
963 (void) chgreel(INPUT);
964 eof = 0; /* reset the eof after chgreel */
965
966 /*
967 * if spans multiple volume, skip the volume header of
968 * the next volume so that the file currently being
969 * extracted can continue to be extracted.
970 */
971 if (Hdr_type == BAR) {
972 skip_bar_volhdr();
973 }
974
975 continue;
976 } else {
977 eof = 0;
978 SBlocks += (ulong_t)rv;
979 }
980 } /* (Buffr.b_end_p - Buffr.b_in_p) <= Bufsize */
981
982 } else { /* Dflag */
983 errno = 0;
984 if ((rv = g_read(Device, Archive, Buffr.b_in_p, Bufsize)) < 0) {
985 return (-1);
986 } /* (rv = g_read(Device, Archive ... */
987 Buffr.b_in_p += rv;
988 Buffr.b_cnt += rv;
989 if (rv == Bufsize) {
990 eof = 0;
991 Blocks++;
992 } else if (!rv) {
993 if (!eof) {
994 eof = 1;
995 return (rv);
996 }
997 return (-1);
998 } else {
999 eof = 0;
1000 SBlocks += (ulong_t)rv;
1001 }
1002 }
1003 return (rv);
1004 }
1005
1006 /*
1007 * bflush: Move wr_cnt bytes from data_p into the I/O buffer. When the
1008 * I/O buffer is full, Flushbuf is set and the buffer is written out.
1009 */
1010
1011 static void
bflush(void)1012 bflush(void)
1013 {
1014 int rv;
1015
1016 while (Buffr.b_cnt >= Bufsize) {
1017 errno = 0;
1018 if ((rv = g_write(Device, Archive, Buffr.b_out_p,
1019 Bufsize)) < 0) {
1020 if (errno == ENOSPC && !Dflag)
1021 rv = chgreel(OUTPUT);
1022 else
1023 ioerror(OUTPUT);
1024 }
1025 Buffr.b_out_p += rv;
1026 Buffr.b_cnt -= rv;
1027 if (rv == Bufsize)
1028 Blocks++;
1029 else if (rv > 0)
1030 SBlocks += (ulong_t)rv;
1031 }
1032 rstbuf();
1033 }
1034
1035 /*
1036 * chgreel: Determine if end-of-medium has been reached. If it has,
1037 * close the current medium and prompt the user for the next medium.
1038 */
1039
1040 static int
chgreel(int dir)1041 chgreel(int dir)
1042 {
1043 int lastchar, tryagain, askagain, rv;
1044 int tmpdev;
1045 char str[APATH];
1046 struct stat statb;
1047
1048 rv = 0;
1049 if (fstat(Archive, &statb) < 0)
1050 msg(EXTN, "Error during stat() of archive");
1051 if ((statb.st_mode & S_IFMT) != S_IFCHR) {
1052 if (dir == INPUT) {
1053 msg(EXT, "%s%s\n",
1054 "Can't read input: end of file encountered ",
1055 "prior to expected end of archive.");
1056 }
1057 }
1058 msg(EPOST, "\007End of medium on \"%s\".", dir ? "output" : "input");
1059 if (is_floppy(Archive))
1060 (void) ioctl(Archive, FDEJECT, NULL);
1061 if ((close(Archive) != 0) && (dir == OUTPUT))
1062 msg(EXTN, "close error");
1063 Archive = 0;
1064 Volcnt++;
1065 for (;;) {
1066 if (Rtty_p == NULL)
1067 Rtty_p = fopen(Ttyname, "r");
1068 do { /* tryagain */
1069 if (IOfil_p) {
1070 do {
1071 msg(EPOST, Eom_p, Volcnt);
1072 if (!Rtty_p || fgets(str, sizeof (str),
1073 Rtty_p) == NULL)
1074 msg(EXT, "Cannot read tty.");
1075 askagain = 0;
1076 switch (*str) {
1077 case '\n':
1078 (void) strcpy(str, IOfil_p);
1079 break;
1080 case 'q':
1081 exit(EXIT_CODE);
1082 default:
1083 askagain = 1;
1084 }
1085 } while (askagain);
1086 } else {
1087
1088 if (Hdr_type == BAR)
1089 Bar_vol_num++;
1090
1091 msg(EPOST,
1092 "To continue, type device/file name when "
1093 "ready.");
1094 if (!Rtty_p || fgets(str, sizeof (str),
1095 Rtty_p) == NULL)
1096 msg(EXT, "Cannot read tty.");
1097 lastchar = strlen(str) - 1;
1098 if (*(str + lastchar) == '\n') /* remove '\n' */
1099 *(str + lastchar) = '\0';
1100 if (!*str)
1101 exit(EXIT_CODE);
1102 }
1103 tryagain = 0;
1104 if ((Archive = open(str, dir)) < 0) {
1105 msg(ERRN, "Cannot open \"%s\"", str);
1106 tryagain = 1;
1107 }
1108 } while (tryagain);
1109 (void) g_init(&tmpdev, &Archive);
1110 if (tmpdev != Device)
1111 msg(EXT, "Cannot change media types in mid-stream.");
1112 if (dir == INPUT)
1113 break;
1114 else { /* dir == OUTPUT */
1115 errno = 0;
1116 if ((rv = g_write(Device, Archive, Buffr.b_out_p,
1117 Bufsize)) == Bufsize)
1118 break;
1119 else
1120 msg(ERR,
1121 "Unable to write this medium, try "
1122 "another.");
1123 }
1124 } /* ;; */
1125 Eomflag = 0;
1126 return (rv);
1127 }
1128
1129 /*
1130 * ckname: Check filenames against user specified patterns,
1131 * and/or ask the user for new name when -r is used.
1132 */
1133
1134 static int
ckname(int flag)1135 ckname(int flag)
1136 {
1137 int lastchar;
1138 size_t rename_bufsz = Max_namesz + 1;
1139
1140 if (Hdr_type != TAR && Hdr_type != USTAR && Hdr_type != BAR) {
1141 /* Re-visit tar size issues later */
1142 if (G_p->g_namesz - 1 > Max_namesz) {
1143 msg(ERR, "Name exceeds maximum length - skipped.");
1144 return (F_SKIP);
1145 }
1146 }
1147
1148 if (Pat_pp && !matched())
1149 return (F_SKIP);
1150
1151 /* rename interactively */
1152 if ((Args & OCr) && !Adir && !G_p->g_rw_sysattr) {
1153 (void) fprintf(Wtty_p, gettext("Rename \"%s%s%s\"? "),
1154 (G_p->g_attrnam_p == NULL) ? G_p->g_nam_p : Renam_p,
1155 (G_p->g_attrnam_p == NULL) ? "" : gettext(" Attribute "),
1156 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p);
1157 (void) fflush(Wtty_p);
1158 if (fgets(Renametmp_p, rename_bufsz, Rtty_p) == NULL)
1159 msg(EXT, "Cannot read tty.");
1160 if (feof(Rtty_p))
1161 exit(EXIT_CODE);
1162 lastchar = strlen(Renametmp_p) - 1;
1163
1164 /* remove trailing '\n' */
1165 if (*(Renametmp_p + lastchar) == '\n')
1166 *(Renametmp_p + lastchar) = '\0';
1167 if (*Renametmp_p == '\0') {
1168 msg(POST, "%s%s%s Skipped.",
1169 (G_p->g_attrnam_p == NULL) ? G_p->g_nam_p :
1170 G_p->g_attrfnam_p,
1171 (G_p->g_attrnam_p == NULL) ? "" :
1172 gettext(" Attribute "),
1173 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p);
1174 if (G_p->g_attrparent_p == NULL) {
1175 *G_p->g_nam_p = '\0';
1176 }
1177 if (Renam_attr_p) {
1178 *Renam_attr_p = '\0';
1179 }
1180 return (F_SKIP);
1181 } else if (strcmp(Renametmp_p, ".") != 0) {
1182 if (G_p->g_attrnam_p == NULL) {
1183 if (strlen(Renametmp_p) > strlen(
1184 G_p->g_nam_p)) {
1185 if ((G_p->g_nam_p != &nambuf[0]) &&
1186 (G_p->g_nam_p != &fullnam[0])) {
1187 free(G_p->g_nam_p);
1188 G_p->g_nam_p = e_zalloc(E_EXIT,
1189 rename_bufsz);
1190 }
1191 }
1192 if (Renam_attr_p) {
1193 *Renam_attr_p = '\0';
1194 }
1195 if ((strlcpy(Renam_p, Renametmp_p,
1196 rename_bufsz) > rename_bufsz) ||
1197 (strlcpy(G_p->g_nam_p, Renametmp_p,
1198 rename_bufsz) > rename_bufsz)) {
1199 msg(EXTN, "buffer overflow");
1200 }
1201 } else {
1202 if (G_p->g_attrnam_p != NULL) {
1203 free(G_p->g_attrnam_p);
1204 G_p->g_attrnam_p = e_strdup(E_EXIT,
1205 Renametmp_p);
1206 (void) strcpy(G_p->g_nam_p, Renam_p);
1207 if (Renam_attr_p) {
1208 if (strlcpy(Renam_attr_p,
1209 Renametmp_p, rename_bufsz) >
1210 rename_bufsz) {
1211 msg(EXTN,
1212 "buffer overflow");
1213 }
1214 }
1215 }
1216 }
1217 } else {
1218 if (G_p->g_attrnam_p == NULL) {
1219 *Renam_p = '\0';
1220 }
1221 if (Renam_attr_p) {
1222 *Renam_attr_p = '\0';
1223 }
1224 }
1225 }
1226 if (flag != 0 || Onecopy == 0) {
1227 VERBOSE((Args & OCt), G_p->g_nam_p);
1228 }
1229 if (Args & OCt)
1230 return (F_SKIP);
1231 return (F_EXTR);
1232 }
1233
1234 /*
1235 * ckopts: Check the validity of all command line options.
1236 */
1237
1238 static void
ckopts(long mask)1239 ckopts(long mask)
1240 {
1241 int oflag;
1242 char *t_p;
1243 long errmsk;
1244 uid_t Euid = geteuid(); /* Effective uid of invoker */
1245 #ifdef SOLARIS_PRIVS
1246 priv_set_t *privset;
1247 priv_set_t *zones_privset;
1248 #endif /* SOLARIS_PRIVS */
1249
1250 if (mask & OCi) {
1251 errmsk = mask & INV_MSK4i;
1252 } else if (mask & OCo) {
1253 errmsk = mask & INV_MSK4o;
1254 } else if (mask & OCp) {
1255 errmsk = mask & INV_MSK4p;
1256 } else {
1257 msg(ERR, "One of -i, -o or -p must be specified.");
1258 errmsk = 0;
1259 }
1260
1261 if (errmsk) {
1262 /* if non-zero, invalid options were specified */
1263 Error_cnt++;
1264 }
1265
1266 if ((mask & OCa) && (mask & OCm) && ((mask & OCi) ||
1267 (mask & OCo))) {
1268 msg(ERR, "-a and -m are mutually exclusive.");
1269 }
1270
1271 if ((mask & OCc) && (mask & OCH) &&
1272 (strcmp("odc", Hdr_p) != 0 && strcmp("odc_sparse", Hdr_p) != 0)) {
1273 msg(ERR, "-c and -H are mutually exclusive.");
1274 }
1275
1276 if ((mask & OCv) && (mask & OCV)) {
1277 msg(ERR, "-v and -V are mutually exclusive.");
1278 }
1279
1280 if ((mask & OCt) && (mask & OCV)) {
1281 msg(ERR, "-t and -V are mutually exclusive.");
1282 }
1283
1284 if ((mask & OCB) && (mask & OCC)) {
1285 msg(ERR, "-B and -C are mutually exclusive.");
1286 }
1287
1288 if ((mask & OCH) && (mask & OC6)) {
1289 msg(ERR, "-H and -6 are mutually exclusive.");
1290 }
1291
1292 if ((mask & OCM) && !((mask & OCI) || (mask & OCO))) {
1293 msg(ERR, "-M not meaningful without -O or -I.");
1294 }
1295
1296 if ((mask & OCA) && !(mask & OCO)) {
1297 msg(ERR, "-A requires the -O option.");
1298 }
1299
1300 if (Bufsize <= 0) {
1301 msg(ERR, "Illegal size given for -C option.");
1302 }
1303
1304 if (mask & OCH) {
1305 t_p = Hdr_p;
1306
1307 while (*t_p != '\0') {
1308 if (isupper(*t_p)) {
1309 *t_p = 'a' + (*t_p - 'A');
1310 }
1311
1312 t_p++;
1313 }
1314
1315 if (!(strcmp("odc", Hdr_p))) {
1316 Hdr_type = CHR;
1317 Max_namesz = CPATH;
1318 Onecopy = 0;
1319 Use_old_stat = 1;
1320 } else if (!(strcmp("odc_sparse", Hdr_p))) {
1321 Hdr_type = CHR;
1322 Max_namesz = CPATH;
1323 Onecopy = 0;
1324 Use_old_stat = 1;
1325 Compress_sparse = 1;
1326 } else if (!(strcmp("ascii_sparse", Hdr_p))) {
1327 Hdr_type = ASC;
1328 Max_namesz = APATH;
1329 Onecopy = 1;
1330 Compress_sparse = 1;
1331 } else if (!(strcmp("crc", Hdr_p))) {
1332 Hdr_type = CRC;
1333 Max_namesz = APATH;
1334 Onecopy = 1;
1335 } else if (!(strcmp("tar", Hdr_p))) {
1336 if (Args & OCo) {
1337 Hdr_type = USTAR;
1338 Max_namesz = HNAMLEN - 1;
1339 } else {
1340 Hdr_type = TAR;
1341 Max_namesz = TNAMLEN - 1;
1342 }
1343 Onecopy = 0;
1344 } else if (!(strcmp("ustar", Hdr_p))) {
1345 Hdr_type = USTAR;
1346 Max_namesz = HNAMLEN - 1;
1347 Onecopy = 0;
1348 } else if (!(strcmp("bar", Hdr_p))) {
1349 if ((Args & OCo) || (Args & OCp)) {
1350 msg(ERR,
1351 "Header type bar can only be used with -i");
1352 }
1353
1354 if (Args & OCP) {
1355 msg(ERR,
1356 "Can't preserve using bar header");
1357 }
1358
1359 Hdr_type = BAR;
1360 Max_namesz = TNAMLEN - 1;
1361 Onecopy = 0;
1362 } else {
1363 msg(ERR, "Invalid header \"%s\" specified", Hdr_p);
1364 }
1365 }
1366
1367 if (mask & OCr) {
1368 Rtty_p = fopen(Ttyname, "r");
1369 Wtty_p = fopen(Ttyname, "w");
1370
1371 if (Rtty_p == NULL || Wtty_p == NULL) {
1372 msg(ERR, "Cannot rename, \"%s\" missing", Ttyname);
1373 }
1374 }
1375
1376 if ((mask & OCE) && (Ef_p = fopen(Efil_p, "r")) == NULL) {
1377 msg(ERR, "Cannot open \"%s\" to read patterns", Efil_p);
1378 }
1379
1380 if ((mask & OCI) && (Archive = open(IOfil_p, O_RDONLY)) < 0) {
1381 msg(ERR, "Cannot open \"%s\" for input", IOfil_p);
1382 }
1383
1384 if (mask & OCO) {
1385 if (mask & OCA) {
1386 if ((Archive = open(IOfil_p, O_RDWR)) < 0) {
1387 msg(ERR,
1388 "Cannot open \"%s\" for append",
1389 IOfil_p);
1390 }
1391 } else {
1392 oflag = (O_WRONLY | O_CREAT | O_TRUNC);
1393
1394 if ((Archive = open(IOfil_p, oflag, 0777)) < 0) {
1395 msg(ERR,
1396 "Cannot open \"%s\" for output",
1397 IOfil_p);
1398 }
1399 }
1400 }
1401
1402 #ifdef SOLARIS_PRIVS
1403 if ((privset = priv_allocset()) == NULL) {
1404 msg(ERR, "Unable to allocate privilege set");
1405 } else if (getppriv(PRIV_EFFECTIVE, privset) != 0) {
1406 msg(ERR, "Unable to obtain privilege set");
1407 } else {
1408 zones_privset = priv_str_to_set("zone", "", NULL);
1409 if (zones_privset != NULL) {
1410 privileged = (priv_issubset(zones_privset,
1411 privset) == B_TRUE);
1412 priv_freeset(zones_privset);
1413 } else {
1414 msg(ERR, "Unable to map privilege to privilege set");
1415 }
1416 }
1417 if (privset != NULL) {
1418 priv_freeset(privset);
1419 }
1420 #else
1421 privileged = (Euid == 0);
1422 #endif /* SOLARIS_PRIVS */
1423
1424 if (mask & OCR) {
1425 if ((Rpw_p = getpwnam(Own_p)) == NULL) {
1426 msg(ERR, "\"%s\" is not a valid user id", Own_p);
1427 } else if ((Euid != Rpw_p->pw_uid) && !privileged) {
1428 msg(ERR, "R option only valid for super-user or "
1429 "id matches login id of user executing cpio");
1430 }
1431 }
1432
1433 if ((mask & OCo) && !(mask & OCO)) {
1434 Out_p = stderr;
1435 }
1436
1437 if ((mask & OCp) && ((mask & (OCB|OCC)) == 0)) {
1438 /*
1439 * We are in pass mode with no block size specified. Use the
1440 * larger of the native page size and 8192.
1441 */
1442
1443 Bufsize = (PageSize > 8192) ? PageSize : 8192;
1444 }
1445 }
1446
1447 /*
1448 * cksum: Calculate the simple checksum of a file (CRC) or header
1449 * (TARTYP (TAR and USTAR)). For -o and the CRC header, the file is opened and
1450 * the checksum is calculated. For -i and the CRC header, the checksum
1451 * is calculated as each block is transferred from the archive I/O buffer
1452 * to the file system I/O buffer. The TARTYP (TAR and USTAR) headers calculate
1453 * the simple checksum of the header (with the checksum field of the
1454 * header initialized to all spaces (\040).
1455 */
1456
1457 static uint_t
cksum(char hdr,int byt_cnt,int * err)1458 cksum(char hdr, int byt_cnt, int *err)
1459 {
1460 char *crc_p, *end_p;
1461 int cnt;
1462 uint_t checksum = 0, have;
1463 off_t lcnt;
1464
1465 if (err != NULL)
1466 *err = 0;
1467 switch (hdr) {
1468 case CRC:
1469 if (Args & OCi) { /* do running checksum */
1470 end_p = Buffr.b_out_p + byt_cnt;
1471 for (crc_p = Buffr.b_out_p; crc_p < end_p; crc_p++)
1472 checksum += (uint_t)*crc_p;
1473 break;
1474 }
1475 /* OCo - do checksum of file */
1476 lcnt = G_p->g_filesz;
1477
1478 while (lcnt > 0) {
1479 have = (lcnt < Bufsize) ? lcnt : Bufsize;
1480 errno = 0;
1481 if (read(Ifile, Buf_p, have) != have) {
1482 msg(ERR, "Error computing checksum.");
1483 if (err != NULL)
1484 *err = 1;
1485 break;
1486 }
1487 end_p = Buf_p + have;
1488 for (crc_p = Buf_p; crc_p < end_p; crc_p++)
1489 checksum += (long)*crc_p;
1490 lcnt -= have;
1491 }
1492 if (lseek(Ifile, (off_t)0, SEEK_ABS) < 0)
1493 msg(ERRN, "Cannot reset file after checksum");
1494 break;
1495 case TARTYP: /* TAR and USTAR */
1496 crc_p = Thdr_p->tbuf.t_cksum;
1497 for (cnt = 0; cnt < TCRCLEN; cnt++) {
1498 *crc_p = '\040';
1499 crc_p++;
1500 }
1501 crc_p = (char *)Thdr_p;
1502 for (cnt = 0; cnt < TARSZ; cnt++) {
1503 /*
1504 * tar uses unsigned checksum, so we must use unsigned
1505 * here in order to be able to read tar archives.
1506 */
1507 checksum += (long)((unsigned char)(*crc_p));
1508 crc_p++;
1509 }
1510 break;
1511 default:
1512 msg(EXT, "Impossible header type.");
1513 } /* hdr */
1514 return (checksum);
1515 }
1516
1517 /*
1518 * creat_hdr: Fill in the generic header structure with the specific
1519 * header information based on the value of Hdr_type.
1520 *
1521 * return (1) if this process was successful, and (0) otherwise.
1522 */
1523
1524 static int
creat_hdr(void)1525 creat_hdr(void)
1526 {
1527 ushort_t ftype;
1528 int fullnamesize;
1529 dev_t dev;
1530 ino_t ino;
1531
1532 ftype = SrcSt.st_mode & Ftype;
1533 Adir = (ftype == S_IFDIR);
1534 Aspec = (ftype == S_IFBLK || ftype == S_IFCHR || ftype == S_IFIFO ||
1535 ftype == S_IFSOCK);
1536 switch (Hdr_type) {
1537 case BIN:
1538 Gen.g_magic = CMN_BIN;
1539 break;
1540 case CHR:
1541 Gen.g_magic = CMN_BIN;
1542 break;
1543 case ASC:
1544 Gen.g_magic = CMN_ASC;
1545 break;
1546 case CRC:
1547 Gen.g_magic = CMN_CRC;
1548 break;
1549 case USTAR:
1550 /*
1551 * If the length of the full name is greater than 256,
1552 * print out a message and return.
1553 */
1554 if ((fullnamesize = strlen(Gen.g_nam_p)) > MAXNAM) {
1555 msg(ERR,
1556 "%s: file name too long", Gen.g_nam_p);
1557 return (0);
1558 } else if (fullnamesize > NAMSIZ) {
1559 /*
1560 * The length of the full name is greater than
1561 * 100, so we must split the filename from the
1562 * path
1563 */
1564 char namebuff[NAMSIZ+1];
1565 char prebuff[PRESIZ+1];
1566 char *lastslash;
1567 int presize, namesize;
1568
1569 (void) memset(namebuff, '\0',
1570 sizeof (namebuff));
1571 (void) memset(prebuff, '\0', sizeof (prebuff));
1572
1573 lastslash = strrchr(Gen.g_nam_p, '/');
1574
1575 if (lastslash != NULL) {
1576 namesize = strlen(++lastslash);
1577 presize = fullnamesize - namesize - 1;
1578 } else {
1579 namesize = fullnamesize;
1580 lastslash = Gen.g_nam_p;
1581 presize = 0;
1582 }
1583
1584 /*
1585 * If the filename is greater than 100 we can't
1586 * archive the file
1587 */
1588 if (namesize > NAMSIZ) {
1589 msg(ERR,
1590 "%s: filename is greater than %d",
1591 lastslash, NAMSIZ);
1592 return (0);
1593 }
1594 (void) strncpy(&namebuff[0], lastslash,
1595 namesize);
1596 /*
1597 * If the prefix is greater than 155 we can't
1598 * archive the file.
1599 */
1600 if (presize > PRESIZ) {
1601 msg(ERR,
1602 "%s: prefix is greater than %d",
1603 Gen.g_nam_p, PRESIZ);
1604 return (0);
1605 }
1606 (void) strncpy(&prebuff[0], Gen.g_nam_p,
1607 presize);
1608
1609 Gen.g_tname = e_zalloc(E_EXIT, namesize + 1);
1610 (void) strcpy(Gen.g_tname, namebuff);
1611
1612 Gen.g_prefix = e_zalloc(E_EXIT, presize + 1);
1613 (void) strcpy(Gen.g_prefix, prebuff);
1614 } else {
1615 Gen.g_tname = Gen.g_nam_p;
1616 }
1617 (void) strcpy(Gen.g_tmagic, "ustar");
1618 (void) memcpy(Gen.g_version, "00", 2);
1619
1620 dpasswd = getpwuid(SrcSt.st_uid);
1621 if (dpasswd == NULL) {
1622 msg(EPOST,
1623 "cpio: could not get passwd information "
1624 "for %s%s%s",
1625 (Gen.g_attrnam_p == NULL) ?
1626 Gen.g_nam_p : Gen.g_attrfnam_p,
1627 (Gen.g_attrnam_p == NULL) ?
1628 "" : Gen.g_rw_sysattr ?
1629 gettext(" System Attribute ") :
1630 gettext(" Attribute "),
1631 (Gen.g_attrnam_p == NULL) ?
1632 "" : Gen.g_attrnam_p);
1633 /* make name null string */
1634 Gen.g_uname[0] = '\0';
1635 } else {
1636 (void) strncpy(&Gen.g_uname[0],
1637 dpasswd->pw_name, 32);
1638 }
1639 dgroup = getgrgid(SrcSt.st_gid);
1640 if (dgroup == NULL) {
1641 msg(EPOST,
1642 "cpio: could not get group information "
1643 "for %s%s%s",
1644 (Gen.g_attrnam_p == NULL) ?
1645 Gen.g_nam_p : Gen.g_attrfnam_p,
1646 (Gen.g_attrnam_p == NULL) ?
1647 "" : Gen.g_rw_sysattr ?
1648 gettext(" System Attribute ") :
1649 gettext(" Attribute "),
1650 (Gen.g_attrnam_p == NULL) ?
1651 "" : Gen.g_attrnam_p);
1652 /* make name null string */
1653 Gen.g_gname[0] = '\0';
1654 } else {
1655 (void) strncpy(&Gen.g_gname[0],
1656 dgroup->gr_name, 32);
1657 }
1658 Gen.g_typeflag = tartype(ftype);
1659 /* FALLTHROUGH */
1660 case TAR:
1661 (void) memset(T_lname, '\0', sizeof (T_lname));
1662 break;
1663 default:
1664 msg(EXT, "Impossible header type.");
1665 }
1666
1667 if (Use_old_stat && (Gen.g_attrnam_p != NULL)) {
1668 /*
1669 * When processing extended attributes, creat_hdr()
1670 * can get called multiple times which means that
1671 * SrcSt.st.st_dev would have gotten converted to
1672 * -Hodc format. We should always use the original
1673 * device here as we need to be able to match on
1674 * the original device id from the file that was
1675 * previewed in sl_preview_synonyms().
1676 */
1677 dev = Savedev;
1678 } else {
1679 dev = SrcSt.st_dev;
1680 }
1681 ino = SrcSt.st_ino;
1682
1683 if (Use_old_stat) {
1684 SrcSt = *OldSt;
1685 }
1686
1687 Gen.g_namesz = strlen(Gen.g_nam_p) + 1;
1688 Gen.g_uid = SrcSt.st_uid;
1689 Gen.g_gid = SrcSt.st_gid;
1690 Gen.g_dev = SrcSt.st_dev;
1691
1692 if (Use_old_stat) {
1693 /* -Hodc */
1694
1695 sl_info_t *p = sl_search(dev, ino, ftype);
1696 Gen.g_ino = p ? p->sl_ino2 : -1;
1697
1698 if (Gen.g_ino == UINT_MAX) {
1699 msg(ERR, "%s%s%s: cannot be archived - inode too big "
1700 "for -Hodc format",
1701 (Gen.g_attrnam_p == NULL) ?
1702 Gen.g_nam_p : Gen.g_attrfnam_p,
1703 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_rw_sysattr ?
1704 gettext(" System Attribute ") :
1705 gettext(" Attribute "),
1706 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_attrnam_p);
1707 return (0);
1708 }
1709 } else {
1710 Gen.g_ino = SrcSt.st_ino;
1711 }
1712
1713 Gen.g_mode = SrcSt.st_mode;
1714 Gen.g_mtime = SrcSt.st_mtime;
1715 Gen.g_nlink = Adir ? SrcSt.st_nlink : sl_numlinks(dev, ino, ftype);
1716
1717 if (ftype == S_IFREG || ftype == S_IFLNK)
1718 Gen.g_filesz = (off_t)SrcSt.st_size;
1719 else
1720 Gen.g_filesz = (off_t)0;
1721 Gen.g_rdev = SrcSt.st_rdev;
1722 return (1);
1723 }
1724
1725 /*
1726 * creat_lnk: Create a link from the existing name1_p to name2_p.
1727 */
1728
1729 static
1730 int
creat_lnk(int dirfd,char * name1_p,char * name2_p)1731 creat_lnk(int dirfd, char *name1_p, char *name2_p)
1732 {
1733 int cnt = 0;
1734
1735 do {
1736 errno = 0;
1737 if (!link(name1_p, name2_p)) {
1738 if (aclp != NULL) {
1739 acl_free(aclp);
1740 aclp = NULL;
1741 acl_is_set = 0;
1742 }
1743 cnt = 0;
1744 break;
1745 } else if ((errno == EEXIST) && (cnt == 0)) {
1746 struct stat lsb1;
1747 struct stat lsb2;
1748
1749 /*
1750 * Check to see if we are trying to link this
1751 * file to itself. If so, count the effort as
1752 * successful. If the two files are different,
1753 * or if either lstat is unsuccessful, proceed
1754 * as we would have otherwise; the appropriate
1755 * error will be reported subsequently.
1756 */
1757
1758 if (lstat(name1_p, &lsb1) != 0) {
1759 msg(ERR, "Cannot lstat source file %s",
1760 name1_p);
1761 } else {
1762 if (lstat(name2_p, &lsb2) != 0) {
1763 msg(ERR, "Cannot lstat "
1764 "destination file %s", name2_p);
1765 } else {
1766 if (lsb1.st_dev == lsb2.st_dev &&
1767 lsb1.st_ino == lsb2.st_ino) {
1768 VERBOSE((Args & (OCv | OCV)),
1769 name2_p);
1770 return (0);
1771 }
1772 }
1773 }
1774
1775 if (!(Args & OCu) && G_p->g_mtime <= DesSt.st_mtime)
1776 msg(ERR, "Existing \"%s\" same age or newer",
1777 name2_p);
1778 else if (unlinkat(dirfd, get_component(name2_p), 0) < 0)
1779 msg(ERRN, "Error cannot unlink \"%s\"",
1780 name2_p);
1781 }
1782 cnt++;
1783 } while ((cnt < 2) && missdir(name2_p) == 0);
1784 if (!cnt) {
1785 char *newname;
1786 char *fromname;
1787 char *attrname;
1788
1789 newname = name2_p;
1790 fromname = name1_p;
1791 attrname = Gen.g_attrnam_p;
1792 if (attrname) {
1793 if (Args & OCp) {
1794 newname = fromname = Fullnam_p;
1795 } else {
1796 newname = Gen.g_attrfnam_p;
1797 }
1798 }
1799 if (Args & OCv) {
1800 (void) fprintf(Err_p,
1801 gettext("%s%s%s linked to %s%s%s\n"), newname,
1802 (attrname == NULL) ? "" : gettext(" attribute "),
1803 (attrname == NULL) ? "" : attrname,
1804 (attrname == NULL) ? fromname : newname,
1805 (attrname == NULL) ? "" : gettext(" attribute "),
1806 (attrname == NULL) ? "" : name1_p);
1807 } else {
1808 VERBOSE((Args & (OCv | OCV)), newname);
1809 }
1810 } else if (cnt == 1)
1811 msg(ERRN,
1812 "Unable to create directory for \"%s\"", name2_p);
1813 else if (cnt == 2)
1814 msg(ERRN,
1815 "Cannot link \"%s\" and \"%s\"", name1_p, name2_p);
1816 return (cnt);
1817 }
1818
1819 /*
1820 * creat_spec:
1821 * Create one of the following:
1822 * directory
1823 * character special file
1824 * block special file
1825 * fifo
1826 * socket
1827 */
1828
1829 static int
creat_spec(int dirfd)1830 creat_spec(int dirfd)
1831 {
1832 char *nam_p;
1833 int cnt, result, rv = 0;
1834 char *curdir;
1835 char *lastslash;
1836
1837 Do_rename = 0; /* creat_tmp() may reset this */
1838
1839 if (Args & OCp) {
1840 nam_p = Fullnam_p;
1841 } else {
1842 nam_p = G_p->g_nam_p;
1843 }
1844
1845 /*
1846 * Is this the extraction of the hidden attribute directory?
1847 * If we are processing the hidden attribute directory of an
1848 * attribute, then just return as modes and times cannot be set.
1849 * Otherwise, if we are processing a hidden attribute, just set
1850 * the mode/times correctly and return.
1851 */
1852
1853 if (Hiddendir) {
1854 if (G_p->g_attrparent_p == NULL) {
1855 if (Args & OCR) {
1856 if (fchownat(dirfd, ".", Rpw_p->pw_uid,
1857 Rpw_p->pw_gid, 0) != 0) {
1858 msg(ERRN,
1859 "Cannot chown() \"attribute "
1860 "directory of file %s\"",
1861 G_p->g_attrfnam_p);
1862 }
1863 } else if ((fchownat(dirfd, ".", G_p->g_uid,
1864 G_p->g_gid, 0) != 0) && privileged) {
1865 msg(ERRN,
1866 "Cannot chown() \"attribute directory of "
1867 "file %s\"", G_p->g_attrfnam_p);
1868 }
1869
1870 if (fchmod(dirfd, G_p->g_mode) != 0) {
1871 msg(ERRN,
1872 "Cannot chmod() \"attribute directory of "
1873 "file %s\"", G_p->g_attrfnam_p);
1874 }
1875
1876 acl_is_set = 0;
1877 if (Pflag && aclp != NULL) {
1878 if (facl_set(dirfd, aclp) < 0) {
1879 msg(ERRN,
1880 "failed to set acl on attribute"
1881 " directory of %s ",
1882 G_p->g_attrfnam_p);
1883 } else {
1884 acl_is_set = 1;
1885 }
1886 acl_free(aclp);
1887 aclp = NULL;
1888 }
1889 }
1890
1891 return (1);
1892 }
1893
1894 result = stat(nam_p, &DesSt);
1895
1896 if (ustar_dir() || Adir) {
1897 /*
1898 * The archive file is a directory.
1899 * Skip "." and ".."
1900 */
1901
1902 curdir = strrchr(nam_p, '.');
1903
1904 if (curdir != NULL && curdir[1] == '\0') {
1905 lastslash = strrchr(nam_p, '/');
1906
1907 if (lastslash != NULL) {
1908 lastslash++;
1909 } else {
1910 lastslash = nam_p;
1911 }
1912
1913 if (!(strcmp(lastslash, ".")) ||
1914 !(strcmp(lastslash, ".."))) {
1915 return (1);
1916 }
1917 }
1918
1919 if (result == 0) {
1920 /* A file by the same name exists. */
1921
1922 /* Take care of ACLs */
1923 acl_is_set = 0;
1924
1925 if (Pflag && aclp != NULL) {
1926 if (acl_set(nam_p, aclp) < 0) {
1927 msg(ERRN,
1928 "\"%s\": failed to set acl",
1929 nam_p);
1930 } else {
1931 acl_is_set = 1;
1932 }
1933
1934 acl_free(aclp);
1935 aclp = NULL;
1936 }
1937 if (Args & OCd) {
1938 /*
1939 * We are creating directories. Keep the
1940 * existing file.
1941 */
1942
1943 rstfiles(U_KEEP, dirfd);
1944 }
1945
1946 /* Report success. */
1947
1948 return (1);
1949 }
1950 } else {
1951 /* The archive file is not a directory. */
1952
1953 if (result == 0) {
1954 /*
1955 * A file by the same name exists. Move it to a
1956 * temporary file.
1957 */
1958
1959 if (creat_tmp(nam_p) < 0) {
1960 /*
1961 * We weren't able to create the temp file.
1962 * Report failure.
1963 */
1964
1965 return (0);
1966 }
1967 }
1968 }
1969
1970 /*
1971 * This pile tries to create the file directly, and, if there is a
1972 * problem, creates missing directories, and then tries to create the
1973 * file again. Two strikes and you're out.
1974 */
1975
1976 cnt = 0;
1977
1978 do {
1979 if (ustar_dir() || Adir) {
1980 /* The archive file is a directory. */
1981
1982 result = mkdir(nam_p, G_p->g_mode);
1983 } else if (ustar_spec() || Aspec) {
1984 /*
1985 * The archive file is block special,
1986 * char special, socket, or a fifo.
1987 * Note that, for a socket, the third
1988 * parameter to mknod() is ignored.
1989 */
1990 result = mknod(nam_p, (int)G_p->g_mode,
1991 (int)G_p->g_rdev);
1992 }
1993
1994 if (result >= 0) {
1995 /*
1996 * The file creation succeeded. Take care of the ACLs.
1997 */
1998
1999 acl_is_set = 0;
2000
2001 if (Pflag && aclp != NULL) {
2002 if (acl_set(nam_p, aclp) < 0) {
2003 msg(ERRN,
2004 "\"%s\": failed to set acl", nam_p);
2005 } else {
2006 acl_is_set = 1;
2007 }
2008
2009 acl_free(aclp);
2010 aclp = NULL;
2011 }
2012
2013 cnt = 0;
2014 break;
2015 }
2016
2017 cnt++;
2018 } while (cnt < 2 && missdir(nam_p) == 0);
2019
2020 switch (cnt) {
2021 case 0:
2022 rv = 1;
2023 rstfiles(U_OVER, dirfd);
2024 break;
2025
2026 case 1:
2027 msg(ERRN,
2028 "Cannot create directory for \"%s\"", nam_p);
2029
2030 if (*Over_p == '\0') {
2031 rstfiles(U_KEEP, dirfd);
2032 }
2033
2034 break;
2035
2036 case 2:
2037 if (ustar_dir() || Adir) {
2038 msg(ERRN, "Cannot create directory \"%s\"", nam_p);
2039 } else if (ustar_spec() || Aspec) {
2040 msg(ERRN, "Cannot mknod() \"%s\"", nam_p);
2041 }
2042
2043 if (*Over_p == '\0') {
2044 rstfiles(U_KEEP, dirfd);
2045 }
2046
2047 break;
2048
2049 default:
2050 msg(EXT, "Impossible case.");
2051 }
2052
2053 return (rv);
2054 }
2055
2056 /*
2057 * creat_tmp:
2058 */
2059
2060 static int
creat_tmp(char * nam_p)2061 creat_tmp(char *nam_p)
2062 {
2063 char *t_p;
2064 int cwd = -1;
2065
2066 if ((Args & OCp) && G_p->g_ino == DesSt.st_ino &&
2067 G_p->g_dev == DesSt.st_dev) {
2068 msg(ERR, "Attempt to pass a file to itself.");
2069 return (-1);
2070 }
2071
2072 if (G_p->g_mtime <= DesSt.st_mtime && !(Args & OCu)) {
2073 msg(ERR, "Existing \"%s\" same age or newer", nam_p);
2074 return (-1);
2075 }
2076
2077 /* Make the temporary file name. */
2078
2079 (void) strcpy(Over_p, nam_p);
2080 t_p = Over_p + strlen(Over_p);
2081
2082 while (t_p != Over_p) {
2083 if (*(t_p - 1) == '/')
2084 break;
2085 t_p--;
2086 }
2087
2088 (void) strcpy(t_p, "XXXXXX");
2089
2090 if (G_p->g_attrnam_p != NULL) {
2091 /*
2092 * Save our current directory, so we can go into
2093 * the attribute directory to make the temp file
2094 * and then return.
2095 */
2096
2097 cwd = save_cwd();
2098 (void) fchdir(G_p->g_dirfd);
2099 }
2100
2101 (void) mktemp(Over_p);
2102
2103 if (G_p->g_attrnam_p != NULL) {
2104 /* Return to the current directory. */
2105
2106 rest_cwd(cwd);
2107 }
2108
2109 if (*Over_p == '\0') {
2110 /* mktemp reports a failure. */
2111
2112 msg(ERR, "Cannot get temporary file name.");
2113 return (-1);
2114 }
2115
2116 /*
2117 * If it's a regular file, write to the temporary file, and then rename
2118 * in order to accommodate potential executables.
2119 *
2120 * Note: g_typeflag is only defined (set) for USTAR archive types. It
2121 * defaults to 0 in the cpio-format-regular file case, so this test
2122 * succeeds.
2123 */
2124
2125 if (G_p->g_typeflag == 0 &&
2126 (DesSt.st_mode & (uint_t)Ftype) == S_IFREG &&
2127 (G_p->g_mode & (uint_t)Ftype) == S_IFREG) {
2128 /*
2129 * The archive file and the filesystem file are both regular
2130 * files. We write to the temporary file in this case.
2131 */
2132
2133 if (Args & OCp) {
2134 if (G_p->g_attrnam_p == NULL) {
2135 Fullnam_p = Over_p;
2136 } else {
2137 Attrfile_p = Over_p;
2138 }
2139 } else {
2140 G_p->g_nam_p = Over_p;
2141 if (G_p->g_attrnam_p != NULL) {
2142 Attrfile_p = Over_p;
2143 }
2144 }
2145
2146 if (G_p->g_attrnam_p == NULL) {
2147 Over_p = nam_p;
2148 } else {
2149 Over_p = G_p->g_attrnam_p;
2150 }
2151
2152 Do_rename = 1;
2153 } else {
2154 /*
2155 * Either the archive file or the filesystem file is not a
2156 * regular file.
2157 */
2158
2159 Do_rename = 0;
2160
2161 if (S_ISDIR(DesSt.st_mode)) {
2162 /*
2163 * The filesystem file is a directory.
2164 *
2165 * Save the current working directory because we will
2166 * want to restore it back just in case remove_dir()
2167 * fails or get confused about where we should be.
2168 */
2169
2170 *Over_p = '\0';
2171 cwd = save_cwd();
2172
2173 if (remove_dir(nam_p) < 0) {
2174 msg(ERRN,
2175 "Cannot remove the directory \"%s\"",
2176 nam_p);
2177 /*
2178 * Restore working directory back to the one
2179 * saved earlier.
2180 */
2181
2182 rest_cwd(cwd);
2183 return (-1);
2184 }
2185
2186 /*
2187 * Restore working directory back to the one
2188 * saved earlier
2189 */
2190
2191 rest_cwd(cwd);
2192 } else {
2193 /*
2194 * The file is not a directory. Will use the original
2195 * link/unlink construct, however, if the file is
2196 * namefs, link would fail with EXDEV. Therefore, we
2197 * use rename() first to back up the file.
2198 */
2199 if (rename(nam_p, Over_p) < 0) {
2200 /*
2201 * If rename failed, try old construction
2202 * method.
2203 */
2204 if (link(nam_p, Over_p) < 0) {
2205 msg(ERRN,
2206 "Cannot rename temporary file "
2207 "\"%s\" to \"%s\"", Over_p, nam_p);
2208 *Over_p = '\0';
2209 return (-1);
2210 }
2211
2212 if (unlink(nam_p) < 0) {
2213 msg(ERRN,
2214 "Cannot unlink() current \"%s\"",
2215 nam_p);
2216 (void) unlink(Over_p);
2217 *Over_p = '\0';
2218 return (-1);
2219 }
2220 }
2221 }
2222 }
2223
2224 return (1);
2225 }
2226
2227 /*
2228 * Copy the datasize amount of data from the input file to buffer.
2229 *
2230 * ifd - Input file descriptor.
2231 * buffer - Buffer (allocated by caller) to copy data to.
2232 * datasize - The amount of data to read from the input file
2233 * and copy to the buffer.
2234 * error - When reading from an Archive file, indicates unreadable
2235 * data was encountered, otherwise indicates errno.
2236 * data_in_info - Information needed when called from data_in().
2237 */
2238 static ssize_t
read_chunk(int ifd,char * buffer,size_t datasize,data_in_t * data_in_info)2239 read_chunk(int ifd, char *buffer, size_t datasize, data_in_t *data_in_info)
2240 {
2241 if (Args & OCp) {
2242 return (read(ifd, buffer, datasize));
2243 } else {
2244 FILL(datasize);
2245 if (data_in_info->data_in_proc_mode != P_SKIP) {
2246 if (Hdr_type == CRC)
2247 data_in_info->data_in_cksumval += cksum(CRC,
2248 datasize, NULL);
2249 if (data_in_info->data_in_swapfile)
2250 swap(Buffr.b_out_p, datasize);
2251
2252
2253 /*
2254 * if the bar archive is compressed, set up a pipe and
2255 * do the de-compression while reading in the file
2256 */
2257 if (Hdr_type == BAR) {
2258 if (data_in_info->data_in_compress_flag == 0 &&
2259 Compressed) {
2260 setup_uncompress(
2261 &(data_in_info->data_in_pipef));
2262 data_in_info->data_in_compress_flag++;
2263 }
2264 }
2265 }
2266 (void) memcpy(buffer, Buffr.b_out_p, datasize);
2267 Buffr.b_out_p += datasize;
2268 Buffr.b_cnt -= datasize;
2269 return (datasize);
2270 }
2271 }
2272
2273 /*
2274 * Read as much data as we can.
2275 *
2276 * ifd - input file descriptor.
2277 * buf - Buffer (allocated by caller) to copy data to.
2278 * bytes - The amount of data to read from the input file
2279 * and copy to the buffer.
2280 * rdblocksz - The size of the chunk of data to read.
2281 *
2282 * Return number of bytes failed to read.
2283 * Return -1 when buffer is empty and read failed.
2284 */
2285 static int
read_bytes(int ifd,char * buf,size_t bytes,size_t rdblocksz,data_in_t * data_in_info)2286 read_bytes(int ifd, char *buf, size_t bytes, size_t rdblocksz,
2287 data_in_t *data_in_info)
2288 {
2289 size_t bytesread;
2290 ssize_t got;
2291
2292 for (bytesread = 0; bytesread < bytes; bytesread += got) {
2293 /*
2294 * Read the data from either the input file descriptor
2295 * or the archive file. read_chunk() will only return
2296 * <= 0 if data_copy() was called from data_pass().
2297 */
2298 if ((got = read_chunk(ifd, buf + bytesread,
2299 min(bytes - bytesread, rdblocksz),
2300 data_in_info)) <= 0) {
2301 /*
2302 * We come here only in the pass mode.
2303 * If data couldn't be read from the input file
2304 * descriptor, return number of bytes in the buf.
2305 * If buffer is empty, return -1.
2306 */
2307 if (bytesread == 0) {
2308 if (got == 0) /* EOF */
2309 data_in_info->data_in_rd_eof = 1;
2310 return (-1);
2311 }
2312 return (bytes - bytesread);
2313 }
2314 }
2315 return (0);
2316 }
2317
2318 /*
2319 * Write as much data as we can.
2320 *
2321 * ofd - output file descriptor.
2322 * buf - Source buffer to output data from.
2323 * maxwrite - The amount of data to write to the output.
2324 *
2325 * return 0 upon success.
2326 */
2327 static int
write_bytes(int ofd,char * buf,size_t maxwrite,data_in_t * data_in_info)2328 write_bytes(int ofd, char *buf, size_t maxwrite, data_in_t *data_in_info)
2329 {
2330 ssize_t cnt;
2331
2332 errno = 0;
2333 if ((cnt = write(ofd, buf, maxwrite)) < (ssize_t)maxwrite) {
2334 data_in_info->data_in_errno = errno;
2335 /*
2336 * data_in() needs to know if it was an actual write(2)
2337 * failure, or if we just couldn't write all of the data
2338 * requested so that we know that the rest of the file's
2339 * data can be read but not written.
2340 */
2341 if (cnt != -1)
2342 data_in_info->data_in_wr_part = 1;
2343 return (1);
2344 } else if (Args & OCp) {
2345 Blocks += (ulong_t)((cnt + (Bufsize - 1)) / Bufsize);
2346 }
2347 return (0);
2348 }
2349
2350 /*
2351 * Perform I/O for given byte size with using limited i/o block size
2352 * and supplied buffer.
2353 *
2354 * ifd/ofd - i/o file descriptor
2355 * buf - buffer to be used for i/o
2356 * bytes - Amount to read/write
2357 * wrblocksz - Output block size.
2358 * rdblocksz - Read block size.
2359 *
2360 * Return 0 upon success. Return negative if read failed.
2361 * Return positive non-zero if write failed.
2362 */
2363 static int
rdwr_bytes(int ifd,int ofd,char * buf,off_t bytes,size_t wrblocksz,size_t rdblocksz,data_in_t * data_in_info)2364 rdwr_bytes(int ifd, int ofd, char *buf, off_t bytes,
2365 size_t wrblocksz, size_t rdblocksz, data_in_t *data_in_info)
2366 {
2367 int rv, sz;
2368 int error = 0;
2369 int write_it = (data_in_info->data_in_proc_mode != P_SKIP);
2370
2371 while (bytes > 0) {
2372 /*
2373 * If the number of bytes left to write is smaller than
2374 * the preferred I/O size, then we're about to do our final
2375 * write to the file, so just set wrblocksz to the number of
2376 * bytes left to write.
2377 */
2378 if (bytes < wrblocksz)
2379 wrblocksz = bytes;
2380
2381 /* Read input till satisfy output block size */
2382 sz = read_bytes(ifd, buf, wrblocksz, rdblocksz, data_in_info);
2383 if (sz < 0)
2384 return (sz);
2385
2386 if (write_it) {
2387 rv = write_bytes(ofd, buf,
2388 wrblocksz - sz, data_in_info);
2389 if (rv != 0) {
2390 /*
2391 * If we wrote partial, we return and quits.
2392 * Otherwise, read through the rest of input
2393 * to go to the next file.
2394 */
2395 if ((Args & OCp) ||
2396 data_in_info->data_in_wr_part) {
2397 return (rv);
2398 } else {
2399 write_it = 0;
2400 }
2401 error = 1;
2402 }
2403 }
2404 bytes -= (wrblocksz - sz);
2405 }
2406 return (error);
2407 }
2408
2409 /*
2410 * Write zeros for give size.
2411 *
2412 * ofd - output file descriptor
2413 * buf - buffer to fill with zeros
2414 * bytes - Amount to write
2415 * wrblocksz - Write block size
2416 *
2417 * return 0 upon success.
2418 */
2419 static int
write_zeros(int ofd,char * buf,off_t bytes,size_t wrblocksz,data_in_t * data_in_info)2420 write_zeros(int ofd, char *buf, off_t bytes, size_t wrblocksz,
2421 data_in_t *data_in_info)
2422 {
2423 int rv;
2424
2425 (void) memset(buf, 0, min(bytes, wrblocksz));
2426 while (bytes > 0) {
2427 if (bytes < wrblocksz)
2428 wrblocksz = bytes;
2429 rv = write_bytes(ofd, buf, wrblocksz, data_in_info);
2430 if (rv != 0)
2431 return (rv);
2432 bytes -= wrblocksz;
2433 }
2434 return (0);
2435 }
2436
2437 /*
2438 * To figure out the size of the buffer used to accumulate data from
2439 * readtape() and to write to the file, we need to determine the largest
2440 * chunk of data to be written to the file at one time. This is determined
2441 * based on the following three things:
2442 * 1) The size of the archived file.
2443 * 2) The preferred I/O size of the file.
2444 * 3) If the file is a read-write system attribute file.
2445 * If the size of the file is less than the preferred I/O size or it's a
2446 * read-write system attribute file, which must be written in one operation,
2447 * then set the maximum write size to the size of the archived file.
2448 * Otherwise, the maximum write size is preferred I/O size.
2449 */
2450 static int
calc_maxwrite(int ofd,int rw_sysattr,off_t bytes,size_t blocksize)2451 calc_maxwrite(int ofd, int rw_sysattr, off_t bytes, size_t blocksize)
2452 {
2453 struct stat tsbuf;
2454 size_t maxwrite;
2455 size_t piosize; /* preferred I/O size */
2456
2457 if (rw_sysattr || bytes < blocksize) {
2458 maxwrite = bytes;
2459 } else {
2460 if (fstat(ofd, &tsbuf) == 0) {
2461 piosize = tsbuf.st_blksize;
2462 } else {
2463 piosize = blocksize;
2464 }
2465 maxwrite = min(bytes, piosize);
2466 }
2467 return (maxwrite);
2468 }
2469 /*
2470 * data_copy() and data_copy_with_holes() copy data from the input
2471 * file to output file descriptor. If ifd is -1, then the input file is
2472 * the archive file.
2473 *
2474 * Parameters
2475 * ifd - Input file descriptor to read from.
2476 * ofd - Output file descriptor of extracted file.
2477 * rw_sysattr - Flag indicating if a file is an extended
2478 * system attribute file.
2479 * bytes - Amount of data (file size) of copy/write.
2480 * blocksize - Amount of data to read at a time from either
2481 * the input file descriptor or from the archive.
2482 * data_in_info - information needed while reading data when
2483 * called by data_in().
2484 * holes - Information of holes in the input file.
2485 *
2486 * Return code
2487 * 0 Success
2488 * < 0 An error occurred during the read of the input
2489 * file
2490 * > 0 An error occurred during the write of the output
2491 * file descriptor.
2492 */
2493 static int
data_copy(int ifd,int ofd,int rw_sysattr,off_t bytes,size_t blocksize,data_in_t * data_in_info)2494 data_copy(int ifd, int ofd, int rw_sysattr, off_t bytes,
2495 size_t blocksize, data_in_t *data_in_info)
2496 {
2497 char *buf;
2498 size_t maxwrite;
2499 int rv;
2500
2501 /* No data to copy. */
2502 if (bytes == 0)
2503 return (0);
2504
2505 maxwrite = calc_maxwrite(ofd, rw_sysattr, bytes, blocksize);
2506 buf = e_zalloc(E_EXIT, maxwrite);
2507
2508 rv = rdwr_bytes(ifd, ofd, buf, bytes, maxwrite,
2509 blocksize, data_in_info);
2510
2511 free(buf);
2512 return (rv);
2513 }
2514
2515 static int
data_copy_with_holes(int ifd,int ofd,int rw_sysattr,off_t bytes,size_t blocksize,data_in_t * data_in_info,holes_info_t * holes)2516 data_copy_with_holes(int ifd, int ofd, int rw_sysattr, off_t bytes,
2517 size_t blocksize, data_in_t *data_in_info, holes_info_t *holes)
2518 {
2519 holes_list_t *hl;
2520 off_t curpos, noff, datasize;
2521 char *buf;
2522 size_t maxwrite;
2523 int rv, error;
2524
2525 if (bytes == 0)
2526 return (0);
2527
2528 maxwrite = calc_maxwrite(ofd, rw_sysattr, bytes, blocksize);
2529 buf = e_zalloc(E_EXIT, maxwrite);
2530
2531 error = 0;
2532 curpos = 0;
2533 for (hl = holes->holes_list; hl != NULL; hl = hl->hl_next) {
2534 if (curpos != hl->hl_data) {
2535 /* adjust output position */
2536 noff = lseek(ofd, hl->hl_data, SEEK_SET);
2537 if (noff != hl->hl_data) {
2538 /*
2539 * Can't seek to the target, try to adjust
2540 * position by filling with zeros.
2541 */
2542 datasize = hl->hl_data - curpos;
2543 rv = write_zeros(ofd, buf, datasize,
2544 maxwrite, data_in_info);
2545 if (rv != 0)
2546 goto errout;
2547 }
2548 /*
2549 * Data is contiguous in the archive, but fragmented
2550 * in the regular file, so we also adjust the input
2551 * file position in pass mode.
2552 */
2553 if (Args & OCp) {
2554 /* adjust input position */
2555 (void) lseek(ifd, hl->hl_data, SEEK_SET);
2556 }
2557 curpos = hl->hl_data;
2558 }
2559 datasize = hl->hl_hole - hl->hl_data;
2560 if (datasize == 0) {
2561 /*
2562 * There is a hole at the end of file. To create
2563 * such hole, we append one byte, and truncate the
2564 * last block. This is necessary because ftruncate(3C)
2565 * alone allocates one block on the end of file.
2566 */
2567 rv = write_zeros(ofd, buf, 1, maxwrite, data_in_info);
2568 if (rv != 0)
2569 goto errout;
2570 (void) ftruncate(ofd, hl->hl_data);
2571 break;
2572 }
2573 rv = rdwr_bytes(ifd, ofd, buf, datasize, maxwrite,
2574 blocksize, data_in_info);
2575 if (rv != 0) {
2576 errout:
2577 /*
2578 * Return if we got a read error or in pass mode,
2579 * or failed with partial write. Otherwise, we'll
2580 * read through the input stream till next file.
2581 */
2582 if (rv < 0 || (Args & OCp) ||
2583 data_in_info->data_in_wr_part) {
2584 free(buf);
2585 return (rv);
2586 }
2587 error = 1;
2588 hl = hl->hl_next;
2589 break;
2590 }
2591 curpos += datasize;
2592 }
2593
2594 /*
2595 * We should read through the input data to go to the next
2596 * header when non-fatal error occured.
2597 */
2598 if (error && !(Args & OCp)) {
2599 data_in_info->data_in_proc_mode = P_SKIP;
2600 while (hl != NULL) {
2601 datasize = hl->hl_hole - hl->hl_data;
2602 rv = rdwr_bytes(ifd, ofd, buf, datasize, maxwrite,
2603 blocksize, data_in_info);
2604 if (rv != 0)
2605 break;
2606 hl = hl->hl_next;
2607 }
2608 }
2609
2610 free(buf);
2611 return (error);
2612 }
2613
2614 /*
2615 * Strip off the sparse file information that is prepended to
2616 * the compressed sparse file. The information is in the following
2617 * format:
2618 * <prepended info size><SP><orig file size><SP><holes info>
2619 * where prepended info size is long right justified in 10 bytes.
2620 * Holesdata consists of the series of offset pairs:
2621 * <data offset><SP><hole offset><SP><data offset><SP><hole offset>...
2622 * prepended info size and original file size have been read in gethdr().
2623 * We read the rest of holes information here in this function.
2624 */
2625 static int
read_holesdata(holes_info_t * holes,off_t * fileszp,char * nam_p,data_in_t * data_in_info)2626 read_holesdata(holes_info_t *holes, off_t *fileszp,
2627 char *nam_p, data_in_t *data_in_info)
2628 {
2629 char *holesdata;
2630 size_t holesdata_sz;
2631
2632 /* We've already read the header. */
2633 holesdata_sz = holes->holesdata_sz - MIN_HOLES_HDRSIZE;
2634
2635 if ((holesdata = e_zalloc(E_NORMAL, holesdata_sz)) == NULL) {
2636 msg(ERRN, "Could not allocate memory for "
2637 "sparse file information", nam_p);
2638 return (1);
2639 }
2640 /*
2641 * This function is called only in OCi mode. Therefore,
2642 * read_bytes() won't fail, and won't return if error occurs in
2643 * input stream. See rstbuf().
2644 */
2645 (void) read_bytes(-1, holesdata, holesdata_sz, CPIOBSZ, data_in_info);
2646 *fileszp -= holesdata_sz;
2647
2648 /* The string should be terminated. */
2649 if (holesdata[holesdata_sz - 1] != '\0') {
2650 invalid:
2651 free(holesdata);
2652 msg(ERR, "invalid sparse file information", nam_p);
2653 return (1);
2654 }
2655 if (parse_holesdata(holes, holesdata) != 0)
2656 goto invalid;
2657
2658 /* sanity check */
2659 if (*fileszp != holes->data_size)
2660 goto invalid;
2661
2662 free(holesdata);
2663 return (0);
2664 }
2665
2666 /*
2667 * data_in: If proc_mode == P_PROC, bread() the file's data from the archive
2668 * and write(2) it to the open fdes gotten from openout(). If proc_mode ==
2669 * P_SKIP, or becomes P_SKIP (due to errors etc), bread(2) the file's data
2670 * and ignore it. If the user specified any of the "swap" options (b, s or S),
2671 * and the length of the file is not appropriate for that action, do not
2672 * perform the "swap", otherwise perform the action on a buffer by buffer basis.
2673 * If the CRC header was selected, calculate a running checksum as each buffer
2674 * is processed.
2675 */
2676 static void
data_in(int proc_mode)2677 data_in(int proc_mode)
2678 {
2679 char *nam_p;
2680 int pad, rv;
2681 int error = 0;
2682 int swapfile = 0;
2683 int cstatus = 0;
2684 off_t filesz;
2685 data_in_t *data_in_info;
2686
2687 if (G_p->g_attrnam_p != NULL) {
2688 nam_p = G_p->g_attrnam_p;
2689 } else {
2690 nam_p = G_p->g_nam_p;
2691 }
2692
2693 if (((G_p->g_mode & Ftype) == S_IFLNK && proc_mode != P_SKIP) ||
2694 (Hdr_type == BAR && bar_linkflag == '2' && proc_mode != P_SKIP)) {
2695 proc_mode = P_SKIP;
2696 VERBOSE((Args & (OCv | OCV)), nam_p);
2697 }
2698 if (Args & (OCb | OCs | OCS)) { /* verify that swapping is possible */
2699 swapfile = 1;
2700 if (Args & (OCs | OCb) && G_p->g_filesz % 2) {
2701 msg(ERR,
2702 "Cannot swap bytes of \"%s\", odd number of bytes",
2703 nam_p);
2704 swapfile = 0;
2705 }
2706 if (Args & (OCS | OCb) && G_p->g_filesz % 4) {
2707 msg(ERR,
2708 "Cannot swap halfwords of \"%s\", odd number "
2709 "of halfwords", nam_p);
2710 swapfile = 0;
2711 }
2712 }
2713
2714 data_in_info = e_zalloc(E_EXIT, sizeof (data_in_t));
2715 data_in_info->data_in_swapfile = swapfile;
2716 data_in_info->data_in_proc_mode = proc_mode;
2717
2718 filesz = G_p->g_filesz;
2719
2720 if (S_ISSPARSE(G_p->g_mode) && G_p->g_holes != NULL) {
2721 /* We've already read the header in gethdr() */
2722 filesz -= MIN_HOLES_HDRSIZE;
2723
2724 /*
2725 * Strip rest of the sparse file information. This includes
2726 * the data/hole offset pairs which will be used to restore
2727 * the holes in the file.
2728 */
2729 if (proc_mode == P_SKIP) {
2730 /* holes info isn't necessary to skip file */
2731 free_holes_info(G_p->g_holes);
2732 G_p->g_holes = NULL;
2733 } else {
2734 rv = read_holesdata(G_p->g_holes, &filesz,
2735 nam_p, data_in_info);
2736 if (rv != 0) {
2737 /*
2738 * We got an error. Skip this file. holes info
2739 * is no longer necessary.
2740 */
2741 free_holes_info(G_p->g_holes);
2742 G_p->g_holes = NULL;
2743
2744 data_in_info->data_in_proc_mode = P_SKIP;
2745 error = 1;
2746 }
2747 }
2748 }
2749
2750 if (G_p->g_holes != NULL) {
2751 rv = data_copy_with_holes(-1, Ofile,
2752 (G_p->g_attrnam_p == NULL) ? 0 : G_p->g_rw_sysattr,
2753 G_p->g_holes->orig_size,
2754 CPIOBSZ, data_in_info, G_p->g_holes);
2755
2756 free_holes_info(G_p->g_holes);
2757 G_p->g_holes = NULL;
2758 } else {
2759 rv = data_copy(-1, Ofile,
2760 (G_p->g_attrnam_p == NULL) ? 0 : G_p->g_rw_sysattr,
2761 filesz, CPIOBSZ, data_in_info);
2762 }
2763
2764 /* This writes out the file from the archive */
2765 if (rv != 0 || error) {
2766 errno = data_in_info->data_in_errno;
2767
2768 if (!error) {
2769 msg(data_in_info->data_in_wr_part ? EXTN : ERRN,
2770 "Cannot write \"%s%s%s\"",
2771 (G_p->g_attrnam_p == NULL) ? "" :
2772 G_p->g_attrfnam_p,
2773 (G_p->g_attrnam_p == NULL) ? "" :
2774 G_p->g_rw_sysattr ?
2775 gettext(" System Attribute ") :
2776 gettext(" Attribute "), nam_p);
2777 }
2778 /*
2779 * We've failed to write to the file, and input data
2780 * has been skiped to the next file. We'll need to restore
2781 * the original file, and skip the rest of work.
2782 */
2783 proc_mode = P_SKIP;
2784 rstfiles(U_KEEP, G_p->g_dirfd);
2785 cstatus = close(Ofile);
2786 Ofile = 0;
2787 if (cstatus != 0) {
2788 msg(EXTN, "close error");
2789 }
2790 }
2791
2792 /* we must use g_filesz for the amount of padding */
2793 pad = (Pad_val + 1 - (G_p->g_filesz & Pad_val)) & Pad_val;
2794 if (pad != 0) {
2795 FILL(pad);
2796 Buffr.b_out_p += pad;
2797 Buffr.b_cnt -= pad;
2798 }
2799 if (proc_mode != P_SKIP) {
2800 if (Hdr_type == CRC &&
2801 Gen.g_cksum != data_in_info->data_in_cksumval) {
2802 msg(ERR, "\"%s\" - checksum error", nam_p);
2803 rstfiles(U_KEEP, G_p->g_dirfd);
2804 } else
2805 rstfiles(U_OVER, G_p->g_dirfd);
2806 if (Hdr_type == BAR && data_in_info->data_in_compress_flag) {
2807 (void) pclose(data_in_info->data_in_pipef);
2808 } else {
2809 cstatus = close(Ofile);
2810 }
2811 Ofile = 0;
2812 if (cstatus != 0) {
2813 msg(EXTN, "close error");
2814 }
2815 }
2816 (void) free(data_in_info);
2817
2818 VERBOSE((proc_mode != P_SKIP && (Args & (OCv | OCV))),
2819 (G_p->g_attrparent_p == NULL) ? G_p->g_nam_p : G_p->g_attrpath_p);
2820 Finished = 1;
2821 }
2822
2823 /*
2824 * Read regular file. Return number of bytes which weren't read.
2825 * Upon return, real_filesz will be real file size of input file.
2826 * When read_exact is specified, read size is adjusted to the given
2827 * file size.
2828 */
2829 static off_t
read_file(char * nam_p,off_t file_size,off_t * real_filesz,boolean_t read_exact)2830 read_file(char *nam_p, off_t file_size, off_t *real_filesz,
2831 boolean_t read_exact)
2832 {
2833 int amount_read;
2834 off_t amt_to_read;
2835 off_t readsz;
2836
2837 if (file_size == 0)
2838 return (0);
2839
2840 amt_to_read = file_size;
2841 do {
2842 if (read_exact && amt_to_read < CPIOBSZ)
2843 readsz = amt_to_read;
2844 else
2845 readsz = CPIOBSZ;
2846
2847 FLUSH(readsz);
2848 errno = 0;
2849
2850 if ((amount_read = read(Ifile, Buffr.b_in_p, readsz)) < 0) {
2851 msg(EXTN, "Cannot read \"%s%s%s\"",
2852 (Gen.g_attrnam_p == NULL) ?
2853 nam_p : Gen.g_attrfnam_p,
2854 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_rw_sysattr ?
2855 gettext(" System Attribute ") :
2856 gettext(" Attribute "),
2857 (Gen.g_attrnam_p == NULL) ? "" : nam_p);
2858 break;
2859 }
2860
2861 if (amount_read == 0) {
2862 /* got EOF. the file has shrunk */
2863 *real_filesz = file_size - amt_to_read;
2864 break;
2865 } else if (amount_read > amt_to_read) {
2866 /* the file has grown */
2867 *real_filesz = file_size +
2868 (amount_read - amt_to_read);
2869 amount_read = amt_to_read;
2870 } else if (amount_read == amt_to_read) {
2871 /* the file is the same size */
2872 *real_filesz = file_size;
2873 }
2874
2875 Buffr.b_in_p += amount_read;
2876 Buffr.b_cnt += amount_read;
2877
2878 amt_to_read -= (off_t)amount_read;
2879 if (!read_exact &&
2880 amt_to_read == 0 && amount_read == CPIOBSZ) {
2881 /*
2882 * If the file size is multiple of CPIOBSZ, we may
2883 * be able to read more from the file even though
2884 * amt_to_read already gets 0.
2885 */
2886 FLUSH(CPIOBSZ);
2887 amount_read = read(Ifile, Buffr.b_in_p, CPIOBSZ);
2888 if (amount_read != 0) {
2889 /* the file has grown */
2890 *real_filesz = file_size + amount_read;
2891 }
2892 }
2893 } while (amt_to_read != 0);
2894
2895 return (amt_to_read);
2896 }
2897
2898 /*
2899 * Read through the data in files skipping holes.
2900 */
2901 static off_t
read_compress_holes(char * nam_p,off_t file_size,off_t * real_filesz,holes_info_t * holes,int * hole_changed)2902 read_compress_holes(char *nam_p, off_t file_size, off_t *real_filesz,
2903 holes_info_t *holes, int *hole_changed)
2904 {
2905 off_t left;
2906 off_t datasize, realsz;
2907 off_t curpos, npos;
2908 holes_list_t *hl = holes->holes_list;
2909
2910 curpos = 0;
2911 for (hl = holes->holes_list; hl != NULL; hl = hl->hl_next) {
2912 datasize = hl->hl_hole - hl->hl_data;
2913
2914 npos = lseek(Ifile, curpos, SEEK_DATA);
2915 if (npos == -1 && errno == ENXIO) {
2916 /*
2917 * No more data. There are two cases.
2918 * - we have a hole toward the end of file.
2919 * - file has been shrunk, and we've reached EOF.
2920 */
2921 *real_filesz = lseek(Ifile, 0, SEEK_END);
2922 if (hl->hl_data == file_size)
2923 return (0);
2924 /*
2925 * File has been shrunk. Check the amount of data
2926 * left.
2927 */
2928 left = 0;
2929 while (hl != NULL) {
2930 left += (hl->hl_hole - hl->hl_data);
2931 hl = hl->hl_next;
2932 }
2933 return (left);
2934 }
2935
2936 /* found data */
2937 curpos = npos;
2938 if (curpos != hl->hl_data) {
2939 /*
2940 * File has been changed. We shouldn't read data
2941 * from different offset since we've already put
2942 * the holes data.
2943 */
2944 *hole_changed = 1;
2945 (void) lseek(Ifile, hl->hl_data, SEEK_SET);
2946 curpos = hl->hl_data;
2947 }
2948 left = read_file(nam_p, datasize, &realsz, B_TRUE);
2949 if (left != 0) {
2950 /* file has been shrunk */
2951 *real_filesz = curpos + datasize - left;
2952 left = file_size - *real_filesz;
2953 return (left);
2954 }
2955 curpos += datasize;
2956 }
2957 /*
2958 * We've read exact size of holes. We need to make sure
2959 * that file hasn't grown by reading from the EOF.
2960 */
2961 realsz = 0;
2962 (void) read_file(nam_p, CPIOBSZ, &realsz, B_FALSE);
2963
2964 *real_filesz = curpos + realsz;
2965 return (0);
2966 }
2967
2968 /*
2969 * data_out: open(2) the file to be archived, compute the checksum
2970 * of it's data if the CRC header was specified and write the header.
2971 * read(2) each block of data and bwrite() it to the archive. For TARTYP (TAR
2972 * and USTAR) archives, pad the data with NULLs to the next 512 byte boundary.
2973 */
2974 static void
data_out(void)2975 data_out(void)
2976 {
2977 char *nam_p;
2978 int cnt, pad;
2979 off_t amt_to_read;
2980 off_t real_filesz;
2981 int errret = 0;
2982 int hole_changed = 0;
2983 off_t orig_filesz;
2984 holes_info_t *holes = NULL;
2985
2986 nam_p = G_p->g_nam_p;
2987 if (Aspec) {
2988 if (Pflag && aclp != NULL) {
2989 char *secinfo = NULL;
2990 int len = 0;
2991
2992 /* append security attributes */
2993 if (append_secattr(&secinfo, &len, aclp) == -1) {
2994 msg(ERR,
2995 "can create security information");
2996 }
2997 /* call append_secattr() if more than one */
2998
2999 if (len > 0) {
3000 /* write ancillary only if there is sec info */
3001 write_hdr(ARCHIVE_ACL, (off_t)len);
3002 write_ancillary(secinfo, len, B_TRUE);
3003 }
3004 }
3005 write_hdr(ARCHIVE_NORMAL, (off_t)0);
3006 rstfiles(U_KEEP, G_p->g_dirfd);
3007 VERBOSE((Args & (OCv | OCV)), nam_p);
3008 return;
3009 }
3010 if ((G_p->g_mode & Ftype) == S_IFLNK && (Hdr_type !=
3011 USTAR && Hdr_type != TAR)) { /* symbolic link */
3012 int size;
3013 write_hdr(ARCHIVE_NORMAL, (off_t)0);
3014
3015 FLUSH(G_p->g_filesz);
3016 errno = 0;
3017
3018 /* Note that "size" and G_p->g_filesz are the same number */
3019
3020 if ((size = readlink(nam_p, Buffr.b_in_p, G_p->g_filesz)) <
3021 0) {
3022 msg(ERRN, "Cannot read symbolic link \"%s\"", nam_p);
3023 return;
3024 }
3025
3026 /*
3027 * Note that it is OK not to add the NUL after the name read by
3028 * readlink, because it is not being used subsequently.
3029 */
3030
3031 Buffr.b_in_p += size;
3032 Buffr.b_cnt += size;
3033 pad = (Pad_val + 1 - (size & Pad_val)) & Pad_val;
3034 if (pad != 0) {
3035 FLUSH(pad);
3036 (void) memset(Buffr.b_in_p, 0, pad);
3037 Buffr.b_in_p += pad;
3038 Buffr.b_cnt += pad;
3039 }
3040 VERBOSE((Args & (OCv | OCV)), nam_p);
3041 return;
3042 } else if ((G_p->g_mode & Ftype) == S_IFLNK &&
3043 (Hdr_type == USTAR || Hdr_type == TAR)) {
3044 int size;
3045
3046 /*
3047 * G_p->g_filesz is the length of the right-hand side of
3048 * the symlink "x -> y".
3049 * The tar link field is only NAMSIZ long.
3050 */
3051
3052 if (G_p->g_filesz > NAMSIZ) {
3053 msg(ERRN,
3054 "Symbolic link too long \"%s\"", nam_p);
3055 return;
3056 }
3057 if ((size = readlink(nam_p, T_lname, G_p->g_filesz)) < 0) {
3058 msg(ERRN,
3059 "Cannot read symbolic link \"%s\"", nam_p);
3060 return;
3061 }
3062 T_lname[size] = '\0';
3063 G_p->g_filesz = (off_t)0;
3064 write_hdr(ARCHIVE_NORMAL, (off_t)0);
3065 VERBOSE((Args & (OCv | OCV)), nam_p);
3066 return;
3067 }
3068 if ((Ifile = openfile(O_RDONLY)) < 0) {
3069 msg(ERR, "\"%s%s%s\" ?",
3070 (Gen.g_attrnam_p == NULL) ? nam_p : Gen.g_attrfnam_p,
3071 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_rw_sysattr ?
3072 gettext(" System Attribute ") : gettext(" Attribute "),
3073 (Gen.g_attrnam_p == NULL) ? "" :
3074 (Gen.g_attrparent_p == NULL) ? Gen.g_attrnam_p :
3075 Gen.g_attrparent_p);
3076 return;
3077 }
3078
3079 /* save original file size */
3080 orig_filesz = G_p->g_filesz;
3081
3082 /*
3083 * Calculate the new compressed file size of a sparse file
3084 * before any of the header information is written
3085 * to the archive.
3086 */
3087 if (Compress_sparse && S_ISREG(G_p->g_mode)) {
3088 /*
3089 * If the file being processed is a sparse file, gather the
3090 * hole information and the compressed file size.
3091 * G_p->g_filesz will need to be changed to be the size of
3092 * the compressed sparse file plus the the size of the hole
3093 * information that will be prepended to the compressed file
3094 * in the archive.
3095 */
3096 holes = get_holes_info(Ifile, G_p->g_filesz, B_FALSE);
3097 if (holes != NULL)
3098 G_p->g_filesz = holes->holesdata_sz + holes->data_size;
3099
3100 if (G_p->g_filesz > Max_offset) {
3101 msg(ERR, "%s%s%s: too large to archive "
3102 "in current mode",
3103 G_p->g_nam_p,
3104 (G_p->g_attrnam_p == NULL) ? "" :
3105 G_p->g_rw_sysattr ?
3106 gettext(" System Attribute ") :
3107 gettext(" Attribute "),
3108 (G_p->g_attrnam_p == NULL) ? "" :
3109 ((G_p->g_attrparent_p == NULL) ?
3110 G_p->g_attrnam_p:
3111 G_p->g_attrpath_p));
3112
3113 (void) close(Ifile);
3114 if (holes != NULL)
3115 free_holes_info(holes);
3116 return; /* do not archive if it's too big */
3117 }
3118 }
3119
3120 /*
3121 * Dump extended attribute header.
3122 */
3123
3124 if (Gen.g_attrnam_p != NULL) {
3125 write_xattr_hdr();
3126 }
3127
3128 if (Hdr_type == CRC) {
3129 uint_t csum = cksum(CRC, 0, &errret);
3130 if (errret != 0) {
3131 G_p->g_cksum = UINT_MAX;
3132 msg(POST, "\"%s%s%s\" skipped",
3133 (Gen.g_attrnam_p == NULL) ?
3134 nam_p : Gen.g_attrfnam_p,
3135 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_rw_sysattr ?
3136 gettext(" System Attribute ") :
3137 gettext(" Attribute "),
3138 (Gen.g_attrnam_p == NULL) ? "" : nam_p);
3139 if (holes != NULL)
3140 free_holes_info(holes);
3141 (void) close(Ifile);
3142 return;
3143 }
3144 G_p->g_cksum = csum;
3145 } else {
3146 G_p->g_cksum = 0;
3147 }
3148
3149 /*
3150 * ACL has been retrieved in getname().
3151 */
3152 if (Pflag) {
3153 char *secinfo = NULL;
3154 int len = 0;
3155
3156 /* append security attributes */
3157 if ((append_secattr(&secinfo, &len, aclp)) == -1)
3158 msg(ERR, "can create security information");
3159
3160 /* call append_secattr() if more than one */
3161
3162 if (len > 0) {
3163 /* write ancillary only if there is sec info */
3164 write_hdr(ARCHIVE_ACL, (off_t)len);
3165 write_ancillary(secinfo, len, B_TRUE);
3166 }
3167 }
3168
3169 if (holes != NULL) {
3170 /*
3171 * Write the header info with a modified c_mode field to
3172 * indicate a compressed sparse file is being archived,
3173 * as well as the new file size, including the size of the
3174 * compressed file as well as all the prepended data.
3175 */
3176 write_hdr(ARCHIVE_SPARSE, (off_t)0);
3177 /* Prepend sparse file info */
3178 write_ancillary(holes->holesdata,
3179 holes->holesdata_sz, B_FALSE);
3180 } else {
3181 write_hdr(ARCHIVE_NORMAL, (off_t)0);
3182 }
3183
3184 real_filesz = 0;
3185
3186 if (holes != NULL) {
3187 amt_to_read = read_compress_holes(nam_p, G_p->g_filesz,
3188 &real_filesz, holes, &hole_changed);
3189 } else {
3190 amt_to_read = read_file(nam_p, G_p->g_filesz,
3191 &real_filesz, B_FALSE);
3192 }
3193
3194 while (amt_to_read > 0) {
3195 cnt = (amt_to_read > CPIOBSZ) ? CPIOBSZ : (int)amt_to_read;
3196 FLUSH(cnt);
3197 (void) memset(Buffr.b_in_p, 0, cnt);
3198 Buffr.b_in_p += cnt;
3199 Buffr.b_cnt += cnt;
3200 amt_to_read -= cnt;
3201 }
3202
3203 pad = (Pad_val + 1 - (G_p->g_filesz & Pad_val)) & Pad_val;
3204 if (pad != 0) {
3205 FLUSH(pad);
3206 (void) memset(Buffr.b_in_p, 0, pad);
3207 Buffr.b_in_p += pad;
3208 Buffr.b_cnt += pad;
3209 }
3210
3211 if (hole_changed == 1) {
3212 msg(ERR,
3213 "File data and hole offsets of \"%s%s%s\" have changed",
3214 (Gen.g_attrnam_p == NULL) ?
3215 G_p->g_nam_p : Gen.g_attrfnam_p,
3216 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_rw_sysattr ?
3217 gettext(" System Attribute ") : gettext(" Attribute "),
3218 (Gen.g_attrnam_p == NULL) ? "" : G_p->g_nam_p);
3219 }
3220 if (real_filesz > orig_filesz) {
3221 msg(ERR, "File size of \"%s%s%s\" has increased by %lld",
3222 (Gen.g_attrnam_p == NULL) ?
3223 G_p->g_nam_p : Gen.g_attrfnam_p,
3224 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_rw_sysattr ?
3225 gettext(" System Attribute ") : gettext(" Attribute "),
3226 (Gen.g_attrnam_p == NULL) ? "" : G_p->g_nam_p,
3227 (real_filesz - orig_filesz));
3228 }
3229 if (real_filesz < orig_filesz) {
3230 msg(ERR, "File size of \"%s%s%s\" has decreased by %lld",
3231 (Gen.g_attrnam_p == NULL) ?
3232 G_p->g_nam_p : Gen.g_attrfnam_p,
3233 (Gen.g_attrnam_p == NULL) ? "" : Gen.g_rw_sysattr ?
3234 gettext(" System Attribute ") : gettext(" Attribute "),
3235 (Gen.g_attrnam_p == NULL) ? "" : G_p->g_nam_p,
3236 (orig_filesz - real_filesz));
3237 }
3238
3239 if (holes != NULL)
3240 free_holes_info(holes);
3241
3242 (void) close(Ifile);
3243 rstfiles(U_KEEP, G_p->g_dirfd);
3244 VERBOSE((Args & (OCv | OCV)), G_p->g_nam_p);
3245 }
3246
3247 /*
3248 * data_pass: If not a special file (Aspec), open(2) the file to be
3249 * transferred, read(2) each block of data and write(2) it to the output file
3250 * Ofile, which was opened in file_pass().
3251 */
3252 static void
data_pass(void)3253 data_pass(void)
3254 {
3255 int rv;
3256 int cstatus;
3257 char *namep = Nam_p;
3258 holes_info_t *holes = NULL;
3259 data_in_t *data_in_info;
3260
3261 if (G_p->g_attrnam_p != NULL) {
3262 namep = G_p->g_attrnam_p;
3263 }
3264 if (Aspec) {
3265 rstfiles(U_KEEP, G_p->g_passdirfd);
3266 cstatus = close(Ofile);
3267 Ofile = 0;
3268 VERBOSE((Args & (OCv | OCV)), Nam_p);
3269 if (cstatus != 0) {
3270 msg(EXTN, "close error");
3271 }
3272 return;
3273 }
3274 if ((Ifile = openat(G_p->g_dirfd, get_component(namep), 0)) < 0) {
3275 msg(ERRN, "Cannot open \"%s%s%s\", skipped",
3276 (G_p->g_attrnam_p == NULL) ? Nam_p : G_p->g_attrfnam_p,
3277 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_rw_sysattr ?
3278 gettext(" System Attribute ") : gettext(" Attribute "),
3279 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p);
3280 rstfiles(U_KEEP, G_p->g_passdirfd);
3281 cstatus = close(Ofile);
3282 Ofile = 0;
3283 if (cstatus != 0) {
3284 msg(EXTN, "close error");
3285 }
3286 return;
3287 }
3288
3289 data_in_info = e_zalloc(E_EXIT, sizeof (data_in_t));
3290 data_in_info->data_in_proc_mode = P_PROC;
3291
3292 if (S_ISREG(G_p->g_mode))
3293 holes = get_holes_info(Ifile, G_p->g_filesz, B_TRUE);
3294
3295 if (holes != NULL) {
3296 rv = data_copy_with_holes(Ifile, Ofile,
3297 (G_p->g_attrnam_p == NULL) ? 0 : G_p->g_rw_sysattr,
3298 G_p->g_filesz, Bufsize, data_in_info, holes);
3299
3300 free_holes_info(holes);
3301 } else {
3302 rv = data_copy(Ifile, Ofile,
3303 (G_p->g_attrnam_p == NULL) ? 0 : G_p->g_rw_sysattr,
3304 G_p->g_filesz, Bufsize, data_in_info);
3305 }
3306
3307 if (rv < 0) {
3308 /* read error or unexpected EOF */
3309 if (data_in_info->data_in_rd_eof) {
3310 /*
3311 * read has reached EOF unexpectedly, but this isn't
3312 * an error since it's the latest shape of the file.
3313 */
3314 msg(EPOST, "File size of \"%s%s%s\" has decreased",
3315 (G_p->g_attrnam_p == NULL) ?
3316 Nam_p : G_p->g_attrfnam_p,
3317 (G_p->g_attrnam_p == NULL) ? "" :
3318 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
3319 gettext(" Attribute "),
3320 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p);
3321
3322 /* It's not error. We'll use the new file */
3323 rv = 0;
3324 } else {
3325 /* read error */
3326 msg(ERRN, "Cannot read \"%s%s%s\"",
3327 (G_p->g_attrnam_p == NULL) ?
3328 Nam_p : G_p->g_attrfnam_p,
3329 (G_p->g_attrnam_p == NULL) ? "" :
3330 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
3331 gettext(" Attribute "),
3332 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p);
3333 }
3334 } else if (rv > 0) {
3335 /* write error */
3336 if (Do_rename) {
3337 msg(ERRN, "Cannot write \"%s%s%s\"", Over_p,
3338 (G_p->g_attrnam_p == NULL) ? "" :
3339 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
3340 gettext(" Attribute "),
3341 (G_p->g_attrnam_p == NULL) ? "" : Over_p);
3342 } else {
3343 msg(ERRN, "Cannot write \"%s%s%s\"",
3344 Fullnam_p,
3345 (G_p->g_attrnam_p == NULL) ? "" :
3346 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
3347 gettext(" Attribute "),
3348 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p);
3349 }
3350 }
3351
3352 free(data_in_info);
3353
3354 if (rv == 0) {
3355 rstfiles(U_OVER, G_p->g_passdirfd);
3356 } else {
3357 rstfiles(U_KEEP, G_p->g_passdirfd);
3358 }
3359
3360 (void) close(Ifile);
3361 cstatus = close(Ofile);
3362 Ofile = 0;
3363 if (cstatus != 0) {
3364 msg(EXTN, "close error");
3365 }
3366 VERBOSE((Args & (OCv | OCV)), Fullnam_p);
3367 Finished = 1;
3368 }
3369
3370 /*
3371 * file_in: Process an object from the archive. If a TARTYP (TAR or USTAR)
3372 * archive and g_nlink == 1, link this file to the file name in t_linkname
3373 * and return. Handle linked files in one of two ways. If Onecopy == 0, this
3374 * is an old style (binary or -c) archive, create and extract the data for the
3375 * first link found, link all subsequent links to this file and skip their data.
3376 * If Oncecopy == 1, save links until all have been processed, and then
3377 * process the links first to last checking their names against the patterns
3378 * and/or asking the user to rename them. The first link that is accepted
3379 * for xtraction is created and the data is read from the archive.
3380 * All subsequent links that are accepted are linked to this file.
3381 */
3382 static void
file_in(void)3383 file_in(void)
3384 {
3385 struct Lnk *l_p, *tl_p;
3386 int lnkem = 0, cleanup = 0;
3387 int proc_file;
3388 struct Lnk *ttl_p;
3389 int typeflag;
3390 char savacl;
3391 int cwd;
3392
3393 G_p = &Gen;
3394
3395 /*
3396 * Now that we've read the extended header,
3397 * determine if we should restore attributes.
3398 * Don't restore the attribute if we are extracting
3399 * a file from an archive (as opposed to doing a table of
3400 * contents) and any of the following are true:
3401 * 1. neither -@ or -/ was specified.
3402 * 2. -@ was specified, -/ wasn't specified, and we're
3403 * processing a hidden attribute directory of an attribute
3404 * or we're processing a read-write system attribute file.
3405 * 3. -@ wasn't specified, -/ was specified, and the file
3406 * we're processing it not a read-write system attribute file,
3407 * or we're processing the hidden attribute directory of an
3408 * attribute.
3409 *
3410 * We always process the attributes if we're just generating
3411 * generating a table of contents, or if both -@ and -/ were
3412 * specified.
3413 */
3414 if (G_p->g_attrnam_p != NULL) {
3415 if (((Args & OCt) == 0) &&
3416 ((!Atflag && !SysAtflag) ||
3417 (Atflag && !SysAtflag && ((G_p->g_attrparent_p != NULL) ||
3418 G_p->g_rw_sysattr)) ||
3419 (!Atflag && SysAtflag && ((G_p->g_attrparent_p != NULL) ||
3420 !G_p->g_rw_sysattr)))) {
3421 proc_file = F_SKIP;
3422 data_in(P_SKIP);
3423 return;
3424 }
3425 }
3426
3427 /*
3428 * Open target directory if this isn't a skipped file
3429 * and g_nlink == 1
3430 *
3431 * Links are handled further down in this function.
3432 */
3433
3434 proc_file = ckname(0);
3435
3436 if (proc_file == F_SKIP && G_p->g_nlink == 1) {
3437 /*
3438 * Normally ckname() prints out the file as a side
3439 * effect except for table of contents listing
3440 * when its parameter is zero and Onecopy isn't
3441 * Zero. Due to this we need to force the name
3442 * to be printed here.
3443 */
3444 if (Onecopy == 1) {
3445 VERBOSE((Args & OCt), G_p->g_nam_p);
3446 }
3447 data_in(P_SKIP);
3448 return;
3449 }
3450
3451 if (proc_file != F_SKIP && open_dirfd() != 0) {
3452 data_in(P_SKIP);
3453 return;
3454 }
3455
3456 if (Hdr_type == BAR) {
3457 bar_file_in();
3458 close_dirfd();
3459 return;
3460 }
3461
3462 /*
3463 * For archives in USTAR format, the files are extracted according
3464 * to the typeflag.
3465 */
3466 if (Hdr_type == USTAR || Hdr_type == TAR) {
3467 typeflag = Thdr_p->tbuf.t_typeflag;
3468 if (G_p->g_nlink == 1) { /* hard link */
3469 if (proc_file != F_SKIP) {
3470 int i;
3471 char lname[NAMSIZ+1];
3472 (void) memset(lname, '\0', sizeof (lname));
3473
3474 (void) strncpy(lname, Thdr_p->tbuf.t_linkname,
3475 NAMSIZ);
3476 for (i = 0; i <= NAMSIZ && lname[i] != 0; i++)
3477 ;
3478
3479 lname[i] = 0;
3480 (void) creat_lnk(G_p->g_dirfd,
3481 &lname[0], G_p->g_nam_p);
3482 }
3483 close_dirfd();
3484 return;
3485 }
3486 if (typeflag == '3' || typeflag == '4' || typeflag == '5' ||
3487 typeflag == '6') {
3488 if (proc_file != F_SKIP &&
3489 creat_spec(G_p->g_dirfd) > 0) {
3490 VERBOSE((Args & (OCv | OCV)),
3491 (G_p->g_attrparent_p == NULL) ?
3492 G_p->g_nam_p : G_p->g_attrpath_p);
3493 }
3494 close_dirfd();
3495 return;
3496 } else if (Adir || Aspec) {
3497 if ((proc_file == F_SKIP) ||
3498 (Ofile = openout(G_p->g_dirfd)) < 0) {
3499 data_in(P_SKIP);
3500 } else {
3501 data_in(P_PROC);
3502 }
3503 close_dirfd();
3504 return;
3505 }
3506 }
3507
3508 if (Adir) {
3509 if (proc_file != F_SKIP && creat_spec(G_p->g_dirfd) > 0) {
3510 VERBOSE((Args & (OCv | OCV)), G_p->g_nam_p);
3511 }
3512 close_dirfd();
3513 if (Onecopy == 1) {
3514 VERBOSE((Args & OCt), G_p->g_nam_p);
3515 }
3516 return;
3517 }
3518 if (G_p->g_nlink == 1 || (Hdr_type == TAR ||
3519 Hdr_type == USTAR)) {
3520 if (Aspec) {
3521 if (proc_file != F_SKIP && creat_spec(G_p->g_dirfd) > 0)
3522 VERBOSE((Args & (OCv | OCV)), G_p->g_nam_p);
3523 } else {
3524 if ((proc_file == F_SKIP) ||
3525 (Ofile = openout(G_p->g_dirfd)) < 0) {
3526 data_in(P_SKIP);
3527 } else {
3528 data_in(P_PROC);
3529 }
3530 }
3531 close_dirfd();
3532 return;
3533 }
3534 close_dirfd();
3535
3536 tl_p = add_lnk(&ttl_p);
3537 l_p = ttl_p;
3538 if (l_p->L_cnt == l_p->L_gen.g_nlink)
3539 cleanup = 1;
3540 if (!Onecopy || G_p->g_attrnam_p != NULL) {
3541 lnkem = (tl_p != l_p) ? 1 : 0;
3542 G_p = &tl_p->L_gen;
3543 if (proc_file == F_SKIP) {
3544 data_in(P_SKIP);
3545 } else {
3546 if (open_dirfd() != 0)
3547 return;
3548 if (!lnkem) {
3549 if (Aspec) {
3550 if (creat_spec(G_p->g_dirfd) > 0)
3551 VERBOSE((Args & (OCv | OCV)),
3552 G_p->g_nam_p);
3553 } else if ((Ofile =
3554 openout(G_p->g_dirfd)) < 0) {
3555 data_in(P_SKIP);
3556 close_dirfd();
3557 reclaim(l_p);
3558 } else {
3559 data_in(P_PROC);
3560 close_dirfd();
3561 }
3562 } else {
3563 /*
3564 * Are we linking an attribute?
3565 */
3566 cwd = -1;
3567 if (l_p->L_gen.g_attrnam_p != NULL) {
3568 (void) strcpy(Lnkend_p,
3569 l_p->L_gen.g_attrnam_p);
3570 (void) strcpy(Full_p,
3571 tl_p->L_gen.g_attrnam_p);
3572 cwd = save_cwd();
3573 (void) fchdir(G_p->g_dirfd);
3574 } else {
3575 (void) strcpy(Lnkend_p,
3576 l_p->L_gen.g_nam_p);
3577 (void) strcpy(Full_p,
3578 tl_p->L_gen.g_nam_p);
3579 }
3580 (void) creat_lnk(G_p->g_dirfd,
3581 Lnkend_p, Full_p);
3582 data_in(P_SKIP);
3583 close_dirfd();
3584 l_p->L_lnk_p = NULL;
3585 free(tl_p->L_gen.g_nam_p);
3586 free(tl_p);
3587 if (cwd != -1)
3588 rest_cwd(cwd);
3589 }
3590 }
3591 } else { /* Onecopy */
3592 if (tl_p->L_gen.g_filesz)
3593 cleanup = 1;
3594 if (!cleanup) {
3595 close_dirfd();
3596 return; /* don't do anything yet */
3597 }
3598 tl_p = l_p;
3599 /*
3600 * ckname will clear aclchar. We need to keep aclchar for
3601 * all links.
3602 */
3603 savacl = aclchar;
3604 while (tl_p != NULL) {
3605 G_p = &tl_p->L_gen;
3606 aclchar = savacl;
3607 if ((proc_file = ckname(1)) != F_SKIP) {
3608 if (open_dirfd() != 0) {
3609 return;
3610 }
3611 if (l_p->L_data) {
3612 (void) creat_lnk(G_p->g_dirfd,
3613 l_p->L_gen.g_nam_p,
3614 G_p->g_nam_p);
3615 } else if (Aspec) {
3616 (void) creat_spec(G_p->g_dirfd);
3617 l_p->L_data = 1;
3618 VERBOSE((Args & (OCv | OCV)),
3619 G_p->g_nam_p);
3620 } else if ((Ofile =
3621 openout(G_p->g_dirfd)) < 0) {
3622 proc_file = F_SKIP;
3623 } else {
3624 data_in(P_PROC);
3625 l_p->L_data = 1;
3626 }
3627 } /* (proc_file = ckname(1)) != F_SKIP */
3628
3629 tl_p = tl_p->L_lnk_p;
3630
3631 close_dirfd();
3632
3633 if (proc_file == F_SKIP && !cleanup) {
3634 tl_p->L_nxt_p = l_p->L_nxt_p;
3635 tl_p->L_bck_p = l_p->L_bck_p;
3636 l_p->L_bck_p->L_nxt_p = tl_p;
3637 l_p->L_nxt_p->L_bck_p = tl_p;
3638 free(l_p->L_gen.g_nam_p);
3639 free(l_p);
3640 }
3641 } /* tl_p->L_lnk_p != NULL */
3642 if (l_p->L_data == 0) {
3643 data_in(P_SKIP);
3644 }
3645 }
3646 if (cleanup) {
3647 reclaim(l_p);
3648 }
3649 }
3650
3651 /*
3652 * file_out: If the current file is not a special file (!Aspec) and it
3653 * is identical to the archive, skip it (do not archive the archive if it
3654 * is a regular file). If creating a TARTYP (TAR or USTAR) archive, the first
3655 * time a link to a file is encountered, write the header and file out normally.
3656 * Subsequent links to this file put this file name in their t_linkname field.
3657 * Otherwise, links are handled in one of two ways, for the old headers
3658 * (i.e. binary and -c), linked files are written out as they are encountered.
3659 * For the new headers (ASC and CRC), links are saved up until all the links
3660 * to each file are found. For a file with n links, write n - 1 headers with
3661 * g_filesz set to 0, write the final (nth) header with the correct g_filesz
3662 * value and write the data for the file to the archive.
3663 */
3664 static
3665 int
file_out(void)3666 file_out(void)
3667 {
3668 struct Lnk *l_p, *tl_p;
3669 int cleanup = 0;
3670 struct Lnk *ttl_p;
3671
3672 G_p = &Gen;
3673 if (!Aspec && IDENT(SrcSt, ArchSt))
3674 return (1); /* do not archive the archive if it's a reg file */
3675 /*
3676 * If compressing sparse files, wait until the compressed file size
3677 * is known to check if file size is too big.
3678 */
3679 if (Compress_sparse == 0 && G_p->g_filesz > Max_offset) {
3680 msg(ERR, "%s%s%s: too large to archive in current mode",
3681 G_p->g_nam_p,
3682 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_rw_sysattr ?
3683 gettext(" System Attribute ") : gettext(" Attribute "),
3684 (G_p->g_attrnam_p == NULL) ? "" :
3685 ((G_p->g_attrparent_p == NULL) ? G_p->g_attrnam_p:
3686 G_p->g_attrpath_p));
3687 return (1); /* do not archive if it's too big */
3688 }
3689 if (Hdr_type == TAR || Hdr_type == USTAR) { /* TAR and USTAR */
3690 if (Adir) {
3691 if (Gen.g_attrnam_p != NULL) {
3692 write_xattr_hdr();
3693 }
3694 write_hdr(ARCHIVE_NORMAL, 0);
3695 return (0);
3696 }
3697 if (G_p->g_nlink == 1) {
3698 data_out();
3699 return (0);
3700 }
3701 tl_p = add_lnk(&ttl_p);
3702 l_p = ttl_p;
3703 if (tl_p == l_p) { /* first link to this file encountered */
3704 data_out();
3705 return (0);
3706 }
3707 (void) strncpy(T_lname, l_p->L_gen.g_nam_p,
3708 l_p->L_gen.g_namesz);
3709
3710 /*
3711 * check if linkname is greater than 100 characters
3712 */
3713 if (strlen(T_lname) > NAMSIZ) {
3714 msg(EPOST, "cpio: %s: linkname %s is greater than %d",
3715 G_p->g_nam_p, T_lname, NAMSIZ);
3716 return (1);
3717 }
3718
3719 write_hdr(ARCHIVE_NORMAL, (off_t)0);
3720 VERBOSE((Args & (OCv | OCV)), tl_p->L_gen.g_nam_p);
3721
3722 /* find the lnk entry in sublist, unlink it, and free it */
3723 for (; ttl_p->L_lnk_p != NULL;
3724 ttl_p = ttl_p->L_lnk_p) {
3725 if (ttl_p->L_lnk_p == tl_p) {
3726 ttl_p->L_lnk_p = tl_p->L_lnk_p;
3727 free(tl_p->L_gen.g_nam_p);
3728 free(tl_p);
3729 break;
3730 }
3731 }
3732
3733 return (0);
3734 }
3735 if (Adir) {
3736 /*
3737 * ACL has been retrieved in getname().
3738 */
3739 if (Pflag) {
3740 char *secinfo = NULL;
3741 int len = 0;
3742
3743 /* append security attributes */
3744 if ((append_secattr(&secinfo, &len, aclp)) == -1)
3745 msg(ERR, "can create security information");
3746
3747 /* call append_secattr() if more than one */
3748
3749 if (len > 0) {
3750 /* write ancillary */
3751 write_hdr(ARCHIVE_ACL, (off_t)len);
3752 write_ancillary(secinfo, len, B_TRUE);
3753 }
3754 }
3755
3756 if (Gen.g_attrnam_p != NULL) {
3757 write_xattr_hdr();
3758 }
3759 write_hdr(ARCHIVE_NORMAL, (off_t)0);
3760 VERBOSE((Args & (OCv | OCV)), G_p->g_nam_p);
3761 return (0);
3762 }
3763 if (G_p->g_nlink == 1) {
3764 data_out();
3765 return (0);
3766 } else {
3767 tl_p = add_lnk(&ttl_p);
3768 l_p = ttl_p;
3769
3770 if (l_p->L_cnt == l_p->L_gen.g_nlink)
3771 cleanup = 1;
3772 else if (Onecopy && G_p->g_attrnam_p == NULL) {
3773 return (0); /* don't process data yet */
3774 }
3775 }
3776 if (Onecopy && G_p->g_attrnam_p == NULL) {
3777 tl_p = l_p;
3778 while (tl_p->L_lnk_p != NULL) {
3779 G_p = &tl_p->L_gen;
3780 G_p->g_filesz = (off_t)0;
3781 /* one link with the acl is sufficient */
3782 write_hdr(ARCHIVE_NORMAL, (off_t)0);
3783 VERBOSE((Args & (OCv | OCV)), G_p->g_nam_p);
3784 tl_p = tl_p->L_lnk_p;
3785 }
3786 G_p = &tl_p->L_gen;
3787 if (open_dirfd() != 0)
3788 return (1);
3789 }
3790 /* old style: has acl and data for every link */
3791 data_out();
3792 if (cleanup)
3793 reclaim(l_p);
3794 return (0);
3795 }
3796
3797 /*
3798 * Verify the underlying file system supports the attribute type.
3799 * Only archive extended attribute files when '-@' was specified.
3800 * Only archive system extended attribute files if '-/' was specified.
3801 */
3802 #if defined(O_XATTR)
3803 static attr_status_t
verify_attr_support(char * filename,int attrflg,arc_action_t actflag,int * ext_attrflg)3804 verify_attr_support(char *filename, int attrflg, arc_action_t actflag,
3805 int *ext_attrflg)
3806 {
3807 /*
3808 * Verify extended attributes are supported/exist. We only
3809 * need to check if we are processing a base file, not an
3810 * extended attribute.
3811 */
3812 if (attrflg) {
3813 *ext_attrflg = (pathconf(filename, (actflag == ARC_CREATE) ?
3814 _PC_XATTR_EXISTS : _PC_XATTR_ENABLED) == 1);
3815 }
3816 if (Atflag) {
3817 #if defined(_PC_SATTR_ENABLED)
3818 if (!*ext_attrflg) {
3819 if (SysAtflag) {
3820 /* Verify system attributes are supported */
3821 if (sysattr_support(filename,
3822 (actflag == ARC_CREATE) ?_PC_SATTR_EXISTS :
3823 _PC_SATTR_ENABLED) != 1) {
3824 return (ATTR_SATTR_ERR);
3825 }
3826 } else
3827 return (ATTR_XATTR_ERR);
3828 #else
3829 return (ATTR_XATTR_ERR);
3830 #endif /* _PC_SATTR_ENABLED */
3831 }
3832
3833 #if defined(_PC_SATTR_ENABLED)
3834 } else if (SysAtflag) {
3835 /* Verify system attributes are supported */
3836 if (sysattr_support(filename, (actflag == ARC_CREATE) ?
3837 _PC_SATTR_EXISTS : _PC_SATTR_ENABLED) != 1) {
3838 return (ATTR_SATTR_ERR);
3839 }
3840 #endif /* _PC_SATTR_ENABLED */
3841 } else {
3842 return (ATTR_SKIP);
3843 }
3844
3845 return (ATTR_OK);
3846 }
3847 #endif
3848
3849 #if defined(O_XATTR)
3850 /*
3851 * Verify the attribute, attrname, is an attribute we want to restore.
3852 * Never restore read-only system attribute files. Only restore read-write
3853 * system attributes files when -/ was specified, and only traverse into
3854 * the 2nd level attribute directory containing only system attributes if
3855 * -@ was specified. This keeps us from archiving
3856 * <attribute name>/<read-write system attribute file>
3857 * when -/ was specified without -@.
3858 *
3859 * attrname - attribute file name
3860 * attrparent - attribute's parent name within the base file's
3861 * attribute digrectory hierarchy
3862 * arc_rwsysattr - flag that indicates that read-write system attribute
3863 * file should be archived as it contains other than
3864 * the default system attributes.
3865 * rw_sysattr - on return, flag will indicate if attrname is a
3866 * read-write system attribute file.
3867 */
3868 static attr_status_t
verify_attr(char * attrname,char * attrparent,int arc_rwsysattr,int * rw_sysattr)3869 verify_attr(char *attrname, char *attrparent, int arc_rwsysattr,
3870 int *rw_sysattr)
3871 {
3872 #if defined(_PC_SATTR_ENABLED)
3873 int attr_supported;
3874
3875 /* Never restore read-only system attribute files */
3876 if ((attr_supported = sysattr_type(attrname)) == _RO_SATTR) {
3877 *rw_sysattr = 0;
3878 return (ATTR_SKIP);
3879 } else {
3880 *rw_sysattr = (attr_supported == _RW_SATTR);
3881 }
3882
3883 /*
3884 * Don't archive a read-write system attribute file if
3885 * it contains only the default system attributes.
3886 */
3887 if (*rw_sysattr && !arc_rwsysattr) {
3888 return (ATTR_SKIP);
3889 }
3890
3891 #else
3892 /* Never restore read-only system attribute files */
3893 if ((*rw_sysattr = is_sysattr(attrname)) == 1) {
3894 return (ATTR_SKIP);
3895 }
3896 #endif /* _PC_SATTR_ENABLED */
3897
3898 /*
3899 * Only restore read-write system attribute files
3900 * when -/ was specified. Only restore extended
3901 * attributes when -@ was specified.
3902 */
3903 if (Atflag) {
3904 if (!SysAtflag) {
3905 /*
3906 * Only archive/restore the hidden directory "." if
3907 * we're processing the top level hidden attribute
3908 * directory. We don't want to process the
3909 * hidden attribute directory of the attribute
3910 * directory that contains only extended system
3911 * attributes.
3912 */
3913 if (*rw_sysattr || (Hiddendir &&
3914 (attrparent != NULL))) {
3915 return (ATTR_SKIP);
3916 }
3917 }
3918 } else if (SysAtflag) {
3919 /*
3920 * Only archive/restore read-write extended system attribute
3921 * files of the base file.
3922 */
3923 if (!*rw_sysattr || (attrparent != NULL)) {
3924 return (ATTR_SKIP);
3925 }
3926 } else {
3927 return (ATTR_SKIP);
3928 }
3929
3930 return (ATTR_OK);
3931 }
3932 #endif
3933
3934 #if defined(O_XATTR)
3935 static int
retry_open_attr(int pdirfd,int cwd,char * fullname,char * pattr,char * name,int oflag,mode_t mode)3936 retry_open_attr(int pdirfd, int cwd, char *fullname, char *pattr, char *name,
3937 int oflag, mode_t mode)
3938 {
3939 int dirfd;
3940 int ofilefd = -1;
3941 struct timeval times[2];
3942 mode_t newmode;
3943 struct stat parentstat;
3944 acl_t *aclp = NULL;
3945 int error;
3946
3947 /*
3948 * We couldn't get to attrdir. See if its
3949 * just a mode problem on the parent file.
3950 * for example: a mode such as r-xr--r--
3951 * on a ufs file system without extended
3952 * system attribute support won't let us
3953 * create an attribute dir if it doesn't
3954 * already exist, and on a ufs file system
3955 * with extended system attribute support
3956 * won't let us open the attribute for
3957 * write.
3958 *
3959 * If file has a non-trivial ACL, then save it
3960 * off so that we can place it back on after doing
3961 * chmod's.
3962 */
3963 if ((dirfd = openat(cwd, (pattr == NULL) ? fullname : pattr,
3964 O_RDONLY)) == -1) {
3965 return (-1);
3966 }
3967 if (fstat(dirfd, &parentstat) == -1) {
3968 msg(ERRN, "Cannot stat %sfile %s",
3969 (pdirfd == -1) ? "" : gettext("parent of "),
3970 (pdirfd == -1) ? fullname : name);
3971 (void) close(dirfd);
3972 return (-1);
3973 }
3974 if ((error = facl_get(dirfd, ACL_NO_TRIVIAL, &aclp)) != 0) {
3975 msg(ERRN, "Failed to retrieve ACL on %sfile %s",
3976 (pdirfd == -1) ? "" : gettext("parent of "),
3977 (pdirfd == -1) ? fullname : name);
3978 (void) close(dirfd);
3979 return (-1);
3980 }
3981
3982 newmode = S_IWUSR | parentstat.st_mode;
3983 if (fchmod(dirfd, newmode) == -1) {
3984 msg(ERRN, "Cannot change mode of %sfile %s to %o",
3985 (pdirfd == -1) ? "" : gettext("parent of "),
3986 (pdirfd == -1) ? fullname : name, newmode);
3987 if (aclp)
3988 acl_free(aclp);
3989 (void) close(dirfd);
3990 return (-1);
3991 }
3992
3993
3994 if (pdirfd == -1) {
3995 /*
3996 * We weren't able to create the attribute directory before.
3997 * Now try again.
3998 */
3999 ofilefd = attropen(fullname, ".", oflag);
4000 } else {
4001 /*
4002 * We weren't able to create open the attribute before.
4003 * Now try again.
4004 */
4005 ofilefd = openat(pdirfd, name, oflag, mode);
4006 }
4007
4008 /*
4009 * Put mode back to original
4010 */
4011 if (fchmod(dirfd, parentstat.st_mode) == -1) {
4012 msg(ERRN, "Cannot restore permissions of %sfile %s to %o",
4013 (pdirfd == -1) ? "" : gettext("parent of "),
4014 (pdirfd == -1) ? fullname : name, newmode);
4015 }
4016
4017 if (aclp) {
4018 error = facl_set(dirfd, aclp);
4019 if (error) {
4020 msg(ERRN, "failed to set acl entries on %sfile %s\n",
4021 (pdirfd == -1) ? "" : gettext("parent of "),
4022 (pdirfd == -1) ? fullname : name);
4023 }
4024 acl_free(aclp);
4025 }
4026
4027 /*
4028 * Put back time stamps
4029 */
4030
4031 times[0].tv_sec = parentstat.st_atime;
4032 times[0].tv_usec = 0;
4033 times[1].tv_sec = parentstat.st_mtime;
4034 times[1].tv_usec = 0;
4035
4036 (void) futimesat(cwd, (pattr == NULL) ? fullname : pattr, times);
4037
4038 (void) close(dirfd);
4039
4040 return (ofilefd);
4041 }
4042 #endif
4043
4044 #if defined(O_XATTR)
4045 /*
4046 * Recursively open attribute directories until the attribute directory
4047 * containing the specified attribute, attrname, is opened.
4048 *
4049 * Currently, only 2 directory levels of attributes are supported, (i.e.,
4050 * extended system attributes on extended attributes). The following are
4051 * the possible input combinations:
4052 * 1. Open the attribute directory of the base file (don't change
4053 * into it).
4054 * attr_parent = NULL
4055 * attrname = '.'
4056 * 2. Open the attribute directory of the base file and change into it.
4057 * attr_parent = NULL
4058 * attrname = <attr> | <sys_attr>
4059 * 3. Open the attribute directory of the base file, change into it,
4060 * then recursively call open_attr_dir() to open the attribute's
4061 * parent directory (don't change into it).
4062 * attr_parent = <attr>
4063 * attrname = '.'
4064 * 4. Open the attribute directory of the base file, change into it,
4065 * then recursively call open_attr_dir() to open the attribute's
4066 * parent directory and change into it.
4067 * attr_parent = <attr>
4068 * attrname = <attr> | <sys_attr>
4069 *
4070 * An attribute directory will be opened only if the underlying file system
4071 * supports the attribute type, and if the command line specifications
4072 * (f_extended_attr and f_sys_attr) enable the processing of the attribute
4073 * type.
4074 *
4075 * On succesful return, attr_parentfd will be the file descriptor of the
4076 * opened attribute directory. In addition, if the attribute is a read-write
4077 * extended system attribute, rw_sysattr will be set to 1, otherwise
4078 * it will be set to 0.
4079 *
4080 * Possible return values:
4081 * ATTR_OK Successfully opened and, if needed, changed into the
4082 * attribute directory containing attrname.
4083 * ATTR_SKIP The command line specifications don't enable the
4084 * processing of the attribute type.
4085 * ATTR_CHDIR_ERR An error occurred while trying to change into an
4086 * attribute directory.
4087 * ATTR_OPEN_ERR An error occurred while trying to open an
4088 * attribute directory.
4089 * ATTR_XATTR_ERR The underlying file system doesn't support extended
4090 * attributes.
4091 * ATTR_SATTR_ERR The underlying file system doesn't support extended
4092 * system attributes.
4093 */
4094 static int
open_attr_dir(char * attrname,char * dirp,int cwd,char * attr_parent,int * attr_parentfd,int * rw_sysattr)4095 open_attr_dir(char *attrname, char *dirp, int cwd, char *attr_parent,
4096 int *attr_parentfd, int *rw_sysattr)
4097 {
4098 attr_status_t rc;
4099 int firsttime = (*attr_parentfd == -1);
4100 int saveerrno;
4101 int ext_attr;
4102
4103 /*
4104 * open_attr_dir() was recursively called (input combination number 4),
4105 * close the previously opened file descriptor as we've already changed
4106 * into it.
4107 */
4108 if (!firsttime) {
4109 (void) close(*attr_parentfd);
4110 *attr_parentfd = -1;
4111 }
4112
4113 /*
4114 * Verify that the underlying file system supports the restoration
4115 * of the attribute.
4116 */
4117 if ((rc = verify_attr_support(dirp, firsttime, ARC_RESTORE,
4118 &ext_attr)) != ATTR_OK) {
4119 return (rc);
4120 }
4121
4122 /* Open the base file's attribute directory */
4123 if ((*attr_parentfd = attropen(dirp, ".", O_RDONLY)) == -1) {
4124 /*
4125 * Save the errno from the attropen so it can be reported
4126 * if the retry of the attropen fails.
4127 */
4128 saveerrno = errno;
4129 if ((*attr_parentfd = retry_open_attr(-1, cwd, dirp,
4130 NULL, ".", O_RDONLY, 0)) == -1) {
4131 (void) close(*attr_parentfd);
4132 *attr_parentfd = -1;
4133 errno = saveerrno;
4134 return (ATTR_OPEN_ERR);
4135 }
4136 }
4137
4138 /*
4139 * Change into the parent attribute's directory unless we are
4140 * processing the hidden attribute directory of the base file itself.
4141 */
4142 if ((Hiddendir == 0) || (firsttime && (attr_parent != NULL))) {
4143 if (fchdir(*attr_parentfd) != 0) {
4144 saveerrno = errno;
4145 (void) close(*attr_parentfd);
4146 *attr_parentfd = -1;
4147 errno = saveerrno;
4148 return (ATTR_CHDIR_ERR);
4149 }
4150 }
4151
4152 /* Determine if the attribute should be processed */
4153 if ((rc = verify_attr(attrname, attr_parent, 1,
4154 rw_sysattr)) != ATTR_OK) {
4155 saveerrno = errno;
4156 (void) close(*attr_parentfd);
4157 *attr_parentfd = -1;
4158 errno = saveerrno;
4159 return (rc);
4160 }
4161
4162 /*
4163 * If the attribute is an extended system attribute of an attribute
4164 * (i.e., <attr>/<sys_attr>), then recursively call open_attr_dir() to
4165 * open the attribute directory of the parent attribute.
4166 */
4167 if (firsttime && (attr_parent != NULL)) {
4168 return (open_attr_dir(attrname, attr_parent, *attr_parentfd,
4169 attr_parent, attr_parentfd, rw_sysattr));
4170 }
4171
4172 return (ATTR_OK);
4173 }
4174 #endif
4175
4176 /*
4177 * file_pass: If the -l option is set (link files when possible), and the
4178 * source and destination file systems are the same, link the source file
4179 * (G_p->g_nam_p) to the destination file (Fullnam) and return. If not a
4180 * linked file, transfer the data. Otherwise, the first link to a file
4181 * encountered is transferred normally and subsequent links are linked to it.
4182 */
4183
4184 static int
file_pass(void)4185 file_pass(void)
4186 {
4187 struct Lnk *l_p, *tl_p;
4188 struct Lnk *ttl_p;
4189 char *save_name;
4190 int size;
4191 int cwd;
4192 char *lfrom, *lto;
4193
4194 G_p = &Gen;
4195
4196 if (Adir && !(Args & OCd)) {
4197 msg(ERR, "Use -d option to copy \"%s\"", G_p->g_nam_p);
4198 return (FILE_PASS_ERR);
4199 }
4200
4201 save_name = G_p->g_nam_p;
4202
4203 while (*(G_p->g_nam_p) == '/') {
4204 G_p->g_nam_p++;
4205 }
4206
4207 (void) strcpy(Full_p, (G_p->g_attrfnam_p == NULL) ?
4208 G_p->g_nam_p : G_p->g_attrfnam_p);
4209
4210 if (G_p->g_attrnam_p == NULL) {
4211 G_p->g_passdirfd = open_dir(Fullnam_p);
4212
4213 if (G_p->g_passdirfd == -1) {
4214 msg(ERRN,
4215 "Cannot open/create \"%s\"", Fullnam_p);
4216 return (FILE_PASS_ERR);
4217 }
4218 } else {
4219 int rw_sysattr;
4220
4221 /*
4222 * Open the file's attribute directory.
4223 * Change into the base file's starting directory then call
4224 * open_attr_dir() to open the attribute directory of either
4225 * the base file (if G_p->g_attrparent_p is NULL) or the
4226 * attribute (if G_p->g_attrparent_p is set) of the base file.
4227 */
4228
4229 G_p->g_passdirfd = -1;
4230 (void) fchdir(G_p->g_baseparent_fd);
4231 (void) open_attr_dir(G_p->g_attrnam_p, Fullnam_p,
4232 G_p->g_baseparent_fd, (G_p->g_attrparent_p == NULL) ? NULL :
4233 G_p->g_attrparent_p, &G_p->g_passdirfd, &rw_sysattr);
4234 if (G_p->g_passdirfd == -1) {
4235 msg(ERRN,
4236 "Cannot open attribute directory of "
4237 "%s%s%sfile \"%s\"",
4238 (G_p->g_attrparent_p == NULL) ? "" :
4239 gettext("attribute \""),
4240 (G_p->g_attrparent_p == NULL) ? "" :
4241 G_p->g_attrparent_p,
4242 (G_p->g_attrparent_p == NULL) ? "" :
4243 gettext("\" of "), Fullnam_p);
4244 return (FILE_PASS_ERR);
4245 }
4246 }
4247
4248 if (Args & OCl) {
4249 /* We are linking back to the source directory. */
4250
4251 if (!Adir) {
4252 char *existingfile = save_name;
4253
4254 if ((Args & OCL) && issymlink) {
4255 /* We are chasing symlinks. */
4256
4257 if ((size = readlink(save_name, Symlnk_p,
4258 MAXPATHLEN)) < 0) {
4259 msg(ERRN,
4260 "Cannot read symbolic link \"%s\"",
4261 save_name);
4262 return (FILE_PASS_ERR);
4263 }
4264
4265 Symlnk_p[size] = '\0';
4266 existingfile = Symlnk_p;
4267 }
4268
4269 if (G_p->g_attrnam_p == NULL) {
4270 if (creat_lnk(G_p->g_passdirfd,
4271 existingfile, Fullnam_p) == 0) {
4272 return (FILE_LINKED);
4273 }
4274 }
4275 }
4276 }
4277
4278 if ((G_p->g_mode & Ftype) == S_IFLNK && !(Args & OCL)) {
4279 /* The archive file is a symlink. */
4280
4281 errno = 0;
4282
4283 if ((size = readlink(save_name, Symlnk_p, MAXPATHLEN)) < 0) {
4284 msg(ERRN,
4285 "Cannot read symbolic link \"%s\"", save_name);
4286 return (FILE_PASS_ERR);
4287 }
4288
4289 errno = 0;
4290 (void) missdir(Fullnam_p);
4291 *(Symlnk_p + size) = '\0';
4292
4293 if (symlink(Symlnk_p, Fullnam_p) < 0) {
4294 if (errno == EEXIST) {
4295 if (openout(G_p->g_passdirfd) < 0) {
4296 if (errno != EEXIST) {
4297 msg(ERRN,
4298 "Cannot create \"%s\"",
4299 Fullnam_p);
4300 }
4301 return (FILE_PASS_ERR);
4302 }
4303 } else {
4304 msg(ERRN, "Cannot create \"%s\"", Fullnam_p);
4305 return (FILE_PASS_ERR);
4306 }
4307 } else {
4308 if (Args & OCR) {
4309 if (lchown(Fullnam_p, (int)Rpw_p->pw_uid,
4310 (int)Rpw_p->pw_gid) < 0) {
4311 msg(ERRN,
4312 "Error during chown() of \"%s\"",
4313 Fullnam_p);
4314 }
4315 } else if ((lchown(Fullnam_p, (int)G_p->g_uid,
4316 (int)G_p->g_gid) < 0) && privileged) {
4317 msg(ERRN,
4318 "Error during chown() of \"%s\"",
4319 Fullnam_p);
4320 }
4321 }
4322
4323 VERBOSE((Args & (OCv | OCV)), Fullnam_p);
4324 return (FILE_PASS_ERR);
4325 }
4326
4327 if (!Adir && G_p->g_nlink > 1) {
4328 /* The archive file has hard links. */
4329
4330 tl_p = add_lnk(&ttl_p);
4331 l_p = ttl_p;
4332
4333 if (tl_p == l_p) {
4334 /* The archive file was not found. */
4335
4336 G_p = &tl_p->L_gen;
4337 } else {
4338 /* The archive file was found. */
4339
4340 cwd = -1;
4341
4342 if (l_p->L_gen.g_attrnam_p != NULL) {
4343 /* We are linking an attribute */
4344
4345 (void) strcpy(Lnkend_p, l_p->L_gen.g_attrnam_p);
4346 cwd = save_cwd();
4347 (void) fchdir(G_p->g_passdirfd);
4348 lfrom = get_component(Lnknam_p);
4349 lto = tl_p->L_gen.g_attrnam_p;
4350 } else {
4351 /* We are not linking an attribute */
4352
4353 (void) strcpy(Lnkend_p, l_p->L_gen.g_nam_p);
4354 (void) strcpy(Full_p, tl_p->L_gen.g_nam_p);
4355 lfrom = Lnknam_p;
4356 lto = Fullnam_p;
4357 }
4358
4359 (void) creat_lnk(G_p->g_passdirfd, lfrom, lto);
4360
4361 if (cwd) {
4362 rest_cwd(cwd);
4363 }
4364
4365 l_p->L_lnk_p = NULL;
4366 free(tl_p->L_gen.g_nam_p);
4367 free(tl_p);
4368
4369 if (l_p->L_cnt == G_p->g_nlink) {
4370 reclaim(l_p);
4371 }
4372
4373 return (FILE_LINKED);
4374 }
4375 }
4376
4377 if (Adir || Aspec) {
4378 /*
4379 * The archive file is a directory, block special, char
4380 * special or a fifo.
4381 */
4382
4383 if (creat_spec(G_p->g_passdirfd) > 0) {
4384 VERBOSE((Args & (OCv | OCV)), Fullnam_p);
4385 }
4386 } else if ((Ofile = openout(G_p->g_passdirfd)) > 0) {
4387 data_pass();
4388 }
4389
4390 return (FILE_COPIED);
4391 }
4392
4393 /*
4394 * flush_lnks: With new linked file handling, linked files are not archived
4395 * until all links have been collected. When the end of the list of filenames
4396 * to archive has been reached, all files that did not encounter all their links
4397 * are written out with actual (encountered) link counts. A file with n links
4398 * (that are archived) will be represented by n headers (one for each link (the
4399 * first n - 1 have g_filesz set to 0)) followed by the data for the file.
4400 */
4401
4402 static void
flush_lnks(void)4403 flush_lnks(void)
4404 {
4405 struct Lnk *l_p, *tl_p;
4406 off_t tfsize;
4407
4408 l_p = Lnk_hd.L_nxt_p;
4409 while (l_p != &Lnk_hd) {
4410 (void) strcpy(Gen.g_nam_p, l_p->L_gen.g_nam_p);
4411 if (stat(Gen.g_nam_p, &SrcSt) == 0) { /* check if file exists */
4412 tl_p = l_p;
4413 (void) creat_hdr();
4414 Gen.g_nlink = l_p->L_cnt; /* "actual" link count */
4415 tfsize = Gen.g_filesz;
4416 Gen.g_filesz = (off_t)0;
4417 G_p = &Gen;
4418 while (tl_p != NULL) {
4419 Gen.g_nam_p = tl_p->L_gen.g_nam_p;
4420 Gen.g_namesz = tl_p->L_gen.g_namesz;
4421 if (tl_p->L_lnk_p == NULL) {
4422 Gen.g_filesz = tfsize;
4423 if (open_dirfd() != 0) {
4424 break;
4425 }
4426 data_out();
4427 break;
4428 }
4429 write_hdr(ARCHIVE_NORMAL,
4430 (off_t)0); /* header only */
4431 VERBOSE((Args & (OCv | OCV)), Gen.g_nam_p);
4432 tl_p = tl_p->L_lnk_p;
4433 }
4434 Gen.g_nam_p = Nam_p;
4435 } else /* stat(Gen.g_nam_p, &SrcSt) == 0 */
4436 msg(ERR, "\"%s%s%s\" has disappeared",
4437 (Gen.g_attrnam_p == NULL) ?
4438 Gen.g_nam_p : Gen.g_attrfnam_p,
4439 (Gen.g_attrnam_p == NULL) ?
4440 "" : Gen.g_rw_sysattr ?
4441 gettext(" System Attribute ") :
4442 gettext(" Attribute "),
4443 (Gen.g_attrnam_p == NULL) ?
4444 "" : Gen.g_attrnam_p);
4445 tl_p = l_p;
4446 l_p = l_p->L_nxt_p;
4447 reclaim(tl_p);
4448 } /* l_p != &Lnk_hd */
4449 }
4450
4451 #if defined(O_XATTR)
4452 static int
is_sysattr(char * name)4453 is_sysattr(char *name)
4454 {
4455 return ((strcmp(name, VIEW_READONLY) == 0) ||
4456 (strcmp(name, VIEW_READWRITE) == 0));
4457 }
4458 #endif
4459
4460 /*
4461 * gethdr: Get a header from the archive, validate it and check for the trailer.
4462 * Any user specified Hdr_type is ignored (set to NONE in main). Hdr_type is
4463 * set appropriately after a valid header is found. Unless the -k option is
4464 * set a corrupted header causes an exit with an error. I/O errors during
4465 * examination of any part of the header cause gethdr to throw away any current
4466 * data and start over. Other errors during examination of any part of the
4467 * header cause gethdr to advance one byte and continue the examination.
4468 */
4469
4470 static int
gethdr(void)4471 gethdr(void)
4472 {
4473 ushort_t ftype;
4474 int hit = NONE, cnt = 0;
4475 int goodhdr, hsize = 0, offset;
4476 int bswap = 0;
4477 char *preptr;
4478 int k = 0;
4479 int j;
4480 int error;
4481 int aclcnt;
4482
4483 Gen.g_nam_p = Nam_p;
4484 do { /* hit == NONE && (Args & OCk) && Buffr.b_cnt > 0 */
4485 FILL(Hdrsz);
4486 switch (Hdr_type) {
4487 case NONE:
4488 case BIN:
4489 Binmag.b_byte[0] = Buffr.b_out_p[0];
4490 Binmag.b_byte[1] = Buffr.b_out_p[1];
4491 if ((Binmag.b_half == CMN_BIN) ||
4492 (Binmag.b_half == CMN_BBS)) {
4493 hit = read_hdr(BIN);
4494 if (Hdr_type == NONE)
4495 bswap = 1;
4496 hsize = HDRSZ + Gen.g_namesz;
4497 break;
4498 }
4499 if (Hdr_type != NONE)
4500 break;
4501 /*FALLTHROUGH*/
4502 case CHR:
4503 if (!(strncmp(Buffr.b_out_p, CMS_CHR, CMS_LEN))) {
4504 hit = read_hdr(CHR);
4505 hsize = CHRSZ + Gen.g_namesz;
4506 break;
4507 }
4508 if (Hdr_type != NONE)
4509 break;
4510 /*FALLTHROUGH*/
4511 case ASC:
4512 if (!(strncmp(Buffr.b_out_p, CMS_ASC, CMS_LEN))) {
4513 hit = read_hdr(ASC);
4514 hsize = ASCSZ + Gen.g_namesz;
4515 Max_namesz = APATH;
4516 break;
4517 }
4518 if (Hdr_type != NONE)
4519 break;
4520 /*FALLTHROUGH*/
4521 case CRC:
4522 if (!(strncmp(Buffr.b_out_p, CMS_CRC, CMS_LEN))) {
4523 hit = read_hdr(CRC);
4524 hsize = ASCSZ + Gen.g_namesz;
4525 Max_namesz = APATH;
4526 break;
4527 }
4528 if (Hdr_type != NONE)
4529 break;
4530 /*FALLTHROUGH*/
4531
4532 case BAR:
4533 if (Hdr_p != NULL && strcmp(Hdr_p, "bar") == 0) {
4534 Hdrsz = BARSZ;
4535 FILL(Hdrsz);
4536 if ((hit = read_hdr(BAR)) == NONE) {
4537 Hdrsz = ASCSZ;
4538 break;
4539 }
4540 hit = BAR;
4541 hsize = BARSZ;
4542 break;
4543 }
4544 /*FALLTHROUGH*/
4545
4546 case USTAR:
4547 if (Hdr_p != NULL && strcmp(Hdr_p, "ustar") == 0) {
4548 Hdrsz = TARSZ;
4549 FILL(Hdrsz);
4550 if ((hit = read_hdr(USTAR)) == NONE) {
4551 Hdrsz = ASCSZ;
4552 break;
4553 }
4554 hit = USTAR;
4555 hsize = TARSZ;
4556 break;
4557 }
4558 /*FALLTHROUGH*/
4559 case TAR:
4560 if (Hdr_p != NULL && strcmp(Hdr_p, "tar") == 0) {
4561 Hdrsz = TARSZ;
4562 FILL(Hdrsz);
4563 if ((hit = read_hdr(TAR)) == NONE) {
4564 Hdrsz = ASCSZ;
4565 break;
4566 }
4567 hit = TAR;
4568 hsize = TARSZ;
4569 break;
4570 }
4571 /*FALLTHROUGH*/
4572 default:
4573 msg(EXT, "Impossible header type.");
4574 } /* Hdr_type */
4575
4576 if (hit == TAR || hit == USTAR) {
4577 Gen.g_nam_p = &nambuf[0];
4578 }
4579
4580 if (hit != NONE) {
4581 FILL(hsize);
4582 goodhdr = 1;
4583 if (Gen.g_filesz < (off_t)0 || Gen.g_namesz < 1)
4584 goodhdr = 0;
4585 if ((hit != USTAR) && (hit != TAR))
4586 if (Gen.g_namesz - 1 > Max_namesz)
4587 goodhdr = 0;
4588 /* TAR and USTAR */
4589 if ((hit == USTAR) || (hit == TAR)) {
4590 if (*Gen.g_nam_p == '\0') { /* tar trailer */
4591 goodhdr = 1;
4592 } else {
4593
4594 G_p = &Gen;
4595 if (G_p->g_cksum !=
4596 cksum(TARTYP, 0, NULL)) {
4597 goodhdr = 0;
4598 msg(ERR,
4599 "Bad header - checksum "
4600 "error.");
4601 }
4602 }
4603 } else if (hit != BAR) { /* binary, -c, ASC and CRC */
4604 if (Gen.g_nlink <= 0)
4605 goodhdr = 0;
4606 if (*(Buffr.b_out_p + hsize - 1) != '\0')
4607 goodhdr = 0;
4608 }
4609 if (!goodhdr) {
4610 hit = NONE;
4611 if (!(Args & OCk))
4612 break;
4613 msg(ERR,
4614 "Corrupt header, file(s) may be lost.");
4615 } else {
4616 FILL(hsize);
4617 }
4618 } /* hit != NONE */
4619 if (hit == NONE) {
4620 Buffr.b_out_p++;
4621 Buffr.b_cnt--;
4622 if (!(Args & OCk))
4623 break;
4624 if (!cnt++)
4625 msg(ERR, "Searching for magic number/header.");
4626 }
4627 } while (hit == NONE);
4628 if (hit == NONE) {
4629 if (Hdr_type == NONE)
4630 msg(EXT, "Not a cpio file, bad header.");
4631 else
4632 msg(EXT, "Bad magic number/header.");
4633 } else if (cnt > 0) {
4634 msg(EPOST, "Re-synchronized on magic number/header.");
4635 }
4636 if (Hdr_type == NONE) {
4637 Hdr_type = hit;
4638 switch (Hdr_type) {
4639 case BIN:
4640 if (bswap)
4641 Args |= BSM;
4642 Hdrsz = HDRSZ;
4643 Max_namesz = CPATH;
4644 Pad_val = HALFWD;
4645 Onecopy = 0;
4646 break;
4647 case CHR:
4648 Hdrsz = CHRSZ;
4649 Max_namesz = CPATH;
4650 Pad_val = 0;
4651 Onecopy = 0;
4652 break;
4653 case ASC:
4654 case CRC:
4655 Hdrsz = ASCSZ;
4656 Max_namesz = APATH;
4657 Pad_val = FULLWD;
4658 Onecopy = 1;
4659 break;
4660 case USTAR:
4661 Hdrsz = TARSZ;
4662 Max_namesz = HNAMLEN - 1;
4663 Pad_val = FULLBK;
4664 Onecopy = 0;
4665 break;
4666 case BAR:
4667 case TAR:
4668 Hdrsz = TARSZ;
4669 Max_namesz = TNAMLEN - 1;
4670 Pad_val = FULLBK;
4671 Onecopy = 0;
4672 break;
4673 default:
4674 msg(EXT, "Impossible header type.");
4675 } /* Hdr_type */
4676 } /* Hdr_type == NONE */
4677 if ((Hdr_type == USTAR) || (Hdr_type == TAR) ||
4678 (Hdr_type == BAR)) { /* TAR, USTAR, BAR */
4679 Gen.g_namesz = 0;
4680 if (Gen.g_nam_p[0] == '\0')
4681 return (0);
4682 else {
4683 preptr = &prebuf[0];
4684 if (*preptr != '\0') {
4685 k = strlen(&prebuf[0]);
4686 if (k < PRESIZ) {
4687 (void) strcpy(&fullnam[0], &prebuf[0]);
4688 j = 0;
4689 fullnam[k++] = '/';
4690 while ((j < NAMSIZ) && (nambuf[j] !=
4691 '\0')) {
4692 fullnam[k] = nambuf[j];
4693 k++; j++;
4694 }
4695 fullnam[k] = '\0';
4696 } else if (k >= PRESIZ) {
4697 k = 0;
4698 while ((k < PRESIZ) && (prebuf[k] !=
4699 '\0')) {
4700 fullnam[k] = prebuf[k];
4701 k++;
4702 }
4703 fullnam[k++] = '/';
4704 j = 0;
4705 while ((j < NAMSIZ) && (nambuf[j] !=
4706 '\0')) {
4707 fullnam[k] = nambuf[j];
4708 k++; j++;
4709 }
4710 fullnam[k] = '\0';
4711 }
4712 Gen.g_nam_p = &fullnam[0];
4713 } else
4714 Gen.g_nam_p = &nambuf[0];
4715
4716 /*
4717 * initialize the buffer so that the prefix will not
4718 * applied to the next entry in the archive
4719 */
4720 (void) memset(prebuf, 0, sizeof (prebuf));
4721 }
4722 } else if (Hdr_type != BAR) {
4723 (void) memcpy(Gen.g_nam_p, Buffr.b_out_p + Hdrsz, Gen.g_namesz);
4724 if (!(strcmp(Gen.g_nam_p, "TRAILER!!!")))
4725 return (0);
4726 }
4727 offset = ((hsize + Pad_val) & ~Pad_val);
4728 FILL(offset + Hdrsz);
4729 Thdr_p = (union tblock *)Buffr.b_out_p;
4730 Buffr.b_out_p += offset;
4731 Buffr.b_cnt -= (off_t)offset;
4732 ftype = Gen.g_mode & Ftype;
4733
4734 #if defined(O_XATTR)
4735 /* extended attribute support */
4736 if (((Gen.g_mode & S_IFMT) == _XATTR_CPIO_MODE) ||
4737 ((Hdr_type == USTAR || Hdr_type == TAR) &&
4738 Thdr_p->tbuf.t_typeflag == _XATTR_HDRTYPE)) {
4739 char *aname;
4740 char *attrparent = NULL;
4741 char *attrpath = NULL;
4742 char *tapath;
4743 char *taname;
4744
4745 if (xattrp != NULL) {
4746 if (xattrbadhead) {
4747 free(xattrhead);
4748 xattrp = NULL;
4749 xattr_linkp = NULL;
4750 xattrhead = NULL;
4751 return (1);
4752 }
4753
4754 /*
4755 * At this point, the attribute path contains
4756 * the path to the attribute rooted at the hidden
4757 * attribute directory of the base file. This can
4758 * be a simple attribute or extended attribute name,
4759 * or it can be something like <attr>/<sys attr> if
4760 * we are processing a system attribute of an attribute.
4761 * Determine the attribute name and attribute parent
4762 * (if there is one). When we are processing a simple
4763 * attribute or extended attribute name, the attribute
4764 * parent will be set to NULL. When we are processing
4765 * something like <attr>/<sys attr>, the attribute
4766 * parent will be contain <attr>, and the attribute
4767 * name will contain <sys attr>.
4768 */
4769 tapath = xattrp->h_names +
4770 strlen(xattrp->h_names) + 1;
4771 attrpath = e_strdup(E_EXIT, tapath);
4772 if ((taname = strpbrk(tapath, "/")) != NULL) {
4773 aname = taname + 1;
4774 *taname = '\0';
4775 attrparent = tapath;
4776 } else {
4777 aname = tapath;
4778 }
4779
4780 Gen.g_rw_sysattr = is_sysattr(aname);
4781 Gen.g_baseparent_fd = attr_baseparent_fd;
4782
4783 if (Gen.g_attrfnam_p != NULL) {
4784 free(Gen.g_attrfnam_p);
4785 Gen.g_attrfnam_p = NULL;
4786 }
4787 if (Gen.g_attrnam_p != NULL) {
4788 free(Gen.g_attrnam_p);
4789 Gen.g_attrnam_p = NULL;
4790 }
4791 if (Gen.g_attrparent_p != NULL) {
4792 free(Gen.g_attrparent_p);
4793 Gen.g_attrparent_p = NULL;
4794 }
4795 if (Gen.g_attrpath_p != NULL) {
4796 free(Gen.g_attrpath_p);
4797 Gen.g_attrpath_p = NULL;
4798 }
4799 if (Renam_p && Renam_p[0] != '\0') {
4800 Gen.g_attrfnam_p = e_strdup(E_EXIT, Renam_p);
4801 } else {
4802 Gen.g_attrfnam_p = e_strdup(E_EXIT,
4803 xattrp->h_names);
4804 }
4805 Gen.g_attrnam_p = e_strdup(E_EXIT, aname);
4806
4807 if (attrparent != NULL) {
4808 if (Renam_attr_p && Renam_attr_p[0] != '\0') {
4809 size_t apathlen = strlen(attrparent) +
4810 strlen(aname) + 2;
4811 Gen.g_attrparent_p = e_strdup(E_EXIT,
4812 Renam_attr_p);
4813 Gen.g_attrpath_p = e_zalloc(E_EXIT,
4814 apathlen);
4815 (void) snprintf(Gen.g_attrpath_p,
4816 apathlen, "%s/%s", Renam_attr_p,
4817 aname);
4818 (void) free(attrparent);
4819 (void) free(attrpath);
4820 } else {
4821 Gen.g_attrparent_p = attrparent;
4822 Gen.g_attrpath_p = attrpath;
4823 }
4824 } else {
4825 Gen.g_attrpath_p = attrpath;
4826 }
4827
4828 if (xattr_linkp != NULL) {
4829 if (Gen.g_linktoattrfnam_p != NULL) {
4830 free(Gen.g_linktoattrfnam_p);
4831 Gen.g_linktoattrfnam_p = NULL;
4832 }
4833 if (Gen.g_linktoattrnam_p != NULL) {
4834 free(Gen.g_linktoattrnam_p);
4835 Gen.g_linktoattrnam_p = NULL;
4836 }
4837 if (Renam_attr_p && Renam_attr_p[0] != '\0') {
4838 Gen.g_linktoattrfnam_p = e_strdup(
4839 E_EXIT, Renam_attr_p);
4840 } else {
4841 Gen.g_linktoattrfnam_p = e_strdup(
4842 E_EXIT, xattr_linkp->h_names);
4843 }
4844 Gen.g_linktoattrnam_p = e_strdup(E_EXIT,
4845 aname);
4846 xattr_linkp = NULL;
4847 }
4848 if (Hdr_type != USTAR && Hdr_type != TAR) {
4849 Gen.g_mode = Gen.g_mode & (~_XATTR_CPIO_MODE);
4850 Gen.g_mode |= attrmode(xattrp->h_typeflag);
4851 } else if (Hdr_type == USTAR || Hdr_type == TAR) {
4852 Thdr_p->tbuf.t_typeflag = xattrp->h_typeflag;
4853 }
4854
4855 ftype = Gen.g_mode & Ftype;
4856 Adir = ftype == S_IFDIR;
4857 Aspec = (ftype == S_IFBLK || ftype == S_IFCHR ||
4858 ftype == S_IFIFO || ftype == S_IFSOCK);
4859
4860 if (Gen.g_attrnam_p[0] == '.' &&
4861 Gen.g_attrnam_p[1] == '\0' &&
4862 xattrp->h_typeflag == DIRTYPE) {
4863 Hiddendir = 1;
4864 } else {
4865 Hiddendir = 0;
4866 }
4867
4868 free(xattrhead);
4869 xattrhead = NULL;
4870 xattrp = NULL;
4871 } else {
4872 if (xattrbadhead == 0) {
4873 (void) read_xattr_hdr();
4874 return (2);
4875 }
4876 }
4877 } else {
4878 Hiddendir = 0;
4879 }
4880 #endif /* O_XATTR */
4881
4882 /* acl support: grab acl info */
4883 if ((Gen.g_mode == SECMODE) || ((Hdr_type == USTAR ||
4884 Hdr_type == TAR) && Thdr_p->tbuf.t_typeflag == 'A')) {
4885 /* this is an ancillary file */
4886 off_t bytes;
4887 char *secp;
4888 int pad;
4889 int cnt;
4890 char *tp;
4891 int attrsize = 0;
4892
4893 if (Pflag) {
4894 bytes = Gen.g_filesz;
4895 secp = e_zalloc(E_EXIT, (uint_t)bytes);
4896 tp = secp;
4897
4898 while (bytes > 0) {
4899 cnt = (int)(bytes > CPIOBSZ) ? CPIOBSZ : bytes;
4900 FILL(cnt);
4901 (void) memcpy(tp, Buffr.b_out_p, cnt);
4902 tp += cnt;
4903 Buffr.b_out_p += cnt;
4904 Buffr.b_cnt -= (off_t)cnt;
4905 bytes -= (off_t)cnt;
4906 }
4907
4908 pad = (Pad_val + 1 - (Gen.g_filesz & Pad_val)) &
4909 Pad_val;
4910 if (pad != 0) {
4911 FILL(pad);
4912 Buffr.b_out_p += pad;
4913 Buffr.b_cnt -= (off_t)pad;
4914 }
4915
4916 /* got all attributes in secp */
4917 tp = secp;
4918 do {
4919 attr = (struct sec_attr *)tp;
4920 switch (attr->attr_type) {
4921 case UFSD_ACL:
4922 case ACE_ACL:
4923 (void) sscanf(attr->attr_len, "%7o",
4924 (uint_t *)&aclcnt);
4925 /* header is 8 */
4926 attrsize = 8 +
4927 strlen(&attr->attr_info[0])
4928 + 1;
4929
4930 error =
4931 acl_fromtext(&attr->attr_info[0],
4932 &aclp);
4933
4934 if (error != 0) {
4935 msg(ERR,
4936 "aclfromtext failed: %s",
4937 acl_strerror(error));
4938 bytes -= attrsize;
4939 break;
4940 }
4941
4942 if (aclcnt != acl_cnt(aclp)) {
4943 msg(ERR, "acl count error");
4944 bytes -= attrsize;
4945 break;
4946 }
4947 bytes -= attrsize;
4948 break;
4949
4950 /* SunFed case goes here */
4951
4952 default:
4953 msg(EXT, "unrecognized attr type");
4954 break;
4955 }
4956 /* next attributes */
4957 tp += attrsize;
4958 } while (bytes > 0);
4959 free(secp);
4960 } else {
4961 /* skip security info */
4962 G_p = &Gen;
4963 data_in(P_SKIP);
4964 }
4965 /*
4966 * We already got the file content, dont call file_in()
4967 * when return. The new return code(2) is used to
4968 * indicate that.
4969 */
4970 VERBOSE((Args & OCt), Gen.g_nam_p);
4971 return (2);
4972 } /* acl */
4973
4974 /*
4975 * Sparse file support
4976 * Read header of holesdata to get original file size.
4977 * This is necessary because ckname() or file_in() shows file size
4978 * with OCt before data_in() extracts the holesdata. data_in()
4979 * actually doesn't extract the holesdata since proc_mode will be
4980 * P_SKIP in the OCt mode.
4981 */
4982 if ((Hdr_type == CHR || Hdr_type == ASC) &&
4983 S_ISSPARSE(Gen.g_mode) && Gen.g_filesz > MIN_HOLES_HDRSIZE) {
4984 char holesdata[MIN_HOLES_HDRSIZE + 1];
4985
4986 FILL(MIN_HOLES_HDRSIZE);
4987 (void) memcpy(holesdata, Buffr.b_out_p, MIN_HOLES_HDRSIZE);
4988 holesdata[MIN_HOLES_HDRSIZE] = '\0';
4989
4990 Gen.g_holes = read_holes_header(holesdata, Gen.g_filesz);
4991 if (Gen.g_holes == NULL) {
4992 msg(EXT, "invalid sparse file information");
4993 } else {
4994 Buffr.b_out_p += MIN_HOLES_HDRSIZE;
4995 Buffr.b_cnt -= MIN_HOLES_HDRSIZE;
4996 }
4997 }
4998
4999 Adir = (ftype == S_IFDIR);
5000 Aspec = (ftype == S_IFBLK || ftype == S_IFCHR || ftype == S_IFIFO ||
5001 ftype == S_IFSOCK);
5002
5003 /*
5004 * Skip any trailing slashes
5005 */
5006 chop_endslashes(Gen.g_nam_p);
5007 return (1);
5008 }
5009
5010 /*
5011 * getname: Get file names for inclusion in the archive. When end of file
5012 * on the input stream of file names is reached, flush the link buffer out.
5013 * For each filename, remove leading "./"s and multiple "/"s, and remove
5014 * any trailing newline "\n". Finally, verify the existence of the file,
5015 * and call creat_hdr() to fill in the gen_hdr structure.
5016 */
5017
5018 static int
getname(void)5019 getname(void)
5020 {
5021 int goodfile = 0, lastchar, err;
5022 char *s;
5023 char *dir;
5024
5025 Gen.g_nam_p = Nam_p;
5026 Hiddendir = 0;
5027
5028 while (!goodfile) {
5029 err = 0;
5030
5031 while ((s = fgets(Gen.g_nam_p, APATH+1, In_p)) != NULL) {
5032 lastchar = strlen(s) - 1;
5033 issymlink = 0;
5034
5035 if (s[lastchar] != '\n') {
5036 if (lastchar == APATH - 1) {
5037 if (!err) {
5038 msg(ERR,
5039 "%s name too long.",
5040 Nam_p);
5041 }
5042 goodfile = 0;
5043 err = 1;
5044 } else {
5045 break;
5046 }
5047 } else {
5048 s[lastchar] = '\0';
5049 break;
5050 }
5051 }
5052
5053 if (s == NULL) {
5054 if (Gen.g_dirfd != -1) {
5055 (void) close(Gen.g_dirfd);
5056 Gen.g_dirfd = -1;
5057 }
5058 if (Onecopy && (Args & OCo)) {
5059 flush_lnks();
5060 }
5061 return (0);
5062 }
5063
5064 while (*Gen.g_nam_p == '.' && Gen.g_nam_p[1] == '/') {
5065 Gen.g_nam_p += 2;
5066 while (*Gen.g_nam_p == '/')
5067 Gen.g_nam_p++;
5068 }
5069
5070 /*
5071 * Skip any trailing slashes
5072 */
5073 chop_endslashes(Gen.g_nam_p);
5074
5075 /*
5076 * Figure out parent directory
5077 */
5078
5079 if (Gen.g_attrnam_p != NULL) {
5080 if (Gen.g_dirfd != -1) {
5081 (void) close(Gen.g_dirfd);
5082 }
5083 Gen.g_dirfd = attropen(Gen.g_attrfnam_p, ".", O_RDONLY);
5084 if (Gen.g_dirfd == -1) {
5085 msg(ERRN,
5086 "Cannot open attribute directory"
5087 " of file %s", Gen.g_attrfnam_p);
5088 continue;
5089 }
5090 } else {
5091 #ifdef O_XATTR
5092 char dirpath[PATH_MAX];
5093
5094 get_parent(Gen.g_nam_p, dirpath);
5095 if (Atflag || SysAtflag) {
5096 dir = dirpath;
5097 if (Gen.g_dirfd != -1) {
5098 (void) close(Gen.g_dirfd);
5099 }
5100 Gen.g_dirfd = open(dir, O_RDONLY);
5101 if (Gen.g_dirfd == -1) {
5102 msg(ERRN,
5103 "Cannot open directory %s", dir);
5104 continue;
5105 }
5106 } else {
5107 /*
5108 * g_dirpath is the pathname cache maintaining
5109 * the dirname which is currently opened.
5110 * We first check the g_dirpath to see if the
5111 * given dirname matches. If so, we don't need
5112 * to open the dir, but we can use the g_dirfd
5113 * as is if it is still available.
5114 */
5115 dir = NULL;
5116 if (Gen.g_dirpath == NULL ||
5117 Gen.g_dirfd == -1) {
5118 /*
5119 * It's the first time or it has
5120 * all gone.
5121 */
5122 dir = e_strdup(E_EXIT, dirpath);
5123 } else {
5124 if (strcmp(Gen.g_dirpath,
5125 dirpath) != 0) {
5126 /* different directory */
5127 dir = e_strdup(E_EXIT, dirpath);
5128 }
5129 }
5130 if (dir != NULL) {
5131 /*
5132 * We need to open the new directory.
5133 * discard the pathname and dirfd
5134 * for the previous directory.
5135 */
5136 if (Gen.g_dirpath != NULL) {
5137 free(Gen.g_dirpath);
5138 Gen.g_dirpath = NULL;
5139 }
5140 if (Gen.g_dirfd != -1) {
5141 (void) close(Gen.g_dirfd);
5142 }
5143 /* open the new dir */
5144 Gen.g_dirfd = open(dir, O_RDONLY);
5145 if (Gen.g_dirfd == -1) {
5146 msg(ERRN, "Cannot open "
5147 "directory %s", dir);
5148 continue;
5149 }
5150 Gen.g_dirpath = dir;
5151 }
5152 }
5153 #else
5154 Gen.g_dirfd = -1;
5155 #endif
5156 }
5157
5158 /* creat_hdr checks for USTAR filename length */
5159
5160 if (Hdr_type != USTAR && strlen(Gen.g_nam_p) >
5161 Max_namesz) {
5162 if (!err) {
5163 msg(ERR, "%s%s%s name too long.",
5164 (Gen.g_attrnam_p == NULL) ?
5165 Nam_p : Gen.g_attrfnam_p,
5166 (Gen.g_attrnam_p == NULL) ?
5167 "" : Gen.g_rw_sysattr ?
5168 gettext(" System Attribute ") :
5169 gettext(" Attribute "),
5170 (Gen.g_attrnam_p == NULL) ?
5171 "" : Gen.g_attrnam_p);
5172 }
5173 goodfile = 0;
5174 err = 1;
5175 }
5176
5177 if (err) {
5178 continue;
5179 } else {
5180 G_p = &Gen;
5181 if (!LSTAT(Gen.g_dirfd, Gen.g_nam_p, &SrcSt)) {
5182 goodfile = 1;
5183
5184 if ((SrcSt.st_mode & Ftype) == S_IFLNK) {
5185 issymlink = 1;
5186
5187 if ((Args & OCL)) {
5188 errno = 0;
5189 if (STAT(Gen.g_dirfd,
5190 G_p->g_nam_p,
5191 &SrcSt) < 0) {
5192 msg(ERRN,
5193 "Cannot follow"
5194 " \"%s%s%s\"",
5195 (Gen.g_attrnam_p ==
5196 NULL) ?
5197 Gen.g_nam_p :
5198 Gen.g_attrfnam_p,
5199 (Gen.g_attrnam_p ==
5200 NULL) ? "" :
5201 Gen.g_rw_sysattr ?
5202 gettext(
5203 " System "
5204 "Attribute ") :
5205 gettext(
5206 " Attribute "),
5207 (Gen.g_attrnam_p ==
5208 NULL) ? "" :
5209 Gen.g_attrnam_p);
5210 goodfile = 0;
5211 }
5212 }
5213 }
5214
5215 if (Use_old_stat) {
5216 OldSt = convert_to_old_stat(&SrcSt,
5217 Gen.g_nam_p, Gen.g_attrnam_p);
5218
5219 if (OldSt == NULL) {
5220 goodfile = 0;
5221 }
5222 }
5223 } else {
5224 msg(ERRN,
5225 "Error with fstatat() of \"%s%s%s\"",
5226 (Gen.g_attrnam_p == NULL) ?
5227 Gen.g_nam_p : Gen.g_attrfnam_p,
5228 (Gen.g_attrnam_p == NULL) ? "" :
5229 Gen.g_rw_sysattr ?
5230 gettext(" System Attribute ") :
5231 gettext(" Attribute "),
5232 (Gen.g_attrnam_p == NULL) ?
5233 "" : Gen.g_attrnam_p);
5234 }
5235 }
5236 }
5237
5238 /*
5239 * Get ACL info: dont bother allocating space if there are only
5240 * standard permissions, i.e. ACL count < 4
5241 */
5242 if ((SrcSt.st_mode & Ftype) != S_IFLNK && Pflag) {
5243 if (acl_get(Gen.g_nam_p, ACL_NO_TRIVIAL, &aclp) != 0)
5244 msg(ERRN, "Error with acl() of \"%s\"", Gen.g_nam_p);
5245 }
5246 /* else: only traditional permissions, so proceed as usual */
5247 if (creat_hdr())
5248 return (1);
5249 else return (2);
5250 }
5251
5252 /*
5253 * getpats: Save any filenames/patterns specified as arguments.
5254 * Read additional filenames/patterns from the file specified by the
5255 * user. The filenames/patterns must occur one per line.
5256 */
5257
5258 static void
getpats(int largc,char ** largv)5259 getpats(int largc, char **largv)
5260 {
5261 char **t_pp;
5262 size_t len;
5263 unsigned numpat = largc, maxpat = largc + 2;
5264
5265 Pat_pp = e_zalloc(E_EXIT, maxpat * sizeof (char *));
5266 t_pp = Pat_pp;
5267 while (*largv) {
5268 *t_pp = e_zalloc(E_EXIT, strlen(*largv) + 1);
5269 (void) strcpy(*t_pp, *largv);
5270 t_pp++;
5271 largv++;
5272 }
5273 while (fgets(Nam_p, Max_namesz + 1, Ef_p) != NULL) {
5274 if (numpat == maxpat - 1) {
5275 maxpat += 10;
5276 Pat_pp = e_realloc(E_EXIT, Pat_pp,
5277 maxpat * sizeof (char *));
5278 t_pp = Pat_pp + numpat;
5279 }
5280 len = strlen(Nam_p); /* includes the \n */
5281 *(Nam_p + len - 1) = '\0'; /* remove the \n */
5282 *t_pp = e_zalloc(E_EXIT, len);
5283 (void) strcpy(*t_pp, Nam_p);
5284 t_pp++;
5285 numpat++;
5286 }
5287 *t_pp = NULL;
5288 }
5289
5290 static void
ioerror(int dir)5291 ioerror(int dir)
5292 {
5293 int t_errno;
5294
5295 t_errno = errno;
5296 errno = 0;
5297 if (fstat(Archive, &ArchSt) < 0)
5298 msg(EXTN, "Error during stat() of archive");
5299 errno = t_errno;
5300 if ((ArchSt.st_mode & Ftype) != S_IFCHR) {
5301 if (dir) {
5302 if (errno == EFBIG)
5303 msg(EXT, "ulimit reached for output file.");
5304 else if (errno == ENOSPC)
5305 msg(EXT, "No space left for output file.");
5306 else
5307 msg(EXTN, "I/O error - cannot continue");
5308 } else
5309 msg(EXT, "Unexpected end-of-file encountered.");
5310 } else
5311 msg(EXTN, "\007I/O error on \"%s\"", dir ? "output" : "input");
5312 }
5313
5314 /*
5315 * matched: Determine if a filename matches the specified pattern(s). If the
5316 * pattern is matched (the second return), return 0 if -f was specified, else
5317 * return != 0. If the pattern is not matched (the first and third
5318 * returns), return 0 if -f was not specified, else return != 0.
5319 */
5320
5321 static int
matched(void)5322 matched(void)
5323 {
5324 char *str_p = G_p->g_nam_p;
5325 char **pat_pp = Pat_pp;
5326 int negatep, result;
5327
5328 /*
5329 * Check for attribute
5330 */
5331 if (G_p->g_attrfnam_p != NULL)
5332 str_p = G_p->g_attrfnam_p;
5333
5334 for (pat_pp = Pat_pp; *pat_pp; pat_pp++) {
5335 negatep = (**pat_pp == '!');
5336
5337 result = fnmatch(negatep ? (*pat_pp+1) : *pat_pp, str_p, 0);
5338
5339 if (result != 0 && result != FNM_NOMATCH) {
5340 msg(POST, "error matching file %s with pattern"
5341 " %s\n", str_p, *pat_pp);
5342 return (Args & OCf);
5343 }
5344
5345 if ((result == 0 && ! negatep) ||
5346 (result == FNM_NOMATCH && negatep)) {
5347 /* match occurred */
5348 return (!(Args & OCf));
5349 }
5350 }
5351 return (Args & OCf); /* not matched */
5352 }
5353
5354 /*
5355 * missdir: Create missing directories for files.
5356 * (Possible future performance enhancement, if missdir is called, we know
5357 * that at least the very last directory of the path does not exist, therefore,
5358 * scan the path from the end
5359 */
5360
5361 static int
missdir(char * nam_p)5362 missdir(char *nam_p)
5363 {
5364 char *c_p;
5365 int cnt = 2;
5366 char *lastp;
5367
5368 if (*(c_p = nam_p) == '/') /* skip over 'root slash' */
5369 c_p++;
5370
5371 lastp = c_p + strlen(nam_p) - 1;
5372 if (*lastp == '/')
5373 *lastp = '\0';
5374
5375 for (; *c_p; ++c_p) {
5376 if (*c_p == '/') {
5377 *c_p = '\0';
5378 if (stat(nam_p, &DesSt) < 0) {
5379 if (Args & OCd) {
5380 cnt = mkdir(nam_p, Def_mode);
5381 if (cnt != 0) {
5382 *c_p = '/';
5383 return (cnt);
5384 }
5385 } else {
5386 msg(ERR, "Missing -d option.");
5387 *c_p = '/';
5388 return (-1);
5389 }
5390 }
5391 *c_p = '/';
5392 }
5393 }
5394 if (cnt == 2) /* the file already exists */
5395 cnt = 0;
5396 return (cnt);
5397 }
5398
5399 /*
5400 * mklong: Convert two shorts into one long. For VAX, Interdata ...
5401 */
5402
5403 static long
mklong(short v[])5404 mklong(short v[])
5405 {
5406
5407 union swpbuf swp_b;
5408
5409 swp_b.s_word = 1;
5410 if (swp_b.s_byte[0]) {
5411 swp_b.s_half[0] = v[1];
5412 swp_b.s_half[1] = v[0];
5413 } else {
5414 swp_b.s_half[0] = v[0];
5415 swp_b.s_half[1] = v[1];
5416 }
5417 return (swp_b.s_word);
5418 }
5419
5420 /*
5421 * mkshort: Convert a long into 2 shorts, for VAX, Interdata ...
5422 */
5423
5424 static void
mkshort(short sval[],long v)5425 mkshort(short sval[], long v)
5426 {
5427 union swpbuf *swp_p, swp_b;
5428
5429 swp_p = (union swpbuf *)sval;
5430 swp_b.s_word = 1;
5431 if (swp_b.s_byte[0]) {
5432 swp_b.s_word = v;
5433 swp_p->s_half[0] = swp_b.s_half[1];
5434 swp_p->s_half[1] = swp_b.s_half[0];
5435 } else {
5436 swp_b.s_word = v;
5437 swp_p->s_half[0] = swp_b.s_half[0];
5438 swp_p->s_half[1] = swp_b.s_half[1];
5439 }
5440 }
5441
5442 /*
5443 * msg: Print either a message (no error) (POST), an error message with or
5444 * without the errno (ERRN or ERR), or print an error message with or without
5445 * the errno and exit (EXTN or EXT).
5446 */
5447 void
msg(int severity,const char * fmt,...)5448 msg(int severity, const char *fmt, ...)
5449 {
5450 FILE *file_p;
5451 va_list ap;
5452
5453 if ((Args & OCV) && Verbcnt) { /* clear current line of dots */
5454 (void) fputc('\n', Out_p);
5455 Verbcnt = 0;
5456 }
5457 va_start(ap, fmt);
5458 if (severity == POST)
5459 file_p = Out_p;
5460 else
5461 if (severity == EPOST)
5462 file_p = Err_p;
5463 else {
5464 file_p = Err_p;
5465 Error_cnt++;
5466 }
5467 (void) fflush(Out_p);
5468 (void) fflush(Err_p);
5469 if ((severity != POST) && (severity != EPOST))
5470 (void) fprintf(file_p, "cpio: ");
5471
5472 /* gettext replaces version of string */
5473
5474 (void) vfprintf(file_p, gettext(fmt), ap);
5475 if (severity == ERRN || severity == EXTN) {
5476 if (G_p && (G_p->g_attrnam_p != NULL) && G_p->g_rw_sysattr) {
5477 if (errno == EPERM) {
5478 (void) fprintf(file_p, ", errno %d, %s", errno,
5479 gettext("insufficient privileges\n"));
5480 } else if (errno == EINVAL) {
5481 (void) fprintf(file_p, ", errno %d, %s",
5482 errno, gettext(
5483 "unsupported on underlying file system\n"));
5484 } else {
5485 (void) fprintf(file_p, ", errno %d, ", errno);
5486 perror("");
5487 }
5488 } else {
5489 (void) fprintf(file_p, ", errno %d, ", errno);
5490 perror("");
5491 }
5492 } else
5493 (void) fprintf(file_p, "\n");
5494 (void) fflush(file_p);
5495 va_end(ap);
5496 if (severity == EXT || severity == EXTN) {
5497 (void) fprintf(file_p, gettext("%d errors\n"), Error_cnt);
5498 exit(EXIT_CODE);
5499 }
5500 }
5501
5502 /*
5503 * openout: Open files for output and set all necessary information.
5504 * If the u option is set (unconditionally overwrite existing files),
5505 * and the current file exists, get a temporary file name from mktemp(3C),
5506 * link the temporary file to the existing file, and remove the existing file.
5507 * Finally either creat(2), mkdir(2) or mknod(2) as appropriate.
5508 *
5509 */
5510
5511 static int
openout(int dirfd)5512 openout(int dirfd)
5513 {
5514 char *nam_p;
5515 int cnt, result;
5516
5517 Do_rename = 0; /* creat_tmp() may reset this */
5518
5519 if (G_p->g_attrnam_p != NULL) {
5520 nam_p = G_p->g_attrnam_p;
5521 } else {
5522 if (Args & OCp) {
5523 nam_p = Fullnam_p;
5524 } else {
5525 nam_p = G_p->g_nam_p;
5526 }
5527 }
5528
5529
5530 if ((Max_filesz != RLIM_INFINITY) &&
5531 (Max_filesz < (G_p->g_filesz >> 9))) {
5532 /* ... divided by 512 ... */
5533 msg(ERR, "Skipping \"%s%s%s\": exceeds ulimit by %lld bytes",
5534 (G_p->g_attrnam_p == NULL) ? nam_p : G_p->g_attrfnam_p,
5535 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_rw_sysattr ?
5536 gettext(" System Attribute ") : gettext(" Attribute "),
5537 (G_p->g_attrnam_p == NULL) ? "" : nam_p,
5538 (off_t)(G_p->g_filesz - (Max_filesz << 9)));
5539 return (-1);
5540 }
5541
5542 if (LSTAT(dirfd, nam_p, &DesSt) == 0) {
5543 /*
5544 * A file by the same name exists. Move it to a temporary
5545 * file unless it's a system attribute file. If we are
5546 * restoring a system attribute file on a file system that
5547 * supports system attributes, then the system attribute file
5548 * will already exist (a default system attribute file will
5549 * get created when the file it is associated with is created).
5550 * If we create a temporary system attribute file, we can't
5551 * overwrite the existing system attribute file using
5552 * renameat(). In addition, only system attributes can exist
5553 * for an attribute of a file, therefore, a temporary file
5554 * cannot be created for a system attribute of an attribute.
5555 * Thus, when restoring a system attribute, we won't move it
5556 * to a temporary file, but will attempt to process it as if
5557 * it didn't already exist.
5558 */
5559
5560 #if defined(_PC_SATTR_ENABLED)
5561 if (G_p->g_rw_sysattr == 0)
5562 #endif /* _PC_SATTR_ENABLED */
5563 if (creat_tmp(nam_p) < 0) {
5564 /*
5565 * We weren't able to create the temp file.
5566 * Report failure.
5567 */
5568
5569 return (-1);
5570 }
5571 }
5572
5573 if (Do_rename) {
5574 /* nam_p was changed by creat_tmp() above. */
5575
5576 if (Args & OCp) {
5577 if (G_p->g_attrnam_p != NULL) {
5578 nam_p = Attrfile_p;
5579 } else {
5580 nam_p = Fullnam_p;
5581 }
5582 } else {
5583 nam_p = G_p->g_nam_p;
5584 }
5585 }
5586
5587 /*
5588 * This pile tries to create the file directly, and, if there is a
5589 * problem, creates missing directories, and then tries to create the
5590 * file again. Two strikes and you're out.
5591 *
5592 * On XATTR system, the directory has already been created by
5593 * open_dirfd(), so error shouldn't happen in the loop. However,
5594 * on non-XATTR system, symlink/open may fail with ENOENT. In such
5595 * case, we go to create missing directories.
5596 */
5597
5598 cnt = 0;
5599
5600 do {
5601 errno = 0;
5602
5603 if (Hdr_type == TAR && Thdr_p->tbuf.t_typeflag == SYMTYPE) {
5604 /* The archive file is a TAR symlink. */
5605 if ((result =
5606 symlink(Thdr_p->tbuf.t_linkname, nam_p)) >= 0) {
5607 cnt = 0;
5608 if (Over_p != NULL) {
5609 (void) unlinkat(dirfd,
5610 get_component(Over_p), 0);
5611 *Over_p = '\0';
5612 }
5613 break;
5614 } else if (errno != ENOENT) {
5615 /* The attempt to symlink failed. */
5616 msg(ERRN,
5617 "Cannot create symbolic link \"%s\" -> "
5618 "\"%s\"",
5619 Thdr_p->tbuf.t_linkname, nam_p);
5620
5621 if (*Over_p != '\0') {
5622 rstfiles(U_KEEP, dirfd);
5623 }
5624 return (-1);
5625 }
5626 } else if (Hdr_type == BAR && bar_linkflag == SYMTYPE) {
5627 if ((result = symlink(bar_linkname, nam_p)) >= 0) {
5628 cnt = 0;
5629 if (Over_p != NULL) {
5630 (void) unlinkat(dirfd,
5631 get_component(Over_p), 0);
5632 *Over_p = '\0';
5633 }
5634 break;
5635 } else if (errno != ENOENT) {
5636 /* The attempt to symlink failed. */
5637 msg(ERRN,
5638 "Cannot create symbolic link \"%s\" -> "
5639 "\"%s\"",
5640 bar_linkname, nam_p);
5641 if (*Over_p != '\0') {
5642 rstfiles(U_KEEP, dirfd);
5643 }
5644 return (-1);
5645 }
5646 } else if ((G_p->g_mode & Ftype) == S_IFLNK) {
5647 if ((!(Args & OCp)) && !(Hdr_type == USTAR)) {
5648 FILL(G_p->g_filesz);
5649 (void) strncpy(Symlnk_p,
5650 Buffr.b_out_p, G_p->g_filesz);
5651 *(Symlnk_p + G_p->g_filesz) = '\0';
5652 } else if ((!(Args & OCp)) && (Hdr_type == USTAR)) {
5653 Symlnk_p[NAMSIZ] = '\0';
5654 (void) strncpy(Symlnk_p,
5655 &Thdr_p->tbuf.t_linkname[0], NAMSIZ);
5656 }
5657 if ((result = symlink(Symlnk_p, nam_p)) >= 0) {
5658 cnt = 0;
5659 if (Over_p != NULL) {
5660 (void) unlinkat(dirfd,
5661 get_component(Over_p), 0);
5662 *Over_p = '\0';
5663 }
5664 break;
5665 } else if (errno != ENOENT) {
5666 /* The attempt to symlink failed. */
5667 msg(ERRN,
5668 "Cannot create symbolic link \"%s\" -> "
5669 "\"%s\"",
5670 Symlnk_p, nam_p);
5671
5672 if (*Over_p != '\0') {
5673 rstfiles(U_KEEP, dirfd);
5674 }
5675 return (-1);
5676 }
5677 } else {
5678 int saveerrno;
5679
5680 if ((result = openat(dirfd, get_component(nam_p),
5681 O_CREAT|O_RDWR|O_TRUNC, (int)G_p->g_mode)) < 0) {
5682 saveerrno = errno;
5683 if (G_p->g_attrnam_p != NULL) {
5684 result = retry_open_attr(dirfd,
5685 Gen.g_baseparent_fd, Fullnam_p,
5686 (G_p->g_attrparent_p == NULL) ?
5687 NULL : G_p->g_attrparent_p, nam_p,
5688 O_CREAT|O_RDWR|O_TRUNC,
5689 (int)G_p->g_mode);
5690 }
5691 }
5692 if (result < 0) {
5693 errno = saveerrno;
5694 if (errno != ENOENT) {
5695 /* The attempt to open failed. */
5696 msg(ERRN, "Cannot open file \"%s\"",
5697 nam_p);
5698 if (*Over_p != '\0') {
5699 rstfiles(U_KEEP, dirfd);
5700 }
5701 return (-1);
5702 }
5703 } else {
5704 /* acl support */
5705 acl_is_set = 0;
5706 if (Pflag && aclp != NULL) {
5707 if (facl_set(result, aclp) < 0) {
5708 msg(ERRN,
5709 "\"%s\": failed to set acl",
5710 nam_p);
5711 } else {
5712 acl_is_set = 1;
5713 }
5714 acl_free(aclp);
5715 aclp = NULL;
5716 }
5717 cnt = 0;
5718 break;
5719 }
5720 }
5721 cnt++;
5722 } while (cnt < 2 && missdir(nam_p) == 0);
5723
5724 switch (cnt) {
5725 case 0:
5726 if ((Args & OCi) && (Hdr_type == USTAR)) {
5727 setpasswd(nam_p);
5728 }
5729 if ((G_p->g_mode & Ftype) == S_IFLNK ||
5730 (Hdr_type == BAR && bar_linkflag == SYMTYPE)) {
5731 if (Args & OCR) {
5732 if (fchownat(dirfd,
5733 get_component(nam_p),
5734 (int)Rpw_p->pw_uid,
5735 (int)Rpw_p->pw_gid,
5736 AT_SYMLINK_NOFOLLOW) < 0) {
5737 msg(ERRN,
5738 "Error during chown() of "
5739 "\"%s%s%s\"",
5740 (G_p->g_attrnam_p == NULL) ?
5741 nam_p : G_p->g_attrfnam_p,
5742 (G_p->g_attrnam_p == NULL) ?
5743 "" : G_p->g_rw_sysattr ?
5744 gettext(" System Attribute ") :
5745 gettext(" Attribute "),
5746 (G_p->g_attrnam_p == NULL) ?
5747 "" : nam_p);
5748 }
5749 } else if ((fchownat(dirfd, get_component(nam_p),
5750 (int)G_p->g_uid, (int)G_p->g_gid,
5751 AT_SYMLINK_NOFOLLOW) < 0) && privileged) {
5752 msg(ERRN,
5753 "Error during chown() of \"%s%s%s\"",
5754 (G_p->g_attrnam_p == NULL) ?
5755 nam_p : G_p->g_attrfnam_p,
5756 (G_p->g_attrnam_p == NULL) ? "" :
5757 G_p->g_rw_sysattr ?
5758 gettext(" System Attribute ") :
5759 gettext(" Attribute "),
5760 (G_p->g_attrnam_p == NULL) ? "" : nam_p);
5761 }
5762 }
5763 break;
5764
5765 case 1:
5766 if (Do_rename) {
5767 msg(ERRN, "Cannot create directory for \"%s%s%s\"",
5768 (G_p->g_attrnam_p == NULL) ? Over_p :
5769 G_p->g_attrfnam_p,
5770 (G_p->g_attrnam_p == NULL) ? "" :
5771 G_p->g_rw_sysattr ?
5772 gettext(" System Attribute ") :
5773 gettext(" Attribute "),
5774 (G_p->g_attrnam_p == NULL) ? "" : Over_p);
5775 } else {
5776 msg(ERRN, "Cannot create directory for \"%s%s%s\"",
5777 (G_p->g_attrnam_p == NULL) ? nam_p :
5778 G_p->g_attrfnam_p,
5779 (G_p->g_attrnam_p == NULL) ? "" :
5780 G_p->g_rw_sysattr ?
5781 gettext(" System Attribute ") :
5782 gettext(" Attribute "),
5783 (G_p->g_attrnam_p == NULL) ? "" : nam_p);
5784 }
5785 break;
5786
5787 case 2:
5788 if (Do_rename) {
5789 msg(ERRN, "Cannot create \"%s%s%s\"",
5790 (G_p->g_attrnam_p == NULL) ? Over_p :
5791 G_p->g_attrfnam_p,
5792 (G_p->g_attrnam_p == NULL) ? "" :
5793 G_p->g_rw_sysattr ?
5794 gettext(" System Attribute ") :
5795 gettext(" Attribute "),
5796 (G_p->g_attrnam_p == NULL) ? "" :
5797 Over_p);
5798 } else {
5799 msg(ERRN, "Cannot create \"%s%s%s\"",
5800 (G_p->g_attrnam_p == NULL) ? nam_p :
5801 G_p->g_attrfnam_p,
5802 (G_p->g_attrnam_p == NULL) ? "" :
5803 G_p->g_rw_sysattr ?
5804 gettext(" System Attribute ") :
5805 gettext(" Attribute "),
5806 (G_p->g_attrnam_p == NULL) ? "" : nam_p);
5807 }
5808 break;
5809
5810 default:
5811 msg(EXT, "Impossible case.");
5812 }
5813
5814 Finished = 0;
5815 return (result);
5816 }
5817
5818 /*
5819 * read_hdr: Transfer headers from the selected format
5820 * in the archive I/O buffer to the generic structure.
5821 */
5822
5823 static
5824 int
read_hdr(int hdr)5825 read_hdr(int hdr)
5826 {
5827 int rv = NONE;
5828 major_t maj, rmaj;
5829 minor_t min, rmin;
5830 char tmpnull = '\0';
5831 static int bar_read_cnt = 0;
5832
5833 if (hdr != BAR) {
5834 if (Buffr.b_end_p != (Buffr.b_out_p + Hdrsz)) {
5835 tmpnull = *(Buffr.b_out_p + Hdrsz);
5836 *(Buffr.b_out_p + Hdrsz) = '\0';
5837 }
5838 }
5839
5840 switch (hdr) {
5841 case BIN:
5842 (void) memcpy(&Hdr, Buffr.b_out_p, HDRSZ);
5843 if (Hdr.h_magic == (short)CMN_BBS) {
5844 swap((char *)&Hdr, HDRSZ);
5845 }
5846 Gen.g_magic = Hdr.h_magic;
5847 Gen.g_mode = Hdr.h_mode;
5848 Gen.g_uid = Hdr.h_uid;
5849 Gen.g_gid = Hdr.h_gid;
5850 Gen.g_nlink = Hdr.h_nlink;
5851 Gen.g_mtime = mklong(Hdr.h_mtime);
5852 Gen.g_ino = Hdr.h_ino;
5853 Gen.g_dev = Hdr.h_dev;
5854 Gen.g_rdev = Hdr.h_rdev;
5855 maj = SVR3_MAJOR(Gen.g_dev);
5856 rmaj = SVR3_MAJOR(Gen.g_rdev);
5857 min = SVR3_MINOR(Gen.g_dev);
5858 rmin = SVR3_MINOR(Gen.g_rdev);
5859 Gen.g_dev = makedev(maj, min);
5860 Gen.g_rdev = makedev(rmaj, rmin);
5861 Gen.g_cksum = 0L;
5862 Gen.g_filesz = (off_t)mklong(Hdr.h_filesize);
5863 Gen.g_namesz = Hdr.h_namesize;
5864 rv = BIN;
5865 break;
5866 case CHR:
5867 if (sscanf(Buffr.b_out_p,
5868 "%6o%6o%6o%6o"
5869 "%6" _SCNoID "%6" _SCNoID
5870 "%6o%6o%11o%6o%11lo",
5871 &Gen.g_magic, &Gen.g_dev, &Gen.g_ino, &Gen.g_mode,
5872 &Gen.g_uid, &Gen.g_gid,
5873 &Gen.g_nlink, &Gen.g_rdev,
5874 (uint_t *)&Gen.g_mtime, (uint_t *)&Gen.g_namesz,
5875 (u_off_t *)&Gen.g_filesz) == CHR_CNT) {
5876 rv = CHR;
5877 maj = SVR3_MAJOR(Gen.g_dev);
5878 rmaj = SVR3_MAJOR(Gen.g_rdev);
5879 min = SVR3_MINOR(Gen.g_dev);
5880 rmin = SVR3_MINOR(Gen.g_rdev);
5881 Gen.g_dev = makedev(maj, min);
5882 Gen.g_rdev = makedev(rmaj, rmin);
5883 }
5884 break;
5885 case ASC:
5886 case CRC:
5887 if (sscanf(Buffr.b_out_p,
5888 "%6x%8x%8x"
5889 "%8" _SCNxID "%8" _SCNxID
5890 "%8x%8x%8lx%8x%8x%8x%8x%8x%8x",
5891 &Gen.g_magic, &Gen.g_ino, &Gen.g_mode,
5892 &Gen.g_uid, &Gen.g_gid,
5893 &Gen.g_nlink, &Gen.g_mtime,
5894 (u_off_t *)&Gen.g_filesz, (uint_t *)&maj, (uint_t *)&min,
5895 (uint_t *)&rmaj, (uint_t *)&rmin, (uint_t *)&Gen.g_namesz,
5896 &Gen.g_cksum) == ASC_CNT) {
5897 Gen.g_dev = makedev(maj, min);
5898 Gen.g_rdev = makedev(rmaj, rmin);
5899 rv = hdr;
5900 }
5901 break;
5902 case USTAR: /* TAR and USTAR */
5903 if (*Buffr.b_out_p == '\0') {
5904 *Gen.g_nam_p = '\0';
5905 nambuf[0] = '\0';
5906 } else {
5907 Thdr_p = (union tblock *)Buffr.b_out_p;
5908 Gen.g_nam_p[0] = '\0';
5909 (void) strncpy((char *)&nambuf,
5910 Thdr_p->tbuf.t_name, NAMSIZ);
5911 (void) sscanf(Thdr_p->tbuf.t_mode, "%8o",
5912 &Gen.g_mode);
5913 (void) sscanf(Thdr_p->tbuf.t_uid, "%8" _SCNoID,
5914 &Gen.g_uid);
5915 (void) sscanf(Thdr_p->tbuf.t_gid, "%8" _SCNoID,
5916 &Gen.g_gid);
5917 (void) sscanf(Thdr_p->tbuf.t_size, "%12lo",
5918 (u_off_t *)&Gen.g_filesz);
5919 (void) sscanf(Thdr_p->tbuf.t_mtime, "%12o",
5920 (uint_t *)&Gen.g_mtime);
5921 (void) sscanf(Thdr_p->tbuf.t_cksum, "%8o",
5922 (uint_t *)&Gen.g_cksum);
5923 if (Thdr_p->tbuf.t_linkname[0] != '\0')
5924 Gen.g_nlink = 1;
5925 else
5926 Gen.g_nlink = 0;
5927
5928 switch (Thdr_p->tbuf.t_typeflag) {
5929 case SYMTYPE:
5930 /* Symbolic Link */
5931 Gen.g_nlink = 2;
5932 break;
5933 case CHRTYPE:
5934 Gen.g_mode |= (S_IFMT & S_IFCHR);
5935 break;
5936 case BLKTYPE:
5937 Gen.g_mode |= (S_IFMT & S_IFBLK);
5938 break;
5939 case DIRTYPE:
5940 Gen.g_mode |= (S_IFMT & S_IFDIR);
5941 break;
5942 case FIFOTYPE:
5943 Gen.g_mode |= (S_IFMT & S_IFIFO);
5944 break;
5945 }
5946
5947 (void) sscanf(Thdr_p->tbuf.t_magic, "%6o",
5948 (uint_t *)&Gen.g_tmagic);
5949 (void) sscanf(Thdr_p->tbuf.t_version, "%2o",
5950 (uint_t *)&Gen.g_version);
5951 (void) sscanf(Thdr_p->tbuf.t_uname, "%32s",
5952 (char *)&Gen.g_uname);
5953 (void) sscanf(Thdr_p->tbuf.t_gname, "%32s",
5954 (char *)&Gen.g_gname);
5955 (void) sscanf(Thdr_p->tbuf.t_devmajor, "%8o", &maj);
5956 (void) sscanf(Thdr_p->tbuf.t_devminor, "%8o", &min);
5957 (void) strncpy((char *)&prebuf,
5958 Thdr_p->tbuf.t_prefix, PRESIZ);
5959 Gen.g_namesz = strlen(Gen.g_nam_p) + 1;
5960 Gen.g_rdev = makedev(maj, min);
5961 }
5962 rv = USTAR;
5963 break;
5964 case TAR:
5965 if (*Buffr.b_out_p == '\0') {
5966 *Gen.g_nam_p = '\0';
5967 nambuf[0] = '\0';
5968 } else {
5969 Thdr_p = (union tblock *)Buffr.b_out_p;
5970 Gen.g_nam_p[0] = '\0';
5971 (void) sscanf(Thdr_p->tbuf.t_mode, "%8o", &Gen.g_mode);
5972 (void) sscanf(Thdr_p->tbuf.t_uid, "%8" _SCNoID,
5973 &Gen.g_uid);
5974 (void) sscanf(Thdr_p->tbuf.t_gid, "%8" _SCNoID,
5975 &Gen.g_gid);
5976 (void) sscanf(Thdr_p->tbuf.t_size, "%12" SCNo64,
5977 (u_off_t *)&Gen.g_filesz);
5978 (void) sscanf(Thdr_p->tbuf.t_mtime, "%12o",
5979 &Gen.g_mtime);
5980 (void) sscanf(Thdr_p->tbuf.t_cksum, "%8o",
5981 &Gen.g_cksum);
5982 if (Thdr_p->tbuf.t_typeflag == '1') /* hardlink */
5983 Gen.g_nlink = 1;
5984 else
5985 Gen.g_nlink = 0;
5986 (void) strncpy(Gen.g_nam_p,
5987 Thdr_p->tbuf.t_name, NAMSIZ);
5988 Gen.g_namesz = strlen(Gen.g_nam_p) + 1;
5989 (void) strcpy(nambuf, Gen.g_nam_p);
5990 }
5991 rv = TAR;
5992 break;
5993 case BAR:
5994 if (Bar_vol_num == 0 && bar_read_cnt == 0) {
5995 read_bar_vol_hdr();
5996 bar_read_cnt++;
5997 } else {
5998 read_bar_file_hdr();
5999 }
6000 rv = BAR;
6001 break;
6002 default:
6003 msg(EXT, "Impossible header type.");
6004 }
6005
6006 if (hdr != BAR) {
6007 if (Buffr.b_end_p != (Buffr.b_out_p + Hdrsz))
6008 *(Buffr.b_out_p + Hdrsz) = tmpnull;
6009 }
6010
6011 return (rv);
6012 }
6013
6014 /*
6015 * reclaim: Reclaim linked file structure storage.
6016 */
6017
6018 static void
reclaim(struct Lnk * p)6019 reclaim(struct Lnk *p)
6020 {
6021 p->L_bck_p->L_nxt_p = p->L_nxt_p;
6022 p->L_nxt_p->L_bck_p = p->L_bck_p;
6023
6024 while (p != NULL) {
6025 struct Lnk *new_p = p->L_lnk_p;
6026
6027 free(p->L_gen.g_nam_p);
6028 free(p);
6029 p = new_p;
6030 }
6031 }
6032
6033 /*
6034 * rstbuf: Reset the I/O buffer, move incomplete potential headers to
6035 * the front of the buffer and force bread() to refill the buffer. The
6036 * return value from bread() is returned (to identify I/O errors). On the
6037 * 3B2, reads must begin on a word boundary, therefore, with the -i option,
6038 * any remaining bytes in the buffer must be moved to the base of the buffer
6039 * in such a way that the destination locations of subsequent reads are
6040 * word aligned.
6041 */
6042
6043 static void
rstbuf(void)6044 rstbuf(void)
6045 {
6046 int pad;
6047
6048 if ((Args & OCi) || Append) {
6049 if (Buffr.b_out_p != Buffr.b_base_p) {
6050 pad = ((Buffr.b_cnt + FULLWD) & ~FULLWD);
6051 Buffr.b_in_p = Buffr.b_base_p + pad;
6052 pad -= Buffr.b_cnt;
6053 (void) memcpy(Buffr.b_base_p + pad, Buffr.b_out_p,
6054 (int)Buffr.b_cnt);
6055 Buffr.b_out_p = Buffr.b_base_p + pad;
6056 }
6057 if (bfill() < 0)
6058 msg(EXT, "Unexpected end-of-archive encountered.");
6059 } else { /* OCo */
6060 (void) memcpy(Buffr.b_base_p, Buffr.b_out_p, (int)Buffr.b_cnt);
6061 Buffr.b_out_p = Buffr.b_base_p;
6062 Buffr.b_in_p = Buffr.b_base_p + Buffr.b_cnt;
6063 }
6064 }
6065
6066 static void
setpasswd(char * nam)6067 setpasswd(char *nam)
6068 {
6069 if ((dpasswd = getpwnam(&Gen.g_uname[0])) == NULL) {
6070 msg(EPOST, "cpio: problem reading passwd entry");
6071 msg(EPOST, "cpio: %s: owner not changed", nam);
6072 if (Gen.g_uid == UID_NOBODY && S_ISREG(Gen.g_mode))
6073 Gen.g_mode &= ~S_ISUID;
6074 } else
6075 Gen.g_uid = dpasswd->pw_uid;
6076
6077 if ((dgroup = getgrnam(&Gen.g_gname[0])) == NULL) {
6078 msg(EPOST, "cpio: problem reading group entry");
6079 msg(EPOST, "cpio: %s: group not changed", nam);
6080 if (Gen.g_gid == GID_NOBODY && S_ISREG(Gen.g_mode))
6081 Gen.g_mode &= ~S_ISGID;
6082 } else
6083 Gen.g_gid = dgroup->gr_gid;
6084 G_p = &Gen;
6085 }
6086
6087 /*
6088 * rstfiles: Perform final changes to the file. If the -u option is set,
6089 * and overwrite == U_OVER, remove the temporary file, else if overwrite
6090 * == U_KEEP, unlink the current file, and restore the existing version
6091 * of the file. In addition, where appropriate, set the access or modification
6092 * times, change the owner and change the modes of the file.
6093 *
6094 * Note that if Do_rename is set, then the roles of original and temporary
6095 * file are reversed. If all went well, we will rename() the temporary file
6096 * over the original in order to accommodate potentially executing files.
6097 */
6098 static void
rstfiles(int over,int dirfd)6099 rstfiles(int over, int dirfd)
6100 {
6101 char *inam_p, *onam_p, *nam_p;
6102 int error;
6103
6104 #if defined(_PC_SATTR_ENABLED)
6105 /* Time or permissions cannot be set on system attribute files */
6106 if ((Gen.g_attrnam_p != NULL) && (Gen.g_rw_sysattr == 1)) {
6107 return;
6108 }
6109 #endif /* _PC_SATTR_ENABLED */
6110
6111 if (Args & OCp) {
6112 if (G_p->g_attrnam_p == NULL) {
6113 nam_p = Fullnam_p;
6114 } else {
6115 nam_p = G_p->g_attrnam_p;
6116 }
6117 } else {
6118 if (Gen.g_nlink > 0) {
6119 nam_p = G_p->g_nam_p;
6120 } else {
6121 nam_p = Gen.g_nam_p;
6122 }
6123 }
6124 if (Gen.g_attrnam_p != NULL) {
6125 nam_p = Gen.g_attrnam_p;
6126 }
6127
6128 if ((Args & OCi) && (Hdr_type == USTAR)) {
6129 setpasswd(nam_p);
6130 }
6131 if (over == U_KEEP && *Over_p != '\0') {
6132 if (Do_rename) {
6133 msg(POST, "Restoring existing \"%s%s%s\"",
6134 (G_p->g_attrnam_p == NULL) ? Over_p : Fullnam_p,
6135 (G_p->g_attrnam_p == NULL) ? "" :
6136 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
6137 gettext(" Attribute "),
6138 (G_p->g_attrnam_p == NULL) ? "" : Over_p);
6139 } else {
6140 msg(POST, "Restoring existing \"%s%s%s\"",
6141 (G_p->g_attrnam_p == NULL) ? nam_p : Fullnam_p,
6142 (G_p->g_attrnam_p == NULL) ? "" :
6143 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
6144 gettext(" Attribute "),
6145 (G_p->g_attrnam_p == NULL) ? "" : nam_p);
6146 }
6147
6148 /* delete what we just built */
6149 (void) unlinkat(dirfd, get_component(nam_p), 0);
6150
6151 /* If the old file needs restoring, do the necessary links */
6152 if (Do_rename) {
6153 char *tmp_ptr;
6154
6155 if (Args & OCp) {
6156 tmp_ptr = Fullnam_p;
6157 Fullnam_p = Over_p;
6158 } else {
6159 tmp_ptr = G_p->g_nam_p;
6160 G_p->g_nam_p = Over_p;
6161 }
6162 Over_p = tmp_ptr;
6163
6164 Do_rename = 0; /* names now have original values */
6165 } else {
6166 if (rename(Over_p, nam_p) < 0) {
6167 if (link(Over_p, nam_p) < 0) {
6168 msg(EXTN,
6169 "Cannot recover original version"
6170 " of \"%s%s%s\"",
6171 (G_p->g_attrnam_p == NULL) ?
6172 nam_p : Fullnam_p,
6173 (G_p->g_attrnam_p == NULL) ? "" :
6174 G_p->g_rw_sysattr ?
6175 gettext(" System Attribute ") :
6176 gettext(" Attribute "),
6177 (G_p->g_attrnam_p == NULL) ?
6178 "" : nam_p);
6179 }
6180 if (unlinkat(dirfd, get_component(Over_p), 0)) {
6181 msg(ERRN,
6182 "Cannot remove temp file "
6183 "\"%s%s%s\"",
6184 (G_p->g_attrnam_p == NULL) ?
6185 Over_p : Fullnam_p,
6186 (G_p->g_attrnam_p == NULL) ? "" :
6187 G_p->g_rw_sysattr ?
6188 gettext(" System Attribute ") :
6189 gettext(" Attribute "),
6190 (G_p->g_attrnam_p == NULL) ?
6191 "" : Over_p);
6192 }
6193 }
6194 }
6195 *Over_p = '\0';
6196 return;
6197 } else if (over == U_OVER && *Over_p != '\0') {
6198 if (Do_rename) {
6199 char *tmp_ptr;
6200
6201 (void) renameat(dirfd, get_component(nam_p),
6202 dirfd, get_component(Over_p));
6203 if (Args & OCp) {
6204 if (G_p->g_attrnam_p == NULL) {
6205 tmp_ptr = Fullnam_p;
6206 Fullnam_p = Over_p;
6207 Over_p = tmp_ptr;
6208 } else {
6209 /*
6210 * Over_p is pointing at g_attrnam_p
6211 * which must be preserved.
6212 *
6213 * We don't want the tmp_ptr and so
6214 * on to throw away our only copy of
6215 * the name.
6216 */
6217 Over_p = Attrfile_p;
6218 }
6219 } else {
6220 tmp_ptr = G_p->g_nam_p;
6221 G_p->g_nam_p = Over_p;
6222 Over_p = tmp_ptr;
6223 }
6224 Do_rename = 0; /* names now have original values */
6225 } else {
6226 if (unlinkat(dirfd, get_component(Over_p), 0) < 0) {
6227 msg(ERRN,
6228 "Cannot unlink() temp file \"%s%s%s\"",
6229 (G_p->g_attrnam_p == NULL) ?
6230 Over_p : Fullnam_p,
6231 (G_p->g_attrnam_p == NULL) ? "" :
6232 G_p->g_rw_sysattr ?
6233 gettext(" System Attribute ") :
6234 gettext(" Attribute "),
6235 (G_p->g_attrnam_p == NULL) ? "" : Over_p);
6236 }
6237 }
6238 *Over_p = '\0';
6239 }
6240 if (Args & OCp) {
6241 if (G_p->g_attrnam_p != NULL) {
6242 inam_p = G_p->g_attrfnam_p;
6243 onam_p = G_p->g_attrnam_p;
6244 } else {
6245 inam_p = Nam_p;
6246 onam_p = Fullnam_p;
6247 }
6248 } else /* OCi only uses onam_p, OCo only uses inam_p */
6249 if (G_p->g_attrnam_p != NULL) {
6250 inam_p = onam_p = G_p->g_attrnam_p;
6251 } else {
6252 inam_p = onam_p = G_p->g_nam_p;
6253 }
6254
6255 /*
6256 * Change the owner, time, and mode to those of the file
6257 * originally created in the archive. Note: time and
6258 * mode do not need to be restored for a symbolic link
6259 * since rstfiles() is not called when the archived file
6260 * is a symlink.
6261 */
6262 if (!(Args & OCo)) {
6263 if (Args & OCR) {
6264 if (fchownat(dirfd, get_component(onam_p),
6265 Rpw_p->pw_uid, Rpw_p->pw_gid,
6266 AT_SYMLINK_NOFOLLOW) < 0) {
6267 msg(ERRN, "Cannot chown() \"%s%s%s\"",
6268 onam_p,
6269 (G_p->g_attrnam_p == NULL) ? "" :
6270 G_p->g_rw_sysattr ?
6271 gettext(" System Attribute ") :
6272 gettext(" Attribute "),
6273 (G_p->g_attrnam_p == NULL) ? "" : onam_p);
6274 }
6275 } else {
6276 if ((fchownat(dirfd, get_component(onam_p),
6277 G_p->g_uid, G_p->g_gid,
6278 AT_SYMLINK_NOFOLLOW) < 0) && privileged) {
6279 msg(ERRN, "Cannot chown() \"%s%s%s\"",
6280 onam_p,
6281 (G_p->g_attrnam_p == NULL) ? "" :
6282 G_p->g_rw_sysattr ?
6283 gettext(" System Attribute ") :
6284 gettext(" Attribute "),
6285 (G_p->g_attrnam_p == NULL) ? "" : onam_p);
6286 }
6287 }
6288
6289 if (Args & OCm) {
6290 set_tym(dirfd, get_component(onam_p),
6291 G_p->g_mtime, G_p->g_mtime);
6292 }
6293
6294 /* Acl was not set, so we must chmod */
6295 if (!acl_is_set) {
6296 mode_t orig_mask = 0, new_mask;
6297
6298 /*
6299 * use fchmod for attributes, since
6300 * we known they are always regular
6301 * files, whereas when it isn't an
6302 * attribute it could be for a fifo
6303 * or something other that we don't
6304 * open and don't have a valid Ofile
6305 * for.
6306 */
6307 if (privileged) {
6308 new_mask = G_p->g_mode;
6309 } else {
6310 orig_mask = umask(0);
6311 new_mask = G_p->g_mode & ~orig_mask;
6312 }
6313
6314 if (G_p->g_attrnam_p != NULL) {
6315 error = fchmod(Ofile, new_mask);
6316 } else {
6317 error = chmod(onam_p, new_mask);
6318 }
6319 if (error < 0) {
6320 msg(ERRN,
6321 "Cannot chmod() \"%s%s%s\"",
6322 (G_p->g_attrnam_p == NULL) ?
6323 onam_p : G_p->g_attrfnam_p,
6324 (G_p->g_attrnam_p == NULL) ? "" :
6325 G_p->g_rw_sysattr ?
6326 gettext(" System Attribute ") :
6327 gettext(" Attribute "),
6328 (G_p->g_attrnam_p == NULL) ? "" : onam_p);
6329 }
6330 if (!privileged) {
6331 (void) umask(orig_mask);
6332 }
6333 }
6334 }
6335
6336 if (!(Args & OCi) && (Args & OCa)) {
6337 /*
6338 * Use dirfd since we are updating original file
6339 * and not just created file
6340 */
6341 set_tym(G_p->g_dirfd, get_component(inam_p),
6342 (uint_t)SrcSt.st_atime, (uint_t)SrcSt.st_mtime);
6343 }
6344 }
6345
6346 /*
6347 * scan4trail: Scan the archive looking for the trailer.
6348 * When found, back the archive up over the trailer and overwrite
6349 * the trailer with the files to be added to the archive.
6350 */
6351
6352 static void
scan4trail(void)6353 scan4trail(void)
6354 {
6355 int rv;
6356 off_t off1, off2;
6357
6358 Append = 1;
6359 Hdr_type = NONE;
6360 G_p = NULL;
6361 while (gethdr()) {
6362 G_p = &Gen;
6363 data_in(P_SKIP);
6364 }
6365 off1 = Buffr.b_cnt;
6366 off2 = Bufsize - (Buffr.b_cnt % Bufsize);
6367 Buffr.b_out_p = Buffr.b_in_p = Buffr.b_base_p;
6368 Buffr.b_cnt = (off_t)0;
6369 if (lseek(Archive, -(off1 + off2), SEEK_REL) < 0)
6370 msg(EXTN, "Unable to append to this archive");
6371 if ((rv = g_read(Device, Archive, Buffr.b_in_p, Bufsize)) < 0)
6372 msg(EXTN, "Cannot append to this archive");
6373 if (lseek(Archive, (off_t)-rv, SEEK_REL) < 0)
6374 msg(EXTN, "Unable to append to this archive");
6375 Buffr.b_cnt = off2;
6376 Buffr.b_in_p = Buffr.b_base_p + Buffr.b_cnt;
6377 Append = 0;
6378 }
6379
6380 /*
6381 * setup: Perform setup and initialization functions. Parse the options
6382 * using getopt(3C), call ckopts to check the options and initialize various
6383 * structures and pointers. Specifically, for the -i option, save any
6384 * patterns, for the -o option, check (via stat(2)) the archive, and for
6385 * the -p option, validate the destination directory.
6386 */
6387
6388 static void
setup(int largc,char ** largv)6389 setup(int largc, char **largv)
6390 {
6391 extern int optind;
6392 extern char *optarg;
6393
6394 #if defined(O_XATTR)
6395 #if defined(_PC_SATTR_ENABLED)
6396 #ifdef WAITAROUND
6397 char *opts_p = "zabcdfiklmopqrstuvABC:DE:H:I:LM:O:PR:SV6@/";
6398 #else
6399 char *opts_p = "abcdfiklmopqrstuvABC:DE:H:I:LM:O:PR:SV6@/";
6400 #endif /* WAITAROUND */
6401
6402 #else /* _PC_SATTR_ENABLED */
6403 #ifdef WAITAROUND
6404 char *opts_p = "zabcdfiklmopqrstuvABC:DE:H:I:LM:O:PR:SV6@";
6405 #else
6406 char *opts_p = "abcdfiklmopqrstuvABC:DE:H:I:LM:O:PR:SV6@";
6407 #endif /* WAITAROUND */
6408 #endif /* _PC_SATTR_ENABLED */
6409
6410 #else /* O_XATTR */
6411 #ifdef WAITAROUND
6412 char *opts_p = "zabcdfiklmopqrstuvABC:DE:H:I:LM:O:PR:SV6";
6413 #else
6414 char *opts_p = "abcdfiklmopqrstuvABC:DE:H:I:LM:O:PR:SV6";
6415 #endif /* WAITAROUND */
6416 #endif /* O_XATTR */
6417
6418 char *dupl_p = "Only one occurrence of -%c allowed";
6419 int option;
6420 int blk_cnt, blk_cnt_max;
6421 struct rlimit rlim;
6422
6423 /* Remember the native page size. */
6424
6425 PageSize = sysconf(_SC_PAGESIZE);
6426
6427 if (PageSize == -1) {
6428 /*
6429 * This sysconf call will almost certainly never fail. The
6430 * symbol PAGESIZE itself resolves to the above sysconf call,
6431 * so we should go ahead and define our own constant.
6432 */
6433 PageSize = 8192;
6434 }
6435
6436 Hdr_type = BIN;
6437 Max_offset = (off_t)(BIN_OFFSET_MAX);
6438 Efil_p = Hdr_p = Own_p = IOfil_p = NULL;
6439 while ((option = getopt(largc, largv, opts_p)) != EOF) {
6440 switch (option) {
6441 #ifdef WAITAROUND
6442 case 'z':
6443 /* rendezvous with the debugger */
6444 waitaround = 1;
6445 break;
6446 #endif
6447 case 'a': /* reset access time */
6448 Args |= OCa;
6449 break;
6450 case 'b': /* swap bytes and halfwords */
6451 Args |= OCb;
6452 break;
6453 case 'c': /* select character header */
6454 Args |= OCc;
6455 Hdr_type = ASC;
6456 Max_namesz = APATH;
6457 Onecopy = 1;
6458 break;
6459 case 'd': /* create directories as needed */
6460 Args |= OCd;
6461 break;
6462 case 'f': /* select files not in patterns */
6463 Args |= OCf;
6464 break;
6465 case 'i': /* "copy in" */
6466 Args |= OCi;
6467 Archive = 0;
6468 break;
6469 case 'k': /* retry after I/O errors */
6470 Args |= OCk;
6471 break;
6472 case 'l': /* link files when possible */
6473 Args |= OCl;
6474 break;
6475 case 'm': /* retain modification time */
6476 Args |= OCm;
6477 break;
6478 case 'o': /* "copy out" */
6479 Args |= OCo;
6480 Archive = 1;
6481 break;
6482 case 'p': /* "pass" */
6483 Max_namesz = APATH;
6484 Args |= OCp;
6485 break;
6486 case 'q': /* "quiet" */
6487 Args |= OCq;
6488 break;
6489 case 'r': /* rename files interactively */
6490 Args |= OCr;
6491 break;
6492 case 's': /* swap bytes */
6493 Args |= OCs;
6494 break;
6495 case 't': /* table of contents */
6496 Args |= OCt;
6497 break;
6498 case 'u': /* copy unconditionally */
6499 Args |= OCu;
6500 break;
6501 case 'v': /* verbose - print file names */
6502 Args |= OCv;
6503 break;
6504 case 'A': /* append to existing archive */
6505 Args |= OCA;
6506 break;
6507 case 'B': /* set block size to 5120 bytes */
6508 Args |= OCB;
6509 Bufsize = 5120;
6510 break;
6511 case 'C': /* set arbitrary block size */
6512 if (Args & OCC)
6513 msg(ERR, dupl_p, 'C');
6514 else {
6515 Args |= OCC;
6516 Bufsize = atoi(optarg);
6517 }
6518 break;
6519 case 'D':
6520 Dflag = 1;
6521 break;
6522 case 'E': /* alternate file for pattern input */
6523 if (Args & OCE)
6524 msg(ERR, dupl_p, 'E');
6525 else {
6526 Args |= OCE;
6527 Efil_p = optarg;
6528 }
6529 break;
6530 case 'H': /* select header type */
6531 if (Args & OCH)
6532 msg(ERR, dupl_p, 'H');
6533 else {
6534 Args |= OCH;
6535 Hdr_p = optarg;
6536 }
6537 break;
6538 case 'I': /* alternate file for archive input */
6539 if (Args & OCI)
6540 msg(ERR, dupl_p, 'I');
6541 else {
6542 Args |= OCI;
6543 IOfil_p = optarg;
6544 }
6545 break;
6546 case 'L': /* follow symbolic links */
6547 Args |= OCL;
6548 break;
6549 case 'M': /* specify new end-of-media message */
6550 if (Args & OCM)
6551 msg(ERR, dupl_p, 'M');
6552 else {
6553 Args |= OCM;
6554 Eom_p = optarg;
6555 }
6556 break;
6557 case 'O': /* alternate file for archive output */
6558 if (Args & OCO)
6559 msg(ERR, dupl_p, 'O');
6560 else {
6561 Args |= OCO;
6562 IOfil_p = optarg;
6563 }
6564 break;
6565 case 'P': /* preserve acls */
6566 Args |= OCP;
6567 Pflag++;
6568 break;
6569 case 'R': /* change owner/group of files */
6570 if (Args & OCR)
6571 msg(ERR, dupl_p, 'R');
6572 else {
6573 Args |= OCR;
6574 Own_p = optarg;
6575 }
6576 break;
6577 case 'S': /* swap halfwords */
6578 Args |= OCS;
6579 break;
6580 case 'V': /* print a dot '.' for each file */
6581 Args |= OCV;
6582 break;
6583 case '6': /* for old, sixth-edition files */
6584 Args |= OC6;
6585 Ftype = SIXTH;
6586 break;
6587 #if defined(O_XATTR)
6588 case '@':
6589 Atflag++;
6590 break;
6591 #if defined(_PC_SATTR_ENABLED)
6592 case '/':
6593 SysAtflag++;
6594 break;
6595 #endif /* _PC_SATTR_ENABLED */
6596 #endif /* O_XATTR */
6597 default:
6598 Error_cnt++;
6599 } /* option */
6600 } /* (option = getopt(largc, largv, opts_p)) != EOF */
6601
6602 #ifdef WAITAROUND
6603 if (waitaround) {
6604 (void) fprintf(stderr, gettext("Rendezvous with cpio on pid"
6605 " %d\n"), getpid());
6606
6607 while (waitaround) {
6608 (void) sleep(10);
6609 }
6610 }
6611 #endif
6612
6613 largc -= optind;
6614 largv += optind;
6615 ckopts(Args);
6616 if (!Error_cnt) {
6617 if (Args & OCr) {
6618 Renam_p = e_zalloc(E_EXIT, APATH + 1);
6619 Renametmp_p = e_zalloc(E_EXIT, APATH + 1);
6620 #if defined(_PC_SATTR_ENABLED)
6621 Renam_attr_p = e_zalloc(E_EXIT, APATH + 1);
6622 #endif
6623 }
6624 Symlnk_p = e_zalloc(E_EXIT, APATH);
6625 Over_p = e_zalloc(E_EXIT, APATH);
6626 Nam_p = e_zalloc(E_EXIT, APATH + 1);
6627 if (Args & OCp) {
6628 Savenam_p = e_zalloc(E_EXIT, APATH + 1);
6629 }
6630 Fullnam_p = e_zalloc(E_EXIT, APATH);
6631 Lnknam_p = e_zalloc(E_EXIT, APATH);
6632 Gen.g_nam_p = Nam_p;
6633 if ((Fullnam_p = getcwd(NULL, APATH)) == NULL)
6634 msg(EXT, "Unable to determine current directory.");
6635 if (Args & OCi) {
6636 if (largc > 0) /* save patterns for -i option, if any */
6637 Pat_pp = largv;
6638 if (Args & OCE)
6639 getpats(largc, largv);
6640 } else if (Args & OCo) {
6641 if (largc != 0) /* error if arguments left with -o */
6642 Error_cnt++;
6643 else if (fstat(Archive, &ArchSt) < 0)
6644 msg(ERRN, "Error during stat() of archive");
6645 switch (Hdr_type) {
6646 case BIN:
6647 Hdrsz = HDRSZ;
6648 Pad_val = HALFWD;
6649 break;
6650 case CHR:
6651 Hdrsz = CHRSZ;
6652 Pad_val = 0;
6653 Max_offset = (off_t)(CHAR_OFFSET_MAX);
6654 break;
6655 case ASC:
6656 case CRC:
6657 Hdrsz = ASCSZ;
6658 Pad_val = FULLWD;
6659 Max_offset = (off_t)(ASC_OFFSET_MAX);
6660 break;
6661 case TAR:
6662 /* FALLTHROUGH */
6663 case USTAR: /* TAR and USTAR */
6664 Hdrsz = TARSZ;
6665 Pad_val = FULLBK;
6666 Max_offset = (off_t)(CHAR_OFFSET_MAX);
6667 break;
6668 default:
6669 msg(EXT, "Impossible header type.");
6670 }
6671 } else { /* directory must be specified */
6672 if (largc != 1)
6673 Error_cnt++;
6674 else if (access(*largv, 2) < 0 && (errno != EACCES))
6675 /*
6676 * EACCES is ignored here as it may occur
6677 * when any directory component of the path
6678 * does not have write permission, even though
6679 * the destination subdirectory has write
6680 * access. Writing to a read only directory
6681 * is handled later, as in "copy in" mode.
6682 */
6683 msg(ERRN,
6684 "Error during access() of \"%s\"", *largv);
6685 }
6686 }
6687 if (Error_cnt)
6688 usage(); /* exits! */
6689 if (Args & (OCi | OCo)) {
6690 if (!Dflag) {
6691 if (Args & (OCB | OCC)) {
6692 if (g_init(&Device, &Archive) < 0)
6693 msg(EXTN,
6694 "Error during initialization");
6695 } else {
6696 if ((Bufsize = g_init(&Device, &Archive)) < 0)
6697 msg(EXTN,
6698 "Error during initialization");
6699 }
6700 }
6701
6702 blk_cnt_max = _20K / Bufsize;
6703 if (blk_cnt_max < MX_BUFS) {
6704 blk_cnt_max = MX_BUFS;
6705 }
6706
6707 Buffr.b_base_p = NULL;
6708
6709 for (blk_cnt = blk_cnt_max; blk_cnt > 1; blk_cnt--) {
6710 Buffr.b_size = (size_t)(Bufsize * blk_cnt);
6711 Buffr.b_base_p = e_valloc(E_NORMAL, Buffr.b_size);
6712 if (Buffr.b_base_p != NULL) {
6713 break;
6714 }
6715 }
6716 if (Buffr.b_base_p == NULL || Buffr.b_size < (2 * CPIOBSZ)) {
6717 msg(EXT, "Out of memory");
6718 }
6719
6720 Buffr.b_out_p = Buffr.b_in_p = Buffr.b_base_p;
6721 Buffr.b_cnt = 0L;
6722 Buffr.b_end_p = Buffr.b_base_p + Buffr.b_size;
6723 }
6724
6725 /*
6726 * Now that Bufsize has stabilized, we can allocate our i/o buffer
6727 */
6728 Buf_p = e_valloc(E_EXIT, Bufsize);
6729
6730 if (Args & OCp) { /* get destination directory */
6731 (void) strcpy(Fullnam_p, *largv);
6732 if (stat(Fullnam_p, &DesSt) < 0)
6733 msg(EXTN, "Error during stat() of \"%s\"", Fullnam_p);
6734 if ((DesSt.st_mode & Ftype) != S_IFDIR)
6735 msg(EXT, "\"%s\" is not a directory", Fullnam_p);
6736 }
6737 Full_p = Fullnam_p + strlen(Fullnam_p) - 1;
6738 if (*Full_p != '/') {
6739 Full_p++;
6740 *Full_p = '/';
6741 }
6742 Full_p++;
6743 *Full_p = '\0';
6744 (void) strcpy(Lnknam_p, Fullnam_p);
6745 Lnkend_p = Lnknam_p + strlen(Lnknam_p);
6746 (void) getrlimit(RLIMIT_FSIZE, &rlim);
6747 Max_filesz = (off_t)rlim.rlim_cur;
6748 Lnk_hd.L_nxt_p = Lnk_hd.L_bck_p = &Lnk_hd;
6749 Lnk_hd.L_lnk_p = NULL;
6750 }
6751
6752 /*
6753 * set_tym: Set the access and/or modification times for a file.
6754 */
6755
6756 static void
set_tym(int dirfd,char * nam_p,time_t atime,time_t mtime)6757 set_tym(int dirfd, char *nam_p, time_t atime, time_t mtime)
6758 {
6759 struct timeval times[2];
6760
6761 times[0].tv_sec = atime;
6762 times[0].tv_usec = 0;
6763 times[1].tv_sec = mtime;
6764 times[1].tv_usec = 0;
6765
6766 if (futimesat(dirfd, nam_p, times) < 0) {
6767 if (Args & OCa) {
6768 msg(ERRN,
6769 "Unable to reset access time for \"%s%s%s\"",
6770 (G_p->g_attrnam_p == NULL) ? nam_p : Fullnam_p,
6771 (G_p->g_attrnam_p == NULL) ? "" :
6772 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
6773 gettext(" Attribute "),
6774 (G_p->g_attrnam_p == NULL) ? "" : nam_p);
6775 } else {
6776 msg(ERRN,
6777 "Unable to reset modification time for \"%s%s%s\"",
6778 (G_p->g_attrnam_p == NULL) ? nam_p : Fullnam_p,
6779 (G_p->g_attrnam_p == NULL) ? "" :
6780 G_p->g_rw_sysattr ? gettext(" System Attribute ") :
6781 gettext(" Attribute "),
6782 (G_p->g_attrnam_p == NULL) ? "" : nam_p);
6783 }
6784 }
6785 }
6786
6787 /*
6788 * sigint: Catch interrupts. If an interrupt occurs during the extraction
6789 * of a file from the archive with the -u option set, and the filename did
6790 * exist, remove the current file and restore the original file. Then exit.
6791 */
6792
6793 /*ARGSUSED*/
6794 static void
sigint(int sig)6795 sigint(int sig)
6796 {
6797 char *nam_p;
6798
6799 (void) signal(SIGINT, SIG_IGN); /* block further signals */
6800 if (!Finished) {
6801 if (Args & OCi)
6802 nam_p = G_p->g_nam_p;
6803 else /* OCp */
6804 nam_p = Fullnam_p;
6805 if (*Over_p != '\0') { /* There is a temp file */
6806 if (unlink(nam_p) < 0) {
6807 msg(ERRN,
6808 "Cannot remove incomplete \"%s\"", nam_p);
6809 }
6810 if (rename(Over_p, nam_p) < 0) {
6811 if (link(Over_p, nam_p) < 0) {
6812 msg(ERRN,
6813 "Cannot recover original \"%s\"",
6814 nam_p);
6815 }
6816 if (unlink(Over_p)) {
6817 msg(ERRN,
6818 "Cannot remove temp file \"%s\"",
6819 Over_p);
6820 }
6821 }
6822 } else if (unlink(nam_p))
6823 msg(ERRN, "Cannot remove incomplete \"%s\"", nam_p);
6824 *Over_p = '\0';
6825 }
6826 exit(EXIT_CODE);
6827 }
6828
6829 /*
6830 * swap: Swap bytes (-s), halfwords (-S) or or both halfwords and bytes (-b).
6831 */
6832
6833 static void
swap(char * buf_p,int cnt)6834 swap(char *buf_p, int cnt)
6835 {
6836 unsigned char tbyte;
6837 int tcnt;
6838 int rcnt;
6839 ushort_t thalf;
6840
6841 rcnt = cnt % 4;
6842 cnt /= 4;
6843 if (Args & (OCb | OCs | BSM)) {
6844 tcnt = cnt;
6845 Swp_p = (union swpbuf *)buf_p;
6846 while (tcnt-- > 0) {
6847 tbyte = Swp_p->s_byte[0];
6848 Swp_p->s_byte[0] = Swp_p->s_byte[1];
6849 Swp_p->s_byte[1] = tbyte;
6850 tbyte = Swp_p->s_byte[2];
6851 Swp_p->s_byte[2] = Swp_p->s_byte[3];
6852 Swp_p->s_byte[3] = tbyte;
6853 Swp_p++;
6854 }
6855 if (rcnt >= 2) {
6856 tbyte = Swp_p->s_byte[0];
6857 Swp_p->s_byte[0] = Swp_p->s_byte[1];
6858 Swp_p->s_byte[1] = tbyte;
6859 tbyte = Swp_p->s_byte[2];
6860 }
6861 }
6862 if (Args & (OCb | OCS)) {
6863 tcnt = cnt;
6864 Swp_p = (union swpbuf *)buf_p;
6865 while (tcnt-- > 0) {
6866 thalf = Swp_p->s_half[0];
6867 Swp_p->s_half[0] = Swp_p->s_half[1];
6868 Swp_p->s_half[1] = thalf;
6869 Swp_p++;
6870 }
6871 }
6872 }
6873
6874 /*
6875 * usage: Print the usage message on stderr and exit.
6876 */
6877
6878 static void
usage(void)6879 usage(void)
6880 {
6881
6882 (void) fflush(stdout);
6883 #if defined(O_XATTR)
6884 (void) fprintf(stderr, gettext("USAGE:\n"
6885 "\tcpio -i[bcdfkmqrstuv@BSV6] [-C size] "
6886 "[-E file] [-H hdr] [-I file [-M msg]] "
6887 "[-R id] [patterns]\n"
6888 "\tcpio -o[acv@ABLV] [-C size] "
6889 "[-H hdr] [-O file [-M msg]]\n"
6890 "\tcpio -p[adlmuv@LV] [-R id] directory\n"));
6891 #else
6892 (void) fprintf(stderr, gettext("USAGE:\n"
6893 "\tcpio -i[bcdfkmqrstuvBSV6] [-C size] "
6894 "[-E file] [-H hdr] [-I file [-M msg]] "
6895 "[-R id] [patterns]\n"
6896 "\tcpio -o[acvABLV] [-C size] "
6897 "[-H hdr] [-O file [-M msg]]\n"
6898 "\tcpio -p[adlmuvLV] [-R id] directory\n"));
6899 #endif
6900 (void) fflush(stderr);
6901 exit(EXIT_CODE);
6902 }
6903
6904 /*
6905 * verbose: For each file, print either the filename (-v) or a dot (-V).
6906 * If the -t option (table of contents) is set, print either the filename,
6907 * or if the -v option is also set, print an "ls -l"-like listing.
6908 */
6909
6910 static void
verbose(char * nam_p)6911 verbose(char *nam_p)
6912 {
6913 int i, j, temp;
6914 mode_t mode;
6915 char modestr[12];
6916 time_t ttime;
6917
6918 /*
6919 * The printf format and associated arguments to print the current
6920 * filename. Normally, just nam_p. If we're processing an extended
6921 * attribute, these are overridden.
6922 */
6923 char *name_fmt = "%s";
6924 const char *name = nam_p;
6925 const char *attribute = NULL;
6926
6927 if (Gen.g_attrnam_p != NULL) {
6928 /*
6929 * Translation note:
6930 * 'attribute' is a noun.
6931 */
6932
6933 if (Gen.g_rw_sysattr) {
6934 name_fmt = gettext("%s system attribute %s");
6935 } else if ((Args & OCt) &&
6936 (is_sysattr(basename(Gen.g_attrnam_p)))) {
6937 name_fmt = gettext("%s system attribute %s");
6938 } else {
6939 name_fmt = gettext("%s attribute %s");
6940 }
6941
6942 name = (Args & OCp) ? nam_p : Gen.g_attrfnam_p;
6943 if (Gen.g_attrparent_p == NULL) {
6944 attribute = Gen.g_attrnam_p;
6945 } else {
6946 attribute = Gen.g_attrpath_p;
6947 }
6948 }
6949
6950 if ((Gen.g_mode == SECMODE) || ((Hdr_type == USTAR ||
6951 Hdr_type == TAR) && Thdr_p->tbuf.t_typeflag == 'A')) {
6952 /* dont print ancillary file */
6953 aclchar = '+';
6954 return;
6955 }
6956 for (i = 0; i < 11; i++)
6957 modestr[i] = '-';
6958 modestr[i] = '\0';
6959 modestr[i-1] = aclchar;
6960 aclchar = ' ';
6961
6962 if ((Args & OCt) && (Args & OCv)) {
6963 mode = Gen.g_mode;
6964 for (i = 0; i < 3; i++) {
6965 temp = (mode >> (6 - (i * 3)));
6966 j = (i * 3) + 1;
6967 if (S_IROTH & temp)
6968 modestr[j] = 'r';
6969 if (S_IWOTH & temp)
6970 modestr[j + 1] = 'w';
6971 if (S_IXOTH & temp)
6972 modestr[j + 2] = 'x';
6973 }
6974
6975 if (Hdr_type != BAR) {
6976 temp = Gen.g_mode & Ftype;
6977 switch (temp) {
6978 case (S_IFIFO):
6979 modestr[0] = 'p';
6980 break;
6981 case (S_IFSOCK):
6982 modestr[0] = 's';
6983 break;
6984 case (S_IFCHR):
6985 modestr[0] = 'c';
6986 break;
6987 case (S_IFDIR):
6988 modestr[0] = 'd';
6989 break;
6990 case (S_IFBLK):
6991 modestr[0] = 'b';
6992 break;
6993 case (S_IFREG): /* was initialized to '-' */
6994 break;
6995 case (S_IFLNK):
6996 modestr[0] = 'l';
6997 break;
6998 default:
6999 msg(ERR, "Impossible file type");
7000 }
7001 } else { /* bar */
7002 temp = Gen.g_mode & Ftype;
7003 switch (temp) {
7004 case (S_IFIFO):
7005 modestr[0] = 'p';
7006 break;
7007 case (S_IFSOCK):
7008 modestr[0] = 's';
7009 break;
7010 case (S_IFCHR):
7011 modestr[0] = 'c';
7012 break;
7013 case (S_IFDIR):
7014 modestr[0] = 'd';
7015 break;
7016 case (S_IFBLK):
7017 modestr[0] = 'b';
7018 break;
7019 }
7020 if (bar_linkflag == SYMTYPE)
7021 modestr[0] = 'l';
7022 }
7023 if ((S_ISUID & Gen.g_mode) == S_ISUID)
7024 modestr[3] = 's';
7025 if ((S_ISVTX & Gen.g_mode) == S_ISVTX)
7026 modestr[9] = 't';
7027 if ((S_ISGID & G_p->g_mode) == S_ISGID && modestr[6] == 'x')
7028 modestr[6] = 's';
7029 else if ((S_ENFMT & Gen.g_mode) == S_ENFMT && modestr[6] != 'x')
7030 modestr[6] = 'l';
7031 if ((Hdr_type == TAR || Hdr_type == USTAR) && Gen.g_nlink == 0)
7032 (void) printf("%s%4d ", modestr, (int)Gen.g_nlink+1);
7033 else
7034 (void) printf("%s%4d ", modestr, (int)Gen.g_nlink);
7035 if (Lastuid == (uid_t)Gen.g_uid) {
7036 if (Lastuid == (uid_t)-1)
7037 (void) printf("-1 ");
7038 else
7039 (void) printf("%-9s", Curpw_p->pw_name);
7040 } else {
7041 if ((Curpw_p = getpwuid((int)Gen.g_uid)) != NULL) {
7042 (void) printf("%-9s", Curpw_p->pw_name);
7043 Lastuid = (uid_t)Gen.g_uid;
7044 } else {
7045 (void) printf("%-9d", (int)Gen.g_uid);
7046 Lastuid = (uid_t)-1;
7047 }
7048 }
7049 if (Lastgid == (gid_t)Gen.g_gid) {
7050 if (Lastgid == (gid_t)-1)
7051 (void) printf("-1 ");
7052 else
7053 (void) printf("%-9s", Curgr_p->gr_name);
7054 } else {
7055 if ((Curgr_p = getgrgid((int)Gen.g_gid)) != NULL) {
7056 (void) printf("%-9s", Curgr_p->gr_name);
7057 Lastgid = (gid_t)Gen.g_gid;
7058 } else {
7059 (void) printf("%-9d", (int)Gen.g_gid);
7060 Lastgid = (gid_t)-1;
7061 }
7062 }
7063
7064 /* print file size */
7065 if (!Aspec || ((Gen.g_mode & Ftype) == S_IFIFO) ||
7066 ((Gen.g_mode & Ftype) == S_IFSOCK) ||
7067 (Hdr_type == BAR && bar_linkflag == SYMTYPE)) {
7068 off_t filesz = Gen.g_filesz;
7069
7070 if (S_ISSPARSE(Gen.g_mode) && Gen.g_holes != NULL)
7071 filesz = Gen.g_holes->orig_size;
7072
7073 (void) printf("%*" PRId64 " ",
7074 filesz < (1L << 31) ? 7 : 11, filesz);
7075 } else {
7076 (void) printf("%3" PRId32 ",%3" PRId32 " ",
7077 major(Gen.g_rdev), minor(Gen.g_rdev));
7078 }
7079 ttime = Gen.g_mtime;
7080 (void) strftime(Time, sizeof (Time),
7081 dcgettext(NULL, FORMAT, LC_TIME), localtime(&ttime));
7082 (void) printf("%s, ", Time);
7083 str_fprintf(stdout, name_fmt, name, attribute);
7084 if ((Gen.g_mode & Ftype) == S_IFLNK) {
7085 if (Hdr_type == USTAR || Hdr_type == TAR) {
7086 (void) strcpy(Symlnk_p,
7087 Thdr_p->tbuf.t_linkname);
7088 } else {
7089 FILL(Gen.g_filesz);
7090 (void) strncpy(Symlnk_p, Buffr.b_out_p,
7091 Gen.g_filesz);
7092 *(Symlnk_p + Gen.g_filesz) = '\0';
7093 }
7094 (void) printf(" -> %s", Symlnk_p);
7095 }
7096 if (Hdr_type == BAR) {
7097 if (bar_linkflag == SYMTYPE)
7098 (void) printf(gettext(" symbolic link to %s"),
7099 bar_linkname);
7100 else if (bar_linkflag == '1')
7101 (void) printf(gettext(" linked to %s"),
7102 bar_linkname);
7103 }
7104 if ((Hdr_type == USTAR || Hdr_type == TAR) &&
7105 Thdr_p->tbuf.t_typeflag == '1') {
7106 (void) printf(gettext(" linked to %s%s%s"),
7107 (Gen.g_attrnam_p == NULL) ?
7108 Thdr_p->tbuf.t_linkname : Gen.g_attrfnam_p,
7109 (Gen.g_attrnam_p == NULL) ? "" :
7110 gettext(" attribute "),
7111 (Gen.g_attrnam_p == NULL) ?
7112 "" : Gen.g_linktoattrnam_p);
7113 }
7114 (void) printf("\n");
7115 } else if ((Args & OCt) || (Args & OCv)) {
7116 str_fprintf(Out_p, name_fmt, name, attribute);
7117 (void) fputc('\n', Out_p);
7118 } else { /* OCV */
7119 (void) fputc('.', Out_p);
7120 if (Verbcnt++ >= 49) { /* start a new line of dots */
7121 Verbcnt = 0;
7122 (void) fputc('\n', Out_p);
7123 }
7124 }
7125 (void) fflush(Out_p);
7126 }
7127
7128 /*
7129 * write_hdr: Transfer header information for the generic structure
7130 * into the format for the selected header and bwrite() the header.
7131 */
7132
7133 static void
write_hdr(int arcflag,off_t len)7134 write_hdr(int arcflag, off_t len)
7135 {
7136 int cnt = 0, pad;
7137 mode_t mode = 0;
7138 major_t maj;
7139 minor_t min;
7140 uid_t uid;
7141 gid_t gid;
7142 const char warnfmt[] = "%s%s%s : %s";
7143
7144 switch (arcflag) {
7145 case ARCHIVE_ACL:
7146 mode = SECMODE;
7147 break;
7148
7149 case ARCHIVE_XATTR:
7150 case ARCHIVE_NORMAL:
7151 /*
7152 * If attribute is being archived in cpio format then
7153 * zap off the file type bits since those are truly a
7154 * mask and reset them with _XATTR_CPIO_MODE
7155 */
7156 /*
7157 * len is the value of g_filesz for normal files
7158 * and the length of the special header buffer in
7159 * the case of acl and xattr headers.
7160 */
7161 if (G_p->g_attrnam_p != NULL && Hdr_type != USTAR &&
7162 Hdr_type != TAR) {
7163 mode = (G_p->g_mode & POSIXMODES) | _XATTR_CPIO_MODE;
7164 } else {
7165 mode = G_p->g_mode;
7166 }
7167 if (arcflag != ARCHIVE_XATTR) {
7168 len = G_p->g_filesz;
7169 }
7170 break;
7171
7172 case ARCHIVE_SPARSE:
7173 mode = G_p->g_mode | C_ISSPARSE;
7174 len = G_p->g_filesz;
7175 break;
7176 }
7177
7178 uid = G_p->g_uid;
7179 gid = G_p->g_gid;
7180 /*
7181 * Handle EFT uids and gids. If they get too big
7182 * to be represented in a particular format, force 'em to 'nobody'.
7183 */
7184 switch (Hdr_type) {
7185 case BIN: /* 16-bits of u_short */
7186 if ((uint_t)uid > (uint_t)USHRT_MAX)
7187 uid = UID_NOBODY;
7188 if ((uint_t)gid > (uint_t)USHRT_MAX)
7189 gid = GID_NOBODY;
7190 break;
7191 case CHR: /* %.6o => 262143 base 10 */
7192 if ((uint_t)uid > (uint_t)0777777)
7193 uid = UID_NOBODY;
7194 if ((uint_t)gid > (uint_t)0777777)
7195 gid = GID_NOBODY;
7196 break;
7197 case ASC: /* %.8x => full 32 bits */
7198 case CRC:
7199 break;
7200 case USTAR:
7201 case TAR: /* %.7o => 2097151 base 10 */
7202 if ((uint_t)uid > (uint_t)07777777)
7203 uid = UID_NOBODY;
7204 if ((uint_t)gid > (uint_t)07777777)
7205 gid = GID_NOBODY;
7206 break;
7207 default:
7208 msg(EXT, "Impossible header type.");
7209 }
7210
7211 /*
7212 * Since cpio formats -don't- encode the symbolic names, print
7213 * a warning message when we map the uid or gid this way.
7214 * Also, if the ownership just changed, clear set[ug]id bits
7215 *
7216 * (Except for USTAR format of course, where we have a string
7217 * representation of the username embedded in the header)
7218 */
7219 if (uid != G_p->g_uid && Hdr_type != USTAR) {
7220 msg(ERR, warnfmt,
7221 (G_p->g_attrnam_p == NULL) ?
7222 G_p->g_nam_p : G_p->g_attrfnam_p,
7223 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_rw_sysattr ?
7224 gettext(" System Attribute ") : gettext(" Attribute "),
7225 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p,
7226 gettext("uid too large for archive format"));
7227 if (S_ISREG(mode))
7228 mode &= ~S_ISUID;
7229 }
7230 if (gid != G_p->g_gid && Hdr_type != USTAR) {
7231 msg(ERR, warnfmt,
7232 (G_p->g_attrnam_p == NULL) ?
7233 G_p->g_nam_p : G_p->g_attrfnam_p,
7234 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_rw_sysattr ?
7235 gettext(" System Attribute ") : gettext(" Attribute "),
7236 (G_p->g_attrnam_p == NULL) ? "" : G_p->g_attrnam_p,
7237 gettext("gid too large for archive format"));
7238 if (S_ISREG(mode))
7239 mode &= ~S_ISGID;
7240 }
7241
7242 switch (Hdr_type) {
7243 case BIN:
7244 case CHR:
7245 case ASC:
7246 case CRC:
7247 cnt = Hdrsz + G_p->g_namesz;
7248 break;
7249 case TAR:
7250 /*FALLTHROUGH*/
7251 case USTAR: /* TAR and USTAR */
7252 cnt = TARSZ;
7253 break;
7254 default:
7255 msg(EXT, "Impossible header type.");
7256 }
7257 FLUSH(cnt);
7258
7259 switch (Hdr_type) {
7260 case BIN:
7261 Hdr.h_magic = (short)G_p->g_magic;
7262 Hdr.h_dev = G_p->g_dev;
7263 Hdr.h_ino = G_p->g_ino;
7264 Hdr.h_uid = uid;
7265 Hdr.h_gid = gid;
7266 Hdr.h_mode = mode;
7267 Hdr.h_nlink = G_p->g_nlink;
7268 maj = major(G_p->g_rdev);
7269 min = minor(G_p->g_rdev);
7270 if (maj > (uint_t)OMAXMAJ)
7271 maj = 0;
7272 if (min > (uint_t)OMAXMIN)
7273 min = 0;
7274 Hdr.h_rdev = TO_SVR3(maj, min);
7275 mkshort(Hdr.h_mtime, (long)G_p->g_mtime);
7276 Hdr.h_namesize = (short)G_p->g_namesz;
7277 mkshort(Hdr.h_filesize, (long)len);
7278 (void) strcpy(Hdr.h_name, G_p->g_nam_p);
7279 (void) memcpy(Buffr.b_in_p, &Hdr, cnt);
7280 break;
7281 case CHR:
7282 (void) sprintf(Buffr.b_in_p,
7283 "%.6o%.6o%.6o%.6o"
7284 "%.6" _PRIoID "%.6" _PRIoID "%.6o"
7285 "%.6" PRIo16
7286 "%.11o%.6o%.11" PRIo64
7287 "%s",
7288 G_p->g_magic, G_p->g_dev, G_p->g_ino, mode,
7289 uid, gid, G_p->g_nlink,
7290 (unsigned short)(G_p->g_rdev & 0xffff),
7291 G_p->g_mtime, G_p->g_namesz, len,
7292 G_p->g_nam_p);
7293 break;
7294 case ASC:
7295 case CRC:
7296 (void) sprintf(Buffr.b_in_p,
7297 "%.6x%.8x%.8x"
7298 "%.8" _PRIxID "%.8" _PRIxID
7299 "%.8x%.8x%.8x"
7300 "%.8" PRIx32 "%.8" PRIx32 "%.8" PRIx32 "%.8" PRIx32
7301 "%.8x%.8x%s",
7302 G_p->g_magic, G_p->g_ino, mode,
7303 G_p->g_uid, G_p->g_gid,
7304 G_p->g_nlink, G_p->g_mtime, (uint_t)len,
7305 major(G_p->g_dev), minor(G_p->g_dev),
7306 major(G_p->g_rdev), minor(G_p->g_rdev),
7307 G_p->g_namesz, G_p->g_cksum, G_p->g_nam_p);
7308 break;
7309 case USTAR:
7310 Thdr_p = (union tblock *)Buffr.b_in_p;
7311 (void) memset(Thdr_p, 0, TARSZ);
7312 (void) strncpy(Thdr_p->tbuf.t_name, G_p->g_tname,
7313 strlen(G_p->g_tname));
7314 (void) sprintf(Thdr_p->tbuf.t_mode, "%07o", mode);
7315 (void) sprintf(Thdr_p->tbuf.t_uid, "%07" _PRIoID, uid);
7316 (void) sprintf(Thdr_p->tbuf.t_gid, "%07" _PRIoID, gid);
7317 (void) sprintf(Thdr_p->tbuf.t_size, "%011" PRIo64, len);
7318 (void) sprintf(Thdr_p->tbuf.t_mtime, "%011o", G_p->g_mtime);
7319 if (arcflag == ARCHIVE_ACL) {
7320 Thdr_p->tbuf.t_typeflag = 'A'; /* ACL file type */
7321 } else if (arcflag == ARCHIVE_XATTR ||
7322 (G_p->g_attrnam_p != NULL)) {
7323 Thdr_p->tbuf.t_typeflag = _XATTR_HDRTYPE;
7324 } else {
7325 Thdr_p->tbuf.t_typeflag = G_p->g_typeflag;
7326 }
7327 if (T_lname[0] != '\0') {
7328 /*
7329 * if not a symbolic link
7330 */
7331 if (((G_p->g_mode & Ftype) != S_IFLNK) &&
7332 (G_p->g_attrnam_p == NULL)) {
7333 Thdr_p->tbuf.t_typeflag = LNKTYPE;
7334 (void) sprintf(Thdr_p->tbuf.t_size,
7335 "%011lo", 0L);
7336 }
7337 (void) strncpy(Thdr_p->tbuf.t_linkname, T_lname,
7338 strlen(T_lname));
7339 }
7340 (void) strcpy(Thdr_p->tbuf.t_magic, TMAGIC);
7341 (void) memcpy(Thdr_p->tbuf.t_version, TVERSION, 2);
7342 (void) strcpy(Thdr_p->tbuf.t_uname, G_p->g_uname);
7343 (void) strcpy(Thdr_p->tbuf.t_gname, G_p->g_gname);
7344 (void) sprintf(Thdr_p->tbuf.t_devmajor, "%07" PRIo32,
7345 major(G_p->g_rdev));
7346 (void) sprintf(Thdr_p->tbuf.t_devminor, "%07" PRIo32,
7347 minor(G_p->g_rdev));
7348 if (Gen.g_prefix) {
7349 (void) strcpy(Thdr_p->tbuf.t_prefix, Gen.g_prefix);
7350 free(Gen.g_prefix);
7351 Gen.g_prefix = NULL;
7352 } else {
7353 Thdr_p->tbuf.t_prefix[0] = '\0';
7354 }
7355 (void) sprintf(Thdr_p->tbuf.t_cksum, "%07o",
7356 cksum(TARTYP, 0, NULL));
7357 break;
7358 case TAR:
7359 Thdr_p = (union tblock *)Buffr.b_in_p;
7360 (void) memset(Thdr_p, 0, TARSZ);
7361 (void) strncpy(Thdr_p->tbuf.t_name, G_p->g_nam_p,
7362 G_p->g_namesz);
7363 (void) sprintf(Thdr_p->tbuf.t_mode, "%07o ", mode);
7364 (void) sprintf(Thdr_p->tbuf.t_uid, "%07" _PRIoID " ", uid);
7365 (void) sprintf(Thdr_p->tbuf.t_gid, "%07" _PRIoID " ", gid);
7366 (void) sprintf(Thdr_p->tbuf.t_size, "%011" PRIo64 " ", len);
7367 (void) sprintf(Thdr_p->tbuf.t_mtime, "%011o ", G_p->g_mtime);
7368 if (T_lname[0] != '\0') {
7369 Thdr_p->tbuf.t_typeflag = '1';
7370 } else {
7371 Thdr_p->tbuf.t_typeflag = '\0';
7372 }
7373 (void) strncpy(Thdr_p->tbuf.t_linkname, T_lname,
7374 strlen(T_lname));
7375 break;
7376 default:
7377 msg(EXT, "Impossible header type.");
7378 } /* Hdr_type */
7379
7380 Buffr.b_in_p += cnt;
7381 Buffr.b_cnt += cnt;
7382 pad = ((cnt + Pad_val) & ~Pad_val) - cnt;
7383 if (pad != 0) {
7384 FLUSH(pad);
7385 (void) memset(Buffr.b_in_p, 0, pad);
7386 Buffr.b_in_p += pad;
7387 Buffr.b_cnt += pad;
7388 }
7389 }
7390
7391 /*
7392 * write_trail: Create the appropriate trailer for the selected header type
7393 * and bwrite the trailer. Pad the buffer with nulls out to the next Bufsize
7394 * boundary, and force a write. If the write completes, or if the trailer is
7395 * completely written (but not all of the padding nulls (as can happen on end
7396 * of medium)) return. Otherwise, the trailer was not completely written out,
7397 * so re-pad the buffer with nulls and try again.
7398 */
7399
7400 static void
write_trail(void)7401 write_trail(void)
7402 {
7403 int cnt, need;
7404
7405 switch (Hdr_type) {
7406 case BIN:
7407 Gen.g_magic = CMN_BIN;
7408 break;
7409 case CHR:
7410 Gen.g_magic = CMN_BIN;
7411 break;
7412 case ASC:
7413 Gen.g_magic = CMN_ASC;
7414 break;
7415 case CRC:
7416 Gen.g_magic = CMN_CRC;
7417 break;
7418 }
7419
7420 switch (Hdr_type) {
7421 case BIN:
7422 case CHR:
7423 case ASC:
7424 case CRC:
7425 Gen.g_mode = Gen.g_uid = Gen.g_gid = 0;
7426 Gen.g_nlink = 1;
7427 Gen.g_mtime = Gen.g_ino = Gen.g_dev = 0;
7428 Gen.g_rdev = Gen.g_cksum = 0;
7429 Gen.g_filesz = (off_t)0;
7430 Gen.g_namesz = strlen("TRAILER!!!") + 1;
7431 (void) strcpy(Gen.g_nam_p, "TRAILER!!!");
7432 G_p = &Gen;
7433 write_hdr(ARCHIVE_NORMAL, (off_t)0);
7434 break;
7435 case TAR:
7436 /*FALLTHROUGH*/
7437 case USTAR: /* TAR and USTAR */
7438 for (cnt = 0; cnt < 3; cnt++) {
7439 FLUSH(TARSZ);
7440 (void) memset(Buffr.b_in_p, 0, TARSZ);
7441 Buffr.b_in_p += TARSZ;
7442 Buffr.b_cnt += TARSZ;
7443 }
7444 break;
7445 default:
7446 msg(EXT, "Impossible header type.");
7447 }
7448 need = Bufsize - (Buffr.b_cnt % Bufsize);
7449 if (need == Bufsize)
7450 need = 0;
7451
7452 while (Buffr.b_cnt > 0) {
7453 while (need > 0) {
7454 cnt = (need < TARSZ) ? need : TARSZ;
7455 need -= cnt;
7456 FLUSH(cnt);
7457 (void) memset(Buffr.b_in_p, 0, cnt);
7458 Buffr.b_in_p += cnt;
7459 Buffr.b_cnt += cnt;
7460 }
7461 bflush();
7462 }
7463 }
7464
7465 /*
7466 * if archives in USTAR format, check if typeflag == '5' for directories
7467 */
7468 static int
ustar_dir(void)7469 ustar_dir(void)
7470 {
7471 if (Hdr_type == USTAR || Hdr_type == TAR) {
7472 if (Thdr_p->tbuf.t_typeflag == '5')
7473 return (1);
7474 }
7475 return (0);
7476 }
7477
7478 /*
7479 * if archives in USTAR format, check if typeflag == '3' || '4' || '6'
7480 * for character, block, fifo special files
7481 */
7482 static int
ustar_spec(void)7483 ustar_spec(void)
7484 {
7485 int typeflag;
7486
7487 if (Hdr_type == USTAR || Hdr_type == TAR) {
7488 typeflag = Thdr_p->tbuf.t_typeflag;
7489 if (typeflag == '3' || typeflag == '4' || typeflag == '6')
7490 return (1);
7491 }
7492 return (0);
7493 }
7494
7495 /*
7496 * The return value is a pointer to a converted copy of the information in
7497 * FromStat if the file is representable in -Hodc format, and NULL otherwise.
7498 */
7499
7500 static struct stat *
convert_to_old_stat(struct stat * FromStat,char * namep,char * attrp)7501 convert_to_old_stat(struct stat *FromStat, char *namep, char *attrp)
7502 {
7503 static struct stat ToSt;
7504 cpioinfo_t TmpSt;
7505
7506 (void) memset(&TmpSt, 0, sizeof (cpioinfo_t));
7507 stat_to_svr32_stat(&TmpSt, FromStat);
7508 (void) memset(&ToSt, 0, sizeof (ToSt));
7509
7510 if (TmpSt.st_rdev == (o_dev_t)NODEV &&
7511 (((TmpSt.st_mode & Ftype) == S_IFCHR) ||
7512 ((TmpSt.st_mode & Ftype) == S_IFBLK))) {
7513 /*
7514 * Encountered a problem representing the rdev information.
7515 * Don't archive it.
7516 */
7517
7518 msg(ERR, "Error -Hodc format can't support expanded"
7519 "types on %s%s%s",
7520 namep,
7521 (attrp == NULL) ? "" : gettext(" Attribute"),
7522 (attrp == NULL) ? "" : attrp);
7523 return (NULL);
7524 }
7525
7526 if (TmpSt.st_dev == (o_dev_t)NODEV) {
7527 /*
7528 * Having trouble representing the device/inode pair. We can't
7529 * track links in this case; break them all into separate
7530 * files.
7531 */
7532
7533 TmpSt.st_ino = 0;
7534
7535 if (((TmpSt.st_mode & Ftype) != S_IFDIR) &&
7536 TmpSt.st_nlink > 1)
7537 msg(POST,
7538 "Warning: file %s%s%s has large "
7539 "device number - linked "
7540 "files will be restored as "
7541 "separate files",
7542 namep,
7543 (attrp == NULL) ? "" : gettext(" Attribute"),
7544 (attrp == NULL) ? "" : attrp);
7545
7546 /* ensure no links */
7547
7548 TmpSt.st_nlink = 1;
7549 }
7550
7551 /* Start converting values */
7552
7553 if (TmpSt.st_dev < 0) {
7554 ToSt.st_dev = 0;
7555 } else {
7556 ToSt.st_dev = (dev_t)TmpSt.st_dev;
7557 }
7558
7559 /* -actual- not truncated uid */
7560
7561 ToSt.st_uid = TmpSt.st_uid;
7562
7563 /* -actual- not truncated gid */
7564
7565 ToSt.st_gid = TmpSt.st_gid;
7566 ToSt.st_ino = (ino_t)TmpSt.st_ino;
7567 ToSt.st_mode = (mode_t)TmpSt.st_mode;
7568 ToSt.st_mtime = (uint_t)TmpSt.st_modtime;
7569 ToSt.st_nlink = (nlink_t)TmpSt.st_nlink;
7570 ToSt.st_size = (off_t)TmpSt.st_size;
7571 ToSt.st_rdev = (dev_t)TmpSt.st_rdev;
7572
7573 return (&ToSt);
7574 }
7575
7576 /*
7577 * In the beginning of each bar archive, there is a header which describes the
7578 * current volume being created, followed by a header which describes the
7579 * current file being created, followed by the file itself. If there is
7580 * more than one file to be created, a separate header will be created for
7581 * each additional file. This structure may be repeated if the bar archive
7582 * contains multiple volumes. If a file spans across volumes, its header
7583 * will not be repeated in the next volume.
7584 * +------------------+
7585 * | vol header |
7586 * |------------------|
7587 * | file header i | i = 0
7588 * |------------------|
7589 * | <file i> |
7590 * |------------------|
7591 * | file header i+1 |
7592 * |------------------|
7593 * | <file i+1> |
7594 * |------------------|
7595 * | . |
7596 * | . |
7597 * | . |
7598 * +------------------+
7599 */
7600
7601 /*
7602 * read in the header that describes the current volume of the bar archive
7603 * to be extracted.
7604 */
7605 static void
read_bar_vol_hdr(void)7606 read_bar_vol_hdr(void)
7607 {
7608 union b_block *tmp_hdr;
7609
7610 tmp_hdr = (union b_block *)Buffr.b_out_p;
7611 if (tmp_hdr->dbuf.bar_magic[0] == BAR_VOLUME_MAGIC) {
7612
7613 if (bar_Vhdr == NULL) {
7614 bar_Vhdr = e_zalloc(E_EXIT, TBLOCK);
7615 }
7616 (void) memcpy(&(bar_Vhdr->dbuf), &(tmp_hdr->dbuf), TBLOCK);
7617 } else {
7618 (void) fprintf(stderr, gettext(
7619 "bar error: cannot read volume header\n"));
7620 exit(1);
7621 }
7622
7623 (void) sscanf(bar_Vhdr->dbuf.mode, "%8l", &Gen_bar_vol.g_mode);
7624 (void) sscanf(bar_Vhdr->dbuf.uid, "%8" _SCNdID, &Gen_bar_vol.g_uid);
7625 (void) sscanf(bar_Vhdr->dbuf.gid, "%8" _SCNdID, &Gen_bar_vol.g_gid);
7626 (void) sscanf(bar_Vhdr->dbuf.size, "%12" SCNo64,
7627 (u_off_t *)&Gen_bar_vol.g_filesz);
7628 (void) sscanf(bar_Vhdr->dbuf.mtime, "%12o", &Gen_bar_vol.g_mtime);
7629 (void) sscanf(bar_Vhdr->dbuf.chksum, "%8o", &Gen_bar_vol.g_cksum);
7630
7631 /* set the compress flag */
7632 if (bar_Vhdr->dbuf.compressed == '1')
7633 Compressed = 1;
7634 else
7635 Compressed = 0;
7636
7637 Buffr.b_out_p += 512;
7638 Buffr.b_cnt -= 512;
7639
7640 /*
7641 * not the first volume; exit
7642 */
7643 if (strcmp(bar_Vhdr->dbuf.volume_num, "1") != 0) {
7644 (void) fprintf(stderr,
7645 gettext("error: This is not volume 1. "));
7646 (void) fprintf(stderr, gettext("This is volume %s. "),
7647 bar_Vhdr->dbuf.volume_num);
7648 (void) fprintf(stderr, gettext("Please insert volume 1.\n"));
7649 exit(1);
7650 }
7651
7652 read_bar_file_hdr();
7653 }
7654
7655 /*
7656 * read in the header that describes the current file to be extracted
7657 */
7658 static void
read_bar_file_hdr(void)7659 read_bar_file_hdr(void)
7660 {
7661 union b_block *tmp_hdr;
7662 char *start_of_name, *name_p;
7663 char *tmp;
7664 major_t maj;
7665 minor_t min;
7666
7667 if (*Buffr.b_out_p == '\0') {
7668 *Gen.g_nam_p = '\0';
7669 exit(0);
7670 }
7671
7672 tmp_hdr = (union b_block *)Buffr.b_out_p;
7673
7674 tmp = &tmp_hdr->dbuf.mode[1];
7675 (void) sscanf(tmp, "%8o", &Gen.g_mode);
7676 (void) sscanf(tmp_hdr->dbuf.uid, "%8" _SCNoID, &Gen.g_uid);
7677 (void) sscanf(tmp_hdr->dbuf.gid, "%8" _SCNoID, &Gen.g_gid);
7678 (void) sscanf(tmp_hdr->dbuf.size, "%12" SCNo64,
7679 (u_off_t *)&Gen.g_filesz);
7680 (void) sscanf(tmp_hdr->dbuf.mtime, "%12o", &Gen.g_mtime);
7681 (void) sscanf(tmp_hdr->dbuf.chksum, "%8o", &Gen.g_cksum);
7682 (void) sscanf(tmp_hdr->dbuf.rdev, "%8o", &Gen.g_rdev);
7683
7684 maj = SVR3_MAJOR(Gen.g_rdev);
7685 min = SVR3_MINOR(Gen.g_rdev);
7686 Gen.g_rdev = makedev(maj, min);
7687 bar_linkflag = tmp_hdr->dbuf.linkflag;
7688 start_of_name = &tmp_hdr->dbuf.start_of_name;
7689
7690
7691 name_p = Gen.g_nam_p;
7692 while ((*name_p++ = *start_of_name++) != '\0')
7693 ;
7694 *name_p = '\0';
7695 if (bar_linkflag == LNKTYPE || bar_linkflag == SYMTYPE)
7696 (void) strcpy(bar_linkname, start_of_name);
7697
7698 Gen.g_namesz = strlen(Gen.g_nam_p) + 1;
7699 (void) strcpy(nambuf, Gen.g_nam_p);
7700 }
7701
7702 /*
7703 * if the bar archive is compressed, set up a pipe and do the de-compression
7704 * as the compressed file is read in.
7705 */
7706 static void
setup_uncompress(FILE ** pipef)7707 setup_uncompress(FILE **pipef)
7708 {
7709 char *cmd_buf;
7710 size_t cmdlen;
7711
7712 cmd_buf = e_zalloc(E_EXIT, MAXPATHLEN * 2);
7713
7714 if (access(Gen.g_nam_p, W_OK) != 0) {
7715 cmdlen = snprintf(cmd_buf, MAXPATHLEN * 2,
7716 "chmod +w '%s'; uncompress -c > '%s'; "
7717 "chmod 0%o '%s'",
7718 Gen.g_nam_p, Gen.g_nam_p, G_p->g_mode, Gen.g_nam_p);
7719 } else {
7720 cmdlen = snprintf(cmd_buf, MAXPATHLEN * 2,
7721 "uncompress -c > '%s'", Gen.g_nam_p);
7722 }
7723
7724 if (cmdlen >= MAXPATHLEN * 2 ||
7725 (*pipef = popen(cmd_buf, "w")) == NULL) {
7726 (void) fprintf(stderr, gettext("error\n"));
7727 exit(1);
7728 }
7729
7730 if (close(Ofile) != 0)
7731 msg(EXTN, "close error");
7732 Ofile = fileno(*pipef);
7733
7734 free(cmd_buf);
7735 }
7736
7737 /*
7738 * if the bar archive spans multiple volumes, read in the header that
7739 * describes the next volume.
7740 */
7741 static void
skip_bar_volhdr(void)7742 skip_bar_volhdr(void)
7743 {
7744 char *buff;
7745 union b_block *tmp_hdr;
7746
7747 buff = e_zalloc(E_EXIT, (uint_t)Bufsize);
7748
7749 if (g_read(Device, Archive, buff, Bufsize) < 0) {
7750 (void) fprintf(stderr, gettext(
7751 "error in skip_bar_volhdr\n"));
7752 } else {
7753
7754 tmp_hdr = (union b_block *)buff;
7755 if (tmp_hdr->dbuf.bar_magic[0] == BAR_VOLUME_MAGIC) {
7756
7757 if (bar_Vhdr == NULL) {
7758 bar_Vhdr = e_zalloc(E_EXIT, TBLOCK);
7759 }
7760 (void) memcpy(&(bar_Vhdr->dbuf),
7761 &(tmp_hdr->dbuf), TBLOCK);
7762 } else {
7763 (void) fprintf(stderr,
7764 gettext("cpio error: cannot read bar volume "
7765 "header\n"));
7766 exit(1);
7767 }
7768
7769 (void) sscanf(bar_Vhdr->dbuf.mode, "%8o",
7770 &Gen_bar_vol.g_mode);
7771 (void) sscanf(bar_Vhdr->dbuf.uid, "%8" _SCNoID,
7772 &Gen_bar_vol.g_uid);
7773 (void) sscanf(bar_Vhdr->dbuf.gid, "%8" _SCNoID,
7774 &Gen_bar_vol.g_gid);
7775 (void) sscanf(bar_Vhdr->dbuf.size, "%12" SCNo64,
7776 (u_off_t *)&Gen_bar_vol.g_filesz);
7777 (void) sscanf(bar_Vhdr->dbuf.mtime, "%12o",
7778 &Gen_bar_vol.g_mtime);
7779 (void) sscanf(bar_Vhdr->dbuf.chksum, "%8o",
7780 &Gen_bar_vol.g_cksum);
7781 if (bar_Vhdr->dbuf.compressed == '1')
7782 Compressed = 1;
7783 else
7784 Compressed = 0;
7785 }
7786
7787 /*
7788 * Now put the rest of the bytes read in into the data buffer.
7789 */
7790 (void) memcpy(Buffr.b_in_p, &buff[512], (Bufsize - 512));
7791 Buffr.b_in_p += (Bufsize - 512);
7792 Buffr.b_cnt += (long)(Bufsize - 512);
7793
7794 free(buff);
7795 }
7796
7797 /*
7798 * check the linkflag which indicates the type of the file to be extracted,
7799 * invoke the corresponding routine to extract the file.
7800 */
7801 static void
bar_file_in(void)7802 bar_file_in(void)
7803 {
7804 /*
7805 * the file is a directory
7806 */
7807 if (Adir) {
7808 if (ckname(1) != F_SKIP && creat_spec(G_p->g_dirfd) > 0) {
7809 VERBOSE((Args & (OCv | OCV)), G_p->g_nam_p);
7810 }
7811 return;
7812 }
7813
7814 switch (bar_linkflag) {
7815 case REGTYPE:
7816 /* regular file */
7817 if ((ckname(1) == F_SKIP) ||
7818 (Ofile = openout(G_p->g_dirfd)) < 0) {
7819 data_in(P_SKIP);
7820 } else {
7821 data_in(P_PROC);
7822 }
7823 break;
7824 case LNKTYPE:
7825 /* hard link */
7826 if (ckname(1) == F_SKIP) {
7827 break;
7828 }
7829 (void) creat_lnk(G_p->g_dirfd, bar_linkname, G_p->g_nam_p);
7830 break;
7831 case SYMTYPE:
7832 /* symbolic link */
7833 if ((ckname(1) == F_SKIP) ||
7834 (Ofile = openout(G_p->g_dirfd)) < 0) {
7835 data_in(P_SKIP);
7836 } else {
7837 data_in(P_PROC);
7838 }
7839 break;
7840 case CHRTYPE:
7841 /* character device or FIFO */
7842 if (ckname(1) != F_SKIP && creat_spec(G_p->g_dirfd) > 0) {
7843 VERBOSE((Args & (OCv | OCV)), G_p->g_nam_p);
7844 }
7845 break;
7846 default:
7847 (void) fprintf(stderr, gettext("error: unknown file type\n"));
7848 break;
7849 }
7850 }
7851
7852
7853 /*
7854 * This originally came from libgenIO/g_init.c
7855 * XXX And it is very broken.
7856 */
7857
7858 /* #include <sys/statvfs.h> */
7859 #include <ftw.h>
7860 /* #include <libgenIO.h> */
7861 #define G_TM_TAPE 1 /* Tapemaster controller */
7862 #define G_XY_DISK 3 /* xy disks */
7863 #define G_SD_DISK 7 /* scsi sd disk */
7864 #define G_XT_TAPE 8 /* xt tapes */
7865 #define G_SF_FLOPPY 9 /* sf floppy */
7866 #define G_XD_DISK 10 /* xd disks */
7867 #define G_ST_TAPE 11 /* scsi tape */
7868 #define G_NS 12 /* noswap pseudo-dev */
7869 #define G_RAM 13 /* ram pseudo-dev */
7870 #define G_FT 14 /* tftp */
7871 #define G_HD 15 /* 386 network disk */
7872 #define G_FD 16 /* 386 AT disk */
7873 #define G_FILE 28 /* file, not a device */
7874 #define G_NO_DEV 29 /* device does not require special treatment */
7875 #define G_DEV_MAX 30 /* last valid device type */
7876
7877 /*
7878 * g_init: Determine the device being accessed, set the buffer size,
7879 * and perform any device specific initialization. Since at this point
7880 * Sun has no system call to read the configuration, the major numbers
7881 * are assumed to be static and types are figured out as such. However,
7882 * as a rough estimate, the buffer size for all types is set to 512
7883 * as a default.
7884 */
7885
7886 static int
g_init(int * devtype,int * fdes)7887 g_init(int *devtype, int *fdes)
7888 {
7889 int bufsize;
7890 struct stat st_buf;
7891 struct statvfs stfs_buf;
7892
7893 *devtype = G_NO_DEV;
7894 bufsize = -1;
7895 if (fstat(*fdes, &st_buf) == -1)
7896 return (-1);
7897 if (!S_ISCHR(st_buf.st_mode) && !S_ISBLK(st_buf.st_mode)) {
7898 if (S_ISFIFO(st_buf.st_mode)) {
7899 bufsize = 512;
7900 } else {
7901 /* find block size for this file system */
7902 *devtype = G_FILE;
7903 if (fstatvfs(*fdes, &stfs_buf) < 0) {
7904 bufsize = -1;
7905 errno = ENODEV;
7906 } else
7907 bufsize = stfs_buf.f_bsize;
7908 }
7909
7910 return (bufsize);
7911
7912 /*
7913 * We'll have to add a remote attribute to stat but this
7914 * should work for now.
7915 */
7916 } else if (st_buf.st_dev & 0x8000) /* if remote rdev */
7917 return (512);
7918
7919 bufsize = 512;
7920
7921 if (Hdr_type == BAR) {
7922 if (is_tape(*fdes)) {
7923 bufsize = BAR_TAPE_SIZE;
7924 msg(EPOST, "Archiving to tape blocking factor 126");
7925 } else if (is_floppy(*fdes)) {
7926 bufsize = BAR_FLOPPY_SIZE;
7927 msg(EPOST, "Archiving to floppy blocking factor 18");
7928 }
7929 }
7930
7931 return (bufsize);
7932 }
7933
7934 /*
7935 * This originally came from libgenIO/g_read.c
7936 */
7937
7938 /*
7939 * g_read: Read nbytes of data from fdes (of type devtype) and place
7940 * data in location pointed to by buf. In case of end of medium,
7941 * translate (where necessary) device specific EOM indications into
7942 * the generic EOM indication of rv = -1, errno = ENOSPC.
7943 */
7944
7945 static int
g_read(int devtype,int fdes,char * buf,unsigned nbytes)7946 g_read(int devtype, int fdes, char *buf, unsigned nbytes)
7947 {
7948 int rv;
7949
7950 if (devtype < 0 || devtype >= G_DEV_MAX) {
7951 errno = ENODEV;
7952 return (-1);
7953 }
7954
7955 rv = read(fdes, buf, nbytes);
7956
7957 /* st devices return 0 when no space left */
7958 if ((rv == 0 && errno == 0 && Hdr_type != BAR) ||
7959 (rv == -1 && errno == EIO)) {
7960 errno = 0;
7961 rv = 0;
7962 }
7963
7964 return (rv);
7965 }
7966
7967 /*
7968 * This originally came from libgenIO/g_write.c
7969 */
7970
7971 /*
7972 * g_write: Write nbytes of data to fdes (of type devtype) from
7973 * the location pointed to by buf. In case of end of medium,
7974 * translate (where necessary) device specific EOM indications into
7975 * the generic EOM indication of rv = -1, errno = ENOSPC.
7976 */
7977
7978 static int
g_write(int devtype,int fdes,char * buf,unsigned nbytes)7979 g_write(int devtype, int fdes, char *buf, unsigned nbytes)
7980 {
7981 int rv;
7982
7983 if (devtype < 0 || devtype >= G_DEV_MAX) {
7984 errno = ENODEV;
7985 return (-1);
7986 }
7987
7988 rv = write(fdes, buf, nbytes);
7989
7990 /* st devices return 0 when no more space left */
7991 if ((rv == 0 && errno == 0) || (rv == -1 && errno == EIO)) {
7992 errno = ENOSPC;
7993 rv = -1;
7994 }
7995
7996 return (rv);
7997 }
7998
7999 /*
8000 * Test for tape
8001 */
8002
8003 static int
is_tape(int fd)8004 is_tape(int fd)
8005 {
8006 struct mtget stuff;
8007
8008 /*
8009 * try to do a generic tape ioctl, just to see if
8010 * the thing is in fact a tape drive(er).
8011 */
8012 if (ioctl(fd, MTIOCGET, &stuff) != -1) {
8013 /* the ioctl succeeded, must have been a tape */
8014 return (1);
8015 }
8016 return (0);
8017 }
8018
8019 /*
8020 * Test for floppy
8021 */
8022
8023 static int
is_floppy(int fd)8024 is_floppy(int fd)
8025 {
8026 struct fd_char stuff;
8027
8028 /*
8029 * try to get the floppy drive characteristics, just to see if
8030 * the thing is in fact a floppy drive(er).
8031 */
8032 if (ioctl(fd, FDIOGCHAR, &stuff) != -1) {
8033 /* the ioctl succeeded, must have been a floppy */
8034 return (1);
8035 }
8036
8037 return (0);
8038 }
8039
8040 /*
8041 * New functions for ACLs and other security attributes
8042 */
8043
8044 /*
8045 * The function appends the new security attribute info to the end of
8046 * existing secinfo.
8047 */
8048 static int
append_secattr(char ** secinfo,int * secinfo_len,acl_t * aclp)8049 append_secattr(char **secinfo, /* existing security info */
8050 int *secinfo_len, /* length of existing security info */
8051 acl_t *aclp) /* new attribute data pointer */
8052 {
8053 char *new_secinfo;
8054 char *attrtext;
8055 size_t newattrsize;
8056 int oldsize;
8057
8058 /* no need to add */
8059 if (aclp == NULL) {
8060 return (0);
8061 }
8062
8063 switch (acl_type(aclp)) {
8064 case ACLENT_T:
8065 case ACE_T:
8066 attrtext = acl_totext(aclp, ACL_APPEND_ID | ACL_COMPACT_FMT |
8067 ACL_SID_FMT);
8068 if (attrtext == NULL) {
8069 msg(EPOST, "acltotext failed");
8070 return (-1);
8071 }
8072 /* header: type + size = 8 */
8073 newattrsize = 8 + strlen(attrtext) + 1;
8074 attr = e_zalloc(E_NORMAL, newattrsize);
8075 if (attr == NULL) {
8076 msg(EPOST, "can't allocate memory");
8077 return (-1);
8078 }
8079 attr->attr_type = (acl_type(aclp) == ACLENT_T) ?
8080 UFSD_ACL : ACE_ACL;
8081 /* acl entry count */
8082 (void) sprintf(attr->attr_len, "%06o", acl_cnt(aclp));
8083 (void) strcpy((char *)&attr->attr_info[0], attrtext);
8084 free(attrtext);
8085 break;
8086
8087 /* SunFed's case goes here */
8088
8089 default:
8090 msg(EPOST, "unrecognized attribute type");
8091 return (-1);
8092 }
8093
8094 /* old security info + new attr header(8) + new attr */
8095 oldsize = *secinfo_len;
8096 *secinfo_len += newattrsize;
8097 new_secinfo = e_zalloc(E_NORMAL, (uint_t)*secinfo_len);
8098 if (new_secinfo == NULL) {
8099 msg(EPOST, "can't allocate memory");
8100 *secinfo_len -= newattrsize;
8101 return (-1);
8102 }
8103
8104 (void) memcpy(new_secinfo, *secinfo, oldsize);
8105 (void) memcpy(new_secinfo + oldsize, attr, newattrsize);
8106
8107 free(*secinfo);
8108 *secinfo = new_secinfo;
8109 return (0);
8110 }
8111
8112 /*
8113 * Append size amount of data from buf to the archive.
8114 */
8115 static void
write_ancillary(char * buf,size_t len,boolean_t padding)8116 write_ancillary(char *buf, size_t len, boolean_t padding)
8117 {
8118 int pad, cnt;
8119
8120 if (len == 0)
8121 return;
8122
8123 while (len > 0) {
8124 cnt = (unsigned)(len > CPIOBSZ) ? CPIOBSZ : len;
8125 FLUSH(cnt);
8126 errno = 0;
8127 (void) memcpy(Buffr.b_in_p, buf, (unsigned)cnt);
8128 Buffr.b_in_p += cnt;
8129 Buffr.b_cnt += cnt;
8130 len -= cnt;
8131 buf += cnt;
8132 }
8133 if (padding) {
8134 pad = (Pad_val + 1 - (cnt & Pad_val)) & Pad_val;
8135 if (pad != 0) {
8136 FLUSH(pad);
8137 (void) memset(Buffr.b_in_p, 0, pad);
8138 Buffr.b_in_p += pad;
8139 Buffr.b_cnt += pad;
8140 }
8141 }
8142 }
8143
8144 static int
remove_dir(char * path)8145 remove_dir(char *path)
8146 {
8147 DIR *name;
8148 struct dirent *direct;
8149 struct stat sbuf;
8150 char *path_copy;
8151
8152 #define MSG1 "remove_dir() failed to stat(\"%s\") "
8153 #define MSG2 "remove_dir() failed to remove_dir(\"%s\") "
8154 #define MSG3 "remove_dir() failed to unlink(\"%s\") "
8155
8156 /*
8157 * Open the directory for reading.
8158 */
8159 if ((name = opendir(path)) == NULL) {
8160 msg(ERRN, "remove_dir() failed to opendir(\"%s\") ", path);
8161 return (-1);
8162 }
8163
8164 if (chdir(path) == -1) {
8165 msg(ERRN, "remove_dir() failed to chdir(\"%s\") ", path);
8166 return (-1);
8167 }
8168
8169 /*
8170 * Read every directory entry.
8171 */
8172 while ((direct = readdir(name)) != NULL) {
8173 /*
8174 * Ignore "." and ".." entries.
8175 */
8176 if (strcmp(direct->d_name, ".") == 0 ||
8177 strcmp(direct->d_name, "..") == 0)
8178 continue;
8179
8180 if (lstat(direct->d_name, &sbuf) == -1) {
8181 msg(ERRN, MSG1, direct->d_name);
8182 (void) closedir(name);
8183 return (-1);
8184 }
8185
8186 if (S_ISDIR(sbuf.st_mode)) {
8187 if (remove_dir(direct->d_name) == -1) {
8188 msg(ERRN, MSG2, direct->d_name);
8189 (void) closedir(name);
8190 return (-1);
8191 }
8192 } else {
8193 if (unlink(direct->d_name) == -1) {
8194 msg(ERRN, MSG3, direct->d_name);
8195 (void) closedir(name);
8196 return (-1);
8197 }
8198 }
8199
8200 }
8201
8202 /*
8203 * Close the directory we just finished reading.
8204 */
8205 (void) closedir(name);
8206
8207 /*
8208 * Change directory to the parent directory...
8209 */
8210 if (chdir("..") == -1) {
8211 msg(ERRN, "remove_dir() failed to chdir(\"..\") ");
8212 return (-1);
8213 }
8214
8215 /*
8216 * ...and finally remove the directory; note we have to
8217 * make a copy since basename is free to modify its input.
8218 */
8219 path_copy = e_strdup(E_NORMAL, path);
8220 if (path_copy == NULL) {
8221 msg(ERRN, "cannot strdup() the directory pathname ");
8222 return (-1);
8223 }
8224
8225 if (rmdir(basename(path_copy)) == -1) {
8226 free(path_copy);
8227 msg(ERRN, "remove_dir() failed to rmdir(\"%s\") ", path);
8228 return (-1);
8229 }
8230
8231 free(path_copy);
8232 return (0);
8233
8234 }
8235
8236 static int
save_cwd(void)8237 save_cwd(void)
8238 {
8239 return (open(".", O_RDONLY));
8240 }
8241
8242 static void
rest_cwd(int cwd)8243 rest_cwd(int cwd)
8244 {
8245 (void) fchdir(cwd);
8246 (void) close(cwd);
8247 }
8248
8249 #if defined(O_XATTR)
8250 static void
xattrs_out(int (* func)())8251 xattrs_out(int (*func)())
8252 {
8253 int dirpfd;
8254 int filefd;
8255 int arc_rwsysattr = 0;
8256 int rw_sysattr = 0;
8257 int ext_attr = 0;
8258 DIR *dirp;
8259 struct dirent *dp;
8260 int slen;
8261 int plen;
8262 char *namep, *savenamep;
8263 char *apathp;
8264 char *attrparent = Gen.g_attrparent_p;
8265 char *filename;
8266
8267 if (attrparent == NULL) {
8268 filename = Gen.g_nam_p;
8269 } else {
8270 filename = Gen.g_attrnam_p;
8271 }
8272
8273 /*
8274 * If the underlying file system supports it, then
8275 * archive the extended attributes if -@ was specified,
8276 * and the extended system attributes if -/ was
8277 * specified.
8278 */
8279 if (verify_attr_support(filename, (attrparent == NULL), ARC_CREATE,
8280 &ext_attr) != ATTR_OK) {
8281 return;
8282 }
8283
8284 #if defined(_PC_SATTR_ENABLED)
8285 if (SysAtflag) {
8286 int filefd;
8287 nvlist_t *slist = NULL;
8288
8289 /*
8290 * Determine if there are non-transient system
8291 * attributes.
8292 */
8293 errno = 0;
8294 if ((filefd = open(filename, O_RDONLY)) == -1) {
8295 if (attrparent == NULL) {
8296 msg(EXTN,
8297 "unable to open %s%s%sfile %s",
8298 (attrparent == NULL) ? "" :
8299 gettext("attribute "),
8300 (attrparent == NULL) ? "" : attrparent,
8301 (attrparent == NULL) ? "" : gettext(" of "),
8302 (attrparent == NULL) ? G_p->g_nam_p :
8303 G_p->g_attrfnam_p);
8304 }
8305 }
8306 if (((slist = sysattr_list(myname, filefd,
8307 filename)) != NULL) || (errno != 0)) {
8308 arc_rwsysattr = 1;
8309 }
8310 if (slist != NULL) {
8311 (void) nvlist_free(slist);
8312 slist = NULL;
8313 }
8314 (void) close(filefd);
8315 }
8316
8317 /*
8318 * If we aren't archiving extended system attributes, and we are
8319 * processing an attribute, or if we are archiving extended system
8320 * attributes, and there are are no extended attributes, then there's
8321 * no need to open up the attribute directory of the file unless the
8322 * extended system attributes are not transient (i.e, the system
8323 * attributes are not the default values).
8324 */
8325 if ((arc_rwsysattr == 0) && ((attrparent != NULL) ||
8326 (SysAtflag && !ext_attr))) {
8327 return;
8328 }
8329
8330 #endif /* _PC_SATTR_ENABLED */
8331
8332 /*
8333 * If aclp still exists then free it since it is was set when base
8334 * file was extracted.
8335 */
8336 if (aclp != NULL) {
8337 acl_free(aclp);
8338 aclp = NULL;
8339 acl_is_set = 0;
8340 }
8341
8342 Gen.g_dirfd = attropen(filename, ".", O_RDONLY);
8343 if (Gen.g_dirfd == -1) {
8344 msg(ERRN, "Cannot open attribute directory of file \"%s%s%s\"",
8345 (attrparent == NULL) ? "" : gettext("attribute "),
8346 (attrparent == NULL) ? "" : attrparent,
8347 (attrparent == NULL) ? "" : gettext(" of "), filename);
8348 return;
8349
8350 }
8351
8352 if (attrparent == NULL) {
8353 savenamep = G_p->g_nam_p;
8354 } else {
8355 savenamep = G_p->g_attrfnam_p;
8356 }
8357
8358 if ((dirpfd = dup(Gen.g_dirfd)) == -1) {
8359 msg(ERRN, "Cannot dup(2) attribute directory descriptor");
8360 return;
8361 }
8362
8363 if ((dirp = fdopendir(dirpfd)) == NULL) {
8364 msg(ERRN, "Cannot fdopendir(3C) directory file descriptor");
8365 return;
8366 }
8367
8368 if (attrparent == NULL) {
8369 Gen.g_baseparent_fd = save_cwd();
8370 }
8371
8372 while ((dp = readdir(dirp)) != NULL) {
8373 if (strcmp(dp->d_name, "..") == 0) {
8374 continue;
8375 }
8376 if (verify_attr(dp->d_name, attrparent,
8377 arc_rwsysattr, &rw_sysattr) != ATTR_OK) {
8378 continue;
8379 }
8380
8381 if (strcmp(dp->d_name, ".") == 0) {
8382 Hiddendir = 1;
8383 } else {
8384 Hiddendir = 0;
8385 }
8386
8387 Gen.g_rw_sysattr = rw_sysattr;
8388 Gen.g_attrnam_p = dp->d_name;
8389
8390 if (STAT(Gen.g_dirfd, Gen.g_nam_p, &SrcSt) == -1) {
8391 msg(ERRN,
8392 "Could not fstatat(2) attribute \"%s\" of"
8393 " file \"%s\"", dp->d_name, (attrparent == NULL) ?
8394 savenamep : Gen.g_attrfnam_p);
8395 continue;
8396 }
8397
8398 if (Use_old_stat) {
8399 Savedev = SrcSt.st_dev;
8400 OldSt = convert_to_old_stat(&SrcSt,
8401 Gen.g_nam_p, Gen.g_attrnam_p);
8402
8403 if (OldSt == NULL) {
8404 msg(ERRN,
8405 "Could not convert to old stat format");
8406 continue;
8407 }
8408 }
8409
8410 Gen.g_attrfnam_p = savenamep;
8411
8412 /*
8413 * Set up dummy header name
8414 *
8415 * One piece is written with .hdr, which
8416 * contains the actual xattr hdr or pathing information
8417 * then the name is updated to drop the .hdr off
8418 * and the actual file itself is archived.
8419 */
8420 slen = strlen(Gen.g_attrnam_p) + strlen(DEVNULL) +
8421 strlen(XATTRHDR) + 2; /* add one for '/' */
8422 if ((namep = e_zalloc(E_NORMAL, slen)) == NULL) {
8423 msg(ERRN, "Could not calloc memory for attribute name");
8424 continue;
8425 }
8426 (void) snprintf(namep, slen, "%s/%s%s",
8427 DEVNULL, Gen.g_attrnam_p, XATTRHDR);
8428 Gen.g_nam_p = namep;
8429
8430 plen = strlen(Gen.g_attrnam_p) + 1;
8431 if (Gen.g_attrparent_p != NULL) {
8432 plen += strlen(Gen.g_attrparent_p) + 1;
8433 }
8434 if ((apathp = e_zalloc(E_NORMAL, plen)) == NULL) {
8435 msg(ERRN, "Could not calloc memory for attribute name");
8436 continue;
8437 }
8438 (void) snprintf(apathp, plen, "%s%s%s",
8439 (Gen.g_attrparent_p == NULL) ? "" : Gen.g_attrparent_p,
8440 (Gen.g_attrparent_p == NULL) ? "" : "/", Gen.g_attrnam_p);
8441
8442 if (Gen.g_attrpath_p != NULL) {
8443 free(Gen.g_attrpath_p);
8444 }
8445 Gen.g_attrpath_p = apathp;
8446
8447 /*
8448 * Get attribute's ACL info: don't bother allocating space
8449 * if there are only standard permissions, i.e. ACL count < 4
8450 */
8451 if (Pflag) {
8452 filefd = openat(Gen.g_dirfd, dp->d_name, O_RDONLY);
8453 if (filefd == -1) {
8454 msg(ERRN,
8455 "Could not open attribute \"%s\" of"
8456 " file \"%s\"", dp->d_name, savenamep);
8457 free(namep);
8458 continue;
8459 }
8460 if (facl_get(filefd, ACL_NO_TRIVIAL, &aclp) != 0) {
8461 msg(ERRN,
8462 "Error with acl() on %s",
8463 Gen.g_nam_p);
8464 }
8465 (void) close(filefd);
8466 }
8467
8468 (void) creat_hdr();
8469 (void) (*func)();
8470
8471 #if defined(_PC_SATTR_ENABLED)
8472 /*
8473 * Recursively call xattrs_out() to process the attribute's
8474 * hidden attribute directory and read-write system attributes.
8475 */
8476 if (SysAtflag && !Hiddendir && !rw_sysattr) {
8477 int savedirfd = Gen.g_dirfd;
8478
8479 (void) fchdir(Gen.g_dirfd);
8480 Gen.g_attrparent_p = dp->d_name;
8481 xattrs_out(func);
8482 Gen.g_dirfd = savedirfd;
8483 Gen.g_attrparent_p = NULL;
8484 }
8485 #endif /* _PC_SATTR_ENABLED */
8486
8487 if (Gen.g_passdirfd != -1) {
8488 (void) close(Gen.g_passdirfd);
8489 Gen.g_passdirfd = -1;
8490 }
8491 Gen.g_attrnam_p = NULL;
8492 Gen.g_attrfnam_p = NULL;
8493 Gen.g_linktoattrfnam_p = NULL;
8494 Gen.g_linktoattrnam_p = NULL;
8495 Gen.g_rw_sysattr = 0;
8496 if (Gen.g_attrpath_p != NULL) {
8497 free(Gen.g_attrpath_p);
8498 Gen.g_attrpath_p = NULL;
8499 }
8500
8501 if (aclp != NULL) {
8502 acl_free(aclp);
8503 aclp = NULL;
8504 acl_is_set = 0;
8505 }
8506 free(namep);
8507 }
8508
8509 (void) closedir(dirp);
8510 (void) close(Gen.g_dirfd);
8511 if (attrparent == NULL) {
8512 rest_cwd(Gen.g_baseparent_fd);
8513 Gen.g_dirfd = -1;
8514 }
8515 Hiddendir = 0;
8516 }
8517 #else
8518 static void
xattrs_out(int (* func)())8519 xattrs_out(int (*func)())
8520 {
8521 }
8522 #endif
8523
8524 /*
8525 * Return the parent directory of a given path.
8526 *
8527 * Examples:
8528 * /usr/tmp return /usr
8529 * /usr/tmp/file return /usr/tmp
8530 * / returns .
8531 * /usr returns /
8532 * file returns .
8533 *
8534 * dir is assumed to be at least as big as path.
8535 */
8536 static void
get_parent(char * path,char * dir)8537 get_parent(char *path, char *dir)
8538 {
8539 char *s;
8540 char tmpdir[PATH_MAX + 1];
8541
8542 if (strlen(path) > PATH_MAX) {
8543 msg(EXT, "pathname is too long");
8544 }
8545 (void) strcpy(tmpdir, path);
8546 chop_endslashes(tmpdir);
8547
8548 if ((s = strrchr(tmpdir, '/')) == NULL) {
8549 (void) strcpy(dir, ".");
8550 } else {
8551 s = skipslashes(s, tmpdir);
8552 *s = '\0';
8553 if (s == tmpdir)
8554 (void) strcpy(dir, "/");
8555 else
8556 (void) strcpy(dir, tmpdir);
8557 }
8558 }
8559
8560 #if defined(O_XATTR)
8561 #define ROUNDTOTBLOCK(a) ((a + (TBLOCK -1)) & ~(TBLOCK -1))
8562
8563 static void
prepare_xattr_hdr(char ** attrbuf,char * filename,char * attrpath,char typeflag,struct Lnk * linkinfo,int * rlen)8564 prepare_xattr_hdr(
8565 char **attrbuf,
8566 char *filename,
8567 char *attrpath,
8568 char typeflag,
8569 struct Lnk *linkinfo,
8570 int *rlen)
8571 {
8572 char *bufhead; /* ptr to full buffer */
8573 char *aptr;
8574 struct xattr_hdr *hptr; /* ptr to header in bufhead */
8575 struct xattr_buf *tptr; /* ptr to pathing pieces */
8576 int totalen; /* total buffer length */
8577 int len; /* length returned to user */
8578 int stringlen; /* length of filename + attr */
8579 /*
8580 * length of filename + attr
8581 * in link section
8582 */
8583 int linkstringlen = 0;
8584 int complen; /* length of pathing section */
8585 int linklen; /* length of link section */
8586 int attrnames_index; /* attrnames starting index */
8587
8588 /*
8589 * Release previous buffer if any.
8590 */
8591
8592 if (*attrbuf != NULL) {
8593 free(*attrbuf);
8594 *attrbuf = NULL;
8595 }
8596
8597 /*
8598 * First add in fixed size stuff
8599 */
8600 len = sizeof (struct xattr_hdr) + sizeof (struct xattr_buf);
8601
8602 /*
8603 * Add space for two nulls
8604 */
8605 stringlen = strlen(attrpath) + strlen(filename) + 2;
8606 complen = stringlen + sizeof (struct xattr_buf);
8607
8608 len += stringlen;
8609
8610 /*
8611 * Now add on space for link info if any
8612 */
8613
8614 if (linkinfo != NULL) {
8615 /*
8616 * Again add space for two nulls
8617 */
8618 linkstringlen = strlen(linkinfo->L_gen.g_attrfnam_p) +
8619 strlen(linkinfo->L_gen.g_attrnam_p) + 2;
8620 linklen = linkstringlen + sizeof (struct xattr_buf);
8621 len += linklen;
8622 } else {
8623 linklen = 0;
8624 }
8625
8626 /*
8627 * Now add padding to end to fill out TBLOCK
8628 *
8629 * Function returns size of real data and not size + padding.
8630 */
8631
8632 totalen = ROUNDTOTBLOCK(len);
8633 bufhead = e_zalloc(E_EXIT, totalen);
8634
8635 /*
8636 * Now we can fill in the necessary pieces
8637 */
8638
8639 /*
8640 * first fill in the fixed header
8641 */
8642 hptr = (struct xattr_hdr *)bufhead;
8643 (void) strcpy(hptr->h_version, XATTR_ARCH_VERS);
8644 (void) sprintf(hptr->h_component_len, "%0*d",
8645 sizeof (hptr->h_component_len) - 1, complen);
8646 (void) sprintf(hptr->h_link_component_len, "%0*d",
8647 sizeof (hptr->h_link_component_len) - 1, linklen);
8648 (void) sprintf(hptr->h_size, "%0*d", sizeof (hptr->h_size) - 1, len);
8649
8650 /*
8651 * Now fill in the filename + attrnames section
8652 * The filename and attrnames section can be composed of two or more
8653 * path segments separated by a null character. The first segment
8654 * is the path to the parent file that roots the entire sequence in
8655 * the normal name space. The remaining segments describes a path
8656 * rooted at the hidden extended attribute directory of the leaf file of
8657 * the previous segment, making it possible to name attributes on
8658 * attributes. Thus, if we are just archiving an extended attribute,
8659 * the second segment will contain the attribute name. If we are
8660 * archiving a system attribute of an extended attribute, then the
8661 * second segment will contain the attribute name, and a third segment
8662 * will contain the system attribute name. The attribute pathing
8663 * information is obtained from 'attrpath'.
8664 */
8665
8666 tptr = (struct xattr_buf *)(bufhead + sizeof (struct xattr_hdr));
8667 (void) sprintf(tptr->h_namesz, "%0*d", sizeof (tptr->h_namesz) - 1,
8668 stringlen);
8669 (void) strcpy(tptr->h_names, filename);
8670 attrnames_index = strlen(filename) + 1;
8671 (void) strcpy(&tptr->h_names[attrnames_index], attrpath);
8672 tptr->h_typeflag = typeflag;
8673
8674 /*
8675 * Split the attrnames section into two segments if 'attrpath'
8676 * contains pathing information for a system attribute of an
8677 * extended attribute. We split them by replacing the '/' with
8678 * a '\0'.
8679 */
8680 if ((aptr = strpbrk(&tptr->h_names[attrnames_index], "/")) != NULL) {
8681 *aptr = '\0';
8682 }
8683
8684 /*
8685 * Now fill in the optional link section if we have one
8686 */
8687
8688 if (linkinfo != NULL) {
8689 tptr = (struct xattr_buf *)(bufhead +
8690 sizeof (struct xattr_hdr) + complen);
8691
8692 (void) sprintf(tptr->h_namesz, "%0*d",
8693 sizeof (tptr->h_namesz) - 1, linkstringlen);
8694 (void) strcpy(tptr->h_names, linkinfo->L_gen.g_attrfnam_p);
8695 (void) strcpy(
8696 &tptr->h_names[strlen(linkinfo->L_gen.g_attrfnam_p) + 1],
8697 linkinfo->L_gen.g_attrnam_p);
8698 tptr->h_typeflag = typeflag;
8699 }
8700 *attrbuf = (char *)bufhead;
8701 *rlen = len;
8702 }
8703 #endif /* O_XATTR */
8704
8705 static char
tartype(int type)8706 tartype(int type)
8707 {
8708 switch (type) {
8709
8710 case S_IFDIR:
8711 return (DIRTYPE);
8712
8713 case S_IFLNK:
8714 return (SYMTYPE);
8715
8716 case S_IFIFO:
8717 return (FIFOTYPE);
8718
8719 case S_IFCHR:
8720 return (CHRTYPE);
8721
8722 case S_IFBLK:
8723 return (BLKTYPE);
8724
8725 case S_IFREG:
8726 return (REGTYPE);
8727
8728 default:
8729 return ('\0');
8730 }
8731 }
8732
8733 #if defined(O_XATTR)
8734 static int
openfile(int omode)8735 openfile(int omode)
8736 {
8737 if (G_p->g_attrnam_p != NULL) {
8738 return (openat(G_p->g_dirfd, G_p->g_attrnam_p, omode));
8739 } else {
8740 return (openat(G_p->g_dirfd,
8741 get_component(G_p->g_nam_p), omode));
8742 }
8743 }
8744 #else
8745 static int
openfile(int omode)8746 openfile(int omode)
8747 {
8748 return (openat(G_p->g_dirfd, get_component(G_p->g_nam_p), omode));
8749 }
8750 #endif
8751
8752 #if defined(O_XATTR)
8753 static int
read_xattr_hdr()8754 read_xattr_hdr()
8755 {
8756 off_t bytes;
8757 int comp_len, link_len;
8758 int namelen;
8759 int asz;
8760 int cnt;
8761 char *tp;
8762 char *xattrapath;
8763 int pad;
8764 int parentfilelen;
8765
8766 /*
8767 * Include any padding in the read. We need to be positioned
8768 * at beginning of next header.
8769 */
8770
8771 bytes = Gen.g_filesz;
8772
8773 if ((xattrhead = e_zalloc(E_NORMAL, (size_t)bytes)) == NULL) {
8774 (void) fprintf(stderr, gettext(
8775 "Insufficient memory for extended attribute\n"));
8776 return (1);
8777 }
8778
8779 tp = (char *)xattrhead;
8780 while (bytes > 0) {
8781 cnt = (int)(bytes > CPIOBSZ) ? CPIOBSZ : bytes;
8782 FILL(cnt);
8783 (void) memcpy(tp, Buffr.b_out_p, cnt);
8784 tp += cnt;
8785 Buffr.b_out_p += cnt;
8786 Buffr.b_cnt -= (off_t)cnt;
8787 bytes -= (off_t)cnt;
8788 }
8789
8790 pad = (Pad_val + 1 - (Gen.g_filesz & Pad_val)) &
8791 Pad_val;
8792 if (pad != 0) {
8793 FILL(pad);
8794 Buffr.b_out_p += pad;
8795 Buffr.b_cnt -= (off_t)pad;
8796 }
8797
8798 /*
8799 * Validate that we can handle header format
8800 */
8801
8802 if (strcmp(xattrhead->h_version, XATTR_ARCH_VERS) != 0) {
8803 (void) fprintf(stderr,
8804 gettext("Unknown extended attribute format encountered\n"));
8805 (void) fprintf(stderr,
8806 gettext("Disabling extended attribute header parsing\n"));
8807 xattrbadhead = 1;
8808 return (1);
8809 }
8810 (void) sscanf(xattrhead->h_component_len, "%10d", &comp_len);
8811 (void) sscanf(xattrhead->h_link_component_len, "%10d", &link_len);
8812 xattrp = (struct xattr_buf *)(((char *)xattrhead) +
8813 sizeof (struct xattr_hdr));
8814 (void) sscanf(xattrp->h_namesz, "%7d", &namelen);
8815 if (link_len > 0) {
8816 xattr_linkp = (struct xattr_buf *)((intptr_t)xattrp +
8817 (int)comp_len);
8818 } else {
8819 xattr_linkp = NULL;
8820 }
8821
8822 /*
8823 * Gather the attribute path from the filename and attrnames section.
8824 * The filename and attrnames section can be composed of two or more
8825 * path segments separated by a null character. The first segment
8826 * is the path to the parent file that roots the entire sequence in
8827 * the normal name space. The remaining segments describes a path
8828 * rooted at the hidden extended attribute directory of the leaf file of
8829 * the previous segment, making it possible to name attributes on
8830 * attributes.
8831 */
8832 parentfilelen = strlen(xattrp->h_names);
8833 xattrapath = xattrp->h_names + parentfilelen + 1;
8834 asz = strlen(xattrapath);
8835 if ((asz + parentfilelen + 2) < namelen) {
8836 /*
8837 * The attrnames section contains a system attribute on an
8838 * attribute. Save the name of the attribute for use later,
8839 * and replace the null separating the attribute name from
8840 * the system attribute name with a '/' so that xattrapath can
8841 * be used to display messages with the full attribute path name
8842 * rooted at the hidden attribute directory of the base file
8843 * in normal name space.
8844 */
8845 xattrapath[asz] = '/';
8846 }
8847
8848 return (0);
8849 }
8850 #endif
8851
8852 static mode_t
attrmode(char type)8853 attrmode(char type)
8854 {
8855 mode_t mode;
8856
8857 switch (type) {
8858 case '\0':
8859 case REGTYPE:
8860 case LNKTYPE:
8861 mode = S_IFREG;
8862 break;
8863
8864 case SYMTYPE:
8865 mode = S_IFLNK;
8866 break;
8867
8868 case CHRTYPE:
8869 mode = S_IFCHR;
8870 break;
8871 case BLKTYPE:
8872 mode = S_IFBLK;
8873 break;
8874 case DIRTYPE:
8875 mode = S_IFDIR;
8876 break;
8877 case FIFOTYPE:
8878 mode = S_IFIFO;
8879 break;
8880 case CONTTYPE:
8881 default:
8882 mode = 0;
8883 }
8884
8885 return (mode);
8886 }
8887
8888 #if defined(O_XATTR)
8889 static char *
get_component(char * path)8890 get_component(char *path)
8891 {
8892 char *ptr;
8893
8894 ptr = strrchr(path, '/');
8895 if (ptr == NULL) {
8896 return (path);
8897 } else {
8898 /*
8899 * Handle trailing slash
8900 */
8901 if (*(ptr + 1) == '\0')
8902 return (ptr);
8903 else
8904 return (ptr + 1);
8905 }
8906 }
8907 #else
8908 static char *
get_component(char * path)8909 get_component(char *path)
8910 {
8911 return (path);
8912 }
8913 #endif
8914
8915 static int
open_dir(char * name)8916 open_dir(char *name)
8917 {
8918 int fd = -1;
8919 int cnt = 0;
8920 char *dir;
8921
8922 dir = e_zalloc(E_EXIT, strlen(name) + 1);
8923
8924 /*
8925 * open directory; creating missing directories along the way.
8926 */
8927 get_parent(name, dir);
8928 do {
8929 fd = open(dir, O_RDONLY);
8930 if (fd != -1) {
8931 free(dir);
8932 return (fd);
8933 }
8934 cnt++;
8935 } while (cnt <= 1 && missdir(name) == 0);
8936
8937 free(dir);
8938 return (-1);
8939 }
8940
8941 static int
open_dirfd()8942 open_dirfd()
8943 {
8944 #ifdef O_XATTR
8945 if ((Args & OCt) == 0) {
8946 close_dirfd();
8947 if (G_p->g_attrnam_p != NULL) {
8948 int rw_sysattr;
8949
8950 /*
8951 * Open the file's attribute directory.
8952 * Change into the base file's starting directory then
8953 * call open_attr_dir() to open the attribute directory
8954 * of either the base file (if G_p->g_attrparent_p is
8955 * NULL) or the attribute (if G_p->g_attrparent_p is
8956 * set) of the base file.
8957 */
8958 (void) fchdir(G_p->g_baseparent_fd);
8959 (void) open_attr_dir(G_p->g_attrnam_p,
8960 G_p->g_attrfnam_p, G_p->g_baseparent_fd,
8961 (G_p->g_attrparent_p == NULL) ? NULL :
8962 G_p->g_attrparent_p, &G_p->g_dirfd, &rw_sysattr);
8963 if (Args & OCi) {
8964 int saveerrno = errno;
8965
8966 (void) fchdir(G_p->g_baseparent_fd);
8967 errno = saveerrno;
8968 }
8969 if ((G_p->g_dirfd == -1) && (Args & (OCi | OCp))) {
8970 msg(ERRN,
8971 "Cannot open attribute directory "
8972 "of %s%s%sfile \"%s\"",
8973 (G_p->g_attrparent_p == NULL) ? "" :
8974 gettext("attribute \""),
8975 (G_p->g_attrparent_p == NULL) ? "" :
8976 G_p->g_attrparent_p,
8977 (G_p->g_attrparent_p == NULL) ? "" :
8978 gettext("\" of "),
8979 G_p->g_attrfnam_p);
8980 return (FILE_PASS_ERR);
8981 }
8982 } else {
8983 G_p->g_dirfd = open_dir(G_p->g_nam_p);
8984 if (G_p->g_dirfd == -1) {
8985 msg(ERRN,
8986 "Cannot open/create %s", G_p->g_nam_p);
8987 return (1);
8988 }
8989 }
8990 } else {
8991 G_p->g_dirfd = -1;
8992 }
8993 #else
8994 G_p->g_dirfd = -1;
8995 #endif
8996 return (0);
8997 }
8998
8999 static void
close_dirfd()9000 close_dirfd()
9001 {
9002 if (G_p->g_dirfd != -1) {
9003 (void) close(G_p->g_dirfd);
9004 G_p->g_dirfd = -1;
9005 }
9006 }
9007
9008 static void
write_xattr_hdr()9009 write_xattr_hdr()
9010 {
9011 char *attrbuf = NULL;
9012 int attrlen = 0;
9013 char *namep;
9014 struct Lnk *tl_p, *linkinfo;
9015
9016 /*
9017 * namep was allocated in xattrs_out. It is big enough to hold
9018 * either the name + .hdr on the end or just the attr name
9019 */
9020
9021 #if defined(O_XATTR)
9022 namep = Gen.g_nam_p;
9023 (void) creat_hdr();
9024
9025 if (Args & OCo) {
9026 linkinfo = NULL;
9027 tl_p = Lnk_hd.L_nxt_p;
9028 while (tl_p != &Lnk_hd) {
9029 if (tl_p->L_gen.g_ino == G_p->g_ino &&
9030 tl_p->L_gen.g_dev == G_p->g_dev) {
9031 linkinfo = tl_p;
9032 break; /* found */
9033 }
9034 tl_p = tl_p->L_nxt_p;
9035 }
9036 prepare_xattr_hdr(&attrbuf, Gen.g_attrfnam_p,
9037 Gen.g_attrpath_p,
9038 (linkinfo == NULL) ?
9039 tartype(Gen.g_mode & Ftype) : LNKTYPE,
9040 linkinfo, &attrlen);
9041 Gen.g_filesz = attrlen;
9042 write_hdr(ARCHIVE_XATTR, (off_t)attrlen);
9043 (void) sprintf(namep, "%s/%s", DEVNULL, Gen.g_attrnam_p);
9044 write_ancillary(attrbuf, attrlen, B_TRUE);
9045 }
9046
9047 (void) creat_hdr();
9048 #endif
9049 }
9050
9051 /*
9052 * skip over extra slashes in string.
9053 *
9054 * For example:
9055 * /usr/tmp/////
9056 *
9057 * would return pointer at
9058 * /usr/tmp/////
9059 * ^
9060 */
9061 static char *
skipslashes(char * string,char * start)9062 skipslashes(char *string, char *start)
9063 {
9064 while ((string > start) && *(string - 1) == '/') {
9065 string--;
9066 }
9067
9068 return (string);
9069 }
9070
9071 static sl_info_t *
sl_info_alloc(void)9072 sl_info_alloc(void)
9073 {
9074 static int num_left;
9075 static sl_info_t *slipool;
9076
9077 if (num_left > 0) {
9078 return (&slipool[--num_left]);
9079 }
9080 num_left = SL_INFO_ALLOC_CHUNK;
9081 slipool = e_zalloc(E_EXIT, sizeof (sl_info_t) * num_left);
9082 return (&slipool[--num_left]);
9083 }
9084
9085 /*
9086 * If a match for the key values was found in the tree, return a pointer to it.
9087 * If a match was not found, insert it and return a pointer to it. This is
9088 * based on Knuth's Algorithm A in Vol 3, section 6.2.3.
9089 */
9090
9091 sl_info_t *
sl_insert(dev_t device,ino_t inode,int ftype)9092 sl_insert(dev_t device, ino_t inode, int ftype)
9093 {
9094 sl_info_t *p; /* moves down the tree */
9095 sl_info_t *q = NULL; /* scratch */
9096 sl_info_t *r; /* scratch */
9097 sl_info_t *s; /* pt where rebalancing may be needed */
9098 sl_info_t *t; /* father of s */
9099 sl_info_t *head;
9100
9101 int a; /* used to hold balance factors */
9102 int done; /* loop control */
9103 int cmpflg; /* used to hold the result of a comparison */
9104
9105 /* initialize */
9106
9107 head = sl_devhash_lookup(device);
9108
9109 if (head == NULL) {
9110 head = sl_info_alloc();
9111 head->llink = NULL;
9112 head->bal = 0;
9113
9114 p = head->rlink = sl_info_alloc();
9115 p->sl_ino = inode;
9116 p->sl_ftype = ftype;
9117 p->sl_count = 0;
9118 p->bal = 0;
9119 p->llink = NULL;
9120 p->rlink = NULL;
9121 sl_devhash_insert(device, head);
9122 return (p);
9123 }
9124
9125 t = head;
9126 s = p = head->rlink;
9127
9128 /* compare */
9129
9130 for (done = 0; ! done; ) {
9131 switch (sl_compare(inode, ftype, p->sl_ino, p->sl_ftype)) {
9132 case -1:
9133 /* move left */
9134
9135 q = p->llink;
9136
9137 if (q == NULL) {
9138 q = sl_info_alloc();
9139 p->llink = q;
9140 done = 1;
9141 continue;
9142 }
9143
9144 break;
9145
9146 case 0:
9147 /* found it */
9148 return (p);
9149
9150 case 1:
9151 /* move right */
9152
9153 q = p->rlink;
9154
9155 if (q == NULL) {
9156 q = sl_info_alloc();
9157 p->rlink = q;
9158 done = 1;
9159 continue;
9160 }
9161
9162 break;
9163 }
9164
9165 if (q->bal != 0) {
9166 t = p;
9167 s = q;
9168 }
9169
9170 p = q;
9171 }
9172
9173 /* insert */
9174
9175 q->sl_ino = inode;
9176 q->sl_ftype = ftype;
9177 q->sl_count = 0;
9178 q->llink = q->rlink = NULL;
9179 q->bal = 0;
9180
9181 /* adjust balance factors */
9182
9183 if ((cmpflg = sl_compare(inode, ftype, s->sl_ino, s->sl_ftype)) < 0) {
9184 r = p = s->llink;
9185 } else {
9186 r = p = s->rlink;
9187 }
9188
9189 while (p != q) {
9190 switch (sl_compare(inode, ftype, p->sl_ino, p->sl_ftype)) {
9191 case -1:
9192 p->bal = -1;
9193 p = p->llink;
9194 break;
9195
9196 case 0:
9197 break;
9198
9199 case 1:
9200 p->bal = 1;
9201 p = p->rlink;
9202 break;
9203 }
9204 }
9205
9206 /* balancing act */
9207
9208 if (cmpflg < 0) {
9209 a = -1;
9210 } else {
9211 a = 1;
9212 }
9213
9214 if (s->bal == 0) {
9215 s->bal = a;
9216 head->llink = (sl_info_t *)((intptr_t)head->llink + 1);
9217 return (q);
9218 } else if (s->bal == -a) {
9219 s->bal = 0;
9220 return (q);
9221 }
9222
9223 /*
9224 * (s->bal == a)
9225 */
9226
9227 if (r->bal == a) {
9228 /* single rotation */
9229
9230 p = r;
9231
9232 if (a == -1) {
9233 s->llink = r->rlink;
9234 r->rlink = s;
9235 } else if (a == 1) {
9236 s->rlink = r->llink;
9237 r->llink = s;
9238 }
9239
9240 s->bal = r->bal = 0;
9241
9242 } else if (r->bal == -a) {
9243 /* double rotation */
9244
9245 if (a == -1) {
9246 p = r->rlink;
9247 r->rlink = p->llink;
9248 p->llink = r;
9249 s->llink = p->rlink;
9250 p->rlink = s;
9251 } else if (a == 1) {
9252 p = r->llink;
9253 r->llink = p->rlink;
9254 p->rlink = r;
9255 s->rlink = p->llink;
9256 p->llink = s;
9257 }
9258
9259 if (p->bal == 0) {
9260 s->bal = 0;
9261 r->bal = 0;
9262 } else if (p->bal == -a) {
9263 s->bal = 0;
9264 r->bal = a;
9265 } else if (p->bal == a) {
9266 s->bal = -a;
9267 r->bal = 0;
9268 }
9269
9270 p->bal = 0;
9271 }
9272
9273 /* finishing touch */
9274
9275 if (s == t->rlink) {
9276 t->rlink = p;
9277 } else {
9278 t->llink = p;
9279 }
9280
9281 return (q);
9282 }
9283
9284 /*
9285 * sl_numlinks: return the number of links that we saw during our preview.
9286 */
9287
9288 static uint_t
sl_numlinks(dev_t device,ino_t inode,int ftype)9289 sl_numlinks(dev_t device, ino_t inode, int ftype)
9290 {
9291 sl_info_t *p = sl_search(device, inode, ftype);
9292
9293 if (p) {
9294 return (p->sl_count);
9295 } else {
9296 return (1);
9297 }
9298 }
9299
9300 /*
9301 * Preview extended and extended system attributes.
9302 *
9303 * Return 0 if successful, otherwise return 1.
9304 */
9305 #if defined(O_XATTR)
9306 static int
preview_attrs(char * s,char * attrparent)9307 preview_attrs(char *s, char *attrparent)
9308 {
9309 char *filename = (attrparent == NULL) ? s : attrparent;
9310 int dirfd;
9311 int tmpfd;
9312 int islnk;
9313 int rc = 0;
9314 int arc_rwsysattr = 0;
9315 int rw_sysattr = 0;
9316 int ext_attr = 0;
9317 DIR *dirp;
9318 struct dirent *dp;
9319 struct stat sb;
9320
9321 /*
9322 * If the underlying file system supports it, then
9323 * archive the extended attributes if -@ was specified,
9324 * and the extended system attributes if -/ was
9325 * specified.
9326 */
9327 if (verify_attr_support(filename, (attrparent == NULL), ARC_CREATE,
9328 &ext_attr) != ATTR_OK) {
9329 return (1);
9330 }
9331
9332 #if defined(_PC_SATTR_ENABLED)
9333 if (SysAtflag) {
9334 int filefd;
9335 nvlist_t *slist = NULL;
9336
9337 /* Determine if there are non-transient system attributes. */
9338 errno = 0;
9339 if ((filefd = open(filename, O_RDONLY)) < 0) {
9340 return (1);
9341 }
9342 if (((slist = sysattr_list(myname, filefd,
9343 filename)) != NULL) || (errno != 0)) {
9344 arc_rwsysattr = 1;
9345 }
9346 if (slist != NULL) {
9347 (void) nvlist_free(slist);
9348 slist = NULL;
9349 }
9350 (void) close(filefd);
9351 }
9352
9353 if ((arc_rwsysattr == 0) && ((attrparent != NULL) ||
9354 (SysAtflag && !ext_attr))) {
9355 return (1);
9356 }
9357 #endif /* _PC_SATTR_ENABLED */
9358 /*
9359 * We need to open the attribute directory of the
9360 * file, and preview all of the file's attributes as
9361 * attributes of the file can be hard links to other
9362 * attributes of the file.
9363 */
9364 dirfd = attropen(filename, ".", O_RDONLY);
9365 if (dirfd == -1)
9366 return (1);
9367
9368 tmpfd = dup(dirfd);
9369 if (tmpfd == -1) {
9370 (void) close(dirfd);
9371 return (1);
9372 }
9373 dirp = fdopendir(tmpfd);
9374 if (dirp == NULL) {
9375 (void) close(dirfd);
9376 (void) close(tmpfd);
9377 return (1);
9378 }
9379
9380 while ((dp = readdir(dirp)) != NULL) {
9381 if (dp->d_name[0] == '.') {
9382 if (dp->d_name[1] == '\0') {
9383 Hiddendir = 1;
9384 } else if ((dp->d_name[1] == '.') &&
9385 (dp->d_name[2] == '\0')) {
9386 continue;
9387 } else {
9388 Hiddendir = 0;
9389 }
9390 } else {
9391 Hiddendir = 0;
9392 }
9393
9394 if (fstatat(dirfd, dp->d_name, &sb,
9395 AT_SYMLINK_NOFOLLOW) < 0) {
9396 continue;
9397 }
9398
9399 if (verify_attr(dp->d_name, attrparent,
9400 arc_rwsysattr, &rw_sysattr) != ATTR_OK) {
9401 continue;
9402 }
9403
9404 islnk = 0;
9405 if (S_ISLNK(sb.st_mode)) {
9406 islnk = 1;
9407 if (Args & OCL) {
9408 if (fstatat(dirfd, dp->d_name,
9409 &sb, 0) < 0) {
9410 continue;
9411 }
9412 }
9413 }
9414 sl_remember_tgt(&sb, islnk, rw_sysattr);
9415
9416 /*
9417 * Recursively call preview_attrs() to preview extended
9418 * system attributes of attributes.
9419 */
9420 if (SysAtflag && !Hiddendir && !rw_sysattr) {
9421 int my_cwd = save_cwd();
9422
9423 (void) fchdir(dirfd);
9424 rc = preview_attrs(s, dp->d_name);
9425 rest_cwd(my_cwd);
9426 }
9427 }
9428 (void) closedir(dirp);
9429 (void) close(dirfd);
9430 return (rc);
9431 }
9432 #endif /* O_XATTR */
9433
9434 /*
9435 * sl_preview_synonyms: Read the file list from the input stream, remembering
9436 * each reference to each file.
9437 */
9438
9439 static void
sl_preview_synonyms(void)9440 sl_preview_synonyms(void)
9441 {
9442 char buf [APATH+1];
9443 char *s;
9444
9445 char *suffix = "/cpioXXXXXX";
9446 char *tmpdir = getenv("TMPDIR");
9447 int tmpfd, islnk;
9448 FILE *tmpfile;
9449 char *tmpfname;
9450
9451 if (tmpdir == NULL || *tmpdir == '\0' ||
9452 (strlen(tmpdir) + strlen(suffix)) > APATH) {
9453 struct statvfs tdsb;
9454
9455 tmpdir = "/var/tmp";
9456
9457 /* /var/tmp is read-only in the mini-root environment */
9458
9459 if (statvfs(tmpdir, &tdsb) == -1 || tdsb.f_flag & ST_RDONLY) {
9460 tmpdir = "/tmp";
9461 }
9462 }
9463
9464 tmpfname = e_zalloc(E_EXIT, strlen(tmpdir) + strlen(suffix) + 1);
9465
9466 (void) strcpy(tmpfname, tmpdir);
9467 (void) strcat(tmpfname, suffix);
9468
9469 if ((tmpfd = mkstemp(tmpfname)) == -1) {
9470 msg(EXTN, "cannot open tmpfile %s%s", tmpdir, suffix);
9471 }
9472
9473 if (unlink(tmpfname) == -1) {
9474 msg(EXTN, "cannot unlink tmpfile %s", tmpfname);
9475 }
9476
9477 if ((tmpfile = fdopen(tmpfd, "w+")) == NULL) {
9478 msg(EXTN, "cannot fdopen tmpfile %s", tmpfname);
9479 }
9480
9481 while ((s = fgets(buf, APATH+1, In_p)) != NULL) {
9482 size_t lastchar;
9483 struct stat sb;
9484
9485 if (fputs(buf, tmpfile) == EOF) {
9486 msg(EXTN, "problem writing to tmpfile %s", tmpfname);
9487 }
9488
9489 /* pre-process the name */
9490
9491 lastchar = strlen(s) - 1;
9492
9493 if (s[lastchar] != '\n' && lastchar == APATH - 1) {
9494 continue;
9495 } else {
9496 s[lastchar] = '\0';
9497 }
9498
9499 while (s[0] == '.' && s[1] == '/') {
9500 s += 2;
9501 while (s[0] == '/') {
9502 s++;
9503 }
9504 }
9505
9506 if (lstat(s, &sb) < 0) {
9507 continue;
9508 }
9509 islnk = 0;
9510 if (S_ISLNK(sb.st_mode)) {
9511 islnk = 1;
9512 if (Args & OCL) {
9513 if (stat(s, &sb) < 0) {
9514 continue;
9515 }
9516 }
9517 }
9518 sl_remember_tgt(&sb, islnk, 0);
9519
9520 #if defined(O_XATTR)
9521 if (Atflag || SysAtflag) {
9522 (void) preview_attrs(s, NULL);
9523 }
9524 #endif /* O_XATTR */
9525 }
9526
9527 if (ferror(In_p)) {
9528 msg(EXTN, "error reading stdin");
9529 }
9530
9531 if (fseek(tmpfile, 0L, SEEK_SET) == -1) {
9532 msg(EXTN, "cannot fseek on tmpfile %s", tmpfname);
9533 }
9534
9535 In_p = tmpfile;
9536 free(tmpfname);
9537 }
9538
9539 /*
9540 * sl_remember_tgt: Add the device/inode for lstat or stat info to the list of
9541 * those we've seen before.
9542 *
9543 * This tree (rooted under head) is keyed by the device/inode of the file
9544 * being pointed to. A count is kept of the number of references encountered
9545 * so far.
9546 */
9547
9548 static void
sl_remember_tgt(const struct stat * sbp,int isSymlink,int is_sysattr)9549 sl_remember_tgt(const struct stat *sbp, int isSymlink, int is_sysattr)
9550 {
9551 sl_info_t *p;
9552 dev_t device;
9553 ino_t inode;
9554 int ftype;
9555
9556 device = sbp->st_dev;
9557 inode = sbp->st_ino;
9558 ftype = sbp->st_mode & Ftype;
9559
9560 /* Determine whether we've seen this one before */
9561
9562 p = sl_insert(device, inode, ftype);
9563
9564 if (p->sl_count > 0) {
9565 /*
9566 * It appears as if have seen this file before as we found a
9567 * matching device, inode, and file type as a file already
9568 * processed. Since there can possibly be files with the
9569 * same device, inode, and file type, but aren't hard links
9570 * (e.g., read-write system attribute files will always have
9571 * the same inode), we need to only attempt to add one to the
9572 * link count if the file we are processing is a hard link
9573 * (i.e., st_nlink > 1).
9574 *
9575 * Note that if we are not chasing symlinks, and this one is a
9576 * symlink, it is identically the one we saw before (you cannot
9577 * have hard links to symlinks); in this case, we leave the
9578 * count alone, so that we don't wind up archiving a symlink to
9579 * itself.
9580 */
9581
9582 if (((Args & OCL) || (! isSymlink)) && !is_sysattr) {
9583 p->sl_count++;
9584 }
9585 } else {
9586 /* We have not seen this file before */
9587
9588 p->sl_count = 1;
9589
9590 if (Use_old_stat) {
9591 /* -Hodc: remap inode (-1 on overflow) */
9592
9593 sl_remap_t *q;
9594
9595 for (q = sl_remap_head; q && (q->dev != device);
9596 q = q->next) {
9597 /* do nothing */
9598 }
9599
9600 if (q == NULL) {
9601 q = e_zalloc(E_EXIT, sizeof (sl_remap_t));
9602 q->dev = device;
9603 p->sl_ino2 = q->inode_count = 1;
9604
9605 q->next = (sl_remap_head) ?
9606 sl_remap_head->next : NULL;
9607 sl_remap_head = q;
9608 } else {
9609 if ((size_t)q->inode_count <=
9610 ((1 << (sizeof (o_ino_t) * 8)) - 1)) {
9611 /* fits in o_ino_t */
9612 p->sl_ino2 = ++(q->inode_count);
9613 } else {
9614 p->sl_ino2 = (ino_t)-1;
9615 }
9616 }
9617 }
9618 }
9619 }
9620
9621 /*
9622 * A faster search, which does not insert the key values into the tree.
9623 * If the a match was found in the tree, return a pointer to it. If it was not
9624 * found, return NULL.
9625 */
9626
9627 sl_info_t *
sl_search(dev_t device,ino_t inode,int ftype)9628 sl_search(dev_t device, ino_t inode, int ftype)
9629 {
9630 sl_info_t *p; /* moves down the tree */
9631 int c; /* comparison value */
9632 sl_info_t *retval = NULL; /* return value */
9633 sl_info_t *head;
9634
9635 head = sl_devhash_lookup(device);
9636 if (head != NULL) {
9637 for (p = head->rlink; p; ) {
9638 if ((c = sl_compare(inode, ftype, p->sl_ino,
9639 p->sl_ftype)) == 0) {
9640 retval = p;
9641 break;
9642 } else if (c < 0) {
9643 p = p->llink;
9644 } else {
9645 p = p->rlink;
9646 }
9647 }
9648 }
9649
9650 return (retval);
9651 }
9652
9653 static sl_info_t *
sl_devhash_lookup(dev_t device)9654 sl_devhash_lookup(dev_t device)
9655 {
9656 int key;
9657 sl_info_link_t *lp;
9658 static sl_info_link_t *devcache;
9659
9660 if (devcache != NULL && devcache->dev == device) {
9661 return (devcache->head);
9662 }
9663
9664 key = DEV_HASHKEY(device);
9665 for (lp = sl_devhash[key]; lp; lp = lp->next) {
9666 if (lp->dev == device) {
9667 devcache = lp;
9668 return (lp->head);
9669 }
9670 }
9671 return (NULL);
9672 }
9673
9674 static void
sl_devhash_insert(dev_t device,sl_info_t * head)9675 sl_devhash_insert(dev_t device, sl_info_t *head)
9676 {
9677 int key = DEV_HASHKEY(device);
9678 sl_info_link_t *lp;
9679
9680 lp = e_zalloc(E_EXIT, sizeof (sl_info_link_t));
9681 lp->dev = device;
9682 lp->head = head;
9683 lp->next = sl_devhash[key];
9684 sl_devhash[key] = lp;
9685 }
9686
9687 static void
chop_endslashes(char * path)9688 chop_endslashes(char *path)
9689 {
9690 char *end, *ptr;
9691
9692 end = &path[strlen(path) -1];
9693 if (*end == '/' && end != path) {
9694 ptr = skipslashes(end, path);
9695 if (ptr != NULL && ptr != path) {
9696 *ptr = '\0';
9697 }
9698 }
9699 }
9700
9701 #if !defined(O_XATTR)
9702 int
openat64(int fd,char * name,int oflag,mode_t cmode)9703 openat64(int fd, char *name, int oflag, mode_t cmode)
9704 {
9705 return (open64(name, oflag, cmode));
9706 }
9707
9708 int
openat(int fd,char * name,int oflag,mode_t cmode)9709 openat(int fd, char *name, int oflag, mode_t cmode)
9710 {
9711 return (open(name, oflag, cmode));
9712 }
9713
9714 int
fchownat(int fd,char * name,uid_t owner,gid_t group,int flag)9715 fchownat(int fd, char *name, uid_t owner, gid_t group, int flag)
9716 {
9717 if (flag == AT_SYMLINK_NOFOLLOW)
9718 return (lchown(name, owner, group));
9719 else
9720 return (chown(name, owner, group));
9721 }
9722
9723 int
renameat(int fromfd,char * old,int tofd,char * new)9724 renameat(int fromfd, char *old, int tofd, char *new)
9725 {
9726 return (rename(old, new));
9727 }
9728
9729 int
futimesat(int fd,char * path,struct timeval times[2])9730 futimesat(int fd, char *path, struct timeval times[2])
9731 {
9732 return (utimes(path, times));
9733 }
9734
9735 int
unlinkat(int dirfd,char * path,int flag)9736 unlinkat(int dirfd, char *path, int flag)
9737 {
9738 if (flag == AT_REMOVEDIR) {
9739 return (rmdir(path));
9740 } else {
9741 return (unlink(path));
9742 }
9743 }
9744
9745 int
fstatat(int fd,char * path,struct stat * buf,int flag)9746 fstatat(int fd, char *path, struct stat *buf, int flag)
9747 {
9748 if (flag == AT_SYMLINK_NOFOLLOW)
9749 return (lstat(path, buf));
9750 else
9751 return (stat(path, buf));
9752 }
9753
9754 int
attropen(char * file,char * attr,int omode,mode_t cmode)9755 attropen(char *file, char *attr, int omode, mode_t cmode)
9756 {
9757 errno = ENOTSUP;
9758 return (-1);
9759 }
9760 #endif
9761