1 /* A tar (tape archiver) program.
3 Copyright (C) 1988, 1992, 1993, 1994, 1995, 1996, 1997, 1999, 2000,
4 2001, 2003, 2004, 2005, 2006, 2007 Free Software Foundation, Inc.
6 Written by John Gilmore, starting 1985-08-25.
8 This program is free software; you can redistribute it and/or modify it
9 under the terms of the GNU General Public License as published by the
10 Free Software Foundation; either version 3, or (at your option) any later
13 This program is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
16 Public License for more details.
18 You should have received a copy of the GNU General Public License along
19 with this program; if not, write to the Free Software Foundation, Inc.,
20 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */
26 #include <argp-namefrob.h>
27 #include <argp-fmtstream.h>
30 #if ! defined SIGCHLD && defined SIGCLD
31 # define SIGCHLD SIGCLD
34 /* The following causes "common.h" to produce definitions of all the global
35 variables, rather than just "extern" declarations of them. GNU tar does
36 depend on the system loader to preset all GLOBAL variables to neutral (or
37 zero) values; explicit initialization is usually not done. */
43 #include <configmake.h>
47 #include <rmt-command.h>
50 #include <version-etc.h>
54 /* Local declarations. */
56 #ifndef DEFAULT_ARCHIVE_FORMAT
57 # define DEFAULT_ARCHIVE_FORMAT GNU_FORMAT
60 #ifndef DEFAULT_ARCHIVE
61 # define DEFAULT_ARCHIVE "tar.out"
64 #ifndef DEFAULT_BLOCKING
65 # define DEFAULT_BLOCKING 20
71 /* Name of option using stdin. */
72 static const char *stdin_used_by
;
74 /* Doesn't return if stdin already requested. */
76 request_stdin (const char *option
)
79 USAGE_ERROR ((0, 0, _("Options `-%s' and `-%s' both want standard input"),
80 stdin_used_by
, option
));
82 stdin_used_by
= option
;
85 extern int rpmatch (char const *response
);
87 /* Returns true if and only if the user typed an affirmative response. */
89 confirm (const char *message_action
, const char *message_name
)
91 static FILE *confirm_file
;
92 static int confirm_file_EOF
;
97 if (archive
== 0 || stdin_used_by
)
99 confirm_file
= fopen (TTY_NAME
, "r");
101 open_fatal (TTY_NAME
);
105 request_stdin ("-w");
106 confirm_file
= stdin
;
110 fprintf (stdlis
, "%s %s?", message_action
, quote (message_name
));
113 if (!confirm_file_EOF
)
115 char *response
= NULL
;
116 size_t response_size
= 0;
117 if (getline (&response
, &response_size
, confirm_file
) < 0)
118 confirm_file_EOF
= 1;
120 status
= rpmatch (response
) > 0;
124 if (confirm_file_EOF
)
126 fputc ('\n', stdlis
);
133 static struct fmttab
{
135 enum archive_format fmt
;
138 { "oldgnu", OLDGNU_FORMAT
},
139 { "ustar", USTAR_FORMAT
},
140 { "posix", POSIX_FORMAT
},
141 #if 0 /* not fully supported yet */
142 { "star", STAR_FORMAT
},
144 { "gnu", GNU_FORMAT
},
145 { "pax", POSIX_FORMAT
}, /* An alias for posix */
150 set_archive_format (char const *name
)
152 struct fmttab
const *p
;
154 for (p
= fmttab
; strcmp (p
->name
, name
) != 0; )
156 USAGE_ERROR ((0, 0, _("%s: Invalid archive format"),
157 quotearg_colon (name
)));
159 archive_format
= p
->fmt
;
163 archive_format_string (enum archive_format fmt
)
165 struct fmttab
const *p
;
167 for (p
= fmttab
; p
->name
; p
++)
173 #define FORMAT_MASK(n) (1<<(n))
176 assert_format(unsigned fmt_mask
)
178 if ((FORMAT_MASK (archive_format
) & fmt_mask
) == 0)
180 _("GNU features wanted on incompatible archive format")));
184 subcommand_string (enum subcommand c
)
188 case UNKNOWN_SUBCOMMAND
:
191 case APPEND_SUBCOMMAND
:
197 case CREATE_SUBCOMMAND
:
200 case DELETE_SUBCOMMAND
:
203 case DIFF_SUBCOMMAND
:
206 case EXTRACT_SUBCOMMAND
:
209 case LIST_SUBCOMMAND
:
212 case UPDATE_SUBCOMMAND
:
221 tar_list_quoting_styles (argp_fmtstream_t fs
, char *prefix
)
225 for (i
= 0; quoting_style_args
[i
]; i
++)
226 argp_fmtstream_printf (fs
, "%s%s\n", prefix
, quoting_style_args
[i
]);
230 tar_set_quoting_style (char *arg
)
234 for (i
= 0; quoting_style_args
[i
]; i
++)
235 if (strcmp (arg
, quoting_style_args
[i
]) == 0)
237 set_quoting_style (NULL
, i
);
241 _("Unknown quoting style `%s'. Try `%s --quoting-style=help' to get a list."), arg
, program_invocation_short_name
));
249 ANCHORED_OPTION
= CHAR_MAX
+ 1,
250 ATIME_PRESERVE_OPTION
,
254 CHECKPOINT_ACTION_OPTION
,
255 DELAY_DIRECTORY_RESTORE_OPTION
,
256 HARD_DEREFERENCE_OPTION
,
258 EXCLUDE_CACHES_OPTION
,
259 EXCLUDE_CACHES_UNDER_OPTION
,
260 EXCLUDE_CACHES_ALL_OPTION
,
263 EXCLUDE_TAG_UNDER_OPTION
,
264 EXCLUDE_TAG_ALL_OPTION
,
270 IGNORE_COMMAND_ERROR_OPTION
,
271 IGNORE_FAILED_READ_OPTION
,
273 KEEP_NEWER_FILES_OPTION
,
279 NO_AUTO_COMPRESS_OPTION
,
280 NO_CHECK_DEVICE_OPTION
,
281 NO_DELAY_DIRECTORY_RESTORE_OPTION
,
282 NO_IGNORE_CASE_OPTION
,
283 NO_IGNORE_COMMAND_ERROR_OPTION
,
285 NO_OVERWRITE_DIR_OPTION
,
286 NO_QUOTE_CHARS_OPTION
,
288 NO_SAME_OWNER_OPTION
,
289 NO_SAME_PERMISSIONS_OPTION
,
291 NO_WILDCARDS_MATCH_SLASH_OPTION
,
294 NUMERIC_OWNER_OPTION
,
297 ONE_FILE_SYSTEM_OPTION
,
298 OVERWRITE_DIR_OPTION
,
305 QUOTING_STYLE_OPTION
,
308 RECURSIVE_UNLINK_OPTION
,
314 SHOW_DEFAULTS_OPTION
,
315 SHOW_OMITTED_DIRS_OPTION
,
316 SHOW_TRANSFORMED_NAMES_OPTION
,
317 SPARSE_VERSION_OPTION
,
318 STRIP_COMPONENTS_OPTION
,
326 USE_COMPRESS_PROGRAM_OPTION
,
330 WILDCARDS_MATCH_SLASH_OPTION
,
334 const char *argp_program_version
= "tar (" PACKAGE_NAME
") " VERSION
;
335 const char *argp_program_bug_address
= "<" PACKAGE_BUGREPORT
">";
336 static char const doc
[] = N_("\
337 GNU `tar' saves many files together into a single tape or disk archive, \
338 and can restore individual files from the archive.\n\
341 tar -cf archive.tar foo bar # Create archive.tar from files foo and bar.\n\
342 tar -tvf archive.tar # List all files in archive.tar verbosely.\n\
343 tar -xf archive.tar # Extract all files from archive.tar.\n")
345 N_("The backup suffix is `~', unless set with --suffix or SIMPLE_BACKUP_SUFFIX.\n\
346 The version control may be set with --backup or VERSION_CONTROL, values are:\n\n\
347 none, off never make backups\n\
348 t, numbered make numbered backups\n\
349 nil, existing numbered if numbered backups exist, simple otherwise\n\
350 never, simple always make simple backups\n");
355 Available option letters are DEIQY and eqy. Consider the following
358 [For Solaris tar compatibility =/= Is it important at all?]
359 e exit immediately with a nonzero exit status if unexpected errors occur
360 E use extended headers (--format=posix)
362 [q alias for --occurrence=1 =/= this would better be used for quiet?]
363 [I same as T =/= will harm star compatibility]
365 y per-file gzip compression
366 Y per-block gzip compression */
368 static struct argp_option options
[] = {
371 N_("Main operation mode:"), GRID
},
374 N_("list the contents of an archive"), GRID
+1 },
375 {"extract", 'x', 0, 0,
376 N_("extract files from an archive"), GRID
+1 },
377 {"get", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
378 {"create", 'c', 0, 0,
379 N_("create a new archive"), GRID
+1 },
381 N_("find differences between archive and file system"), GRID
+1 },
382 {"compare", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
383 {"append", 'r', 0, 0,
384 N_("append files to the end of an archive"), GRID
+1 },
385 {"update", 'u', 0, 0,
386 N_("only append files newer than copy in archive"), GRID
+1 },
387 {"catenate", 'A', 0, 0,
388 N_("append tar files to an archive"), GRID
+1 },
389 {"concatenate", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
390 {"delete", DELETE_OPTION
, 0, 0,
391 N_("delete from the archive (not on mag tapes!)"), GRID
+1 },
392 {"test-label", TEST_LABEL_OPTION
, NULL
, 0,
393 N_("test the archive volume label and exit"), GRID
+1 },
398 N_("Operation modifiers:"), GRID
},
400 {"sparse", 'S', 0, 0,
401 N_("handle sparse files efficiently"), GRID
+1 },
402 {"sparse-version", SPARSE_VERSION_OPTION
, N_("MAJOR[.MINOR]"), 0,
403 N_("set version of the sparse format to use (implies --sparse)"), GRID
+1},
404 {"incremental", 'G', 0, 0,
405 N_("handle old GNU-format incremental backup"), GRID
+1 },
406 {"listed-incremental", 'g', N_("FILE"), 0,
407 N_("handle new GNU-format incremental backup"), GRID
+1 },
408 {"ignore-failed-read", IGNORE_FAILED_READ_OPTION
, 0, 0,
409 N_("do not exit with nonzero on unreadable files"), GRID
+1 },
410 {"occurrence", OCCURRENCE_OPTION
, N_("NUMBER"), OPTION_ARG_OPTIONAL
,
411 N_("process only the NUMBERth occurrence of each file in the archive;"
412 " this option is valid only in conjunction with one of the subcommands"
413 " --delete, --diff, --extract or --list and when a list of files"
414 " is given either on the command line or via the -T option;"
415 " NUMBER defaults to 1"), GRID
+1 },
416 {"seek", 'n', NULL
, 0,
417 N_("archive is seekable"), GRID
+1 },
418 {"no-check-device", NO_CHECK_DEVICE_OPTION
, NULL
, 0,
419 N_("do not check device numbers when creating incremental archives"),
421 {"check-device", CHECK_DEVICE_OPTION
, NULL
, 0,
422 N_("check device numbers when creating incremental archives (default)"),
428 N_("Overwrite control:"), GRID
},
430 {"verify", 'W', 0, 0,
431 N_("attempt to verify the archive after writing it"), GRID
+1 },
432 {"remove-files", REMOVE_FILES_OPTION
, 0, 0,
433 N_("remove files after adding them to the archive"), GRID
+1 },
434 {"keep-old-files", 'k', 0, 0,
435 N_("don't replace existing files when extracting"), GRID
+1 },
436 {"keep-newer-files", KEEP_NEWER_FILES_OPTION
, 0, 0,
437 N_("don't replace existing files that are newer than their archive copies"), GRID
+1 },
438 {"overwrite", OVERWRITE_OPTION
, 0, 0,
439 N_("overwrite existing files when extracting"), GRID
+1 },
440 {"unlink-first", 'U', 0, 0,
441 N_("remove each file prior to extracting over it"), GRID
+1 },
442 {"recursive-unlink", RECURSIVE_UNLINK_OPTION
, 0, 0,
443 N_("empty hierarchies prior to extracting directory"), GRID
+1 },
444 {"no-overwrite-dir", NO_OVERWRITE_DIR_OPTION
, 0, 0,
445 N_("preserve metadata of existing directories"), GRID
+1 },
446 {"overwrite-dir", OVERWRITE_DIR_OPTION
, 0, 0,
447 N_("overwrite metadata of existing directories when extracting (default)"),
453 N_("Select output stream:"), GRID
},
455 {"to-stdout", 'O', 0, 0,
456 N_("extract files to standard output"), GRID
+1 },
457 {"to-command", TO_COMMAND_OPTION
, N_("COMMAND"), 0,
458 N_("pipe extracted files to another program"), GRID
+1 },
459 {"ignore-command-error", IGNORE_COMMAND_ERROR_OPTION
, 0, 0,
460 N_("ignore exit codes of children"), GRID
+1 },
461 {"no-ignore-command-error", NO_IGNORE_COMMAND_ERROR_OPTION
, 0, 0,
462 N_("treat non-zero exit codes of children as error"), GRID
+1 },
467 N_("Handling of file attributes:"), GRID
},
469 {"owner", OWNER_OPTION
, N_("NAME"), 0,
470 N_("force NAME as owner for added files"), GRID
+1 },
471 {"group", GROUP_OPTION
, N_("NAME"), 0,
472 N_("force NAME as group for added files"), GRID
+1 },
473 {"mtime", MTIME_OPTION
, N_("DATE-OR-FILE"), 0,
474 N_("set mtime for added files from DATE-OR-FILE"), GRID
+1 },
475 {"mode", MODE_OPTION
, N_("CHANGES"), 0,
476 N_("force (symbolic) mode CHANGES for added files"), GRID
+1 },
477 {"atime-preserve", ATIME_PRESERVE_OPTION
,
478 N_("METHOD"), OPTION_ARG_OPTIONAL
,
479 N_("preserve access times on dumped files, either by restoring the times"
480 " after reading (METHOD='replace'; default) or by not setting the times"
481 " in the first place (METHOD='system')"), GRID
+1 },
483 N_("don't extract file modified time"), GRID
+1 },
484 {"same-owner", SAME_OWNER_OPTION
, 0, 0,
485 N_("try extracting files with the same ownership"), GRID
+1 },
486 {"no-same-owner", NO_SAME_OWNER_OPTION
, 0, 0,
487 N_("extract files as yourself"), GRID
+1 },
488 {"numeric-owner", NUMERIC_OWNER_OPTION
, 0, 0,
489 N_("always use numbers for user/group names"), GRID
+1 },
490 {"preserve-permissions", 'p', 0, 0,
491 N_("extract information about file permissions (default for superuser)"),
493 {"same-permissions", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
494 {"no-same-permissions", NO_SAME_PERMISSIONS_OPTION
, 0, 0,
495 N_("apply the user's umask when extracting permissions from the archive (default for ordinary users)"), GRID
+1 },
496 {"preserve-order", 's', 0, 0,
497 N_("sort names to extract to match archive"), GRID
+1 },
498 {"same-order", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
499 {"preserve", PRESERVE_OPTION
, 0, 0,
500 N_("same as both -p and -s"), GRID
+1 },
501 {"delay-directory-restore", DELAY_DIRECTORY_RESTORE_OPTION
, 0, 0,
502 N_("delay setting modification times and permissions of extracted"
503 " directories until the end of extraction"), GRID
+1 },
504 {"no-delay-directory-restore", NO_DELAY_DIRECTORY_RESTORE_OPTION
, 0, 0,
505 N_("cancel the effect of --delay-directory-restore option"), GRID
+1 },
510 N_("Device selection and switching:"), GRID
},
512 {"file", 'f', N_("ARCHIVE"), 0,
513 N_("use archive file or device ARCHIVE"), GRID
+1 },
514 {"force-local", FORCE_LOCAL_OPTION
, 0, 0,
515 N_("archive file is local even if it has a colon"), GRID
+1 },
516 {"rmt-command", RMT_COMMAND_OPTION
, N_("COMMAND"), 0,
517 N_("use given rmt COMMAND instead of rmt"), GRID
+1 },
518 {"rsh-command", RSH_COMMAND_OPTION
, N_("COMMAND"), 0,
519 N_("use remote COMMAND instead of rsh"), GRID
+1 },
521 {"-[0-7][lmh]", 0, NULL
, OPTION_DOC
, /* It is OK, since `name' will never be
523 N_("specify drive and density"), GRID
+1 },
525 {NULL
, '0', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
526 {NULL
, '1', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
527 {NULL
, '2', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
528 {NULL
, '3', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
529 {NULL
, '4', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
530 {NULL
, '5', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
531 {NULL
, '6', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
532 {NULL
, '7', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
533 {NULL
, '8', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
534 {NULL
, '9', NULL
, OPTION_HIDDEN
, NULL
, GRID
+1 },
536 {"multi-volume", 'M', 0, 0,
537 N_("create/list/extract multi-volume archive"), GRID
+1 },
538 {"tape-length", 'L', N_("NUMBER"), 0,
539 N_("change tape after writing NUMBER x 1024 bytes"), GRID
+1 },
540 {"info-script", 'F', N_("NAME"), 0,
541 N_("run script at end of each tape (implies -M)"), GRID
+1 },
542 {"new-volume-script", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
543 {"volno-file", VOLNO_FILE_OPTION
, N_("FILE"), 0,
544 N_("use/update the volume number in FILE"), GRID
+1 },
549 N_("Device blocking:"), GRID
},
551 {"blocking-factor", 'b', N_("BLOCKS"), 0,
552 N_("BLOCKS x 512 bytes per record"), GRID
+1 },
553 {"record-size", RECORD_SIZE_OPTION
, N_("NUMBER"), 0,
554 N_("NUMBER of bytes per record, multiple of 512"), GRID
+1 },
555 {"ignore-zeros", 'i', 0, 0,
556 N_("ignore zeroed blocks in archive (means EOF)"), GRID
+1 },
557 {"read-full-records", 'B', 0, 0,
558 N_("reblock as we read (for 4.2BSD pipes)"), GRID
+1 },
563 N_("Archive format selection:"), GRID
},
565 {"format", 'H', N_("FORMAT"), 0,
566 N_("create archive of the given format"), GRID
+1 },
568 {NULL
, 0, NULL
, 0, N_("FORMAT is one of the following:"), GRID
+2 },
569 {" v7", 0, NULL
, OPTION_DOC
|OPTION_NO_TRANS
, N_("old V7 tar format"),
571 {" oldgnu", 0, NULL
, OPTION_DOC
|OPTION_NO_TRANS
,
572 N_("GNU format as per tar <= 1.12"), GRID
+3 },
573 {" gnu", 0, NULL
, OPTION_DOC
|OPTION_NO_TRANS
,
574 N_("GNU tar 1.13.x format"), GRID
+3 },
575 {" ustar", 0, NULL
, OPTION_DOC
|OPTION_NO_TRANS
,
576 N_("POSIX 1003.1-1988 (ustar) format"), GRID
+3 },
577 {" pax", 0, NULL
, OPTION_DOC
|OPTION_NO_TRANS
,
578 N_("POSIX 1003.1-2001 (pax) format"), GRID
+3 },
579 {" posix", 0, NULL
, OPTION_DOC
|OPTION_NO_TRANS
, N_("same as pax"), GRID
+3 },
581 {"old-archive", OLD_ARCHIVE_OPTION
, 0, 0, /* FIXME */
582 N_("same as --format=v7"), GRID
+8 },
583 {"portability", 0, 0, OPTION_ALIAS
, NULL
, GRID
+8 },
584 {"posix", POSIX_OPTION
, 0, 0,
585 N_("same as --format=posix"), GRID
+8 },
586 {"pax-option", PAX_OPTION
, N_("keyword[[:]=value][,keyword[[:]=value]]..."), 0,
587 N_("control pax keywords"), GRID
+8 },
588 {"label", 'V', N_("TEXT"), 0,
589 N_("create archive with volume name TEXT; at list/extract time, use TEXT as a globbing pattern for volume name"), GRID
+8 },
594 N_("Compression options:"), GRID
},
595 {"auto-compress", 'a', 0, 0,
596 N_("use archive suffix to determine the compression program"), GRID
+1 },
597 {"no-auto-compress", NO_AUTO_COMPRESS_OPTION
, 0, 0,
598 N_("do not use use archive suffix to determine the compression program"),
601 N_("filter the archive through bzip2"), GRID
+1 },
603 N_("filter the archive through gzip"), GRID
+1 },
604 {"gunzip", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
605 {"ungzip", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
606 {"compress", 'Z', 0, 0,
607 N_("filter the archive through compress"), GRID
+1 },
608 {"uncompress", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
610 N_("filter the archive through lzma"), GRID
+1 },
611 {"lzop", LZOP_OPTION
, 0, 0,
612 N_("filter the archive through lzop"), GRID
+8 },
613 {"use-compress-program", USE_COMPRESS_PROGRAM_OPTION
, N_("PROG"), 0,
614 N_("filter through PROG (must accept -d)"), GRID
+1 },
619 N_("Local file selection:"), GRID
},
621 {"add-file", ARGP_KEY_ARG
, N_("FILE"), 0,
622 N_("add given FILE to the archive (useful if its name starts with a dash)"), GRID
+1 },
623 {"directory", 'C', N_("DIR"), 0,
624 N_("change to directory DIR"), GRID
+1 },
625 {"files-from", 'T', N_("FILE"), 0,
626 N_("get names to extract or create from FILE"), GRID
+1 },
627 {"null", NULL_OPTION
, 0, 0,
628 N_("-T reads null-terminated names, disable -C"), GRID
+1 },
629 {"no-null", NO_NULL_OPTION
, 0, 0,
630 N_("disable the effect of the previous --null option"), GRID
+1 },
631 {"unquote", UNQUOTE_OPTION
, 0, 0,
632 N_("unquote filenames read with -T (default)"), GRID
+1 },
633 {"no-unquote", NO_UNQUOTE_OPTION
, 0, 0,
634 N_("do not unquote filenames read with -T"), GRID
+1 },
635 {"exclude", EXCLUDE_OPTION
, N_("PATTERN"), 0,
636 N_("exclude files, given as a PATTERN"), GRID
+1 },
637 {"exclude-from", 'X', N_("FILE"), 0,
638 N_("exclude patterns listed in FILE"), GRID
+1 },
639 {"exclude-caches", EXCLUDE_CACHES_OPTION
, 0, 0,
640 N_("exclude contents of directories containing CACHEDIR.TAG, "
641 "except for the tag file itself"), GRID
+1 },
642 {"exclude-caches-under", EXCLUDE_CACHES_UNDER_OPTION
, 0, 0,
643 N_("exclude everything under directories containing CACHEDIR.TAG"),
645 {"exclude-caches-all", EXCLUDE_CACHES_ALL_OPTION
, 0, 0,
646 N_("exclude directories containing CACHEDIR.TAG"), GRID
+1 },
647 {"exclude-tag", EXCLUDE_TAG_OPTION
, N_("FILE"), 0,
648 N_("exclude contents of directories containing FILE, except"
649 " for FILE itself"), GRID
+1 },
650 {"exclude-tag-under", EXCLUDE_TAG_UNDER_OPTION
, N_("FILE"), 0,
651 N_("exclude everything under directories containing FILE"), GRID
+1 },
652 {"exclude-tag-all", EXCLUDE_TAG_ALL_OPTION
, N_("FILE"), 0,
653 N_("exclude directories containing FILE"), GRID
+1 },
654 {"exclude-vcs", EXCLUDE_VCS_OPTION
, NULL
, 0,
655 N_("exclude version control system directories"), GRID
+1 },
656 {"no-recursion", NO_RECURSION_OPTION
, 0, 0,
657 N_("avoid descending automatically in directories"), GRID
+1 },
658 {"one-file-system", ONE_FILE_SYSTEM_OPTION
, 0, 0,
659 N_("stay in local file system when creating archive"), GRID
+1 },
660 {"recursion", RECURSION_OPTION
, 0, 0,
661 N_("recurse into directories (default)"), GRID
+1 },
662 {"absolute-names", 'P', 0, 0,
663 N_("don't strip leading `/'s from file names"), GRID
+1 },
664 {"dereference", 'h', 0, 0,
665 N_("follow symlinks; archive and dump the files they point to"), GRID
+1 },
666 {"hard-dereference", HARD_DEREFERENCE_OPTION
, 0, 0,
667 N_("follow hard links; archive and dump the files they refer to"), GRID
+1 },
668 {"starting-file", 'K', N_("MEMBER-NAME"), 0,
669 N_("begin at member MEMBER-NAME in the archive"), GRID
+1 },
670 {"newer", 'N', N_("DATE-OR-FILE"), 0,
671 N_("only store files newer than DATE-OR-FILE"), GRID
+1 },
672 {"after-date", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
673 {"newer-mtime", NEWER_MTIME_OPTION
, N_("DATE"), 0,
674 N_("compare date and time when data changed only"), GRID
+1 },
675 {"backup", BACKUP_OPTION
, N_("CONTROL"), OPTION_ARG_OPTIONAL
,
676 N_("backup before removal, choose version CONTROL"), GRID
+1 },
677 {"suffix", SUFFIX_OPTION
, N_("STRING"), 0,
678 N_("backup before removal, override usual suffix ('~' unless overridden by environment variable SIMPLE_BACKUP_SUFFIX)"), GRID
+1 },
683 N_("File name transformations:"), GRID
},
684 {"strip-components", STRIP_COMPONENTS_OPTION
, N_("NUMBER"), 0,
685 N_("strip NUMBER leading components from file names on extraction"),
687 {"transform", TRANSFORM_OPTION
, N_("EXPRESSION"), 0,
688 N_("use sed replace EXPRESSION to transform file names"), GRID
+1 },
693 N_("File name matching options (affect both exclude and include patterns):"),
695 {"ignore-case", IGNORE_CASE_OPTION
, 0, 0,
696 N_("ignore case"), GRID
+1 },
697 {"anchored", ANCHORED_OPTION
, 0, 0,
698 N_("patterns match file name start"), GRID
+1 },
699 {"no-anchored", NO_ANCHORED_OPTION
, 0, 0,
700 N_("patterns match after any `/' (default for exclusion)"), GRID
+1 },
701 {"no-ignore-case", NO_IGNORE_CASE_OPTION
, 0, 0,
702 N_("case sensitive matching (default)"), GRID
+1 },
703 {"wildcards", WILDCARDS_OPTION
, 0, 0,
704 N_("use wildcards (default for exclusion)"), GRID
+1 },
705 {"no-wildcards", NO_WILDCARDS_OPTION
, 0, 0,
706 N_("verbatim string matching"), GRID
+1 },
707 {"no-wildcards-match-slash", NO_WILDCARDS_MATCH_SLASH_OPTION
, 0, 0,
708 N_("wildcards do not match `/'"), GRID
+1 },
709 {"wildcards-match-slash", WILDCARDS_MATCH_SLASH_OPTION
, 0, 0,
710 N_("wildcards match `/' (default for exclusion)"), GRID
+1 },
715 N_("Informative output:"), GRID
},
717 {"verbose", 'v', 0, 0,
718 N_("verbosely list files processed"), GRID
+1 },
719 {"checkpoint", CHECKPOINT_OPTION
, N_("NUMBER"), OPTION_ARG_OPTIONAL
,
720 N_("display progress messages every NUMBERth record (default 10)"),
722 {"checkpoint-action", CHECKPOINT_ACTION_OPTION
, N_("ACTION"), 0,
723 N_("execute ACTION on each checkpoint"),
725 {"check-links", 'l', 0, 0,
726 N_("print a message if not all links are dumped"), GRID
+1 },
727 {"totals", TOTALS_OPTION
, N_("SIGNAL"), OPTION_ARG_OPTIONAL
,
728 N_("print total bytes after processing the archive; "
729 "with an argument - print total bytes when this SIGNAL is delivered; "
730 "Allowed signals are: SIGHUP, SIGQUIT, SIGINT, SIGUSR1 and SIGUSR2; "
731 "the names without SIG prefix are also accepted"), GRID
+1 },
732 {"utc", UTC_OPTION
, 0, 0,
733 N_("print file modification dates in UTC"), GRID
+1 },
734 {"index-file", INDEX_FILE_OPTION
, N_("FILE"), 0,
735 N_("send verbose output to FILE"), GRID
+1 },
736 {"block-number", 'R', 0, 0,
737 N_("show block number within archive with each message"), GRID
+1 },
738 {"interactive", 'w', 0, 0,
739 N_("ask for confirmation for every action"), GRID
+1 },
740 {"confirmation", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
741 {"show-defaults", SHOW_DEFAULTS_OPTION
, 0, 0,
742 N_("show tar defaults"), GRID
+1 },
743 {"show-omitted-dirs", SHOW_OMITTED_DIRS_OPTION
, 0, 0,
744 N_("when listing or extracting, list each directory that does not match search criteria"), GRID
+1 },
745 {"show-transformed-names", SHOW_TRANSFORMED_NAMES_OPTION
, 0, 0,
746 N_("show file or archive names after transformation"),
748 {"show-stored-names", 0, 0, OPTION_ALIAS
, NULL
, GRID
+1 },
749 {"quoting-style", QUOTING_STYLE_OPTION
, N_("STYLE"), 0,
750 N_("set name quoting style; see below for valid STYLE values"), GRID
+1 },
751 {"quote-chars", QUOTE_CHARS_OPTION
, N_("STRING"), 0,
752 N_("additionally quote characters from STRING"), GRID
+1 },
753 {"no-quote-chars", NO_QUOTE_CHARS_OPTION
, N_("STRING"), 0,
754 N_("disable quoting for characters from STRING"), GRID
+1 },
759 N_("Compatibility options:"), GRID
},
762 N_("when creating, same as --old-archive; when extracting, same as --no-same-owner"), GRID
+1 },
767 N_("Other options:"), GRID
},
769 {"restrict", RESTRICT_OPTION
, 0, 0,
770 N_("disable use of some potentially harmful options"), -1 },
772 {"help", '?', 0, 0, N_("give this help list"), -1},
773 {"usage", USAGE_OPTION
, 0, 0, N_("give a short usage message"), -1},
774 {"version", VERSION_OPTION
, 0, 0, N_("print program version"), -1},
775 /* FIXME -V (--label) conflicts with the default short option for
777 {"HANG", HANG_OPTION
, "SECS", OPTION_ARG_OPTIONAL
| OPTION_HIDDEN
,
778 N_("hang for SECS seconds (default 3600)"), 0},
784 static char const *const atime_preserve_args
[] =
786 "replace", "system", NULL
789 static enum atime_preserve
const atime_preserve_types
[] =
791 replace_atime_preserve
, system_atime_preserve
794 /* Make sure atime_preserve_types has as much entries as atime_preserve_args
795 (minus 1 for NULL guard) */
796 ARGMATCH_VERIFY (atime_preserve_args
, atime_preserve_types
);
798 /* Wildcard matching settings */
801 default_wildcards
, /* For exclusion == enable_wildcards,
802 for inclusion == disable_wildcards */
807 struct tar_args
/* Variables used during option parsing */
809 struct textual_date
*textual_date
; /* Keeps the arguments to --newer-mtime
810 and/or --date option if they are
812 enum wildcards wildcards
; /* Wildcard settings (--wildcards/
814 int matching_flags
; /* exclude_fnmatch options */
815 int include_anchored
; /* Pattern anchoring options used for
817 bool o_option
; /* True if -o option was given */
818 bool pax_option
; /* True if --pax-option was given */
819 char const *backup_suffix_string
; /* --suffix option argument */
820 char const *version_control_string
; /* --backup option argument */
821 bool input_files
; /* True if some input files where given */
822 int compress_autodetect
; /* True if compression autodetection should
823 be attempted when creating archives */
827 #define MAKE_EXCL_OPTIONS(args) \
828 ((((args)->wildcards != disable_wildcards) ? EXCLUDE_WILDCARDS : 0) \
829 | (args)->matching_flags \
832 #define MAKE_INCL_OPTIONS(args) \
833 ((((args)->wildcards == enable_wildcards) ? EXCLUDE_WILDCARDS : 0) \
834 | (args)->include_anchored \
835 | (args)->matching_flags \
842 static char *vcs_file
[] = {
874 for (i
= 0; vcs_file
[i
]; i
++)
875 add_exclude (excluded
, vcs_file
[i
], 0);
880 # define DECL_SHOW_DEFAULT_SETTINGS(stream, printer) \
883 "--format=%s -f%s -b%d --quoting-style=%s --rmt-command=%s", \
884 archive_format_string (DEFAULT_ARCHIVE_FORMAT), \
885 DEFAULT_ARCHIVE, DEFAULT_BLOCKING, \
886 quoting_style_args[DEFAULT_QUOTING_STYLE], \
887 DEFAULT_RMT_COMMAND); \
888 printer (stream, " --rsh-command=%s", REMOTE_SHELL); \
889 printer (stream, "\n"); \
892 # define DECL_SHOW_DEFAULT_SETTINGS(stream, printer) \
895 "--format=%s -f%s -b%d --quoting-style=%s --rmt-command=%s", \
896 archive_format_string (DEFAULT_ARCHIVE_FORMAT), \
897 DEFAULT_ARCHIVE, DEFAULT_BLOCKING, \
898 quoting_style_args[DEFAULT_QUOTING_STYLE], \
899 DEFAULT_RMT_COMMAND); \
900 printer (stream, "\n"); \
905 show_default_settings (FILE *fp
)
906 DECL_SHOW_DEFAULT_SETTINGS(fp
, fprintf
)
909 show_default_settings_fs (argp_fmtstream_t fs
)
910 DECL_SHOW_DEFAULT_SETTINGS(fs
, argp_fmtstream_printf
)
913 set_subcommand_option (enum subcommand subcommand
)
915 if (subcommand_option
!= UNKNOWN_SUBCOMMAND
916 && subcommand_option
!= subcommand
)
918 _("You may not specify more than one `-Acdtrux' option")));
920 subcommand_option
= subcommand
;
924 set_use_compress_program_option (const char *string
)
926 if (use_compress_program_option
927 && strcmp (use_compress_program_option
, string
) != 0)
928 USAGE_ERROR ((0, 0, _("Conflicting compression options")));
930 use_compress_program_option
= string
;
937 print_total_stats ();
938 #ifndef HAVE_SIGACTION
939 signal (signo
, sigstat
);
944 stat_on_signal (int signo
)
946 #ifdef HAVE_SIGACTION
947 struct sigaction act
;
948 act
.sa_handler
= sigstat
;
949 sigemptyset (&act
.sa_mask
);
951 sigaction (signo
, &act
, NULL
);
953 signal (signo
, sigstat
);
958 set_stat_signal (const char *name
)
965 { "SIGUSR1", SIGUSR1
},
967 { "SIGUSR2", SIGUSR2
},
969 { "SIGHUP", SIGHUP
},
971 { "SIGINT", SIGINT
},
973 { "SIGQUIT", SIGQUIT
},
978 for (p
= sigtab
; p
< sigtab
+ sizeof (sigtab
) / sizeof (sigtab
[0]); p
++)
979 if (strcmp (p
->name
, name
) == 0)
981 stat_on_signal (p
->signo
);
984 FATAL_ERROR ((0, 0, _("Unknown signal name: %s"), name
));
990 struct textual_date
*next
;
997 get_date_or_file (struct tar_args
*args
, const char *option
,
998 const char *str
, struct timespec
*ts
)
1000 if (FILE_SYSTEM_PREFIX_LEN (str
) != 0
1005 if (deref_stat (dereference_option
, str
, &st
) != 0)
1008 USAGE_ERROR ((0, 0, _("Date sample file not found")));
1010 *ts
= get_stat_mtime (&st
);
1014 if (! get_date (ts
, str
, NULL
))
1016 WARN ((0, 0, _("Substituting %s for unknown date format %s"),
1017 tartime (*ts
, false), quote (str
)));
1022 struct textual_date
*p
= xmalloc (sizeof (*p
));
1026 p
->next
= args
->textual_date
;
1027 args
->textual_date
= p
;
1033 report_textual_dates (struct tar_args
*args
)
1035 struct textual_date
*p
;
1036 for (p
= args
->textual_date
; p
; )
1038 struct textual_date
*next
= p
->next
;
1039 char const *treated_as
= tartime (*p
->ts
, true);
1040 if (strcmp (p
->date
, treated_as
) != 0)
1041 WARN ((0, 0, _("Option %s: Treating date `%s' as %s"),
1042 p
->option
, p
->date
, treated_as
));
1049 static volatile int _argp_hang
;
1051 /* Either NL or NUL, as decided by the --null option. */
1052 static char filename_terminator
;
1054 enum read_file_list_state
/* Result of reading file name from the list file */
1056 file_list_success
, /* OK, name read successfully */
1057 file_list_end
, /* End of list file */
1058 file_list_zero
, /* Zero separator encountered where it should not */
1059 file_list_skip
/* Empty (zero-length) entry encountered, skip it */
1062 /* Read from FP a sequence of characters up to TERM and put them
1065 static enum read_file_list_state
1066 read_name_from_file (FILE *fp
, struct obstack
*stk
, int term
)
1071 for (c
= getc (fp
); c
!= EOF
&& c
!= term
; c
= getc (fp
))
1075 /* We have read a zero separator. The file possibly is
1077 return file_list_zero
;
1079 obstack_1grow (stk
, c
);
1083 if (counter
== 0 && c
!= EOF
)
1084 return file_list_skip
;
1086 obstack_1grow (stk
, 0);
1088 return (counter
== 0 && c
== EOF
) ? file_list_end
: file_list_success
;
1092 static bool files_from_option
; /* When set, tar will not refuse to create
1094 static struct obstack argv_stk
; /* Storage for additional command line options
1095 read using -T option */
1097 /* Prevent recursive inclusion of the same file */
1100 struct file_id_list
*next
;
1105 static struct file_id_list
*file_id_list
;
1108 add_file_id (const char *filename
)
1110 struct file_id_list
*p
;
1113 if (stat (filename
, &st
))
1114 stat_fatal (filename
);
1115 for (p
= file_id_list
; p
; p
= p
->next
)
1116 if (p
->ino
== st
.st_ino
&& p
->dev
== st
.st_dev
)
1118 FATAL_ERROR ((0, 0, _("%s: file list already read"),
1119 quotearg_colon (filename
)));
1121 p
= xmalloc (sizeof *p
);
1122 p
->next
= file_id_list
;
1128 /* Default density numbers for [0-9][lmh] device specifications */
1130 #ifndef LOW_DENSITY_NUM
1131 # define LOW_DENSITY_NUM 0
1134 #ifndef MID_DENSITY_NUM
1135 # define MID_DENSITY_NUM 8
1138 #ifndef HIGH_DENSITY_NUM
1139 # define HIGH_DENSITY_NUM 16
1143 update_argv (const char *filename
, struct argp_state
*state
)
1146 size_t count
= 0, i
;
1150 bool is_stdin
= false;
1151 enum read_file_list_state read_state
;
1152 int term
= filename_terminator
;
1154 if (!strcmp (filename
, "-"))
1157 request_stdin ("-T");
1162 add_file_id (filename
);
1163 if ((fp
= fopen (filename
, "r")) == NULL
)
1164 open_fatal (filename
);
1167 while ((read_state
= read_name_from_file (fp
, &argv_stk
, term
))
1172 case file_list_success
:
1176 case file_list_end
: /* won't happen, just to pacify gcc */
1179 case file_list_zero
:
1183 WARN ((0, 0, N_("%s: file name read contains nul character"),
1184 quotearg_colon (filename
)));
1186 /* Prepare new stack contents */
1187 size
= obstack_object_size (&argv_stk
);
1188 p
= obstack_finish (&argv_stk
);
1189 for (; size
> 0; size
--, p
++)
1191 obstack_1grow (&argv_stk
, *p
);
1193 obstack_1grow (&argv_stk
, '\n');
1194 obstack_1grow (&argv_stk
, 0);
1196 /* Read rest of files using new filename terminator */
1201 case file_list_skip
:
1212 start
= obstack_finish (&argv_stk
);
1215 for (p
= start
; *p
; p
+= strlen (p
) + 1)
1219 new_argc
= state
->argc
+ count
;
1220 new_argv
= xmalloc (sizeof (state
->argv
[0]) * (new_argc
+ 1));
1221 memcpy (new_argv
, state
->argv
, sizeof (state
->argv
[0]) * (state
->argc
+ 1));
1222 state
->argv
= new_argv
;
1223 memmove (&state
->argv
[state
->next
+ count
], &state
->argv
[state
->next
],
1224 (state
->argc
- state
->next
+ 1) * sizeof (state
->argv
[0]));
1226 state
->argc
= new_argc
;
1228 for (i
= state
->next
, p
= start
; *p
; p
+= strlen (p
) + 1, i
++)
1230 if (term
== 0 && p
[0] == '-')
1231 state
->argv
[i
++] = "--add-file";
1238 tar_help (struct argp_state
*state
)
1240 argp_fmtstream_t fs
;
1241 state
->flags
|= ARGP_NO_EXIT
;
1242 argp_state_help (state
, state
->out_stream
,
1243 ARGP_HELP_STD_HELP
& ~ARGP_HELP_BUG_ADDR
);
1244 /* FIXME: use struct uparams.rmargin (from argp-help.c) instead of 79 */
1245 fs
= argp_make_fmtstream (state
->out_stream
, 0, 79, 0);
1247 argp_fmtstream_printf (fs
, "\n%s\n\n",
1248 _("Valid arguments for --quoting-style options are:"));
1249 tar_list_quoting_styles (fs
, " ");
1251 argp_fmtstream_puts (fs
, _("\n*This* tar defaults to:\n"));
1252 show_default_settings_fs (fs
);
1253 argp_fmtstream_putc (fs
, '\n');
1254 argp_fmtstream_printf (fs
, _("Report bugs to %s.\n"),
1255 argp_program_bug_address
);
1256 argp_fmtstream_free (fs
);
1260 parse_opt (int key
, char *arg
, struct argp_state
*state
)
1262 struct tar_args
*args
= state
->input
;
1267 /* File name or non-parsed option, because of ARGP_IN_ORDER */
1268 name_add_name (arg
, MAKE_INCL_OPTIONS (args
));
1269 args
->input_files
= true;
1273 set_subcommand_option (CAT_SUBCOMMAND
);
1277 args
->compress_autodetect
= true;
1280 case NO_AUTO_COMPRESS_OPTION
:
1281 args
->compress_autodetect
= false;
1287 if (! (xstrtoumax (arg
, 0, 10, &u
, "") == LONGINT_OK
1288 && u
== (blocking_factor
= u
)
1289 && 0 < blocking_factor
1290 && u
== (record_size
= u
* BLOCKSIZE
) / BLOCKSIZE
))
1291 USAGE_ERROR ((0, 0, "%s: %s", quotearg_colon (arg
),
1292 _("Invalid blocking factor")));
1297 /* Try to reblock input records. For reading 4.2BSD pipes. */
1299 /* It would surely make sense to exchange -B and -R, but it seems
1300 that -B has been used for a long while in Sun tar and most
1301 BSD-derived systems. This is a consequence of the block/record
1302 terminology confusion. */
1304 read_full_records_option
= true;
1308 set_subcommand_option (CREATE_SUBCOMMAND
);
1316 set_subcommand_option (DIFF_SUBCOMMAND
);
1320 if (archive_names
== allocated_archive_names
)
1321 archive_name_array
= x2nrealloc (archive_name_array
,
1322 &allocated_archive_names
,
1323 sizeof (archive_name_array
[0]));
1325 archive_name_array
[archive_names
++] = arg
;
1329 /* Since -F is only useful with -M, make it implied. Run this
1330 script at the end of each tape. */
1332 info_script_option
= arg
;
1333 multi_volume_option
= true;
1337 listed_incremental_option
= arg
;
1338 after_date_option
= true;
1342 /* We are making an incremental dump (FIXME: are we?); save
1343 directories at the beginning of the archive, and include in each
1344 directory its contents. */
1346 incremental_option
= true;
1350 /* Follow symbolic links. */
1351 dereference_option
= true;
1354 case HARD_DEREFERENCE_OPTION
:
1355 hard_dereference_option
= true;
1359 /* Ignore zero blocks (eofs). This can't be the default,
1360 because Unix tar writes two blocks of zeros, then pads out
1361 the record with garbage. */
1363 ignore_zeros_option
= true;
1368 _("Warning: the -I option is not supported;"
1369 " perhaps you meant -j or -T?")));
1373 set_use_compress_program_option ("bzip2");
1377 set_use_compress_program_option ("lzma");
1381 /* Don't replace existing files. */
1382 old_files_option
= KEEP_OLD_FILES
;
1386 starting_file_option
= true;
1390 case ONE_FILE_SYSTEM_OPTION
:
1391 /* When dumping directories, don't dump files/subdirectories
1392 that are on other filesystems. */
1393 one_file_system_option
= true;
1397 check_links_option
= 1;
1403 if (xstrtoumax (arg
, 0, 10, &u
, "") != LONGINT_OK
)
1404 USAGE_ERROR ((0, 0, "%s: %s", quotearg_colon (arg
),
1405 _("Invalid tape length")));
1406 tape_length_option
= 1024 * (tarlong
) u
;
1407 multi_volume_option
= true;
1412 set_use_compress_program_option ("lzop");
1416 touch_option
= true;
1420 /* Make multivolume archive: when we can't write any more into
1421 the archive, re-open it, and continue writing. */
1423 multi_volume_option
= true;
1427 get_date_or_file (args
, "--mtime", arg
, &mtime_option
);
1428 set_mtime_option
= true;
1432 seekable_archive
= true;
1436 after_date_option
= true;
1439 case NEWER_MTIME_OPTION
:
1440 if (NEWER_OPTION_INITIALIZED (newer_mtime_option
))
1441 USAGE_ERROR ((0, 0, _("More than one threshold date")));
1442 get_date_or_file (args
,
1443 key
== NEWER_MTIME_OPTION
? "--newer-mtime"
1444 : "--after-date", arg
, &newer_mtime_option
);
1448 args
->o_option
= true;
1452 to_stdout_option
= true;
1456 same_permissions_option
= true;
1460 absolute_names_option
= true;
1464 set_subcommand_option (APPEND_SUBCOMMAND
);
1468 /* Print block numbers for debugging bad tar archives. */
1470 /* It would surely make sense to exchange -B and -R, but it seems
1471 that -B has been used for a long while in Sun tar and most
1472 BSD-derived systems. This is a consequence of the block/record
1473 terminology confusion. */
1475 block_number_option
= true;
1479 /* Names to extract are sorted. */
1481 same_order_option
= true;
1485 sparse_option
= true;
1488 case SPARSE_VERSION_OPTION
:
1489 sparse_option
= true;
1492 tar_sparse_major
= strtoul (arg
, &p
, 10);
1496 USAGE_ERROR ((0, 0, _("Invalid sparse version value")));
1497 tar_sparse_minor
= strtoul (p
+ 1, &p
, 10);
1499 USAGE_ERROR ((0, 0, _("Invalid sparse version value")));
1505 set_subcommand_option (LIST_SUBCOMMAND
);
1509 case TEST_LABEL_OPTION
:
1510 set_subcommand_option (LIST_SUBCOMMAND
);
1511 test_label_option
= true;
1515 update_argv (arg
, state
);
1516 /* Indicate we've been given -T option. This is for backward
1517 compatibility only, so that `tar cfT archive /dev/null will
1519 files_from_option
= true;
1523 set_subcommand_option (UPDATE_SUBCOMMAND
);
1527 old_files_option
= UNLINK_FIRST_OLD_FILES
;
1539 volume_label_option
= arg
;
1543 interactive_option
= true;
1547 verify_option
= true;
1551 set_subcommand_option (EXTRACT_SUBCOMMAND
);
1555 if (add_exclude_file (add_exclude
, excluded
, arg
,
1556 MAKE_EXCL_OPTIONS (args
), '\n')
1560 FATAL_ERROR ((0, e
, "%s", quotearg_colon (arg
)));
1565 set_use_compress_program_option ("gzip");
1569 set_use_compress_program_option ("compress");
1572 case ANCHORED_OPTION
:
1573 args
->matching_flags
|= EXCLUDE_ANCHORED
;
1576 case ATIME_PRESERVE_OPTION
:
1577 atime_preserve_option
=
1579 ? XARGMATCH ("--atime-preserve", arg
,
1580 atime_preserve_args
, atime_preserve_types
)
1581 : replace_atime_preserve
);
1582 if (! O_NOATIME
&& atime_preserve_option
== system_atime_preserve
)
1584 _("--atime-preserve='system' is not supported"
1585 " on this platform")));
1588 case CHECK_DEVICE_OPTION
:
1589 check_device_option
= true;
1592 case NO_CHECK_DEVICE_OPTION
:
1593 check_device_option
= false;
1596 case CHECKPOINT_OPTION
:
1603 checkpoint_compile_action (".");
1606 checkpoint_option
= strtoul (arg
, &p
, 0);
1609 _("--checkpoint value is not an integer")));
1612 checkpoint_option
= DEFAULT_CHECKPOINT
;
1615 case CHECKPOINT_ACTION_OPTION
:
1616 checkpoint_compile_action (arg
);
1620 backup_option
= true;
1622 args
->version_control_string
= arg
;
1625 case DELAY_DIRECTORY_RESTORE_OPTION
:
1626 delay_directory_restore_option
= true;
1629 case NO_DELAY_DIRECTORY_RESTORE_OPTION
:
1630 delay_directory_restore_option
= false;
1634 set_subcommand_option (DELETE_SUBCOMMAND
);
1637 case EXCLUDE_OPTION
:
1638 add_exclude (excluded
, arg
, MAKE_EXCL_OPTIONS (args
));
1641 case EXCLUDE_CACHES_OPTION
:
1642 add_exclusion_tag ("CACHEDIR.TAG", exclusion_tag_contents
,
1646 case EXCLUDE_CACHES_UNDER_OPTION
:
1647 add_exclusion_tag ("CACHEDIR.TAG", exclusion_tag_under
,
1651 case EXCLUDE_CACHES_ALL_OPTION
:
1652 add_exclusion_tag ("CACHEDIR.TAG", exclusion_tag_all
,
1656 case EXCLUDE_TAG_OPTION
:
1657 add_exclusion_tag (arg
, exclusion_tag_contents
, NULL
);
1660 case EXCLUDE_TAG_UNDER_OPTION
:
1661 add_exclusion_tag (arg
, exclusion_tag_under
, NULL
);
1664 case EXCLUDE_TAG_ALL_OPTION
:
1665 add_exclusion_tag (arg
, exclusion_tag_all
, NULL
);
1668 case EXCLUDE_VCS_OPTION
:
1669 exclude_vcs_files ();
1672 case FORCE_LOCAL_OPTION
:
1673 force_local_option
= true;
1677 set_archive_format (arg
);
1680 case INDEX_FILE_OPTION
:
1681 index_file_name
= arg
;
1684 case IGNORE_CASE_OPTION
:
1685 args
->matching_flags
|= FNM_CASEFOLD
;
1688 case IGNORE_COMMAND_ERROR_OPTION
:
1689 ignore_command_error_option
= true;
1692 case IGNORE_FAILED_READ_OPTION
:
1693 ignore_failed_read_option
= true;
1696 case KEEP_NEWER_FILES_OPTION
:
1697 old_files_option
= KEEP_NEWER_FILES
;
1701 if (! (strlen (arg
) < GNAME_FIELD_SIZE
1702 && gname_to_gid (arg
, &group_option
)))
1705 if (xstrtoumax (arg
, 0, 10, &g
, "") == LONGINT_OK
1709 FATAL_ERROR ((0, 0, "%s: %s", quotearg_colon (arg
),
1710 _("%s: Invalid group")));
1715 mode_option
= mode_compile (arg
);
1717 FATAL_ERROR ((0, 0, _("Invalid mode given on option")));
1718 initial_umask
= umask (0);
1719 umask (initial_umask
);
1722 case NO_ANCHORED_OPTION
:
1723 args
->include_anchored
= 0; /* Clear the default for comman line args */
1724 args
->matching_flags
&= ~ EXCLUDE_ANCHORED
;
1727 case NO_IGNORE_CASE_OPTION
:
1728 args
->matching_flags
&= ~ FNM_CASEFOLD
;
1731 case NO_IGNORE_COMMAND_ERROR_OPTION
:
1732 ignore_command_error_option
= false;
1735 case NO_OVERWRITE_DIR_OPTION
:
1736 old_files_option
= NO_OVERWRITE_DIR_OLD_FILES
;
1739 case NO_QUOTE_CHARS_OPTION
:
1741 set_char_quoting (NULL
, *arg
, 0);
1744 case NO_WILDCARDS_OPTION
:
1745 args
->wildcards
= disable_wildcards
;
1748 case NO_WILDCARDS_MATCH_SLASH_OPTION
:
1749 args
->matching_flags
|= FNM_FILE_NAME
;
1753 filename_terminator
= '\0';
1756 case NO_NULL_OPTION
:
1757 filename_terminator
= '\n';
1760 case NUMERIC_OWNER_OPTION
:
1761 numeric_owner_option
= true;
1764 case OCCURRENCE_OPTION
:
1766 occurrence_option
= 1;
1770 if (xstrtoumax (arg
, 0, 10, &u
, "") == LONGINT_OK
)
1771 occurrence_option
= u
;
1773 FATAL_ERROR ((0, 0, "%s: %s", quotearg_colon (arg
),
1774 _("Invalid number")));
1778 case OVERWRITE_DIR_OPTION
:
1779 old_files_option
= DEFAULT_OLD_FILES
;
1782 case OVERWRITE_OPTION
:
1783 old_files_option
= OVERWRITE_OLD_FILES
;
1787 if (! (strlen (arg
) < UNAME_FIELD_SIZE
1788 && uname_to_uid (arg
, &owner_option
)))
1791 if (xstrtoumax (arg
, 0, 10, &u
, "") == LONGINT_OK
1795 FATAL_ERROR ((0, 0, "%s: %s", quotearg_colon (arg
),
1796 _("Invalid owner")));
1800 case QUOTE_CHARS_OPTION
:
1802 set_char_quoting (NULL
, *arg
, 1);
1805 case QUOTING_STYLE_OPTION
:
1806 tar_set_quoting_style (arg
);
1810 args
->pax_option
= true;
1811 xheader_set_option (arg
);
1815 set_archive_format ("posix");
1818 case PRESERVE_OPTION
:
1819 /* FIXME: What it is good for? */
1820 same_permissions_option
= true;
1821 same_order_option
= true;
1824 case RECORD_SIZE_OPTION
:
1827 if (! (xstrtoumax (arg
, 0, 10, &u
, "") == LONGINT_OK
1828 && u
== (size_t) u
))
1829 USAGE_ERROR ((0, 0, "%s: %s", quotearg_colon (arg
),
1830 _("Invalid record size")));
1832 if (record_size
% BLOCKSIZE
!= 0)
1833 USAGE_ERROR ((0, 0, _("Record size must be a multiple of %d."),
1835 blocking_factor
= record_size
/ BLOCKSIZE
;
1839 case RECURSIVE_UNLINK_OPTION
:
1840 recursive_unlink_option
= true;
1843 case REMOVE_FILES_OPTION
:
1844 remove_files_option
= true;
1847 case RESTRICT_OPTION
:
1848 restrict_option
= true;
1851 case RMT_COMMAND_OPTION
:
1855 case RSH_COMMAND_OPTION
:
1856 rsh_command_option
= arg
;
1859 case SHOW_DEFAULTS_OPTION
:
1860 show_default_settings (stdout
);
1864 case STRIP_COMPONENTS_OPTION
:
1867 if (! (xstrtoumax (arg
, 0, 10, &u
, "") == LONGINT_OK
1868 && u
== (size_t) u
))
1869 USAGE_ERROR ((0, 0, "%s: %s", quotearg_colon (arg
),
1870 _("Invalid number of elements")));
1871 strip_name_components
= u
;
1875 case SHOW_OMITTED_DIRS_OPTION
:
1876 show_omitted_dirs_option
= true;
1879 case SHOW_TRANSFORMED_NAMES_OPTION
:
1880 show_transformed_names_option
= true;
1884 backup_option
= true;
1885 args
->backup_suffix_string
= arg
;
1888 case TO_COMMAND_OPTION
:
1889 if (to_command_option
)
1890 USAGE_ERROR ((0, 0, _("Only one --to-command option allowed")));
1891 to_command_option
= arg
;
1896 set_stat_signal (arg
);
1898 totals_option
= true;
1901 case TRANSFORM_OPTION
:
1902 set_transform_expr (arg
);
1905 case USE_COMPRESS_PROGRAM_OPTION
:
1906 set_use_compress_program_option (arg
);
1909 case VOLNO_FILE_OPTION
:
1910 volno_file_option
= arg
;
1913 case WILDCARDS_OPTION
:
1914 args
->wildcards
= enable_wildcards
;
1917 case WILDCARDS_MATCH_SLASH_OPTION
:
1918 args
->matching_flags
&= ~ FNM_FILE_NAME
;
1921 case NO_RECURSION_OPTION
:
1922 recursion_option
= 0;
1925 case NO_SAME_OWNER_OPTION
:
1926 same_owner_option
= -1;
1929 case NO_SAME_PERMISSIONS_OPTION
:
1930 same_permissions_option
= -1;
1933 case RECURSION_OPTION
:
1934 recursion_option
= FNM_LEADING_DIR
;
1937 case SAME_OWNER_OPTION
:
1938 same_owner_option
= 1;
1941 case UNQUOTE_OPTION
:
1942 unquote_option
= true;
1945 case NO_UNQUOTE_OPTION
:
1946 unquote_option
= false;
1958 #ifdef DEVICE_PREFIX
1960 int device
= key
- '0';
1962 static char buf
[sizeof DEVICE_PREFIX
+ 10];
1966 argp_error (state
, _("Malformed density argument: %s"), quote (arg
));
1968 strcpy (buf
, DEVICE_PREFIX
);
1969 cursor
= buf
+ strlen (buf
);
1971 #ifdef DENSITY_LETTER
1973 sprintf (cursor
, "%d%c", device
, arg
[0]);
1975 #else /* not DENSITY_LETTER */
1980 device
+= LOW_DENSITY_NUM
;
1984 device
+= MID_DENSITY_NUM
;
1988 device
+= HIGH_DENSITY_NUM
;
1992 argp_error (state
, _("Unknown density: `%c'"), arg
[0]);
1994 sprintf (cursor
, "%d", device
);
1996 #endif /* not DENSITY_LETTER */
1998 if (archive_names
== allocated_archive_names
)
1999 archive_name_array
= x2nrealloc (archive_name_array
,
2000 &allocated_archive_names
,
2001 sizeof (archive_name_array
[0]));
2002 archive_name_array
[archive_names
++] = xstrdup (buf
);
2006 #else /* not DEVICE_PREFIX */
2009 _("Options `-[0-7][lmh]' not supported by *this* tar"));
2011 #endif /* not DEVICE_PREFIX */
2019 argp_state_help (state
, state
->out_stream
, ARGP_HELP_USAGE
);
2023 case VERSION_OPTION
:
2024 version_etc (state
->out_stream
, "tar", PACKAGE_NAME
, VERSION
,
2025 "John Gilmore", "Jay Fenlason", (char *) NULL
);
2030 _argp_hang
= atoi (arg
? arg
: "3600");
2031 while (_argp_hang
-- > 0)
2036 return ARGP_ERR_UNKNOWN
;
2041 static struct argp argp
= {
2054 argp_help (&argp
, stderr
, ARGP_HELP_SEE
, (char*) program_name
);
2059 /* Parse the options for tar. */
2061 static struct argp_option
*
2062 find_argp_option (struct argp_option
*o
, int letter
)
2069 && o
->doc
== NULL
); o
++)
2070 if (o
->key
== letter
)
2076 decode_options (int argc
, char **argv
)
2079 struct tar_args args
;
2081 /* Set some default option values. */
2082 args
.textual_date
= NULL
;
2083 args
.wildcards
= default_wildcards
;
2084 args
.matching_flags
= 0;
2085 args
.include_anchored
= EXCLUDE_ANCHORED
;
2086 args
.o_option
= false;
2087 args
.pax_option
= false;
2088 args
.backup_suffix_string
= getenv ("SIMPLE_BACKUP_SUFFIX");
2089 args
.version_control_string
= 0;
2090 args
.input_files
= false;
2091 args
.compress_autodetect
= false;
2093 subcommand_option
= UNKNOWN_SUBCOMMAND
;
2094 archive_format
= DEFAULT_FORMAT
;
2095 blocking_factor
= DEFAULT_BLOCKING
;
2096 record_size
= DEFAULT_BLOCKING
* BLOCKSIZE
;
2097 excluded
= new_exclude ();
2098 newer_mtime_option
.tv_sec
= TYPE_MINIMUM (time_t);
2099 newer_mtime_option
.tv_nsec
= -1;
2100 recursion_option
= FNM_LEADING_DIR
;
2101 unquote_option
= true;
2102 tar_sparse_major
= 1;
2103 tar_sparse_minor
= 0;
2108 check_device_option
= true;
2110 /* Convert old-style tar call by exploding option element and rearranging
2111 options accordingly. */
2113 if (argc
> 1 && argv
[1][0] != '-')
2115 int new_argc
; /* argc value for rearranged arguments */
2116 char **new_argv
; /* argv value for rearranged arguments */
2117 char *const *in
; /* cursor into original argv */
2118 char **out
; /* cursor into rearranged argv */
2119 const char *letter
; /* cursor into old option letters */
2120 char buffer
[3]; /* constructed option buffer */
2122 /* Initialize a constructed option. */
2127 /* Allocate a new argument array, and copy program name in it. */
2129 new_argc
= argc
- 1 + strlen (argv
[1]);
2130 new_argv
= xmalloc ((new_argc
+ 1) * sizeof (char *));
2135 /* Copy each old letter option as a separate option, and have the
2136 corresponding argument moved next to it. */
2138 for (letter
= *in
++; *letter
; letter
++)
2140 struct argp_option
*opt
;
2142 buffer
[1] = *letter
;
2143 *out
++ = xstrdup (buffer
);
2144 opt
= find_argp_option (options
, *letter
);
2145 if (opt
&& opt
->arg
)
2147 if (in
< argv
+ argc
)
2150 USAGE_ERROR ((0, 0, _("Old option `%c' requires an argument."),
2155 /* Copy all remaining options. */
2157 while (in
< argv
+ argc
)
2161 /* Replace the old option list by the new one. */
2167 /* Parse all options and non-options as they appear. */
2169 prepend_default_options (getenv ("TAR_OPTIONS"), &argc
, &argv
);
2171 if (argp_parse (&argp
, argc
, argv
, ARGP_IN_ORDER
|ARGP_NO_HELP
,
2173 exit (TAREXIT_FAILURE
);
2176 /* Special handling for 'o' option:
2178 GNU tar used to say "output old format".
2179 UNIX98 tar says don't chown files after extracting (we use
2180 "--no-same-owner" for this).
2182 The old GNU tar semantics is retained when used with --create
2183 option, otherwise UNIX98 semantics is assumed */
2187 if (subcommand_option
== CREATE_SUBCOMMAND
)
2189 /* GNU Tar <= 1.13 compatibility */
2190 set_archive_format ("v7");
2194 /* UNIX98 compatibility */
2195 same_owner_option
= -1;
2199 /* Handle operands after any "--" argument. */
2200 for (; idx
< argc
; idx
++)
2202 name_add_name (argv
[idx
], MAKE_INCL_OPTIONS (&args
));
2203 args
.input_files
= true;
2206 /* Warn about implicit use of the wildcards in command line arguments.
2208 warn_regex_usage
= args
.wildcards
== default_wildcards
;
2210 /* Derive option values and check option consistency. */
2212 if (archive_format
== DEFAULT_FORMAT
)
2214 if (args
.pax_option
)
2215 archive_format
= POSIX_FORMAT
;
2217 archive_format
= DEFAULT_ARCHIVE_FORMAT
;
2220 if ((volume_label_option
&& subcommand_option
== CREATE_SUBCOMMAND
)
2221 || incremental_option
2222 || multi_volume_option
2224 assert_format (FORMAT_MASK (OLDGNU_FORMAT
)
2225 | FORMAT_MASK (GNU_FORMAT
)
2226 | FORMAT_MASK (POSIX_FORMAT
));
2228 if (occurrence_option
)
2230 if (!args
.input_files
)
2232 _("--occurrence is meaningless without a file list")));
2233 if (subcommand_option
!= DELETE_SUBCOMMAND
2234 && subcommand_option
!= DIFF_SUBCOMMAND
2235 && subcommand_option
!= EXTRACT_SUBCOMMAND
2236 && subcommand_option
!= LIST_SUBCOMMAND
)
2238 _("--occurrence cannot be used in the requested operation mode")));
2241 if (seekable_archive
&& subcommand_option
== DELETE_SUBCOMMAND
)
2243 /* The current code in delete.c is based on the assumption that
2244 skip_member() reads all data from the archive. So, we should
2245 make sure it won't use seeks. On the other hand, the same code
2246 depends on the ability to backspace a record in the archive,
2247 so setting seekable_archive to false is technically incorrect.
2248 However, it is tested only in skip_member(), so it's not a
2250 seekable_archive
= false;
2253 if (archive_names
== 0)
2255 /* If no archive file name given, try TAPE from the environment, or
2256 else, DEFAULT_ARCHIVE from the configuration process. */
2259 archive_name_array
[0] = getenv ("TAPE");
2260 if (! archive_name_array
[0])
2261 archive_name_array
[0] = DEFAULT_ARCHIVE
;
2264 /* Allow multiple archives only with `-M'. */
2266 if (archive_names
> 1 && !multi_volume_option
)
2268 _("Multiple archive files require `-M' option")));
2270 if (listed_incremental_option
2271 && NEWER_OPTION_INITIALIZED (newer_mtime_option
))
2273 _("Cannot combine --listed-incremental with --newer")));
2275 if (volume_label_option
)
2277 if (archive_format
== GNU_FORMAT
|| archive_format
== OLDGNU_FORMAT
)
2279 size_t volume_label_max_len
=
2280 (sizeof current_header
->header
.name
2281 - 1 /* for trailing '\0' */
2282 - (multi_volume_option
2283 ? (sizeof " Volume "
2284 - 1 /* for null at end of " Volume " */
2285 + INT_STRLEN_BOUND (int) /* for volume number */
2286 - 1 /* for sign, as 0 <= volno */)
2288 if (volume_label_max_len
< strlen (volume_label_option
))
2290 ngettext ("%s: Volume label is too long (limit is %lu byte)",
2291 "%s: Volume label is too long (limit is %lu bytes)",
2292 volume_label_max_len
),
2293 quotearg_colon (volume_label_option
),
2294 (unsigned long) volume_label_max_len
));
2297 Label length in PAX format is limited by the volume size. */
2302 if (multi_volume_option
)
2303 USAGE_ERROR ((0, 0, _("Cannot verify multi-volume archives")));
2304 if (use_compress_program_option
)
2305 USAGE_ERROR ((0, 0, _("Cannot verify compressed archives")));
2308 if (use_compress_program_option
)
2310 if (multi_volume_option
)
2311 USAGE_ERROR ((0, 0, _("Cannot use multi-volume compressed archives")));
2312 if (subcommand_option
== UPDATE_SUBCOMMAND
2313 || subcommand_option
== APPEND_SUBCOMMAND
2314 || subcommand_option
== DELETE_SUBCOMMAND
)
2315 USAGE_ERROR ((0, 0, _("Cannot update compressed archives")));
2316 if (subcommand_option
== CAT_SUBCOMMAND
)
2317 USAGE_ERROR ((0, 0, _("Cannot concatenate compressed archives")));
2320 /* It is no harm to use --pax-option on non-pax archives in archive
2321 reading mode. It may even be useful, since it allows to override
2322 file attributes from tar headers. Therefore I allow such usage.
2325 && archive_format
!= POSIX_FORMAT
2326 && (subcommand_option
!= EXTRACT_SUBCOMMAND
2327 || subcommand_option
!= DIFF_SUBCOMMAND
2328 || subcommand_option
!= LIST_SUBCOMMAND
))
2329 USAGE_ERROR ((0, 0, _("--pax-option can be used only on POSIX archives")));
2331 /* If ready to unlink hierarchies, so we are for simpler files. */
2332 if (recursive_unlink_option
)
2333 old_files_option
= UNLINK_FIRST_OLD_FILES
;
2336 if (test_label_option
)
2338 /* --test-label is silent if the user has specified the label name to
2340 if (!args
.input_files
)
2343 else if (utc_option
)
2346 if (tape_length_option
&& tape_length_option
< record_size
)
2347 USAGE_ERROR ((0, 0, _("Volume length cannot be less than record size")));
2349 /* Forbid using -c with no input files whatsoever. Check that `-f -',
2350 explicit or implied, is used correctly. */
2352 switch (subcommand_option
)
2354 case CREATE_SUBCOMMAND
:
2355 if (!args
.input_files
&& !files_from_option
)
2357 _("Cowardly refusing to create an empty archive")));
2358 if (args
.compress_autodetect
&& archive_names
2359 && strcmp (archive_name_array
[0], "-"))
2360 set_comression_program_by_suffix (archive_name_array
[0],
2361 use_compress_program_option
);
2364 case EXTRACT_SUBCOMMAND
:
2365 case LIST_SUBCOMMAND
:
2366 case DIFF_SUBCOMMAND
:
2367 for (archive_name_cursor
= archive_name_array
;
2368 archive_name_cursor
< archive_name_array
+ archive_names
;
2369 archive_name_cursor
++)
2370 if (!strcmp (*archive_name_cursor
, "-"))
2371 request_stdin ("-f");
2374 case CAT_SUBCOMMAND
:
2375 case UPDATE_SUBCOMMAND
:
2376 case APPEND_SUBCOMMAND
:
2377 for (archive_name_cursor
= archive_name_array
;
2378 archive_name_cursor
< archive_name_array
+ archive_names
;
2379 archive_name_cursor
++)
2380 if (!strcmp (*archive_name_cursor
, "-"))
2382 _("Options `-Aru' are incompatible with `-f -'")));
2388 /* Initialize stdlis */
2389 if (index_file_name
)
2391 stdlis
= fopen (index_file_name
, "w");
2393 open_error (index_file_name
);
2396 stdlis
= to_stdout_option
? stderr
: stdout
;
2398 archive_name_cursor
= archive_name_array
;
2400 /* Prepare for generating backup names. */
2402 if (args
.backup_suffix_string
)
2403 simple_backup_suffix
= xstrdup (args
.backup_suffix_string
);
2407 backup_type
= xget_version ("--backup", args
.version_control_string
);
2408 /* No backup is needed either if explicitely disabled or if
2409 the extracted files are not being written to disk. */
2410 if (backup_type
== no_backups
|| EXTRACT_OVER_PIPE
)
2411 backup_option
= false;
2414 checkpoint_finish_compile ();
2417 report_textual_dates (&args
);
2423 /* Main routine for tar. */
2425 main (int argc
, char **argv
)
2428 program_name
= argv
[0];
2430 setlocale (LC_ALL
, "");
2431 bindtextdomain (PACKAGE
, LOCALEDIR
);
2432 textdomain (PACKAGE
);
2434 exit_failure
= TAREXIT_FAILURE
;
2435 exit_status
= TAREXIT_SUCCESS
;
2436 filename_terminator
= '\n';
2437 set_quoting_style (0, DEFAULT_QUOTING_STYLE
);
2439 /* Make sure we have first three descriptors available */
2442 /* Pre-allocate a few structures. */
2444 allocated_archive_names
= 10;
2445 archive_name_array
=
2446 xmalloc (sizeof (const char *) * allocated_archive_names
);
2449 obstack_init (&argv_stk
);
2452 /* System V fork+wait does not work if SIGCHLD is ignored. */
2453 signal (SIGCHLD
, SIG_DFL
);
2456 /* Decode options. */
2458 decode_options (argc
, argv
);
2462 /* Main command execution. */
2464 if (volno_file_option
)
2465 init_volume_number ();
2467 switch (subcommand_option
)
2469 case UNKNOWN_SUBCOMMAND
:
2471 _("You must specify one of the `-Acdtrux' options")));
2473 case CAT_SUBCOMMAND
:
2474 case UPDATE_SUBCOMMAND
:
2475 case APPEND_SUBCOMMAND
:
2479 case DELETE_SUBCOMMAND
:
2480 delete_archive_members ();
2483 case CREATE_SUBCOMMAND
:
2487 case EXTRACT_SUBCOMMAND
:
2489 read_and (extract_archive
);
2491 /* FIXME: should extract_finish () even if an ordinary signal is
2497 case LIST_SUBCOMMAND
:
2498 read_and (list_archive
);
2501 case DIFF_SUBCOMMAND
:
2503 read_and (diff_archive
);
2508 print_total_stats ();
2510 if (check_links_option
)
2513 if (volno_file_option
)
2514 closeout_volume_number ();
2516 /* Dispose of allocated memory, and return. */
2518 free (archive_name_array
);
2521 if (exit_status
== TAREXIT_FAILURE
)
2522 error (0, 0, _("Exiting with failure status due to previous errors"));
2524 if (stdlis
== stdout
)
2526 else if (ferror (stderr
) || fclose (stderr
) != 0)
2527 exit_status
= TAREXIT_FAILURE
;
2533 tar_stat_init (struct tar_stat_info
*st
)
2535 memset (st
, 0, sizeof (*st
));
2539 tar_stat_destroy (struct tar_stat_info
*st
)
2541 free (st
->orig_file_name
);
2542 free (st
->file_name
);
2543 free (st
->link_name
);
2546 free (st
->sparse_map
);
2548 xheader_destroy (&st
->xhdr
);
2549 memset (st
, 0, sizeof (*st
));
2552 /* Format mask for all available formats that support nanosecond
2553 timestamp resolution. */
2554 #define NS_PRECISION_FORMAT_MASK FORMAT_MASK (POSIX_FORMAT)
2556 /* Same as timespec_cmp, but ignore nanoseconds if current archive
2557 format does not provide sufficient resolution. */
2559 tar_timespec_cmp (struct timespec a
, struct timespec b
)
2561 if (!(FORMAT_MASK (current_format
) & NS_PRECISION_FORMAT_MASK
))
2562 a
.tv_nsec
= b
.tv_nsec
= 0;
2563 return timespec_cmp (a
, b
);