123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415 |
- /*-
- * Copyright 2006-2008 Colin Percival
- * All rights reserved.
- *
- * Portions of the file below are covered by the following license:
- *
- * Copyright (c) 2003-2007 Tim Kientzle
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
- * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
- * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
- * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
- * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- #include "bsdtar_platform.h"
- __FBSDID("$FreeBSD: src/usr.bin/tar/read.c,v 1.40 2008/08/21 06:41:14 kientzle Exp $");
- #ifdef HAVE_SYS_TYPES_H
- #include <sys/types.h>
- #endif
- #ifdef HAVE_SYS_PARAM_H
- #include <sys/param.h>
- #endif
- #ifdef HAVE_SYS_STAT_H
- #include <sys/stat.h>
- #endif
- #ifdef HAVE_ERRNO_H
- #include <errno.h>
- #endif
- #ifdef HAVE_GRP_H
- #include <grp.h>
- #endif
- #if HAVE_INTTYPES_H
- #include <inttypes.h>
- #endif
- #ifdef HAVE_LIMITS_H
- #include <limits.h>
- #endif
- #ifdef HAVE_PWD_H
- #include <pwd.h>
- #endif
- #include <signal.h>
- #include <stdio.h>
- #ifdef HAVE_STDLIB_H
- #include <stdlib.h>
- #endif
- #ifdef HAVE_STRING_H
- #include <string.h>
- #endif
- #ifdef HAVE_TIME_H
- #include <time.h>
- #endif
- #ifdef HAVE_UNISTD_H
- #include <unistd.h>
- #endif
- #include "bsdtar.h"
- #include "archive_multitape.h"
- static void read_archive(struct bsdtar *bsdtar, char mode);
- void
- tarsnap_mode_t(struct bsdtar *bsdtar)
- {
- read_archive(bsdtar, 't');
- unmatched_inclusions_warn(bsdtar, "Not found in archive");
- }
- void
- tarsnap_mode_x(struct bsdtar *bsdtar)
- {
- /* We want to catch SIGINFO and SIGUSR1. */
- siginfo_init(bsdtar);
- read_archive(bsdtar, 'x');
- unmatched_inclusions_warn(bsdtar, "Not found in archive");
- /* Restore old SIGINFO + SIGUSR1 handlers. */
- siginfo_done(bsdtar);
- }
- static void
- progress_func(void * cookie)
- {
- struct bsdtar * bsdtar = cookie;
- siginfo_printinfo(bsdtar, 0, 0);
- }
- /*
- * Should we skip over this file if given --resume-extract?
- * 0: skip it. 1: don't skip. -1: error.
- */
- static int
- check_skip_file(const char * filename, const struct stat * archive_st)
- {
- struct stat file_st;
- /* Get info about the file on disk. */
- if (stat(filename, &file_st) == -1) {
- if (errno == ENOENT)
- goto noskip;
- goto err0;
- }
- /*
- * Compare file size and mtime (seconds). Some filesystems don't have
- * sub-second timestamp precision, so comparing the full timespecs
- * would produce a lot of false negatives.
- */
- if (file_st.st_size != archive_st->st_size)
- goto noskip;
- #ifdef POSIXFAIL_STAT_ST_MTIM
- /* POSIX Issue 7. */
- if (file_st.st_mtim.tv_sec != archive_st->st_mtim.tv_sec)
- goto noskip;
- #else
- /* POSIX Issue 6 and below: use time_t st_mtime instead of st_mtim. */
- if (file_st.st_mtime != archive_st->st_mtime)
- goto noskip;
- #endif
- /* Skip file. */
- return (0);
- noskip:
- /* Don't skip. */
- return (1);
- err0:
- /* Failure! */
- return (-1);
- }
- /*
- * Handle 'x' and 't' modes.
- */
- static void
- read_archive(struct bsdtar *bsdtar, char mode)
- {
- FILE *out;
- struct archive *a;
- struct archive_entry *entry;
- const struct stat *st;
- int r;
- while (*bsdtar->argv) {
- include(bsdtar, *bsdtar->argv);
- bsdtar->argv++;
- }
- if (bsdtar->names_from_file != NULL)
- include_from_file(bsdtar, bsdtar->names_from_file);
- if ((a = archive_read_new()) == NULL) {
- bsdtar_warnc(bsdtar, ENOMEM, "Cannot allocate memory");
- goto err0;
- }
- archive_read_support_compression_none(a);
- archive_read_support_format_tar(a);
- if (archive_read_open_multitape(a, bsdtar->machinenum,
- bsdtar->tapenames[0]) == NULL) {
- bsdtar_warnc(bsdtar, 0, "%s", archive_error_string(a));
- goto err1;
- }
- do_chdir(bsdtar);
- if (mode == 'x') {
- /* Set an extract callback so that we can handle SIGINFO. */
- archive_read_extract_set_progress_callback(a, progress_func,
- bsdtar);
- }
- if (mode == 'x' && bsdtar->option_chroot) {
- #if HAVE_CHROOT
- if (chroot(".") != 0) {
- bsdtar_warnc(bsdtar, errno, "Can't chroot to \".\"");
- goto err1;
- }
- #else
- bsdtar_warnc(bsdtar, 0,
- "chroot isn't supported on this platform");
- goto err1;
- #endif
- }
- for (;;) {
- /* Support --fast-read option */
- if (bsdtar->option_fast_read &&
- unmatched_inclusions(bsdtar) == 0)
- break;
- r = archive_read_next_header(a, &entry);
- if (r == ARCHIVE_EOF)
- break;
- if (r < ARCHIVE_OK)
- bsdtar_warnc(bsdtar, 0, "%s", archive_error_string(a));
- if (r <= ARCHIVE_WARN)
- bsdtar->return_value = 1;
- if (r == ARCHIVE_RETRY) {
- /* Retryable error: try again */
- bsdtar_warnc(bsdtar, 0, "Retrying...");
- continue;
- }
- if (r == ARCHIVE_FATAL)
- break;
- if (bsdtar->option_numeric_owner) {
- archive_entry_set_uname(entry, NULL);
- archive_entry_set_gname(entry, NULL);
- }
- /*
- * Exclude entries that are too old.
- */
- st = archive_entry_stat(entry);
- if (bsdtar->newer_ctime_sec > 0) {
- if (st->st_ctime < bsdtar->newer_ctime_sec)
- continue; /* Too old, skip it. */
- if (st->st_ctime == bsdtar->newer_ctime_sec
- && ARCHIVE_STAT_CTIME_NANOS(st)
- <= bsdtar->newer_ctime_nsec)
- continue; /* Too old, skip it. */
- }
- if (bsdtar->newer_mtime_sec > 0) {
- if (st->st_mtime < bsdtar->newer_mtime_sec)
- continue; /* Too old, skip it. */
- if (st->st_mtime == bsdtar->newer_mtime_sec
- && ARCHIVE_STAT_MTIME_NANOS(st)
- <= bsdtar->newer_mtime_nsec)
- continue; /* Too old, skip it. */
- }
- /*
- * Note that pattern exclusions are checked before
- * pathname rewrites are handled. This gives more
- * control over exclusions, since rewrites always lose
- * information. (For example, consider a rewrite
- * s/foo[0-9]/foo/. If we check exclusions after the
- * rewrite, there would be no way to exclude foo1/bar
- * while allowing foo2/bar.)
- */
- if (excluded(bsdtar, archive_entry_pathname(entry)))
- continue; /* Excluded by a pattern test. */
- if (mode == 't') {
- /* Perversely, gtar uses -O to mean "send to stderr"
- * when used with -t. */
- out = bsdtar->option_stdout ? stderr : stdout;
- /*
- * TODO: Provide some reasonable way to
- * preview rewrites. gtar always displays
- * the unedited path in -t output, which means
- * you cannot easily preview rewrites.
- */
- if (bsdtar->verbose < 2)
- safe_fprintf(out, "%s",
- archive_entry_pathname(entry));
- else
- list_item_verbose(bsdtar, out, entry);
- fflush(out);
- r = archive_read_data_skip(a);
- if (r == ARCHIVE_WARN) {
- fprintf(out, "\n");
- bsdtar_warnc(bsdtar, 0, "%s",
- archive_error_string(a));
- }
- if (r == ARCHIVE_RETRY) {
- fprintf(out, "\n");
- bsdtar_warnc(bsdtar, 0, "%s",
- archive_error_string(a));
- }
- if (r == ARCHIVE_FATAL) {
- fprintf(out, "\n");
- bsdtar_warnc(bsdtar, 0, "%s",
- archive_error_string(a));
- bsdtar->return_value = 1;
- break;
- }
- fprintf(out, "\n");
- } else {
- /* Note: some rewrite failures prevent extraction. */
- if (edit_pathname(bsdtar, entry))
- continue; /* Excluded by a rewrite failure. */
- /* Don't extract if the file already matches it. */
- if (bsdtar->option_resume_extract) {
- r = check_skip_file(
- archive_entry_pathname(entry), st);
- if (r == -1) {
- bsdtar_warnc(bsdtar, errno, "stat(%s)",
- archive_entry_pathname(entry));
- goto err1;
- }
- /* Skip file. */
- if (r == 0)
- continue;
- }
- if (bsdtar->option_interactive &&
- !yes("extract '%s'", archive_entry_pathname(entry)))
- continue;
- if (bsdtar->verbose > 1) {
- /* GNU tar uses -tv format with -xvv */
- safe_fprintf(stderr, "x ");
- list_item_verbose(bsdtar, stderr, entry);
- fflush(stderr);
- } else if (bsdtar->verbose > 0) {
- /* Format follows SUSv2, including the
- * deferred '\n'. */
- safe_fprintf(stderr, "x %s",
- archive_entry_pathname(entry));
- fflush(stderr);
- }
- /*
- * Tell the SIGINFO-handler code what we're doing.
- * a->file_count is incremented by
- * archive_read_next_header(), which has already
- * been called for this file. However,
- * siginfo_setinfo() takes the number of files we
- * have already processed (in the past), so we
- * need to subtract 1 from the reported file count.
- */
- siginfo_setinfo(bsdtar, "extracting",
- archive_entry_pathname(entry), 0,
- archive_file_count(a) - 1,
- archive_position_uncompressed(a));
- siginfo_printinfo(bsdtar, 0, 0);
- if (bsdtar->option_stdout)
- r = archive_read_data_into_fd(a, 1);
- else
- r = archive_read_extract(a, entry,
- bsdtar->extract_flags);
- if (r != ARCHIVE_OK) {
- if (!bsdtar->verbose)
- safe_fprintf(stderr, "%s",
- archive_entry_pathname(entry));
- safe_fprintf(stderr, ": %s",
- archive_error_string(a));
- if (!bsdtar->verbose)
- fprintf(stderr, "\n");
- bsdtar->return_value = 1;
- }
- if (bsdtar->verbose)
- fprintf(stderr, "\n");
- if (r == ARCHIVE_FATAL)
- break;
- }
- }
- /* We're not processing any more files. */
- if (mode == 'x') {
- /* siginfo was not initialized in 't' mode. */
- siginfo_setinfo(bsdtar, NULL, NULL, 0, archive_file_count(a),
- archive_position_uncompressed(a));
- }
- r = archive_read_close(a);
- if (r != ARCHIVE_OK)
- bsdtar_warnc(bsdtar, 0, "%s", archive_error_string(a));
- if (r <= ARCHIVE_WARN)
- bsdtar->return_value = 1;
- if (bsdtar->verbose > 2)
- fprintf(stdout, "Archive Format: %s, Compression: %s\n",
- archive_format_name(a), archive_compression_name(a));
- /* Always print a final message for --progress-bytes. */
- if ((mode == 'x') && (bsdtar->option_progress_bytes != 0))
- raise(SIGUSR1);
- /* Print a final update (if desired). */
- if (mode == 'x') {
- /* siginfo was not initialized in 't' mode. */
- siginfo_printinfo(bsdtar, 0, 1);
- }
- archive_read_finish(a);
- /* Success! */
- return;
- err1:
- archive_read_finish(a);
- err0:
- /* Failure! */
- bsdtar->return_value = 1;
- return;
- }
|