diff options
author | Denis Vlasenko <vda.linux@googlemail.com> | 2006-10-27 16:07:20 +0000 |
---|---|---|
committer | Denis Vlasenko <vda.linux@googlemail.com> | 2006-10-27 16:07:20 +0000 |
commit | 51b4c92f80dc232b9d34985f99a4479393644433 (patch) | |
tree | e62b6e2b809db8135d640e3b8222cfd4d52d9098 | |
parent | fefb279ace2da131cad369a3cc6983a1bc220a4a (diff) | |
download | busybox-w32-51b4c92f80dc232b9d34985f99a4479393644433.tar.gz busybox-w32-51b4c92f80dc232b9d34985f99a4479393644433.tar.bz2 busybox-w32-51b4c92f80dc232b9d34985f99a4479393644433.zip |
chown: add -vcf support if CONFIG_DESKTOP
chmod: stop following symlinks
-rw-r--r-- | coreutils/chmod.c | 31 | ||||
-rw-r--r-- | coreutils/chown.c | 58 | ||||
-rw-r--r-- | include/usage.h | 9 |
3 files changed, 71 insertions, 27 deletions
diff --git a/coreutils/chmod.c b/coreutils/chmod.c index 18334b8bb..c4f8fa0b2 100644 --- a/coreutils/chmod.c +++ b/coreutils/chmod.c | |||
@@ -22,9 +22,23 @@ | |||
22 | #define OPT_QUIET (USE_DESKTOP(option_mask32 & 8) SKIP_DESKTOP(0)) | 22 | #define OPT_QUIET (USE_DESKTOP(option_mask32 & 8) SKIP_DESKTOP(0)) |
23 | #define OPT_STR ("-R" USE_DESKTOP("vcf")) | 23 | #define OPT_STR ("-R" USE_DESKTOP("vcf")) |
24 | 24 | ||
25 | /* TODO: | ||
26 | * chmod never changes the permissions of symbolic links; the chmod | ||
27 | * system call cannot change their permissions. This is not a problem | ||
28 | * since the permissions of symbolic links are never used. | ||
29 | * However, for each symbolic link listed on the command line, chmod changes | ||
30 | * the permissions of the pointed-to file. In contrast, chmod ignores | ||
31 | * symbolic links encountered during recursive directory traversals. | ||
32 | */ | ||
33 | |||
25 | static int fileAction(const char *fileName, struct stat *statbuf, void* junk) | 34 | static int fileAction(const char *fileName, struct stat *statbuf, void* junk) |
26 | { | 35 | { |
27 | mode_t newmode = statbuf->st_mode; | 36 | mode_t newmode = statbuf->st_mode; |
37 | |||
38 | // TODO: match GNU behavior: | ||
39 | // if (depth > 0 && S_ISLNK(statbuf->st_mode)) return TRUE; | ||
40 | // if (depth == 0) follow link | ||
41 | |||
28 | if (!bb_parse_mode((char *)junk, &newmode)) | 42 | if (!bb_parse_mode((char *)junk, &newmode)) |
29 | bb_error_msg_and_die("invalid mode: %s", (char *)junk); | 43 | bb_error_msg_and_die("invalid mode: %s", (char *)junk); |
30 | 44 | ||
@@ -33,7 +47,7 @@ static int fileAction(const char *fileName, struct stat *statbuf, void* junk) | |||
33 | || (OPT_CHANGED && statbuf->st_mode != newmode) | 47 | || (OPT_CHANGED && statbuf->st_mode != newmode) |
34 | ) { | 48 | ) { |
35 | printf("mode of '%s' changed to %04o (%s)\n", fileName, | 49 | printf("mode of '%s' changed to %04o (%s)\n", fileName, |
36 | newmode & 7777, bb_mode_string(newmode)+1); | 50 | newmode & 07777, bb_mode_string(newmode)+1); |
37 | } | 51 | } |
38 | return TRUE; | 52 | return TRUE; |
39 | } | 53 | } |
@@ -48,10 +62,11 @@ int chmod_main(int argc, char **argv) | |||
48 | char *arg, **argp; | 62 | char *arg, **argp; |
49 | char *smode; | 63 | char *smode; |
50 | 64 | ||
51 | /* Convert first encountered -r into a-r, etc */ | 65 | /* Convert first encountered -r into a-r, -w into a-w etc */ |
52 | argp = argv + 1; | 66 | argp = argv + 1; |
53 | while ((arg = *argp)) { | 67 | while ((arg = *argp)) { |
54 | /* Protect against mishandling e.g. "chmod 644 -r" */ | 68 | /* Mode spec must be the first arg (sans -R etc) */ |
69 | /* (protect against mishandling e.g. "chmod 644 -r") */ | ||
55 | if (arg[0] != '-') | 70 | if (arg[0] != '-') |
56 | break; | 71 | break; |
57 | /* An option. Not a -- or valid option? */ | 72 | /* An option. Not a -- or valid option? */ |
@@ -72,8 +87,14 @@ int chmod_main(int argc, char **argv) | |||
72 | 87 | ||
73 | /* Ok, ready to do the deed now */ | 88 | /* Ok, ready to do the deed now */ |
74 | do { | 89 | do { |
75 | if (!recursive_action(*argv, OPT_RECURSE, TRUE, FALSE, | 90 | if (!recursive_action(*argv, |
76 | fileAction, fileAction, smode)) { | 91 | OPT_RECURSE, // recurse |
92 | FALSE, // follow links: GNU doesn't | ||
93 | FALSE, // depth first | ||
94 | fileAction, // file action | ||
95 | fileAction, // dir action | ||
96 | smode) // user data | ||
97 | ) { | ||
77 | retval = EXIT_FAILURE; | 98 | retval = EXIT_FAILURE; |
78 | } | 99 | } |
79 | } while (*++argv); | 100 | } while (*++argv); |
diff --git a/coreutils/chown.c b/coreutils/chown.c index 42f973f32..bef89ce86 100644 --- a/coreutils/chown.c +++ b/coreutils/chown.c | |||
@@ -7,14 +7,10 @@ | |||
7 | * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. | 7 | * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. |
8 | */ | 8 | */ |
9 | 9 | ||
10 | /* BB_AUDIT SUSv3 defects - unsupported options -h, -H, -L, and -P. */ | 10 | /* BB_AUDIT SUSv3 defects - unsupported options -H, -L, and -P. */ |
11 | /* BB_AUDIT GNU defects - unsupported options -h, -c, -f, -v, and long options. */ | 11 | /* BB_AUDIT GNU defects - unsupported long options. */ |
12 | /* BB_AUDIT Note: gnu chown does not support -H, -L, or -P. */ | ||
13 | /* http://www.opengroup.org/onlinepubs/007904975/utilities/chown.html */ | 12 | /* http://www.opengroup.org/onlinepubs/007904975/utilities/chown.html */ |
14 | 13 | ||
15 | #include <stdlib.h> | ||
16 | #include <unistd.h> | ||
17 | #include <string.h> | ||
18 | #include "busybox.h" | 14 | #include "busybox.h" |
19 | 15 | ||
20 | static uid_t uid = -1; | 16 | static uid_t uid = -1; |
@@ -22,39 +18,55 @@ static gid_t gid = -1; | |||
22 | 18 | ||
23 | static int (*chown_func)(const char *, uid_t, gid_t) = chown; | 19 | static int (*chown_func)(const char *, uid_t, gid_t) = chown; |
24 | 20 | ||
21 | #define OPT_RECURSE (option_mask32 & 1) | ||
22 | #define OPT_NODEREF (option_mask32 & 2) | ||
23 | #define OPT_VERBOSE (USE_DESKTOP(option_mask32 & 4) SKIP_DESKTOP(0)) | ||
24 | #define OPT_CHANGED (USE_DESKTOP(option_mask32 & 8) SKIP_DESKTOP(0)) | ||
25 | #define OPT_QUIET (USE_DESKTOP(option_mask32 & 0x10) SKIP_DESKTOP(0)) | ||
26 | #define OPT_STR ("Rh" USE_DESKTOP("vcf")) | ||
27 | |||
28 | /* TODO: | ||
29 | * -H if a command line argument is a symbolic link to a directory, traverse it | ||
30 | * -L traverse every symbolic link to a directory encountered | ||
31 | * -P do not traverse any symbolic links (default) | ||
32 | */ | ||
33 | |||
25 | static int fileAction(const char *fileName, struct stat *statbuf, | 34 | static int fileAction(const char *fileName, struct stat *statbuf, |
26 | void ATTRIBUTE_UNUSED *junk) | 35 | void ATTRIBUTE_UNUSED *junk) |
27 | { | 36 | { |
37 | // TODO: -H/-L/-P | ||
38 | // if (depth ... && S_ISLNK(statbuf->st_mode)) .... | ||
39 | |||
28 | if (!chown_func(fileName, | 40 | if (!chown_func(fileName, |
29 | (uid == (uid_t)-1) ? statbuf->st_uid : uid, | 41 | (uid == (uid_t)-1) ? statbuf->st_uid : uid, |
30 | (gid == (gid_t)-1) ? statbuf->st_gid : gid)) { | 42 | (gid == (gid_t)-1) ? statbuf->st_gid : gid)) { |
43 | if (OPT_VERBOSE | ||
44 | || (OPT_CHANGED && (statbuf->st_uid != uid || statbuf->st_gid != gid)) | ||
45 | ) { | ||
46 | printf("changed ownership of '%s' to %u:%u\n", fileName, uid, gid); | ||
47 | } | ||
31 | return TRUE; | 48 | return TRUE; |
32 | } | 49 | } |
33 | bb_perror_msg("%s", fileName); /* A filename can have % in it... */ | 50 | if (!OPT_QUIET) |
51 | bb_perror_msg("%s", fileName); /* A filename can have % in it... */ | ||
34 | return FALSE; | 52 | return FALSE; |
35 | } | 53 | } |
36 | 54 | ||
37 | #define FLAG_R 1 | ||
38 | #define FLAG_h 2 | ||
39 | |||
40 | int chown_main(int argc, char **argv) | 55 | int chown_main(int argc, char **argv) |
41 | { | 56 | { |
42 | int flags; | ||
43 | int retval = EXIT_SUCCESS; | 57 | int retval = EXIT_SUCCESS; |
44 | char *groupName; | 58 | char *groupName; |
45 | 59 | ||
46 | flags = getopt32(argc, argv, "Rh"); | 60 | opt_complementary = "-2"; |
61 | getopt32(argc, argv, OPT_STR); | ||
47 | 62 | ||
48 | if (flags & FLAG_h) chown_func = lchown; | 63 | if (OPT_NODEREF) chown_func = lchown; |
49 | |||
50 | if (argc - optind < 2) { | ||
51 | bb_show_usage(); | ||
52 | } | ||
53 | 64 | ||
54 | argv += optind; | 65 | argv += optind; |
55 | 66 | ||
56 | /* First, check if there is a group name here */ | 67 | /* First, check if there is a group name here */ |
57 | if ((groupName = strchr(*argv, '.')) == NULL) { | 68 | groupName = strchr(*argv, '.'); |
69 | if (!groupName) { | ||
58 | groupName = strchr(*argv, ':'); | 70 | groupName = strchr(*argv, ':'); |
59 | } | 71 | } |
60 | 72 | ||
@@ -68,8 +80,14 @@ int chown_main(int argc, char **argv) | |||
68 | 80 | ||
69 | /* Ok, ready to do the deed now */ | 81 | /* Ok, ready to do the deed now */ |
70 | do { | 82 | do { |
71 | if (! recursive_action (*argv, (flags & FLAG_R), FALSE, FALSE, | 83 | if (!recursive_action(*argv, |
72 | fileAction, fileAction, NULL)) { | 84 | OPT_RECURSE, // recurse |
85 | FALSE, // follow links: TODO: -H/-L/-P | ||
86 | FALSE, // depth first | ||
87 | fileAction, // file action | ||
88 | fileAction, // dir action | ||
89 | NULL) // user data | ||
90 | ) { | ||
73 | retval = EXIT_FAILURE; | 91 | retval = EXIT_FAILURE; |
74 | } | 92 | } |
75 | } while (*++argv); | 93 | } while (*++argv); |
diff --git a/include/usage.h b/include/usage.h index 26122fd79..8d61c2907 100644 --- a/include/usage.h +++ b/include/usage.h | |||
@@ -204,12 +204,17 @@ | |||
204 | "-r--r--r-- 1 root root 0 Apr 12 18:25 /tmp/foo\n" | 204 | "-r--r--r-- 1 root root 0 Apr 12 18:25 /tmp/foo\n" |
205 | 205 | ||
206 | #define chown_trivial_usage \ | 206 | #define chown_trivial_usage \ |
207 | "[ -Rh ]... OWNER[<.|:>[GROUP]] FILE..." | 207 | "[-Rh"USE_DESKTOP("cvf")"]... OWNER[<.|:>[GROUP]] FILE..." |
208 | #define chown_full_usage \ | 208 | #define chown_full_usage \ |
209 | "Change the owner and/or group of each FILE to OWNER and/or GROUP.\n" \ | 209 | "Change the owner and/or group of each FILE to OWNER and/or GROUP.\n" \ |
210 | "\nOptions:\n" \ | 210 | "\nOptions:\n" \ |
211 | "\t-R\tChanges files and directories recursively\n" \ | 211 | "\t-R\tChanges files and directories recursively\n" \ |
212 | "\t-h\tDo not dereference symbolic links" | 212 | "\t-h\tDo not dereference symbolic links" \ |
213 | USE_DESKTOP( \ | ||
214 | "\n\t-c\tList changed files" \ | ||
215 | "\n\t-v\tList all files" \ | ||
216 | "\n\t-f\tHide errors" \ | ||
217 | ) | ||
213 | #define chown_example_usage \ | 218 | #define chown_example_usage \ |
214 | "$ ls -l /tmp/foo\n" \ | 219 | "$ ls -l /tmp/foo\n" \ |
215 | "-r--r--r-- 1 andersen andersen 0 Apr 12 18:25 /tmp/foo\n" \ | 220 | "-r--r--r-- 1 andersen andersen 0 Apr 12 18:25 /tmp/foo\n" \ |