X-Git-Url: https://git.librecmc.org/?a=blobdiff_plain;f=coreutils%2Fmv.c;h=1c0dc3d722a4952be83cb76ae4323c243b35c71d;hb=39440e520c7445e9bafd310816cc152a3192721a;hp=10a08221014eac582b1b7b53106fb45ad089a91a;hpb=9b5871888989b16f94cbba5dd304ac444def3afd;p=oweals%2Fbusybox.git diff --git a/coreutils/mv.c b/coreutils/mv.c index 10a082210..1c0dc3d72 100644 --- a/coreutils/mv.c +++ b/coreutils/mv.c @@ -1,7 +1,8 @@ +/* vi: set sw=4 ts=4: */ /* * Mini mv implementation for busybox * - * Copyright (C) 1998 by Erik Andersen + * Copyright (C) 2000 by Matt Kraai * * 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 @@ -19,65 +20,120 @@ * */ -#include "internal.h" -#include +/* Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org) + * + * Size reduction and improved error checking. + */ + #include #include -#include -#include +#include +#include #include +#include +#include /* struct option */ +#include "busybox.h" +#include "libcoreutils/coreutils.h" -const char mv_usage[] = "source-file [source-file ...] destination-file\n" - "\n" "\tMove the source files to the destination.\n" "\n"; +static const struct option mv_long_options[] = { + { "interactive", 0, NULL, 'i' }, + { "force", 0, NULL, 'f' }, + { 0, 0, 0, 0 } +}; +#define OPT_FILEUTILS_FORCE 1 +#define OPT_FILEUTILS_INTERACTIVE 2 +static const char fmt[] = "cannot overwrite %sdirectory with %sdirectory"; -extern int mv_main (int argc, char **argv) +int mv_main(int argc, char **argv) { - const char *srcName; - const char *destName; - const char *lastArg; - int dirFlag; - - if (argc < 3) { - fprintf (stderr, "Usage: %s %s", *argv, mv_usage); - exit (FALSE); - } - lastArg = argv[argc - 1]; - - dirFlag = isDirectory (lastArg); - - if ((argc > 3) && !dirFlag) { - fprintf (stderr, "%s: not a directory\n", lastArg); - exit (FALSE); - } - - while (argc-- > 2) { - srcName = *(++argv); - - if (access (srcName, 0) < 0) { - perror (srcName); - continue; + struct stat dest_stat; + const char *last; + const char *dest; + unsigned long flags; + int dest_exists; + int status = 0; + + bb_applet_long_options = mv_long_options; + bb_opt_complementally = "f-i:i-f"; + flags = bb_getopt_ulflags(argc, argv, "fi"); + if (optind + 2 > argc) { + bb_show_usage(); } - destName = lastArg; + last = argv[argc - 1]; + argv += optind; - if (dirFlag==TRUE) - destName = buildName (destName, srcName); + if (optind + 2 == argc) { + if ((dest_exists = cp_mv_stat(last, &dest_stat)) < 0) { + return 1; + } - if (rename (srcName, destName) >= 0) - continue; - - if (errno != EXDEV) { - perror (destName); - continue; + if (!(dest_exists & 2)) { + dest = last; + goto DO_MOVE; + } } - if (!copyFile (srcName, destName, TRUE, FALSE)) - continue; - - if (unlink (srcName) < 0) - perror (srcName); - } - exit (TRUE); + do { + dest = concat_path_file(last, bb_get_last_path_component(*argv)); + + if ((dest_exists = cp_mv_stat(dest, &dest_stat)) < 0) { + goto RET_1; + } + +DO_MOVE: + + if (dest_exists && !(flags & OPT_FILEUTILS_FORCE) && + ((access(dest, W_OK) < 0 && isatty(0)) || + (flags & OPT_FILEUTILS_INTERACTIVE))) { + if (fprintf(stderr, "mv: overwrite `%s'? ", dest) < 0) { + goto RET_1; /* Ouch! fprintf failed! */ + } + if (!bb_ask_confirmation()) { + goto RET_0; + } + } + if (rename(*argv, dest) < 0) { + struct stat source_stat; + int source_exists; + + if (errno != EXDEV || + (source_exists = cp_mv_stat(*argv, &source_stat)) < 1) { + bb_perror_msg("unable to rename `%s'", *argv); + } else { + if (dest_exists) { + if (dest_exists == 3) { + if (source_exists != 3) { + bb_error_msg(fmt, "", "non-"); + goto RET_1; + } + } else { + if (source_exists == 3) { + bb_error_msg(fmt, "non-", ""); + goto RET_1; + } + } + if (unlink(dest) < 0) { + bb_perror_msg("cannot remove `%s'", dest); + goto RET_1; + } + } + if ((copy_file(*argv, dest, + FILEUTILS_RECUR | FILEUTILS_PRESERVE_STATUS) >= 0) && + (remove_file(*argv, FILEUTILS_RECUR | FILEUTILS_FORCE) >= 0)) { + goto RET_0; + } + } +RET_1: + status = 1; + } +RET_0: + if (dest != last) { + free((void *) dest); + } + } while (*++argv != last); + + return (status); }