1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34#define FOR_chgrp
35#define FORCE_FLAGS
36#include "toys.h"
37
38GLOBALS(
39 uid_t owner;
40 gid_t group;
41 char *owner_name, *group_name;
42 int symfollow;
43)
44
45static int do_chgrp(struct dirtree *node)
46{
47 int fd, ret, flags = toys.optflags;
48
49
50 if (!dirtree_notdotdot(node)) return 0;
51 if ((flags & FLAG_R) && !node->again && S_ISDIR(node->st.st_mode))
52 return DIRTREE_COMEAGAIN|(DIRTREE_SYMFOLLOW*!!(flags&FLAG_L));
53
54 fd = dirtree_parentfd(node);
55 ret = fchownat(fd, node->name, TT.owner, TT.group,
56 AT_SYMLINK_NOFOLLOW*(!(flags&(FLAG_L|FLAG_H)) && (flags&(FLAG_h|FLAG_R))));
57
58 if (ret || (flags & FLAG_v)) {
59 char *path = dirtree_path(node, 0);
60 if (flags & FLAG_v)
61 xprintf("%s %s%s%s %s\n", toys.which->name,
62 TT.owner_name ? TT.owner_name : "",
63 toys.which->name[2]=='o' && TT.group_name ? ":" : "",
64 TT.group_name ? TT.group_name : "", path);
65 if (ret == -1 && !(toys.optflags & FLAG_f))
66 perror_msg("'%s' to '%s:%s'", path, TT.owner_name, TT.group_name);
67 free(path);
68 }
69 toys.exitval |= ret;
70
71 return 0;
72}
73
74void chgrp_main(void)
75{
76 int ischown = toys.which->name[2] == 'o';
77 char **s, *own;
78
79 TT.owner = TT.group = -1;
80
81
82 if (ischown) {
83 char *grp;
84
85 own = xstrdup(*toys.optargs);
86 if ((grp = strchr(own, ':')) || (grp = strchr(own, '.'))) {
87 *(grp++) = 0;
88 TT.group_name = grp;
89 }
90 if (*own) TT.owner = xgetuid(TT.owner_name = own);
91 } else TT.group_name = *toys.optargs;
92
93 if (TT.group_name && *TT.group_name)
94 TT.group = xgetgid(TT.group_name);
95
96 for (s=toys.optargs+1; *s; s++)
97 dirtree_flagread(*s, DIRTREE_SYMFOLLOW*!!(toys.optflags&(FLAG_H|FLAG_L)),
98 do_chgrp);
99
100 if (CFG_TOYBOX_FREE && ischown) free(own);
101}
102
103void chown_main()
104{
105 chgrp_main();
106}
107