summaryrefslogtreecommitdiffstats
path: root/contrib/tar/src/tar.c
diff options
context:
space:
mode:
Diffstat (limited to 'contrib/tar/src/tar.c')
-rw-r--r--contrib/tar/src/tar.c1366
1 files changed, 0 insertions, 1366 deletions
diff --git a/contrib/tar/src/tar.c b/contrib/tar/src/tar.c
deleted file mode 100644
index 7d872fa..0000000
--- a/contrib/tar/src/tar.c
+++ /dev/null
@@ -1,1366 +0,0 @@
-/* A tar (tape archiver) program.
-
- Copyright (C) 1988, 1992, 1993, 1994, 1995, 1996, 1997, 1999, 2000, 2001
- Free Software Foundation, Inc.
-
- Written by John Gilmore, starting 1985-08-25.
-
- This program is free software; you can redistribute it and/or modify it
- under the terms of the GNU General Public License as published by the
- Free Software Foundation; either version 2, or (at your option) any later
- version.
-
- This program is distributed in the hope that it will be useful, but
- WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
- Public License for more details.
-
- You should have received a copy of the GNU General Public License along
- with this program; if not, write to the Free Software Foundation, Inc.,
- 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
-
-/* $FreeBSD$ */
-
-#include "system.h"
-
-#include <fnmatch.h>
-#include <getopt.h>
-
-#include <signal.h>
-#if ! defined SIGCHLD && defined SIGCLD
-# define SIGCHLD SIGCLD
-#endif
-
-/* The following causes "common.h" to produce definitions of all the global
- variables, rather than just "extern" declarations of them. GNU tar does
- depend on the system loader to preset all GLOBAL variables to neutral (or
- zero) values; explicit initialization is usually not done. */
-#define GLOBAL
-#include "common.h"
-
-#include <print-copyr.h>
-#include <localedir.h>
-#include <prepargs.h>
-#include <quotearg.h>
-#include <xstrtol.h>
-
-time_t get_date ();
-
-/* Local declarations. */
-
-#ifndef DEFAULT_ARCHIVE
-# define DEFAULT_ARCHIVE "tar.out"
-#endif
-
-#ifndef DEFAULT_BLOCKING
-# define DEFAULT_BLOCKING 20
-#endif
-
-static void usage PARAMS ((int)) __attribute__ ((noreturn));
-
-/* Miscellaneous. */
-
-/* Name of option using stdin. */
-static const char *stdin_used_by;
-
-/* Doesn't return if stdin already requested. */
-void
-request_stdin (const char *option)
-{
- if (stdin_used_by)
- USAGE_ERROR ((0, 0, _("Options `-%s' and `-%s' both want standard input"),
- stdin_used_by, option));
-
- stdin_used_by = option;
-}
-
-/* Returns true if and only if the user typed 'y' or 'Y'. */
-int
-confirm (const char *message_action, const char *message_name)
-{
- static FILE *confirm_file;
- static int confirm_file_EOF;
-
- if (!confirm_file)
- {
- if (archive == 0 || stdin_used_by)
- {
- confirm_file = fopen (TTY_NAME, "r");
- if (! confirm_file)
- open_fatal (TTY_NAME);
- }
- else
- {
- request_stdin ("-w");
- confirm_file = stdin;
- }
- }
-
- fprintf (stdlis, "%s %s?", message_action, quote (message_name));
- fflush (stdlis);
-
- {
- int reply = confirm_file_EOF ? EOF : getc (confirm_file);
- int character;
-
- for (character = reply;
- character != '\n';
- character = getc (confirm_file))
- if (character == EOF)
- {
- confirm_file_EOF = 1;
- fputc ('\n', stdlis);
- fflush (stdlis);
- break;
- }
- return reply == 'y' || reply == 'Y';
- }
-}
-
-/* Options. */
-
-/* For long options that unconditionally set a single flag, we have getopt
- do it. For the others, we share the code for the equivalent short
- named option, the name of which is stored in the otherwise-unused `val'
- field of the `struct option'; for long options that have no equivalent
- short option, we use non-characters as pseudo short options,
- starting at CHAR_MAX + 1 and going upwards. */
-
-enum
-{
- ANCHORED_OPTION = CHAR_MAX + 1,
- BACKUP_OPTION,
- DELETE_OPTION,
- EXCLUDE_OPTION,
- GROUP_OPTION,
- IGNORE_CASE_OPTION,
- MODE_OPTION,
- NEWER_MTIME_OPTION,
- NO_ANCHORED_OPTION,
- NO_IGNORE_CASE_OPTION,
- NO_WILDCARDS_OPTION,
- NO_WILDCARDS_MATCH_SLASH_OPTION,
- NULL_OPTION,
- OVERWRITE_OPTION,
- OVERWRITE_DIR_OPTION,
- OWNER_OPTION,
- POSIX_OPTION,
- PRESERVE_OPTION,
- RECORD_SIZE_OPTION,
- RSH_COMMAND_OPTION,
- SUFFIX_OPTION,
- USE_COMPRESS_PROGRAM_OPTION,
- VOLNO_FILE_OPTION,
- WILDCARDS_OPTION,
- WILDCARDS_MATCH_SLASH_OPTION,
-
- /* Some cleanup is being made in GNU tar long options. Using old names is
- allowed for a while, but will also send a warning to stderr. Take old
- names out in 1.14, or in summer 1997, whichever happens last. */
-
- OBSOLETE_ABSOLUTE_NAMES,
- OBSOLETE_BLOCK_COMPRESS,
- OBSOLETE_BLOCKING_FACTOR,
- OBSOLETE_BLOCK_NUMBER,
- OBSOLETE_READ_FULL_RECORDS,
- OBSOLETE_TOUCH,
- OBSOLETE_VERSION_CONTROL
-};
-
-/* If nonzero, display usage information and exit. */
-static int show_help;
-
-/* If nonzero, print the version on standard output and exit. */
-static int show_version;
-
-static struct option long_options[] =
-{
- {"absolute-names", no_argument, 0, 'P'},
- {"absolute-paths", no_argument, 0, OBSOLETE_ABSOLUTE_NAMES},
- {"after-date", required_argument, 0, 'N'},
- {"anchored", no_argument, 0, ANCHORED_OPTION},
- {"append", no_argument, 0, 'r'},
- {"atime-preserve", no_argument, &atime_preserve_option, 1},
- {"backup", optional_argument, 0, BACKUP_OPTION},
- {"block-compress", no_argument, 0, OBSOLETE_BLOCK_COMPRESS},
- {"block-number", no_argument, 0, 'R'},
- {"block-size", required_argument, 0, OBSOLETE_BLOCKING_FACTOR},
- {"blocking-factor", required_argument, 0, 'b'},
- {"bzip", no_argument, 0, 'j'},
- {"bzip2", no_argument, 0, 'j'},
- {"bunzip2", no_argument, 0, 'j'},
- {"catenate", no_argument, 0, 'A'},
- {"checkpoint", no_argument, &checkpoint_option, 1},
- {"compare", no_argument, 0, 'd'},
- {"compress", no_argument, 0, 'Z'},
- {"concatenate", no_argument, 0, 'A'},
- {"confirmation", no_argument, 0, 'w'},
- /* FIXME: --selective as a synonym for --confirmation? */
- {"create", no_argument, 0, 'c'},
- {"delete", no_argument, 0, DELETE_OPTION},
- {"dereference", no_argument, 0, 'h'},
- {"diff", no_argument, 0, 'd'},
- {"directory", required_argument, 0, 'C'},
- {"exclude", required_argument, 0, EXCLUDE_OPTION},
- {"exclude-from", required_argument, 0, 'X'},
- {"extract", no_argument, 0, 'x'},
- {"fast-read", no_argument, &fast_read_option, 1},
- {"file", required_argument, 0, 'f'},
- {"files-from", required_argument, 0, 'T'},
- {"force-local", no_argument, &force_local_option, 1},
- {"get", no_argument, 0, 'x'},
- {"group", required_argument, 0, GROUP_OPTION},
- {"gunzip", no_argument, 0, 'z'},
- {"gzip", no_argument, 0, 'z'},
- {"help", no_argument, &show_help, 1},
- {"ignore-case", no_argument, 0, IGNORE_CASE_OPTION},
- {"ignore-failed-read", no_argument, &ignore_failed_read_option, 1},
- {"ignore-zeros", no_argument, 0, 'i'},
- /* FIXME: --ignore-end as a new name for --ignore-zeros? */
- {"incremental", no_argument, 0, 'G'},
- {"info-script", required_argument, 0, 'F'},
- {"interactive", no_argument, 0, 'w'},
- {"keep-old-files", no_argument, 0, 'k'},
- {"label", required_argument, 0, 'V'},
- {"list", no_argument, 0, 't'},
- {"listed-incremental", required_argument, 0, 'g'},
- {"mode", required_argument, 0, MODE_OPTION},
- {"modification-time", no_argument, 0, OBSOLETE_TOUCH},
- {"multi-volume", no_argument, 0, 'M'},
- {"new-volume-script", required_argument, 0, 'F'},
- {"newer", required_argument, 0, 'N'},
- {"newer-mtime", required_argument, 0, NEWER_MTIME_OPTION},
- {"null", no_argument, 0, NULL_OPTION},
- {"no-anchored", no_argument, 0, NO_ANCHORED_OPTION},
- {"no-ignore-case", no_argument, 0, NO_IGNORE_CASE_OPTION},
- {"no-wildcards", no_argument, 0, NO_WILDCARDS_OPTION},
- {"no-wildcards-match-slash", no_argument, 0, NO_WILDCARDS_MATCH_SLASH_OPTION},
- {"norecurse", no_argument, &recursion_option, 0},
- {"no-recursion", no_argument, &recursion_option, 0},
- {"no-same-owner", no_argument, &same_owner_option, -1},
- {"no-same-permissions", no_argument, &same_permissions_option, -1},
- {"numeric-owner", no_argument, &numeric_owner_option, 1},
- {"old-archive", no_argument, 0, 'o'},
- {"one-file-system", no_argument, 0, 'l'},
- {"overwrite", no_argument, 0, OVERWRITE_OPTION},
- {"overwrite-dir", no_argument, 0, OVERWRITE_DIR_OPTION},
- {"owner", required_argument, 0, OWNER_OPTION},
- {"portability", no_argument, 0, 'o'},
- {"posix", no_argument, 0, POSIX_OPTION},
- {"preserve", no_argument, 0, PRESERVE_OPTION},
- {"preserve-order", no_argument, 0, 's'},
- {"preserve-permissions", no_argument, 0, 'p'},
- {"recursion", no_argument, &recursion_option, FNM_LEADING_DIR},
- {"recursive-unlink", no_argument, &recursive_unlink_option, 1},
- {"read-full-blocks", no_argument, 0, OBSOLETE_READ_FULL_RECORDS},
- {"read-full-records", no_argument, 0, 'B'},
- /* FIXME: --partial-blocks might be a synonym for --read-full-records? */
- {"record-number", no_argument, 0, OBSOLETE_BLOCK_NUMBER},
- {"record-size", required_argument, 0, RECORD_SIZE_OPTION},
- {"remove-files", no_argument, &remove_files_option, 1},
- {"rsh-command", required_argument, 0, RSH_COMMAND_OPTION},
- {"same-order", no_argument, 0, 's'},
- {"same-owner", no_argument, &same_owner_option, 1},
- {"same-permissions", no_argument, 0, 'p'},
- {"show-omitted-dirs", no_argument, &show_omitted_dirs_option, 1},
- {"sparse", no_argument, 0, 'S'},
- {"starting-file", required_argument, 0, 'K'},
- {"suffix", required_argument, 0, SUFFIX_OPTION},
- {"tape-length", required_argument, 0, 'L'},
- {"to-stdout", no_argument, 0, 'O'},
- {"totals", no_argument, &totals_option, 1},
- {"touch", no_argument, 0, 'm'},
- {"uncompress", no_argument, 0, 'Z'},
- {"ungzip", no_argument, 0, 'z'},
- {"unlink", no_argument, 0, 'U'},
- {"unlink-first", no_argument, 0, 'U'},
- {"update", no_argument, 0, 'u'},
- {"use-compress-program", required_argument, 0, USE_COMPRESS_PROGRAM_OPTION},
- {"verbose", no_argument, 0, 'v'},
- {"verify", no_argument, 0, 'W'},
- {"version", no_argument, &show_version, 1},
- {"version-control", required_argument, 0, OBSOLETE_VERSION_CONTROL},
- {"volno-file", required_argument, 0, VOLNO_FILE_OPTION},
- {"wildcards", no_argument, 0, WILDCARDS_OPTION},
- {"wildcards-match-slash", no_argument, 0, WILDCARDS_MATCH_SLASH_OPTION},
-
- {0, 0, 0, 0}
-};
-
-/* Print a usage message and exit with STATUS. */
-static void
-usage (int status)
-{
- if (status != TAREXIT_SUCCESS)
- fprintf (stderr, _("Try `%s --help' for more information.\n"),
- program_name);
- else
- {
- fputs (_("\
-GNU `tar' saves many files together into a single tape or disk archive, and\n\
-can restore individual files from the archive.\n"),
- stdout);
- printf (_("\nUsage: %s [OPTION]... [FILE]...\n\
-\n\
-Examples:\n\
- %s -cf archive.tar foo bar # Create archive.tar from files foo and bar.\n\
- %s -tvf archive.tar # List all files in archive.tar verbosely.\n\
- %s -xf archive.tar # Extract all files from archive.tar.\n"),
- program_name, program_name, program_name, program_name);
- fputs (_("\
-\n\
-If a long option shows an argument as mandatory, then it is mandatory\n\
-for the equivalent short option also. Similarly for optional arguments.\n"),
- stdout);
- fputs(_("\
-\n\
-Main operation mode:\n\
- -t, --list list the contents of an archive\n\
- -x, --extract, --get extract files from an archive\n\
- -c, --create create a new archive\n\
- -d, --diff, --compare find differences between archive and file system\n\
- -r, --append append files to the end of an archive\n\
- -u, --update only append files newer than copy in archive\n\
- -A, --catenate append tar files to an archive\n\
- --concatenate same as -A\n\
- --delete delete from the archive (not on mag tapes!)\n"),
- stdout);
- fputs (_("\
-\n\
-Operation modifiers:\n\
- -W, --verify attempt to verify the archive after writing it\n\
- --remove-files remove files after adding them to the archive\n\
- -k, --keep-old-files don't replace existing files when extracting\n\
- --overwrite overwrite existing files when extracting\n\
- --overwrite-dir overwrite directory metadata when extracting\n\
- -U, --unlink,\n\
- --unlink-first remove each file prior to extracting over it\n\
- --recursive-unlink empty hierarchies prior to extracting directory\n\
- -S, --sparse handle sparse files efficiently\n\
- -O, --to-stdout extract files to standard output\n\
- -G, --incremental handle old GNU-format incremental backup\n\
- -g, --listed-incremental=FILE\n\
- handle new GNU-format incremental backup\n\
- --ignore-failed-read do not exit with nonzero on unreadable files\n\
- --fast-read stop after desired names in archive have been found\n"),
- stdout);
- fputs (_("\
-\n\
-Handling of file attributes:\n\
- --owner=NAME force NAME as owner for added files\n\
- --group=NAME force NAME as group for added files\n\
- --mode=CHANGES force (symbolic) mode CHANGES for added files\n\
- --atime-preserve don't change access times on dumped files\n\
- -m, --modification-time don't extract file modified time\n\
- --same-owner try extracting files with the same ownership\n\
- --show-omitted-dirs show omitted directories while processing the\n\
- archive\n\
- --no-same-owner extract files as yourself\n\
- --numeric-owner always use numbers for user/group names\n\
- -p, --same-permissions extract permissions information\n\
- --no-same-permissions do not extract permissions information\n\
- --preserve-permissions same as -p\n\
- -s, --same-order sort names to extract to match archive\n\
- --preserve-order same as -s\n\
- --preserve same as both -p and -s\n"),
- stdout);
- fputs (_("\
-\n\
-Device selection and switching:\n\
- -f, --file=ARCHIVE use archive file or device ARCHIVE\n\
- --force-local archive file is local even if it has a colon\n\
- --rsh-command=COMMAND use remote COMMAND instead of rsh\n\
- -[0-7][lmh] specify drive and density\n\
- -M, --multi-volume create/list/extract multi-volume archive\n\
- -L, --tape-length=NUM change tape after writing NUM x 1024 bytes\n\
- -F, --info-script=FILE run script at end of each tape (implies -M)\n\
- --new-volume-script=FILE same as -F FILE\n\
- --volno-file=FILE use/update the volume number in FILE\n"),
- stdout);
- fputs (_("\
-\n\
-Device blocking:\n\
- -b, --blocking-factor=BLOCKS BLOCKS x 512 bytes per record\n\
- --record-size=SIZE SIZE bytes per record, multiple of 512\n\
- -i, --ignore-zeros ignore zeroed blocks in archive (means EOF)\n\
- -B, --read-full-records reblock as we read (for 4.2BSD pipes)\n"),
- stdout);
- fputs (_("\
-\n\
-Archive format selection:\n\
- -V, --label=NAME create archive with volume name NAME\n\
- PATTERN at list/extract time, a globbing PATTERN\n\
- -o, --old-archive, --portability write a V7 format archive\n\
- --posix write a POSIX format archive\n\
- -j, -y, --bzip, --bzip2, --bunzip2 filter the archive through bzip2\n\
- -z, --gzip, --ungzip filter the archive through gzip\n\
- -Z, --compress, --uncompress filter the archive through compress\n\
- --use-compress-program=PROG filter through PROG (must accept -d)\n"),
- stdout);
- fputs (_("\
-\n\
-Local file selection:\n\
- -C, --directory=DIR change to directory DIR\n\
- -T, -I, --files-from=NAME get names to extract or create from file NAME\n\
- --null -T reads null-terminated names, disable -C\n\
- --exclude=PATTERN exclude files, given as a a globbing PATTERN\n\
- -X, --exclude-from=FILE exclude patterns listed in FILE\n\
- --anchored exclude patterns match file name start (default)\n\
- --no-anchored exclude patterns match after any /\n\
- --ignore-case exclusion ignores case\n\
- --no-ignore-case exclusion is case sensitive (default)\n\
- --wildcards exclude patterns use wildcards (default)\n\
- --no-wildcards exclude patterns are plain strings\n\
- --wildcards-match-slash exclude pattern wildcards match '/' (default)\n\
- --no-wildcards-match-slash exclude pattern wildcards do not match '/'\n\
- -P, --absolute-names don't strip leading `/'s from file names\n\
- -h, --dereference dump instead the files symlinks point to\n\
- -n, --norecurse\n\
- --no-recursion avoid descending automatically in directories\n\
- -l, --one-file-system stay in local file system when creating archive\n\
- -K, --starting-file=NAME begin at file NAME in the archive\n"),
- stdout);
-#if !MSDOS
- fputs (_("\
- -N, --newer=DATE only store files with creation time newer than\n\
- DATE\n\
- --newer-mtime=DATE only store files with modification time newer\n\
- than DATE\n\
- --after-date=DATE same as -N\n"),
- stdout);
-#endif
- fputs (_("\
- --backup[=CONTROL] backup before removal, choose version control\n\
- --suffix=SUFFIX backup before removal, override usual suffix\n"),
- stdout);
- fputs (_("\
-\n\
-Informative output:\n\
- --help print this help, then exit\n\
- --version print tar program version number, then exit\n\
- -v, --verbose verbosely list files processed\n\
- --checkpoint print number of buffer reads/writes\n\
- --totals print total bytes written while creating archive\n\
- -R, --block-number show block number within archive with each message\n\
- -w, --interactive ask for confirmation for every action\n\
- --confirmation same as -w\n"),
- stdout);
- fputs (_("\
-\n\
-The backup suffix is `~', unless set with --suffix or SIMPLE_BACKUP_SUFFIX.\n\
-The version control may be set with --backup or VERSION_CONTROL, values are:\n\
-\n\
- t, numbered make numbered backups\n\
- nil, existing numbered if numbered backups exist, simple otherwise\n\
- never, simple always make simple backups\n"),
- stdout);
- printf (_("\
-\n\
-GNU tar cannot read nor produce `--posix' archives. If POSIXLY_CORRECT\n\
-is set in the environment, GNU extensions are disallowed with `--posix'.\n\
-Support for POSIX is only partially implemented, don't count on it yet.\n\
-ARCHIVE may be FILE, HOST:FILE or USER@HOST:FILE; DATE may be a textual date\n\
-or a file name starting with `/' or `.', in which case the file's date is used.\n\
-*This* `tar' defaults to `-f%s -b%d'.\n"),
- DEFAULT_ARCHIVE, DEFAULT_BLOCKING);
- fputs (_("\nReport bugs to <bug-tar@gnu.org>.\n"), stdout);
- }
- exit (status);
-}
-
-/* Parse the options for tar. */
-
-/* Available option letters are DEHIJQY and aenqy. Some are reserved:
-
- e exit immediately with a nonzero exit status if unexpected errors occur
- E use extended headers (draft POSIX headers, that is)
- I same as T (for compatibility with Solaris tar)
- n the archive is quickly seekable, so don't worry about random seeks
- q stop after extracting the first occurrence of the named file
- y per-file gzip compression
- Y per-block gzip compression */
-
-#define OPTION_STRING \
- "-01234567ABC:F:GI:K:L:MnN:OPRST:UV:WX:Zb:cdf:g:hijklmoprstuvwxyz"
-
-static void
-set_subcommand_option (enum subcommand subcommand)
-{
- if (subcommand_option != UNKNOWN_SUBCOMMAND
- && subcommand_option != subcommand)
- USAGE_ERROR ((0, 0,
- _("You may not specify more than one `-Acdtrux' option")));
-
- subcommand_option = subcommand;
-}
-
-static void
-set_use_compress_program_option (const char *string)
-{
- if (use_compress_program_option && strcmp (use_compress_program_option, string) != 0)
- USAGE_ERROR ((0, 0, _("Conflicting compression options")));
-
- use_compress_program_option = string;
-}
-
-static void
-decode_options (int argc, char **argv)
-{
- int optchar; /* option letter */
- int input_files; /* number of input files */
- const char *backup_suffix_string;
- const char *version_control_string = 0;
- int exclude_options = EXCLUDE_WILDCARDS;
-
- /* Set some default option values. */
-
- subcommand_option = UNKNOWN_SUBCOMMAND;
- archive_format = DEFAULT_FORMAT;
- blocking_factor = DEFAULT_BLOCKING;
- record_size = DEFAULT_BLOCKING * BLOCKSIZE;
- excluded = new_exclude ();
- newer_mtime_option = TYPE_MINIMUM (time_t);
- recursion_option = FNM_LEADING_DIR;
- namelist_freed = 0;
-
- owner_option = -1;
- group_option = -1;
-
- backup_suffix_string = getenv ("SIMPLE_BACKUP_SUFFIX");
-
- /* Convert old-style tar call by exploding option element and rearranging
- options accordingly. */
-
- if (argc > 1 && argv[1][0] != '-')
- {
- int new_argc; /* argc value for rearranged arguments */
- char **new_argv; /* argv value for rearranged arguments */
- char *const *in; /* cursor into original argv */
- char **out; /* cursor into rearranged argv */
- const char *letter; /* cursor into old option letters */
- char buffer[3]; /* constructed option buffer */
- const char *cursor; /* cursor in OPTION_STRING */
-
- /* Initialize a constructed option. */
-
- buffer[0] = '-';
- buffer[2] = '\0';
-
- /* Allocate a new argument array, and copy program name in it. */
-
- new_argc = argc - 1 + strlen (argv[1]);
- new_argv = xmalloc ((new_argc + 1) * sizeof (char *));
- in = argv;
- out = new_argv;
- *out++ = *in++;
-
- /* Copy each old letter option as a separate option, and have the
- corresponding argument moved next to it. */
-
- for (letter = *in++; *letter; letter++)
- {
- buffer[1] = *letter;
- *out++ = xstrdup (buffer);
- cursor = strchr (OPTION_STRING, *letter);
- if (cursor && cursor[1] == ':')
- {
- if (in < argv + argc)
- *out++ = *in++;
- else
- USAGE_ERROR ((0, 0, _("Old option `%c' requires an argument."),
- *letter));
- }
- }
-
- /* Copy all remaining options. */
-
- while (in < argv + argc)
- *out++ = *in++;
-
- /* And NULL terminate the argv[] array */
- *out++ = NULL;
-
- /* Replace the old option list by the new one. */
-
- argc = new_argc;
- argv = new_argv;
- }
-
- /* Parse all options and non-options as they appear. */
-
- input_files = 0;
-
- prepend_default_options (getenv ("TAR_OPTIONS"), &argc, &argv);
-
- while (optchar = getopt_long (argc, argv, OPTION_STRING, long_options, 0),
- optchar != -1)
- switch (optchar)
- {
- case '?':
- usage (TAREXIT_FAILURE);
-
- case 0:
- break;
-
- case 1:
- /* File name or non-parsed option, because of RETURN_IN_ORDER
- ordering triggered by the leading dash in OPTION_STRING. */
-
- name_add (optarg);
- input_files++;
- break;
-
- case 'A':
- set_subcommand_option (CAT_SUBCOMMAND);
- break;
-
- case OBSOLETE_BLOCK_COMPRESS:
- WARN ((0, 0, _("Obsolete option, now implied by --blocking-factor")));
- break;
-
- case OBSOLETE_BLOCKING_FACTOR:
- WARN ((0, 0, _("Obsolete option name replaced by --blocking-factor")));
- /* Fall through. */
-
- case 'b':
- {
- uintmax_t u;
- if (! (xstrtoumax (optarg, 0, 10, &u, "") == LONGINT_OK
- && u == (blocking_factor = u)
- && 0 < blocking_factor
- && u == (record_size = u * BLOCKSIZE) / BLOCKSIZE))
- USAGE_ERROR ((0, 0, "%s: %s", quotearg_colon (optarg),
- _("Invalid blocking factor")));
- }
- break;
-
- case OBSOLETE_READ_FULL_RECORDS:
- WARN ((0, 0,
- _("Obsolete option name replaced by --read-full-records")));
- /* Fall through. */
-
- case 'B':
- /* Try to reblock input records. For reading 4.2BSD pipes. */
-
- /* It would surely make sense to exchange -B and -R, but it seems
- that -B has been used for a long while in Sun tar ans most
- BSD-derived systems. This is a consequence of the block/record
- terminology confusion. */
-
- read_full_records_option = 1;
- break;
-
- case 'c':
- set_subcommand_option (CREATE_SUBCOMMAND);
- break;
-
- case 'C':
- name_add ("-C");
- name_add (optarg);
- break;
-
- case 'd':
- set_subcommand_option (DIFF_SUBCOMMAND);
- break;
-
- case 'f':
- if (archive_names == allocated_archive_names)
- {
- allocated_archive_names *= 2;
- archive_name_array =
- xrealloc (archive_name_array,
- sizeof (const char *) * allocated_archive_names);
- }
- archive_name_array[archive_names++] = optarg;
- break;
-
- case 'F':
- /* Since -F is only useful with -M, make it implied. Run this
- script at the end of each tape. */
-
- info_script_option = optarg;
- multi_volume_option = 1;
- break;
-
- case 'g':
- listed_incremental_option = optarg;
- after_date_option = 1;
- /* Fall through. */
-
- case 'G':
- /* We are making an incremental dump (FIXME: are we?); save
- directories at the beginning of the archive, and include in each
- directory its contents. */
-
- incremental_option = 1;
- break;
-
- case 'h':
- /* Follow symbolic links. */
-
- dereference_option = 1;
- break;
-
- case 'i':
- /* Ignore zero blocks (eofs). This can't be the default,
- because Unix tar writes two blocks of zeros, then pads out
- the record with garbage. */
-
- ignore_zeros_option = 1;
- break;
-
- case 'j':
- case 'y':
- set_use_compress_program_option ("bzip2");
- break;
-
- case 'k':
- /* Don't replace existing files. */
- old_files_option = KEEP_OLD_FILES;
- break;
-
- case 'K':
- starting_file_option = 1;
- addname (optarg, 0);
- break;
-
- case 'l':
- /* When dumping directories, don't dump files/subdirectories
- that are on other filesystems. */
-
- one_file_system_option = 1;
- break;
-
- case 'L':
- {
- uintmax_t u;
- if (xstrtoumax (optarg, 0, 10, &u, "") != LONGINT_OK)
- USAGE_ERROR ((0, 0, "%s: %s", quotearg_colon (optarg),
- _("Invalid tape length")));
- tape_length_option = 1024 * (tarlong) u;
- multi_volume_option = 1;
- }
- break;
-
- case OBSOLETE_TOUCH:
- WARN ((0, 0, _("Obsolete option name replaced by --touch")));
- /* Fall through. */
-
- case 'm':
- touch_option = 1;
- break;
-
- case 'M':
- /* Make multivolume archive: when we can't write any more into
- the archive, re-open it, and continue writing. */
-
- multi_volume_option = 1;
- break;
-
- case 'n':
- recursion_option = 0;
- break;
-
-#if !MSDOS
- case 'N':
- after_date_option = 1;
- /* Fall through. */
-
- case NEWER_MTIME_OPTION:
- if (newer_mtime_option != TYPE_MINIMUM (time_t))
- USAGE_ERROR ((0, 0, _("More than one threshold date")));
-
- if (FILESYSTEM_PREFIX_LEN (optarg) != 0
- || ISSLASH (*optarg)
- || *optarg == '.')
- {
- struct stat st;
- if (deref_stat (dereference_option, optarg, &st) != 0)
- {
- stat_error (optarg);
- USAGE_ERROR ((0, 0, _("Date file not found")));
- }
- newer_mtime_option = st.st_mtime;
- }
- else
- {
- newer_mtime_option = get_date (optarg, 0);
- if (newer_mtime_option == (time_t) -1)
- WARN ((0, 0, _("Substituting %s for unknown date format %s"),
- tartime (newer_mtime_option), quote (optarg)));
- }
-
- break;
-#endif /* not MSDOS */
-
- case 'o':
- if (archive_format == DEFAULT_FORMAT)
- archive_format = V7_FORMAT;
- else if (archive_format != V7_FORMAT)
- USAGE_ERROR ((0, 0, _("Conflicting archive format options")));
- break;
-
- case 'O':
- to_stdout_option = 1;
- break;
-
- case 'p':
- same_permissions_option = 1;
- break;
-
- case OBSOLETE_ABSOLUTE_NAMES:
- WARN ((0, 0, _("Obsolete option name replaced by --absolute-names")));
- /* Fall through. */
-
- case 'P':
- absolute_names_option = 1;
- break;
-
- case 'r':
- set_subcommand_option (APPEND_SUBCOMMAND);
- break;
-
- case OBSOLETE_BLOCK_NUMBER:
- WARN ((0, 0, _("Obsolete option name replaced by --block-number")));
- /* Fall through. */
-
- case 'R':
- /* Print block numbers for debugging bad tar archives. */
-
- /* It would surely make sense to exchange -B and -R, but it seems
- that -B has been used for a long while in Sun tar ans most
- BSD-derived systems. This is a consequence of the block/record
- terminology confusion. */
-
- block_number_option = 1;
- break;
-
- case 's':
- /* Names to extr are sorted. */
-
- same_order_option = 1;
- break;
-
- case 'S':
- sparse_option = 1;
- break;
-
- case 't':
- set_subcommand_option (LIST_SUBCOMMAND);
- verbose_option++;
- break;
-
- case 'T':
- case 'I':
- files_from_option = optarg;
- break;
-
- case 'u':
- set_subcommand_option (UPDATE_SUBCOMMAND);
- break;
-
- case 'U':
- old_files_option = UNLINK_FIRST_OLD_FILES;
- break;
-
- case 'v':
- verbose_option++;
- break;
-
- case 'V':
- volume_label_option = optarg;
- break;
-
- case 'w':
- interactive_option = 1;
- break;
-
- case 'W':
- verify_option = 1;
- break;
-
- case 'x':
- set_subcommand_option (EXTRACT_SUBCOMMAND);
- break;
-
- case 'X':
- if (add_exclude_file (add_exclude, excluded, optarg,
- exclude_options | recursion_option, '\n')
- != 0)
- {
- int e = errno;
- FATAL_ERROR ((0, e, "%s", quotearg_colon (optarg)));
- }
- break;
-
- case 'z':
- set_use_compress_program_option ("gzip");
- break;
-
- case 'Z':
- set_use_compress_program_option ("compress");
- break;
-
- case OBSOLETE_VERSION_CONTROL:
- WARN ((0, 0, _("Obsolete option name replaced by --backup")));
- /* Fall through. */
-
- case ANCHORED_OPTION:
- exclude_options |= EXCLUDE_ANCHORED;
- break;
-
- case BACKUP_OPTION:
- backup_option = 1;
- if (optarg)
- version_control_string = optarg;
- break;
-
- case DELETE_OPTION:
- set_subcommand_option (DELETE_SUBCOMMAND);
- break;
-
- case EXCLUDE_OPTION:
- add_exclude (excluded, optarg, exclude_options | recursion_option);
- break;
-
- case IGNORE_CASE_OPTION:
- exclude_options |= FNM_CASEFOLD;
- break;
-
- case GROUP_OPTION:
- if (! (strlen (optarg) < GNAME_FIELD_SIZE
- && gname_to_gid (optarg, &group_option)))
- {
- uintmax_t g;
- if (xstrtoumax (optarg, 0, 10, &g, "") == LONGINT_OK
- && g == (gid_t) g)
- group_option = g;
- else
- FATAL_ERROR ((0, 0, "%s: %s", quotearg_colon (optarg),
- _("%s: Invalid group")));
- }
- break;
-
- case MODE_OPTION:
- mode_option
- = mode_compile (optarg,
- MODE_MASK_EQUALS | MODE_MASK_PLUS | MODE_MASK_MINUS);
- if (mode_option == MODE_INVALID)
- FATAL_ERROR ((0, 0, _("Invalid mode given on option")));
- if (mode_option == MODE_MEMORY_EXHAUSTED)
- xalloc_die ();
- break;
-
- case NO_ANCHORED_OPTION:
- exclude_options &= ~ EXCLUDE_ANCHORED;
- break;
-
- case NO_IGNORE_CASE_OPTION:
- exclude_options &= ~ FNM_CASEFOLD;
- break;
-
- case NO_WILDCARDS_OPTION:
- exclude_options &= ~ EXCLUDE_WILDCARDS;
- break;
-
- case NO_WILDCARDS_MATCH_SLASH_OPTION:
- exclude_options |= FNM_FILE_NAME;
- break;
-
- case NULL_OPTION:
- filename_terminator = '\0';
- break;
-
- case OVERWRITE_OPTION:
- old_files_option = OVERWRITE_OLD_FILES;
- break;
-
- case OVERWRITE_DIR_OPTION:
- old_files_option = OVERWRITE_OLD_DIRS;
- break;
-
- case OWNER_OPTION:
- if (! (strlen (optarg) < UNAME_FIELD_SIZE
- && uname_to_uid (optarg, &owner_option)))
- {
- uintmax_t u;
- if (xstrtoumax (optarg, 0, 10, &u, "") == LONGINT_OK
- && u == (uid_t) u)
- owner_option = u;
- else
- FATAL_ERROR ((0, 0, "%s: %s", quotearg_colon (optarg),
- _("Invalid owner")));
- }
- break;
-
- case POSIX_OPTION:
-#if OLDGNU_COMPATIBILITY
- if (archive_format == DEFAULT_FORMAT)
- archive_format = GNU_FORMAT;
- else if (archive_format != GNU_FORMAT)
- USAGE_ERROR ((0, 0, _("Conflicting archive format options")));
-#else
- if (archive_format == DEFAULT_FORMAT)
- archive_format = POSIX_FORMAT;
- else if (archive_format != POSIX_FORMAT)
- USAGE_ERROR ((0, 0, _("Conflicting archive format options")));
-#endif
- break;
-
- case PRESERVE_OPTION:
- same_permissions_option = 1;
- same_order_option = 1;
- break;
-
- case RECORD_SIZE_OPTION:
- {
- uintmax_t u;
- if (! (xstrtoumax (optarg, 0, 10, &u, "") == LONGINT_OK
- && u == (size_t) u))
- USAGE_ERROR ((0, 0, "%s: %s", quotearg_colon (optarg),
- _("Invalid record size")));
- record_size = u;
- if (record_size % BLOCKSIZE != 0)
- USAGE_ERROR ((0, 0, _("Record size must be a multiple of %d."),
- BLOCKSIZE));
- blocking_factor = record_size / BLOCKSIZE;
- }
- break;
-
- case RSH_COMMAND_OPTION:
- rsh_command_option = optarg;
- break;
-
- case SUFFIX_OPTION:
- backup_option = 1;
- backup_suffix_string = optarg;
- break;
-
- case USE_COMPRESS_PROGRAM_OPTION:
- set_use_compress_program_option (optarg);
- break;
-
- case VOLNO_FILE_OPTION:
- volno_file_option = optarg;
- break;
-
- case WILDCARDS_OPTION:
- exclude_options |= EXCLUDE_WILDCARDS;
- break;
-
- case WILDCARDS_MATCH_SLASH_OPTION:
- exclude_options &= ~ FNM_FILE_NAME;
- break;
-
- case '0':
- case '1':
- case '2':
- case '3':
- case '4':
- case '5':
- case '6':
- case '7':
-
-#ifdef DEVICE_PREFIX
- {
- int device = optchar - '0';
- int density;
- static char buf[sizeof DEVICE_PREFIX + 10];
- char *cursor;
-
- density = getopt_long (argc, argv, "lmh", 0, 0);
- strcpy (buf, DEVICE_PREFIX);
- cursor = buf + strlen (buf);
-
-#ifdef DENSITY_LETTER
-
- sprintf (cursor, "%d%c", device, density);
-
-#else /* not DENSITY_LETTER */
-
- switch (density)
- {
- case 'l':
-#ifdef LOW_NUM
- device += LOW_NUM;
-#endif
- break;
-
- case 'm':
-#ifdef MID_NUM
- device += MID_NUM;
-#else
- device += 8;
-#endif
- break;
-
- case 'h':
-#ifdef HGH_NUM
- device += HGH_NUM;
-#else
- device += 16;
-#endif
- break;
-
- default:
- usage (TAREXIT_FAILURE);
- }
- sprintf (cursor, "%d", device);
-
-#endif /* not DENSITY_LETTER */
-
- if (archive_names == allocated_archive_names)
- {
- allocated_archive_names *= 2;
- archive_name_array =
- xrealloc (archive_name_array,
- sizeof (const char *) * allocated_archive_names);
- }
- archive_name_array[archive_names++] = buf;
-
- /* FIXME: How comes this works for many archives when buf is
- not xstrdup'ed? */
- }
- break;
-
-#else /* not DEVICE_PREFIX */
-
- USAGE_ERROR ((0, 0,
- _("Options `-[0-7][lmh]' not supported by *this* tar")));
-
-#endif /* not DEVICE_PREFIX */
- }
-
- /* Handle operands after any "--" argument. */
- for (; optind < argc; optind++)
- {
- name_add (argv[optind]);
- input_files++;
- }
-
- /* Process trivial options. */
-
- if (show_version)
- {
- printf ("tar (GNU %s) %s\n", PACKAGE, VERSION);
- print_copyright ("2001 Free Software Foundation, Inc.");
- puts (_("\
-This program comes with NO WARRANTY, to the extent permitted by law.\n\
-You may redistribute it under the terms of the GNU General Public License;\n\
-see the file named COPYING for details."));
-
- puts (_("Written by John Gilmore and Jay Fenlason."));
-
- exit (TAREXIT_SUCCESS);
- }
-
- if (show_help)
- usage (TAREXIT_SUCCESS);
-
- /* Derive option values and check option consistency. */
-
- if (archive_format == DEFAULT_FORMAT)
- {
-#if OLDGNU_COMPATIBILITY
- archive_format = OLDGNU_FORMAT;
-#else
- archive_format = GNU_FORMAT;
-#endif
- }
-
- if (archive_format == GNU_FORMAT && getenv ("POSIXLY_CORRECT"))
- archive_format = POSIX_FORMAT;
-
- if ((volume_label_option
- || incremental_option || multi_volume_option || sparse_option)
- && archive_format != OLDGNU_FORMAT && archive_format != GNU_FORMAT)
- USAGE_ERROR ((0, 0,
- _("GNU features wanted on incompatible archive format")));
-
- if (archive_names == 0)
- {
- /* If no archive file name given, try TAPE from the environment, or
- else, DEFAULT_ARCHIVE from the configuration process. */
-
- archive_names = 1;
- archive_name_array[0] = getenv ("TAPE");
- if (! archive_name_array[0])
- archive_name_array[0] = DEFAULT_ARCHIVE;
- }
-
- /* Allow multiple archives only with `-M'. */
-
- if (archive_names > 1 && !multi_volume_option)
- USAGE_ERROR ((0, 0,
- _("Multiple archive files requires `-M' option")));
-
- if (listed_incremental_option
- && newer_mtime_option != TYPE_MINIMUM (time_t))
- USAGE_ERROR ((0, 0,
- _("Cannot combine --listed-incremental with --newer")));
-
- if (volume_label_option)
- {
- size_t volume_label_max_len =
- (sizeof current_header->header.name
- - 1 /* for trailing '\0' */
- - (multi_volume_option
- ? (sizeof " Volume "
- - 1 /* for null at end of " Volume " */
- + INT_STRLEN_BOUND (int) /* for volume number */
- - 1 /* for sign, as 0 <= volno */)
- : 0));
- if (volume_label_max_len < strlen (volume_label_option))
- USAGE_ERROR ((0, 0,
- _("%s: Volume label is too long (limit is %lu bytes)"),
- quotearg_colon (volume_label_option),
- (unsigned long) volume_label_max_len));
- }
-
- /* If ready to unlink hierarchies, so we are for simpler files. */
- if (recursive_unlink_option)
- old_files_option = UNLINK_FIRST_OLD_FILES;
-
- /* Forbid using -c with no input files whatsoever. Check that `-f -',
- explicit or implied, is used correctly. */
-
- switch (subcommand_option)
- {
- case CREATE_SUBCOMMAND:
- if (input_files == 0 && !files_from_option)
- USAGE_ERROR ((0, 0,
- _("Cowardly refusing to create an empty archive")));
- break;
-
- case EXTRACT_SUBCOMMAND:
- case LIST_SUBCOMMAND:
- case DIFF_SUBCOMMAND:
- for (archive_name_cursor = archive_name_array;
- archive_name_cursor < archive_name_array + archive_names;
- archive_name_cursor++)
- if (!strcmp (*archive_name_cursor, "-"))
- request_stdin ("-f");
- break;
-
- case CAT_SUBCOMMAND:
- case UPDATE_SUBCOMMAND:
- case APPEND_SUBCOMMAND:
- for (archive_name_cursor = archive_name_array;
- archive_name_cursor < archive_name_array + archive_names;
- archive_name_cursor++)
- if (!strcmp (*archive_name_cursor, "-"))
- USAGE_ERROR ((0, 0,
- _("Options `-Aru' are incompatible with `-f -'")));
-
- default:
- break;
- }
-
- archive_name_cursor = archive_name_array;
-
- /* Prepare for generating backup names. */
-
- if (backup_suffix_string)
- simple_backup_suffix = xstrdup (backup_suffix_string);
-
- if (backup_option)
- backup_type = xget_version ("--backup", version_control_string);
-}
-
-/* Tar proper. */
-
-/* Main routine for tar. */
-int
-main (int argc, char **argv)
-{
-#if HAVE_CLOCK_GETTIME
- if (clock_gettime (CLOCK_REALTIME, &start_timespec) != 0)
-#endif
- start_time = time (0);
- program_name = argv[0];
- (void) setlocale (LC_ALL, "");
- bindtextdomain (PACKAGE, LOCALEDIR);
- textdomain (PACKAGE);
-
- exit_status = TAREXIT_SUCCESS;
- filename_terminator = '\n';
- set_quoting_style (0, escape_quoting_style);
-
- /* Pre-allocate a few structures. */
-
- allocated_archive_names = 10;
- archive_name_array =
- xmalloc (sizeof (const char *) * allocated_archive_names);
- archive_names = 0;
-
-#ifdef SIGCHLD
- /* System V fork+wait does not work if SIGCHLD is ignored. */
- signal (SIGCHLD, SIG_DFL);
-#endif
-
- init_names ();
-
- /* Decode options. */
-
- decode_options (argc, argv);
- name_init (argc, argv);
-
- /* Main command execution. */
-
- if (volno_file_option)
- init_volume_number ();
-
- switch (subcommand_option)
- {
- case UNKNOWN_SUBCOMMAND:
- USAGE_ERROR ((0, 0,
- _("You must specify one of the `-Acdtrux' options")));
-
- case CAT_SUBCOMMAND:
- case UPDATE_SUBCOMMAND:
- case APPEND_SUBCOMMAND:
- update_archive ();
- break;
-
- case DELETE_SUBCOMMAND:
- delete_archive_members ();
- break;
-
- case CREATE_SUBCOMMAND:
- create_archive ();
- name_close ();
-
- if (totals_option)
- print_total_written ();
- break;
-
- case EXTRACT_SUBCOMMAND:
- extr_init ();
- read_and (extract_archive);
- extract_finish ();
- break;
-
- case LIST_SUBCOMMAND:
- read_and (list_archive);
- break;
-
- case DIFF_SUBCOMMAND:
- diff_init ();
- read_and (diff_archive);
- break;
- }
-
- if (volno_file_option)
- closeout_volume_number ();
-
- /* Dispose of allocated memory, and return. */
-
- free (archive_name_array);
- name_term ();
-
- if (stdlis == stdout && (ferror (stdout) || fclose (stdout) != 0))
- FATAL_ERROR ((0, 0, _("Error in writing to standard output")));
- if (exit_status == TAREXIT_FAILURE)
- error (0, 0, _("Error exit delayed from previous errors"));
- exit (exit_status);
-}
OpenPOWER on IntegriCloud