cp: implement -T
authorAaro Koskinen <aaro.koskinen@iki.fi>
Thu, 1 Feb 2018 08:29:05 +0000 (09:29 +0100)
committerDenys Vlasenko <vda.linux@googlemail.com>
Thu, 1 Feb 2018 08:30:59 +0000 (09:30 +0100)
Implement "cp -T". Some Linux kernel Makefiles started using this recently,
so allow also building on systems using busybox cp.

function                                             old     new   delta
cp_main                                              360     428     +68
copy_file                                           1678    1676      -2
packed_usage                                       32290   32259     -31
------------------------------------------------------------------------------
(add/remove: 0/0 grow/shrink: 1/2 up/down: 76/-39)             Total: 35 bytes

Signed-off-by: Aaro Koskinen <aaro.koskinen@iki.fi>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
coreutils/cp.c
include/libbb.h

index 05c725cd0817f26d97a0c34a89a762d3bbbc6032..8d93c6fe4b3014628857e7457b7941640a8bf8f2 100644 (file)
@@ -48,6 +48,7 @@
 //usage:     "\n       -f      Overwrite"
 //usage:     "\n       -i      Prompt before overwrite"
 //usage:     "\n       -l,-s   Create (sym)links"
+//usage:     "\n       -T      Treat DEST as a normal file"
 //usage:     "\n       -u      Copy only newer files"
 
 #include "libbb.h"
@@ -93,6 +94,7 @@ int cp_main(int argc, char **argv)
                "no-dereference\0" No_argument "P"
                "recursive\0"      No_argument "R"
                "symbolic-link\0"  No_argument "s"
+               "no-target-directory\0" No_argument "T"
                "verbose\0"        No_argument "v"
                "update\0"         No_argument "u"
                "remove-destination\0" No_argument "\xff"
@@ -122,6 +124,8 @@ int cp_main(int argc, char **argv)
         *      remove each existing destination file before attempting to open
         * --parents
         *      use full source file name under DIRECTORY
+        * -T, --no-target-directory
+        *      treat DEST as a normal file
         * NOT SUPPORTED IN BBOX:
         * --backup[=CONTROL]
         *      make a backup of each existing destination file
@@ -140,8 +144,6 @@ int cp_main(int argc, char **argv)
         *      override the usual backup suffix
         * -t, --target-directory=DIRECTORY
         *      copy all SOURCE arguments into DIRECTORY
-        * -T, --no-target-directory
-        *      treat DEST as a normal file
         * -x, --one-file-system
         *      stay on this file system
         * -Z, --context=CONTEXT
@@ -176,6 +178,12 @@ int cp_main(int argc, char **argv)
                if (d_flags < 0)
                        return EXIT_FAILURE;
 
+               if (flags & FILEUTILS_NO_TARGET_DIR) { /* -T */
+                       if (!(s_flags & 2) && (d_flags & 2))
+                               /* cp -T NOTDIR DIR */
+                               bb_error_msg_and_die("'%s' is a directory", last);
+               }
+
 #if ENABLE_FEATURE_CP_LONG_OPTIONS
                //bb_error_msg("flags:%x FILEUTILS_RMDEST:%x OPT_parents:%x",
                //      flags, FILEUTILS_RMDEST, OPT_parents);
@@ -193,11 +201,14 @@ int cp_main(int argc, char **argv)
                if (!((s_flags | d_flags) & 2)
                    /* ...or: recursing, the 1st is a directory, and the 2nd doesn't exist... */
                 || ((flags & FILEUTILS_RECUR) && (s_flags & 2) && !d_flags)
+                || (flags & FILEUTILS_NO_TARGET_DIR)
                ) {
                        /* Do a simple copy */
                        dest = last;
                        goto DO_COPY; /* NB: argc==2 -> *++argv==last */
                }
+       } else if (flags & FILEUTILS_NO_TARGET_DIR) {
+               bb_error_msg_and_die("too many arguments");
        }
 
        while (1) {
index 5f25b5ddeb02222e68636514bd7ef55a650a514d..a9386402085e79b647ca6967980ffa9abd131e23 100644 (file)
@@ -404,10 +404,11 @@ enum {    /* cp.c, mv.c, install.c depend on these values. CAREFUL when changing th
        /* -P = -d   (mapped in cp.c) */
        FILEUTILS_VERBOSE         = (1 << 12) * ENABLE_FEATURE_VERBOSE, /* -v */
        FILEUTILS_UPDATE          = 1 << 13, /* -u */
+       FILEUTILS_NO_TARGET_DIR   = 1 << 14, /* -T */
 #if ENABLE_SELINUX
-       FILEUTILS_PRESERVE_SECURITY_CONTEXT = 1 << 14, /* -c */
+       FILEUTILS_PRESERVE_SECURITY_CONTEXT = 1 << 15, /* -c */
 #endif
-       FILEUTILS_RMDEST          = 1 << (15 - !ENABLE_SELINUX), /* --remove-destination */
+       FILEUTILS_RMDEST          = 1 << (16 - !ENABLE_SELINUX), /* --remove-destination */
        /*
         * Hole. cp may have some bits set here,
         * they should not affect remove_file()/copy_file()
@@ -417,7 +418,7 @@ enum {      /* cp.c, mv.c, install.c depend on these values. CAREFUL when changing th
 #endif
        FILEUTILS_IGNORE_CHMOD_ERR = 1 << 31,
 };
-#define FILEUTILS_CP_OPTSTR "pdRfilsLHarPvu" IF_SELINUX("c")
+#define FILEUTILS_CP_OPTSTR "pdRfilsLHarPvuT" IF_SELINUX("c")
 extern int remove_file(const char *path, int flags) FAST_FUNC;
 /* NB: without FILEUTILS_RECUR in flags, it will basically "cat"
  * the source, not copy (unless "source" is a directory).