|  | /* $NetBSD: cp.c,v 1.58 2012/01/04 15:58:37 christos Exp $ */ | 
|  |  | 
|  | /* | 
|  | * Copyright (c) 1988, 1993, 1994 | 
|  | *	The Regents of the University of California.  All rights reserved. | 
|  | * | 
|  | * This code is derived from software contributed to Berkeley by | 
|  | * David Hitz of Auspex Systems Inc. | 
|  | * | 
|  | * 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. | 
|  | * 3. Neither the name of the University nor the names of its contributors | 
|  | *    may be used to endorse or promote products derived from this software | 
|  | *    without specific prior written permission. | 
|  | * | 
|  | * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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 <sys/cdefs.h> | 
|  | #ifndef lint | 
|  | __COPYRIGHT( | 
|  | "@(#) Copyright (c) 1988, 1993, 1994\ | 
|  | The Regents of the University of California.  All rights reserved."); | 
|  | #endif /* not lint */ | 
|  |  | 
|  | #ifndef lint | 
|  | #if 0 | 
|  | static char sccsid[] = "@(#)cp.c	8.5 (Berkeley) 4/29/95"; | 
|  | #else | 
|  | __RCSID("$NetBSD: cp.c,v 1.58 2012/01/04 15:58:37 christos Exp $"); | 
|  | #endif | 
|  | #endif /* not lint */ | 
|  |  | 
|  | /* | 
|  | * Cp copies source files to target files. | 
|  | * | 
|  | * The global PATH_T structure "to" always contains the path to the | 
|  | * current target file.  Since fts(3) does not change directories, | 
|  | * this path can be either absolute or dot-relative. | 
|  | * | 
|  | * The basic algorithm is to initialize "to" and use fts(3) to traverse | 
|  | * the file hierarchy rooted in the argument list.  A trivial case is the | 
|  | * case of 'cp file1 file2'.  The more interesting case is the case of | 
|  | * 'cp file1 file2 ... fileN dir' where the hierarchy is traversed and the | 
|  | * path (relative to the root of the traversal) is appended to dir (stored | 
|  | * in "to") to form the final target path. | 
|  | */ | 
|  |  | 
|  | #include <sys/param.h> | 
|  | #include <sys/stat.h> | 
|  |  | 
|  | #include <assert.h> | 
|  | #include <err.h> | 
|  | #include <errno.h> | 
|  | #include <fts.h> | 
|  | #include <locale.h> | 
|  | #include <signal.h> | 
|  | #include <stdlib.h> | 
|  | #include <stdio.h> | 
|  | #include <string.h> | 
|  | #include <unistd.h> | 
|  |  | 
|  | #include "extern.h" | 
|  |  | 
|  | #define	STRIP_TRAILING_SLASH(p) {					\ | 
|  | while ((p).p_end > (p).p_path + 1 && (p).p_end[-1] == '/')	\ | 
|  | *--(p).p_end = '\0';					\ | 
|  | } | 
|  |  | 
|  | static char empty[] = ""; | 
|  | PATH_T to = { .p_end = to.p_path, .target_end = empty  }; | 
|  |  | 
|  | uid_t myuid; | 
|  | int Hflag, Lflag, Rflag, Pflag, fflag, iflag, lflag, pflag, rflag, vflag, Nflag; | 
|  | mode_t myumask; | 
|  | sig_atomic_t pinfo; | 
|  |  | 
|  | enum op { FILE_TO_FILE, FILE_TO_DIR, DIR_TO_DNE }; | 
|  |  | 
|  | static int copy(char *[], enum op, int); | 
|  |  | 
|  | static void | 
|  | progress(int sig __unused) | 
|  | { | 
|  |  | 
|  | pinfo++; | 
|  | } | 
|  |  | 
|  | int | 
|  | cp_main(int argc, char *argv[]) | 
|  | { | 
|  | struct stat to_stat, tmp_stat; | 
|  | enum op type; | 
|  | int ch, fts_options, r, have_trailing_slash; | 
|  | char *target, **src; | 
|  |  | 
|  | #ifndef ANDROID | 
|  | setprogname(argv[0]); | 
|  | #endif | 
|  | (void)setlocale(LC_ALL, ""); | 
|  |  | 
|  | Hflag = Lflag = Pflag = Rflag = 0; | 
|  | while ((ch = getopt(argc, argv, "HLNPRfailprv")) != -1) | 
|  | switch (ch) { | 
|  | case 'H': | 
|  | Hflag = 1; | 
|  | Lflag = Pflag = 0; | 
|  | break; | 
|  | case 'L': | 
|  | Lflag = 1; | 
|  | Hflag = Pflag = 0; | 
|  | break; | 
|  | case 'N': | 
|  | Nflag = 1; | 
|  | break; | 
|  | case 'P': | 
|  | Pflag = 1; | 
|  | Hflag = Lflag = 0; | 
|  | break; | 
|  | case 'R': | 
|  | Rflag = 1; | 
|  | break; | 
|  | case 'a': | 
|  | Pflag = 1; | 
|  | pflag = 1; | 
|  | Rflag = 1; | 
|  | Hflag = Lflag = 0; | 
|  | break; | 
|  | case 'f': | 
|  | fflag = 1; | 
|  | iflag = 0; | 
|  | break; | 
|  | case 'i': | 
|  | iflag = isatty(fileno(stdin)); | 
|  | fflag = 0; | 
|  | break; | 
|  | case 'l': | 
|  | lflag = 1; | 
|  | break; | 
|  | case 'p': | 
|  | pflag = 1; | 
|  | break; | 
|  | case 'r': | 
|  | rflag = 1; | 
|  | break; | 
|  | case 'v': | 
|  | vflag = 1; | 
|  | break; | 
|  | case '?': | 
|  | default: | 
|  | cp_usage(); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  | argc -= optind; | 
|  | argv += optind; | 
|  |  | 
|  | if (argc < 2) | 
|  | cp_usage(); | 
|  |  | 
|  | fts_options = FTS_NOCHDIR | FTS_PHYSICAL; | 
|  | if (rflag) { | 
|  | if (Rflag) { | 
|  | errx(EXIT_FAILURE, | 
|  | "the -R and -r options may not be specified together."); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  | if (Hflag || Lflag || Pflag) { | 
|  | errx(EXIT_FAILURE, | 
|  | "the -H, -L, and -P options may not be specified with the -r option."); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  | fts_options &= ~FTS_PHYSICAL; | 
|  | fts_options |= FTS_LOGICAL; | 
|  | } | 
|  |  | 
|  | if (Rflag) { | 
|  | if (Hflag) | 
|  | fts_options |= FTS_COMFOLLOW; | 
|  | if (Lflag) { | 
|  | fts_options &= ~FTS_PHYSICAL; | 
|  | fts_options |= FTS_LOGICAL; | 
|  | } | 
|  | } else if (!Pflag) { | 
|  | fts_options &= ~FTS_PHYSICAL; | 
|  | fts_options |= FTS_LOGICAL | FTS_COMFOLLOW; | 
|  | } | 
|  |  | 
|  | myuid = getuid(); | 
|  |  | 
|  | /* Copy the umask for explicit mode setting. */ | 
|  | myumask = umask(0); | 
|  | (void)umask(myumask); | 
|  |  | 
|  | /* Save the target base in "to". */ | 
|  | target = argv[--argc]; | 
|  | if (strlcpy(to.p_path, target, sizeof(to.p_path)) >= sizeof(to.p_path)) | 
|  | errx(EXIT_FAILURE, "%s: name too long", target); | 
|  | to.p_end = to.p_path + strlen(to.p_path); | 
|  | have_trailing_slash = (to.p_end[-1] == '/'); | 
|  | if (have_trailing_slash) | 
|  | STRIP_TRAILING_SLASH(to); | 
|  | to.target_end = to.p_end; | 
|  |  | 
|  | /* Set end of argument list for fts(3). */ | 
|  | argv[argc] = NULL; | 
|  |  | 
|  | #ifndef ANDROID | 
|  | (void)signal(SIGINFO, progress); | 
|  | #endif | 
|  |  | 
|  | /* | 
|  | * Cp has two distinct cases: | 
|  | * | 
|  | * cp [-R] source target | 
|  | * cp [-R] source1 ... sourceN directory | 
|  | * | 
|  | * In both cases, source can be either a file or a directory. | 
|  | * | 
|  | * In (1), the target becomes a copy of the source. That is, if the | 
|  | * source is a file, the target will be a file, and likewise for | 
|  | * directories. | 
|  | * | 
|  | * In (2), the real target is not directory, but "directory/source". | 
|  | */ | 
|  | if (Pflag) | 
|  | r = lstat(to.p_path, &to_stat); | 
|  | else | 
|  | r = stat(to.p_path, &to_stat); | 
|  | if (r == -1 && errno != ENOENT) { | 
|  | err(EXIT_FAILURE, "%s", to.p_path); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  | if (r == -1 || !S_ISDIR(to_stat.st_mode)) { | 
|  | /* | 
|  | * Case (1).  Target is not a directory. | 
|  | */ | 
|  | if (argc > 1) | 
|  | cp_usage(); | 
|  | /* | 
|  | * Need to detect the case: | 
|  | *	cp -R dir foo | 
|  | * Where dir is a directory and foo does not exist, where | 
|  | * we want pathname concatenations turned on but not for | 
|  | * the initial mkdir(). | 
|  | */ | 
|  | if (r == -1) { | 
|  | if (rflag || (Rflag && (Lflag || Hflag))) | 
|  | r = stat(*argv, &tmp_stat); | 
|  | else | 
|  | r = lstat(*argv, &tmp_stat); | 
|  | if (r == -1) { | 
|  | err(EXIT_FAILURE, "%s", *argv); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  |  | 
|  | if (S_ISDIR(tmp_stat.st_mode) && (Rflag || rflag)) | 
|  | type = DIR_TO_DNE; | 
|  | else | 
|  | type = FILE_TO_FILE; | 
|  | } else | 
|  | type = FILE_TO_FILE; | 
|  |  | 
|  | if (have_trailing_slash && type == FILE_TO_FILE) { | 
|  | if (r == -1) | 
|  | errx(1, "directory %s does not exist", | 
|  | to.p_path); | 
|  | else | 
|  | errx(1, "%s is not a directory", to.p_path); | 
|  | } | 
|  | } else { | 
|  | /* | 
|  | * Case (2).  Target is a directory. | 
|  | */ | 
|  | type = FILE_TO_DIR; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * make "cp -rp src/ dst" behave like "cp -rp src dst" not | 
|  | * like "cp -rp src/. dst" | 
|  | */ | 
|  | for (src = argv; *src; src++) { | 
|  | size_t len = strlen(*src); | 
|  | while (len-- > 1 && (*src)[len] == '/') | 
|  | (*src)[len] = '\0'; | 
|  | } | 
|  |  | 
|  | exit(copy(argv, type, fts_options)); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  |  | 
|  | static int dnestack[MAXPATHLEN]; /* unlikely we'll have more nested dirs */ | 
|  | static ssize_t dnesp; | 
|  | static void | 
|  | pushdne(int dne) | 
|  | { | 
|  |  | 
|  | dnestack[dnesp++] = dne; | 
|  | assert(dnesp < MAXPATHLEN); | 
|  | } | 
|  |  | 
|  | static int | 
|  | popdne(void) | 
|  | { | 
|  | int rv; | 
|  |  | 
|  | rv = dnestack[--dnesp]; | 
|  | assert(dnesp >= 0); | 
|  | return rv; | 
|  | } | 
|  |  | 
|  | static int | 
|  | copy(char *argv[], enum op type, int fts_options) | 
|  | { | 
|  | struct stat to_stat; | 
|  | FTS *ftsp; | 
|  | FTSENT *curr; | 
|  | int base, dne, sval; | 
|  | int this_failed, any_failed; | 
|  | size_t nlen; | 
|  | char *p, *target_mid; | 
|  |  | 
|  | base = 0;	/* XXX gcc -Wuninitialized (see comment below) */ | 
|  |  | 
|  | if ((ftsp = fts_open(argv, fts_options, NULL)) == NULL) | 
|  | err(EXIT_FAILURE, "%s", argv[0]); | 
|  | /* NOTREACHED */ | 
|  | for (any_failed = 0; (curr = fts_read(ftsp)) != NULL;) { | 
|  | this_failed = 0; | 
|  | switch (curr->fts_info) { | 
|  | case FTS_NS: | 
|  | case FTS_DNR: | 
|  | case FTS_ERR: | 
|  | warnx("%s: %s", curr->fts_path, | 
|  | strerror(curr->fts_errno)); | 
|  | this_failed = any_failed = 1; | 
|  | continue; | 
|  | case FTS_DC:			/* Warn, continue. */ | 
|  | warnx("%s: directory causes a cycle", curr->fts_path); | 
|  | this_failed = any_failed = 1; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * If we are in case (2) or (3) above, we need to append the | 
|  | * source name to the target name. | 
|  | */ | 
|  | if (type != FILE_TO_FILE) { | 
|  | if ((curr->fts_namelen + | 
|  | to.target_end - to.p_path + 1) > MAXPATHLEN) { | 
|  | warnx("%s/%s: name too long (not copied)", | 
|  | to.p_path, curr->fts_name); | 
|  | this_failed = any_failed = 1; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Need to remember the roots of traversals to create | 
|  | * correct pathnames.  If there's a directory being | 
|  | * copied to a non-existent directory, e.g. | 
|  | *	cp -R a/dir noexist | 
|  | * the resulting path name should be noexist/foo, not | 
|  | * noexist/dir/foo (where foo is a file in dir), which | 
|  | * is the case where the target exists. | 
|  | * | 
|  | * Also, check for "..".  This is for correct path | 
|  | * concatentation for paths ending in "..", e.g. | 
|  | *	cp -R .. /tmp | 
|  | * Paths ending in ".." are changed to ".".  This is | 
|  | * tricky, but seems the easiest way to fix the problem. | 
|  | * | 
|  | * XXX | 
|  | * Since the first level MUST be FTS_ROOTLEVEL, base | 
|  | * is always initialized. | 
|  | */ | 
|  | if (curr->fts_level == FTS_ROOTLEVEL) { | 
|  | if (type != DIR_TO_DNE) { | 
|  | p = strrchr(curr->fts_path, '/'); | 
|  | base = (p == NULL) ? 0 : | 
|  | (int)(p - curr->fts_path + 1); | 
|  |  | 
|  | if (!strcmp(&curr->fts_path[base], | 
|  | "..")) | 
|  | base += 1; | 
|  | } else | 
|  | base = curr->fts_pathlen; | 
|  | } | 
|  |  | 
|  | p = &curr->fts_path[base]; | 
|  | nlen = curr->fts_pathlen - base; | 
|  | target_mid = to.target_end; | 
|  | if (*p != '/' && target_mid[-1] != '/') | 
|  | *target_mid++ = '/'; | 
|  | *target_mid = 0; | 
|  |  | 
|  | if (target_mid - to.p_path + nlen >= PATH_MAX) { | 
|  | warnx("%s%s: name too long (not copied)", | 
|  | to.p_path, p); | 
|  | this_failed = any_failed = 1; | 
|  | continue; | 
|  | } | 
|  | (void)strncat(target_mid, p, nlen); | 
|  | to.p_end = target_mid + nlen; | 
|  | *to.p_end = 0; | 
|  | STRIP_TRAILING_SLASH(to); | 
|  | } | 
|  |  | 
|  | sval = Pflag ? lstat(to.p_path, &to_stat) : stat(to.p_path, &to_stat); | 
|  | /* Not an error but need to remember it happened */ | 
|  | if (sval == -1) | 
|  | dne = 1; | 
|  | else { | 
|  | if (to_stat.st_dev == curr->fts_statp->st_dev && | 
|  | to_stat.st_ino == curr->fts_statp->st_ino) { | 
|  | warnx("%s and %s are identical (not copied).", | 
|  | to.p_path, curr->fts_path); | 
|  | this_failed = any_failed = 1; | 
|  | if (S_ISDIR(curr->fts_statp->st_mode)) | 
|  | (void)fts_set(ftsp, curr, FTS_SKIP); | 
|  | continue; | 
|  | } | 
|  | if (!S_ISDIR(curr->fts_statp->st_mode) && | 
|  | S_ISDIR(to_stat.st_mode)) { | 
|  | warnx("cannot overwrite directory %s with non-directory %s", | 
|  | to.p_path, curr->fts_path); | 
|  | this_failed = any_failed = 1; | 
|  | continue; | 
|  | } | 
|  | dne = 0; | 
|  | } | 
|  |  | 
|  | switch (curr->fts_statp->st_mode & S_IFMT) { | 
|  | case S_IFLNK: | 
|  | /* Catch special case of a non dangling symlink */ | 
|  | if((fts_options & FTS_LOGICAL) || | 
|  | ((fts_options & FTS_COMFOLLOW) && curr->fts_level == 0)) { | 
|  | if (copy_file(curr, dne)) | 
|  | this_failed = any_failed = 1; | 
|  | } else { | 
|  | if (copy_link(curr, !dne)) | 
|  | this_failed = any_failed = 1; | 
|  | } | 
|  | break; | 
|  | case S_IFDIR: | 
|  | if (!Rflag && !rflag) { | 
|  | if (curr->fts_info == FTS_D) | 
|  | warnx("%s is a directory (not copied).", | 
|  | curr->fts_path); | 
|  | (void)fts_set(ftsp, curr, FTS_SKIP); | 
|  | this_failed = any_failed = 1; | 
|  | break; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Directories get noticed twice: | 
|  | *  In the first pass, create it if needed. | 
|  | *  In the second pass, after the children have been copied, set the permissions. | 
|  | */ | 
|  | if (curr->fts_info == FTS_D) /* First pass */ | 
|  | { | 
|  | /* | 
|  | * If the directory doesn't exist, create the new | 
|  | * one with the from file mode plus owner RWX bits, | 
|  | * modified by the umask.  Trade-off between being | 
|  | * able to write the directory (if from directory is | 
|  | * 555) and not causing a permissions race.  If the | 
|  | * umask blocks owner writes, we fail.. | 
|  | */ | 
|  | pushdne(dne); | 
|  | if (dne) { | 
|  | if (mkdir(to.p_path, | 
|  | curr->fts_statp->st_mode | S_IRWXU) < 0) | 
|  | err(EXIT_FAILURE, "%s", | 
|  | to.p_path); | 
|  | /* NOTREACHED */ | 
|  | } else if (!S_ISDIR(to_stat.st_mode)) { | 
|  | errno = ENOTDIR; | 
|  | err(EXIT_FAILURE, "%s", | 
|  | to.p_path); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  | } | 
|  | else if (curr->fts_info == FTS_DP) /* Second pass */ | 
|  | { | 
|  | /* | 
|  | * If not -p and directory didn't exist, set it to be | 
|  | * the same as the from directory, umodified by the | 
|  | * umask; arguably wrong, but it's been that way | 
|  | * forever. | 
|  | */ | 
|  | if (pflag && setfile(curr->fts_statp, 0)) | 
|  | this_failed = any_failed = 1; | 
|  | else if ((dne = popdne())) | 
|  | (void)chmod(to.p_path, | 
|  | curr->fts_statp->st_mode); | 
|  | } | 
|  | else | 
|  | { | 
|  | warnx("directory %s encountered when not expected.", | 
|  | curr->fts_path); | 
|  | this_failed = any_failed = 1; | 
|  | break; | 
|  | } | 
|  |  | 
|  | break; | 
|  | case S_IFBLK: | 
|  | case S_IFCHR: | 
|  | if (Rflag) { | 
|  | if (copy_special(curr->fts_statp, !dne)) | 
|  | this_failed = any_failed = 1; | 
|  | } else | 
|  | if (copy_file(curr, dne)) | 
|  | this_failed = any_failed = 1; | 
|  | break; | 
|  | case S_IFIFO: | 
|  | if (Rflag) { | 
|  | if (copy_fifo(curr->fts_statp, !dne)) | 
|  | this_failed = any_failed = 1; | 
|  | } else | 
|  | if (copy_file(curr, dne)) | 
|  | this_failed = any_failed = 1; | 
|  | break; | 
|  | default: | 
|  | if (copy_file(curr, dne)) | 
|  | this_failed = any_failed = 1; | 
|  | break; | 
|  | } | 
|  | if (vflag && !this_failed) | 
|  | (void)printf("%s -> %s\n", curr->fts_path, to.p_path); | 
|  | } | 
|  | if (errno) { | 
|  | err(EXIT_FAILURE, "fts_read"); | 
|  | /* NOTREACHED */ | 
|  | } | 
|  | (void)fts_close(ftsp); | 
|  | return (any_failed); | 
|  | } |