blob: 47ad85ecf13af845dbdabae57ea7784a5fc4de7a [file] [log] [blame]
Matt Kraai91b28552001-04-23 18:53:07 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Mini cp implementation for busybox
4 *
Matt Kraai91b28552001-04-23 18:53:07 +00005 * Copyright (C) 2000 by Matt Kraai <kraai@alumni.carnegiemellon.edu>
6 *
Rob Landley2f309322005-11-01 21:55:14 +00007 * Licensed under GPL v2 or later, see file LICENSE in this tarball for details.
Matt Kraai91b28552001-04-23 18:53:07 +00008 */
9
Manuel Novoa III cad53642003-03-19 09:13:01 +000010/* http://www.opengroup.org/onlinepubs/007904975/utilities/cp.html */
11
12/* Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org)
13 *
14 * Size reduction.
15 */
16
Matt Kraai91b28552001-04-23 18:53:07 +000017#include "busybox.h"
Manuel Novoa III cad53642003-03-19 09:13:01 +000018#include "libcoreutils/coreutils.h"
19
Rob Landleydfba7412006-03-06 20:47:33 +000020int cp_main(int argc, char **argv)
Matt Kraai91b28552001-04-23 18:53:07 +000021{
Manuel Novoa III cad53642003-03-19 09:13:01 +000022 struct stat source_stat;
23 struct stat dest_stat;
24 const char *last;
25 const char *dest;
26 int s_flags;
27 int d_flags;
28 int flags;
Matt Kraai91b28552001-04-23 18:53:07 +000029 int status = 0;
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000030 enum {
31 OPT_a = 1 << (sizeof(FILEUTILS_CP_OPTSTR)-1),
32 OPT_r = 1 << (sizeof(FILEUTILS_CP_OPTSTR)),
33 OPT_P = 1 << (sizeof(FILEUTILS_CP_OPTSTR)+1),
34 OPT_H = 1 << (sizeof(FILEUTILS_CP_OPTSTR)+2),
35 OPT_L = 1 << (sizeof(FILEUTILS_CP_OPTSTR)+3),
36 };
Matt Kraai91b28552001-04-23 18:53:07 +000037
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000038 // Soft- and hardlinking don't mix
39 // -P and -d are the same (-P is POSIX, -d is GNU)
40 // -r and -R are the same
41 // -a = -pdR
42 opt_complementary = "?:l--s:s--l:Pd:rR:apdR";
43 flags = getopt32(argc, argv, FILEUTILS_CP_OPTSTR "arPHL");
Mike Frysinger2ed05ab2005-04-14 02:52:50 +000044 /* Default behavior of cp is to dereference, so we don't have to do
45 * anything special when we are given -L.
46 * The behavior of -H is *almost* like -L, but not quite, so let's
47 * just ignore it too for fun.
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000048 if (flags & OPT_L) ...
49 if (flags & OPT_H) ... // deref command-line params only
Mike Frysinger2ed05ab2005-04-14 02:52:50 +000050 */
Manuel Novoa III cad53642003-03-19 09:13:01 +000051
52 flags ^= FILEUTILS_DEREFERENCE; /* The sense of this flag was reversed. */
53
54 if (optind + 2 > argc) {
55 bb_show_usage();
56 }
57
58 last = argv[argc - 1];
59 argv += optind;
Matt Kraai91b28552001-04-23 18:53:07 +000060
61 /* If there are only two arguments and... */
62 if (optind + 2 == argc) {
Manuel Novoa III cad53642003-03-19 09:13:01 +000063 s_flags = cp_mv_stat2(*argv, &source_stat,
Denis Vlasenko9213a9e2006-09-17 16:28:10 +000064 (flags & FILEUTILS_DEREFERENCE) ? stat : lstat);
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000065 if (s_flags < 0)
66 return EXIT_FAILURE;
67 d_flags = cp_mv_stat(last, &dest_stat);
68 if (d_flags < 0)
69 return EXIT_FAILURE;
70
Matt Kraai91b28552001-04-23 18:53:07 +000071 /* ...if neither is a directory or... */
Manuel Novoa III cad53642003-03-19 09:13:01 +000072 if ( !((s_flags | d_flags) & 2) ||
73 /* ...recursing, the 1st is a directory, and the 2nd doesn't exist... */
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000074 ((flags & FILEUTILS_RECUR) && (s_flags & 2) && !d_flags)
Manuel Novoa III cad53642003-03-19 09:13:01 +000075 ) {
Matt Kraai91b28552001-04-23 18:53:07 +000076 /* ...do a simple copy. */
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000077 dest = xstrdup(last);
Mike Frysinger2ed05ab2005-04-14 02:52:50 +000078 goto DO_COPY; /* Note: optind+2==argc implies argv[1]==last below. */
Matt Kraai91b28552001-04-23 18:53:07 +000079 }
80 }
81
Manuel Novoa III cad53642003-03-19 09:13:01 +000082 do {
83 dest = concat_path_file(last, bb_get_last_path_component(*argv));
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000084 DO_COPY:
Manuel Novoa III cad53642003-03-19 09:13:01 +000085 if (copy_file(*argv, dest, flags) < 0) {
Matt Kraai91b28552001-04-23 18:53:07 +000086 status = 1;
Manuel Novoa III cad53642003-03-19 09:13:01 +000087 }
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000088 free((void*)dest);
89 } while (*++argv != last);
Matt Kraai91b28552001-04-23 18:53:07 +000090
Denis Vlasenkof24e1f42006-10-21 23:40:20 +000091 return status;
Matt Kraai91b28552001-04-23 18:53:07 +000092}