xref: /aosp_15_r20/external/toybox/toys/other/lsattr.c (revision cf5a6c84e2b8763fc1a7db14496fd4742913b199)
1*cf5a6c84SAndroid Build Coastguard Worker /* lsattr.c - List file attributes on a Linux second extended file system.
2*cf5a6c84SAndroid Build Coastguard Worker  *
3*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2013 Ranjan Kumar <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2013 Kyungwan Han <[email protected]>
5*cf5a6c84SAndroid Build Coastguard Worker  *
6*cf5a6c84SAndroid Build Coastguard Worker  * No Standard.
7*cf5a6c84SAndroid Build Coastguard Worker  *
8*cf5a6c84SAndroid Build Coastguard Worker  * TODO cleanup
9*cf5a6c84SAndroid Build Coastguard Worker 
10*cf5a6c84SAndroid Build Coastguard Worker USE_LSATTR(NEWTOY(lsattr, "ldapvR", TOYFLAG_BIN))
11*cf5a6c84SAndroid Build Coastguard Worker USE_CHATTR(NEWTOY(chattr, "?p#v#R", TOYFLAG_BIN))
12*cf5a6c84SAndroid Build Coastguard Worker 
13*cf5a6c84SAndroid Build Coastguard Worker config LSATTR
14*cf5a6c84SAndroid Build Coastguard Worker   bool "lsattr"
15*cf5a6c84SAndroid Build Coastguard Worker   default y
16*cf5a6c84SAndroid Build Coastguard Worker   help
17*cf5a6c84SAndroid Build Coastguard Worker     usage: lsattr [-Radlpv] [FILE...]
18*cf5a6c84SAndroid Build Coastguard Worker 
19*cf5a6c84SAndroid Build Coastguard Worker     List file attributes on a Linux file system.
20*cf5a6c84SAndroid Build Coastguard Worker     Flag letters are defined in chattr help.
21*cf5a6c84SAndroid Build Coastguard Worker 
22*cf5a6c84SAndroid Build Coastguard Worker     -R	Recursively list attributes of directories and their contents
23*cf5a6c84SAndroid Build Coastguard Worker     -a	List all files in directories, including files that start with '.'
24*cf5a6c84SAndroid Build Coastguard Worker     -d	List directories like other files, rather than listing their contents
25*cf5a6c84SAndroid Build Coastguard Worker     -l	List long flag names
26*cf5a6c84SAndroid Build Coastguard Worker     -p	List the file's project number
27*cf5a6c84SAndroid Build Coastguard Worker     -v	List the file's version/generation number
28*cf5a6c84SAndroid Build Coastguard Worker 
29*cf5a6c84SAndroid Build Coastguard Worker config CHATTR
30*cf5a6c84SAndroid Build Coastguard Worker   bool "chattr"
31*cf5a6c84SAndroid Build Coastguard Worker   default y
32*cf5a6c84SAndroid Build Coastguard Worker   help
33*cf5a6c84SAndroid Build Coastguard Worker     usage: chattr [-R] [-+=AacDdijsStTu] [-p PROJID] [-v VERSION] [FILE...]
34*cf5a6c84SAndroid Build Coastguard Worker 
35*cf5a6c84SAndroid Build Coastguard Worker     Change file attributes on a Linux file system.
36*cf5a6c84SAndroid Build Coastguard Worker 
37*cf5a6c84SAndroid Build Coastguard Worker     -R	Recurse
38*cf5a6c84SAndroid Build Coastguard Worker     -p	Set the file's project number
39*cf5a6c84SAndroid Build Coastguard Worker     -v	Set the file's version/generation number
40*cf5a6c84SAndroid Build Coastguard Worker 
41*cf5a6c84SAndroid Build Coastguard Worker     Operators:
42*cf5a6c84SAndroid Build Coastguard Worker       '-' Remove attributes
43*cf5a6c84SAndroid Build Coastguard Worker       '+' Add attributes
44*cf5a6c84SAndroid Build Coastguard Worker       '=' Set attributes
45*cf5a6c84SAndroid Build Coastguard Worker 
46*cf5a6c84SAndroid Build Coastguard Worker     Attributes:
47*cf5a6c84SAndroid Build Coastguard Worker       A  No atime                     a  Append only
48*cf5a6c84SAndroid Build Coastguard Worker       C  No COW                       c  Compression
49*cf5a6c84SAndroid Build Coastguard Worker       D  Synchronous dir updates      d  No dump
50*cf5a6c84SAndroid Build Coastguard Worker       E  Encrypted                    e  Extents
51*cf5a6c84SAndroid Build Coastguard Worker       F  Case-insensitive (casefold)
52*cf5a6c84SAndroid Build Coastguard Worker       I  Indexed directory            i  Immutable
53*cf5a6c84SAndroid Build Coastguard Worker       j  Journal data
54*cf5a6c84SAndroid Build Coastguard Worker       N  Inline data in inode
55*cf5a6c84SAndroid Build Coastguard Worker       P  Project hierarchy
56*cf5a6c84SAndroid Build Coastguard Worker       S  Synchronous file updates     s  Secure delete
57*cf5a6c84SAndroid Build Coastguard Worker       T  Top of dir hierarchy         t  No tail-merging
58*cf5a6c84SAndroid Build Coastguard Worker       u  Allow undelete
59*cf5a6c84SAndroid Build Coastguard Worker       V  Verity
60*cf5a6c84SAndroid Build Coastguard Worker */
61*cf5a6c84SAndroid Build Coastguard Worker #define FOR_lsattr
62*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
63*cf5a6c84SAndroid Build Coastguard Worker #include <linux/fs.h>
64*cf5a6c84SAndroid Build Coastguard Worker 
65*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
66*cf5a6c84SAndroid Build Coastguard Worker   long v, p;
67*cf5a6c84SAndroid Build Coastguard Worker 
68*cf5a6c84SAndroid Build Coastguard Worker   unsigned add, rm, set;
69*cf5a6c84SAndroid Build Coastguard Worker   // !add and !rm tell us whether they were used, but `chattr =` is meaningful.
70*cf5a6c84SAndroid Build Coastguard Worker   int have_set;
71*cf5a6c84SAndroid Build Coastguard Worker )
72*cf5a6c84SAndroid Build Coastguard Worker 
73*cf5a6c84SAndroid Build Coastguard Worker // Added more recently than the 7 year support horizon. TODO: remove
74*cf5a6c84SAndroid Build Coastguard Worker #ifndef FS_CASEFOLD_FL
75*cf5a6c84SAndroid Build Coastguard Worker #define FS_CASEFOLD_FL    0x40000000 // commit 71e90b4654a92 2019-07-23
76*cf5a6c84SAndroid Build Coastguard Worker #endif
77*cf5a6c84SAndroid Build Coastguard Worker #ifndef FS_VERITY_FL
78*cf5a6c84SAndroid Build Coastguard Worker #define FS_VERITY_FL      0x00100000 // commit fe9918d3b228b 2019-07-22
79*cf5a6c84SAndroid Build Coastguard Worker #endif
80*cf5a6c84SAndroid Build Coastguard Worker 
81*cf5a6c84SAndroid Build Coastguard Worker static struct ext2_attr {
82*cf5a6c84SAndroid Build Coastguard Worker   char *name;
83*cf5a6c84SAndroid Build Coastguard Worker   unsigned flag;
84*cf5a6c84SAndroid Build Coastguard Worker   char opt;
85*cf5a6c84SAndroid Build Coastguard Worker } e2attrs[] = {
86*cf5a6c84SAndroid Build Coastguard Worker   // Do not sort! These are in the order that lsattr outputs them.
87*cf5a6c84SAndroid Build Coastguard Worker   {"Secure_Deletion",               FS_SECRM_FL,        's'},
88*cf5a6c84SAndroid Build Coastguard Worker   {"Undelete",                      FS_UNRM_FL,         'u'},
89*cf5a6c84SAndroid Build Coastguard Worker   {"Synchronous_Updates",           FS_SYNC_FL,         'S'},
90*cf5a6c84SAndroid Build Coastguard Worker   {"Synchronous_Directory_Updates", FS_DIRSYNC_FL,      'D'},
91*cf5a6c84SAndroid Build Coastguard Worker   {"Immutable",                     FS_IMMUTABLE_FL,    'i'},
92*cf5a6c84SAndroid Build Coastguard Worker   {"Append_Only",                   FS_APPEND_FL,       'a'},
93*cf5a6c84SAndroid Build Coastguard Worker   {"No_Dump",                       FS_NODUMP_FL,       'd'},
94*cf5a6c84SAndroid Build Coastguard Worker   {"No_Atime",                      FS_NOATIME_FL,      'A'},
95*cf5a6c84SAndroid Build Coastguard Worker   {"Compression_Requested",         FS_COMPR_FL,        'c'},
96*cf5a6c84SAndroid Build Coastguard Worker   {"Encrypted",                     FS_ENCRYPT_FL,      'E'},
97*cf5a6c84SAndroid Build Coastguard Worker   {"Journaled_Data",                FS_JOURNAL_DATA_FL, 'j'},
98*cf5a6c84SAndroid Build Coastguard Worker   {"Indexed_directory",             FS_INDEX_FL,        'I'},
99*cf5a6c84SAndroid Build Coastguard Worker   {"No_Tailmerging",                FS_NOTAIL_FL,       't'},
100*cf5a6c84SAndroid Build Coastguard Worker   {"Top_of_Directory_Hierarchies",  FS_TOPDIR_FL,       'T'},
101*cf5a6c84SAndroid Build Coastguard Worker   {"Extents",                       FS_EXTENT_FL,       'e'},
102*cf5a6c84SAndroid Build Coastguard Worker   {"No_COW",                        FS_NOCOW_FL,        'C'},
103*cf5a6c84SAndroid Build Coastguard Worker   {"Casefold",                      FS_CASEFOLD_FL,     'F'},
104*cf5a6c84SAndroid Build Coastguard Worker   {"Inline_Data",                   FS_INLINE_DATA_FL,  'N'},
105*cf5a6c84SAndroid Build Coastguard Worker   {"Project_Hierarchy",             FS_PROJINHERIT_FL,  'P'},
106*cf5a6c84SAndroid Build Coastguard Worker   {"Verity",                        FS_VERITY_FL,       'V'},
107*cf5a6c84SAndroid Build Coastguard Worker   {NULL,                            0,                  0},
108*cf5a6c84SAndroid Build Coastguard Worker };
109*cf5a6c84SAndroid Build Coastguard Worker 
110*cf5a6c84SAndroid Build Coastguard Worker // Get file flags on a Linux second extended file system.
ext2_getflag(int fd,struct stat * sb,unsigned * flag)111*cf5a6c84SAndroid Build Coastguard Worker static int ext2_getflag(int fd, struct stat *sb, unsigned *flag)
112*cf5a6c84SAndroid Build Coastguard Worker {
113*cf5a6c84SAndroid Build Coastguard Worker   if(!S_ISREG(sb->st_mode) && !S_ISDIR(sb->st_mode)) {
114*cf5a6c84SAndroid Build Coastguard Worker     errno = EOPNOTSUPP;
115*cf5a6c84SAndroid Build Coastguard Worker     return -1;
116*cf5a6c84SAndroid Build Coastguard Worker   }
117*cf5a6c84SAndroid Build Coastguard Worker   return ioctl(fd, FS_IOC_GETFLAGS, flag);
118*cf5a6c84SAndroid Build Coastguard Worker }
119*cf5a6c84SAndroid Build Coastguard Worker 
attrstr(unsigned attrs,int full)120*cf5a6c84SAndroid Build Coastguard Worker static char *attrstr(unsigned attrs, int full)
121*cf5a6c84SAndroid Build Coastguard Worker {
122*cf5a6c84SAndroid Build Coastguard Worker   struct ext2_attr *a = e2attrs;
123*cf5a6c84SAndroid Build Coastguard Worker   char *s = toybuf;
124*cf5a6c84SAndroid Build Coastguard Worker 
125*cf5a6c84SAndroid Build Coastguard Worker   for (; a->name; a++)
126*cf5a6c84SAndroid Build Coastguard Worker     if (attrs & a->flag) *s++ = a->opt;
127*cf5a6c84SAndroid Build Coastguard Worker     else if (full) *s++ = '-';
128*cf5a6c84SAndroid Build Coastguard Worker   *s = 0;
129*cf5a6c84SAndroid Build Coastguard Worker 
130*cf5a6c84SAndroid Build Coastguard Worker   return toybuf;
131*cf5a6c84SAndroid Build Coastguard Worker }
132*cf5a6c84SAndroid Build Coastguard Worker 
print_file_attr(char * path)133*cf5a6c84SAndroid Build Coastguard Worker static void print_file_attr(char *path)
134*cf5a6c84SAndroid Build Coastguard Worker {
135*cf5a6c84SAndroid Build Coastguard Worker   unsigned flag = 0, version = 0;
136*cf5a6c84SAndroid Build Coastguard Worker   int fd = -1;
137*cf5a6c84SAndroid Build Coastguard Worker   struct stat sb;
138*cf5a6c84SAndroid Build Coastguard Worker 
139*cf5a6c84SAndroid Build Coastguard Worker   if (!stat(path, &sb) && !S_ISREG(sb.st_mode) && !S_ISDIR(sb.st_mode)) {
140*cf5a6c84SAndroid Build Coastguard Worker     errno = EOPNOTSUPP;
141*cf5a6c84SAndroid Build Coastguard Worker     goto error;
142*cf5a6c84SAndroid Build Coastguard Worker   }
143*cf5a6c84SAndroid Build Coastguard Worker   if (-1 == (fd=open(path, O_RDONLY | O_NONBLOCK))) goto error;
144*cf5a6c84SAndroid Build Coastguard Worker 
145*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(p)) {
146*cf5a6c84SAndroid Build Coastguard Worker     struct fsxattr fsx;
147*cf5a6c84SAndroid Build Coastguard Worker 
148*cf5a6c84SAndroid Build Coastguard Worker     if (ioctl(fd, FS_IOC_FSGETXATTR, &fsx)) goto error;
149*cf5a6c84SAndroid Build Coastguard Worker     xprintf("%5u ", fsx.fsx_projid);
150*cf5a6c84SAndroid Build Coastguard Worker   }
151*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(v)) {
152*cf5a6c84SAndroid Build Coastguard Worker     if (ioctl(fd, FS_IOC_GETVERSION, (void*)&version) < 0) goto error;
153*cf5a6c84SAndroid Build Coastguard Worker     xprintf("%-10u ", version);
154*cf5a6c84SAndroid Build Coastguard Worker   }
155*cf5a6c84SAndroid Build Coastguard Worker 
156*cf5a6c84SAndroid Build Coastguard Worker   if (ext2_getflag(fd, &sb, &flag) < 0) perror_msg("reading flags '%s'", path);
157*cf5a6c84SAndroid Build Coastguard Worker   else {
158*cf5a6c84SAndroid Build Coastguard Worker     struct ext2_attr *ptr = e2attrs;
159*cf5a6c84SAndroid Build Coastguard Worker     int name_found = 0;
160*cf5a6c84SAndroid Build Coastguard Worker 
161*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(l)) {
162*cf5a6c84SAndroid Build Coastguard Worker       xprintf("%-50s ", path);
163*cf5a6c84SAndroid Build Coastguard Worker       for (; ptr->name; ptr++) {
164*cf5a6c84SAndroid Build Coastguard Worker         if (flag & ptr->flag) {
165*cf5a6c84SAndroid Build Coastguard Worker           if (name_found) xprintf(", "); //for formatting.
166*cf5a6c84SAndroid Build Coastguard Worker           xprintf("%s", ptr->name);
167*cf5a6c84SAndroid Build Coastguard Worker           name_found = 1;
168*cf5a6c84SAndroid Build Coastguard Worker         }
169*cf5a6c84SAndroid Build Coastguard Worker       }
170*cf5a6c84SAndroid Build Coastguard Worker       if (!name_found) xprintf("---");
171*cf5a6c84SAndroid Build Coastguard Worker       xputc('\n');
172*cf5a6c84SAndroid Build Coastguard Worker     } else xprintf("%s %s\n", attrstr(flag, 1), path);
173*cf5a6c84SAndroid Build Coastguard Worker   }
174*cf5a6c84SAndroid Build Coastguard Worker   path = 0;
175*cf5a6c84SAndroid Build Coastguard Worker error:
176*cf5a6c84SAndroid Build Coastguard Worker   xclose(fd);
177*cf5a6c84SAndroid Build Coastguard Worker   if (path) perror_msg("reading '%s'", path);
178*cf5a6c84SAndroid Build Coastguard Worker }
179*cf5a6c84SAndroid Build Coastguard Worker 
180*cf5a6c84SAndroid Build Coastguard Worker // Get directory information.
retell_dir(struct dirtree * root)181*cf5a6c84SAndroid Build Coastguard Worker static int retell_dir(struct dirtree *root)
182*cf5a6c84SAndroid Build Coastguard Worker {
183*cf5a6c84SAndroid Build Coastguard Worker   char *fpath = 0;
184*cf5a6c84SAndroid Build Coastguard Worker 
185*cf5a6c84SAndroid Build Coastguard Worker   if (root->again) {
186*cf5a6c84SAndroid Build Coastguard Worker     xputc('\n');
187*cf5a6c84SAndroid Build Coastguard Worker 
188*cf5a6c84SAndroid Build Coastguard Worker     return 0;
189*cf5a6c84SAndroid Build Coastguard Worker   }
190*cf5a6c84SAndroid Build Coastguard Worker   if (S_ISDIR(root->st.st_mode) && !root->parent)
191*cf5a6c84SAndroid Build Coastguard Worker     return DIRTREE_RECURSE|DIRTREE_COMEAGAIN;
192*cf5a6c84SAndroid Build Coastguard Worker 
193*cf5a6c84SAndroid Build Coastguard Worker   fpath = dirtree_path(root, NULL);
194*cf5a6c84SAndroid Build Coastguard Worker   if (*root->name != '.' || FLAG(a)) {
195*cf5a6c84SAndroid Build Coastguard Worker     print_file_attr(fpath);
196*cf5a6c84SAndroid Build Coastguard Worker     if (S_ISDIR(root->st.st_mode) && FLAG(R) && dirtree_notdotdot(root)) {
197*cf5a6c84SAndroid Build Coastguard Worker       xprintf("\n%s:\n", fpath);
198*cf5a6c84SAndroid Build Coastguard Worker       free(fpath);
199*cf5a6c84SAndroid Build Coastguard Worker       return DIRTREE_RECURSE|DIRTREE_COMEAGAIN;
200*cf5a6c84SAndroid Build Coastguard Worker     }
201*cf5a6c84SAndroid Build Coastguard Worker   }
202*cf5a6c84SAndroid Build Coastguard Worker   free(fpath);
203*cf5a6c84SAndroid Build Coastguard Worker 
204*cf5a6c84SAndroid Build Coastguard Worker   return 0;
205*cf5a6c84SAndroid Build Coastguard Worker }
206*cf5a6c84SAndroid Build Coastguard Worker 
lsattr_main(void)207*cf5a6c84SAndroid Build Coastguard Worker void lsattr_main(void)
208*cf5a6c84SAndroid Build Coastguard Worker {
209*cf5a6c84SAndroid Build Coastguard Worker   if (!*toys.optargs) dirtree_read(".", retell_dir);
210*cf5a6c84SAndroid Build Coastguard Worker   else for (; *toys.optargs; toys.optargs++) {
211*cf5a6c84SAndroid Build Coastguard Worker     struct stat sb;
212*cf5a6c84SAndroid Build Coastguard Worker 
213*cf5a6c84SAndroid Build Coastguard Worker     if (lstat(*toys.optargs, &sb)) perror_msg("stat '%s'", *toys.optargs);
214*cf5a6c84SAndroid Build Coastguard Worker     else if (S_ISDIR(sb.st_mode) && !FLAG(d))
215*cf5a6c84SAndroid Build Coastguard Worker       dirtree_read(*toys.optargs, retell_dir);
216*cf5a6c84SAndroid Build Coastguard Worker     else print_file_attr(*toys.optargs);// to handle "./Filename" or "./Dir"
217*cf5a6c84SAndroid Build Coastguard Worker   }
218*cf5a6c84SAndroid Build Coastguard Worker }
219*cf5a6c84SAndroid Build Coastguard Worker 
220*cf5a6c84SAndroid Build Coastguard Worker // Switch gears from lsattr to chattr.
221*cf5a6c84SAndroid Build Coastguard Worker #define FOR_chattr
222*cf5a6c84SAndroid Build Coastguard Worker #include "generated/flags.h"
223*cf5a6c84SAndroid Build Coastguard Worker 
224*cf5a6c84SAndroid Build Coastguard Worker // Set file flags on a Linux second extended file system.
ext2_setflag(int fd,struct stat * sb,unsigned flag)225*cf5a6c84SAndroid Build Coastguard Worker static inline int ext2_setflag(int fd, struct stat *sb, unsigned flag)
226*cf5a6c84SAndroid Build Coastguard Worker {
227*cf5a6c84SAndroid Build Coastguard Worker   if (!S_ISREG(sb->st_mode) && !S_ISDIR(sb->st_mode)) {
228*cf5a6c84SAndroid Build Coastguard Worker     errno = EOPNOTSUPP;
229*cf5a6c84SAndroid Build Coastguard Worker     return -1;
230*cf5a6c84SAndroid Build Coastguard Worker   }
231*cf5a6c84SAndroid Build Coastguard Worker   return ioctl(fd, FS_IOC_SETFLAGS, &flag);
232*cf5a6c84SAndroid Build Coastguard Worker }
233*cf5a6c84SAndroid Build Coastguard Worker 
get_flag_val(char ch)234*cf5a6c84SAndroid Build Coastguard Worker static unsigned get_flag_val(char ch)
235*cf5a6c84SAndroid Build Coastguard Worker {
236*cf5a6c84SAndroid Build Coastguard Worker   struct ext2_attr *ptr = e2attrs;
237*cf5a6c84SAndroid Build Coastguard Worker 
238*cf5a6c84SAndroid Build Coastguard Worker   for (; ptr->name; ptr++) if (ptr->opt == ch) return ptr->flag;
239*cf5a6c84SAndroid Build Coastguard Worker   help_exit("bad '%c'", ch);
240*cf5a6c84SAndroid Build Coastguard Worker }
241*cf5a6c84SAndroid Build Coastguard Worker 
242*cf5a6c84SAndroid Build Coastguard Worker // Parse command line argument and fill the chattr structure.
parse_cmdline_arg(char *** argv)243*cf5a6c84SAndroid Build Coastguard Worker static void parse_cmdline_arg(char ***argv)
244*cf5a6c84SAndroid Build Coastguard Worker {
245*cf5a6c84SAndroid Build Coastguard Worker   char *arg = **argv, *ptr;
246*cf5a6c84SAndroid Build Coastguard Worker 
247*cf5a6c84SAndroid Build Coastguard Worker   while (arg) {
248*cf5a6c84SAndroid Build Coastguard Worker     if (*arg=='-') for (ptr = ++arg; *ptr; ptr++) TT.rm |= get_flag_val(*ptr);
249*cf5a6c84SAndroid Build Coastguard Worker     else if (*arg=='+')
250*cf5a6c84SAndroid Build Coastguard Worker       for (ptr = ++arg; *ptr; ptr++) TT.add |= get_flag_val(*ptr);
251*cf5a6c84SAndroid Build Coastguard Worker     else if (*arg=='=') {
252*cf5a6c84SAndroid Build Coastguard Worker       TT.have_set = 1;
253*cf5a6c84SAndroid Build Coastguard Worker       for (ptr = ++arg; *ptr; ptr++) TT.set |= get_flag_val(*ptr);
254*cf5a6c84SAndroid Build Coastguard Worker     } else return;
255*cf5a6c84SAndroid Build Coastguard Worker     arg = *(*argv += 1);
256*cf5a6c84SAndroid Build Coastguard Worker   }
257*cf5a6c84SAndroid Build Coastguard Worker }
258*cf5a6c84SAndroid Build Coastguard Worker 
259*cf5a6c84SAndroid Build Coastguard Worker // Update attribute of given file.
update_attr(struct dirtree * root)260*cf5a6c84SAndroid Build Coastguard Worker static int update_attr(struct dirtree *root)
261*cf5a6c84SAndroid Build Coastguard Worker {
262*cf5a6c84SAndroid Build Coastguard Worker   char *fpath = 0;
263*cf5a6c84SAndroid Build Coastguard Worker   int vv = TT.v, fd;
264*cf5a6c84SAndroid Build Coastguard Worker 
265*cf5a6c84SAndroid Build Coastguard Worker   if (!dirtree_notdotdot(root)) return 0;
266*cf5a6c84SAndroid Build Coastguard Worker 
267*cf5a6c84SAndroid Build Coastguard Worker   // if file is a link and recursive is set or file is not regular+link+dir
268*cf5a6c84SAndroid Build Coastguard Worker   // (like fifo or dev file) then escape the file.
269*cf5a6c84SAndroid Build Coastguard Worker   if ((S_ISLNK(root->st.st_mode) && FLAG(R))
270*cf5a6c84SAndroid Build Coastguard Worker     || (!S_ISREG(root->st.st_mode) && !S_ISLNK(root->st.st_mode)
271*cf5a6c84SAndroid Build Coastguard Worker       && !S_ISDIR(root->st.st_mode)))
272*cf5a6c84SAndroid Build Coastguard Worker     return 0;
273*cf5a6c84SAndroid Build Coastguard Worker 
274*cf5a6c84SAndroid Build Coastguard Worker   fpath = dirtree_path(root, NULL);
275*cf5a6c84SAndroid Build Coastguard Worker   if (-1 == (fd=open(fpath, O_RDONLY | O_NONBLOCK))) {
276*cf5a6c84SAndroid Build Coastguard Worker     free(fpath);
277*cf5a6c84SAndroid Build Coastguard Worker     return DIRTREE_ABORT;
278*cf5a6c84SAndroid Build Coastguard Worker   }
279*cf5a6c84SAndroid Build Coastguard Worker 
280*cf5a6c84SAndroid Build Coastguard Worker   // Any potential flag changes?
281*cf5a6c84SAndroid Build Coastguard Worker   if (TT.have_set | TT.add | TT.rm) {
282*cf5a6c84SAndroid Build Coastguard Worker     unsigned orig, new;
283*cf5a6c84SAndroid Build Coastguard Worker 
284*cf5a6c84SAndroid Build Coastguard Worker     // Read current flags.
285*cf5a6c84SAndroid Build Coastguard Worker     if (ext2_getflag(fd, &(root->st), &orig) < 0) {
286*cf5a6c84SAndroid Build Coastguard Worker       perror_msg("read flags of '%s'", fpath);
287*cf5a6c84SAndroid Build Coastguard Worker       free(fpath);
288*cf5a6c84SAndroid Build Coastguard Worker       xclose(fd);
289*cf5a6c84SAndroid Build Coastguard Worker       return DIRTREE_ABORT;
290*cf5a6c84SAndroid Build Coastguard Worker     }
291*cf5a6c84SAndroid Build Coastguard Worker     // Apply the requested changes.
292*cf5a6c84SAndroid Build Coastguard Worker     if (TT.have_set) new = TT.set; // '='.
293*cf5a6c84SAndroid Build Coastguard Worker     else { // '-' and/or '+'.
294*cf5a6c84SAndroid Build Coastguard Worker       new = orig;
295*cf5a6c84SAndroid Build Coastguard Worker       new &= ~(TT.rm);
296*cf5a6c84SAndroid Build Coastguard Worker       new |= TT.add;
297*cf5a6c84SAndroid Build Coastguard Worker       if (!S_ISDIR(root->st.st_mode)) new &= ~FS_DIRSYNC_FL;
298*cf5a6c84SAndroid Build Coastguard Worker     }
299*cf5a6c84SAndroid Build Coastguard Worker     // Write them back if there was any change.
300*cf5a6c84SAndroid Build Coastguard Worker     if (orig != new && ext2_setflag(fd, &(root->st), new)<0)
301*cf5a6c84SAndroid Build Coastguard Worker       perror_msg("%s: setting flags to =%s failed", fpath, attrstr(new, 0));
302*cf5a6c84SAndroid Build Coastguard Worker   }
303*cf5a6c84SAndroid Build Coastguard Worker 
304*cf5a6c84SAndroid Build Coastguard Worker   // (FS_IOC_SETVERSION works all the way back to 2.6, but FS_IOC_FSSETXATTR
305*cf5a6c84SAndroid Build Coastguard Worker   // isn't available until 4.5.)
306*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(v) && (ioctl(fd, FS_IOC_SETVERSION, &vv)<0))
307*cf5a6c84SAndroid Build Coastguard Worker     perror_msg("%s: setting version to %d failed", fpath, vv);
308*cf5a6c84SAndroid Build Coastguard Worker 
309*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(p)) {
310*cf5a6c84SAndroid Build Coastguard Worker     struct fsxattr fsx;
311*cf5a6c84SAndroid Build Coastguard Worker     int fail = ioctl(fd, FS_IOC_FSGETXATTR, &fsx);
312*cf5a6c84SAndroid Build Coastguard Worker 
313*cf5a6c84SAndroid Build Coastguard Worker     fsx.fsx_projid = TT.p;
314*cf5a6c84SAndroid Build Coastguard Worker     if (fail || ioctl(fd, FS_IOC_FSSETXATTR, &fsx))
315*cf5a6c84SAndroid Build Coastguard Worker       perror_msg("%s: setting projid to %u failed", fpath, fsx.fsx_projid);
316*cf5a6c84SAndroid Build Coastguard Worker   }
317*cf5a6c84SAndroid Build Coastguard Worker 
318*cf5a6c84SAndroid Build Coastguard Worker   free(fpath);
319*cf5a6c84SAndroid Build Coastguard Worker   xclose(fd);
320*cf5a6c84SAndroid Build Coastguard Worker   return (FLAG(R) && S_ISDIR(root->st.st_mode)) ? DIRTREE_RECURSE : 0;
321*cf5a6c84SAndroid Build Coastguard Worker }
322*cf5a6c84SAndroid Build Coastguard Worker 
chattr_main(void)323*cf5a6c84SAndroid Build Coastguard Worker void chattr_main(void)
324*cf5a6c84SAndroid Build Coastguard Worker {
325*cf5a6c84SAndroid Build Coastguard Worker   char **argv = toys.optargs;
326*cf5a6c84SAndroid Build Coastguard Worker 
327*cf5a6c84SAndroid Build Coastguard Worker   parse_cmdline_arg(&argv);
328*cf5a6c84SAndroid Build Coastguard Worker   if (TT.p < 0 || TT.p > UINT_MAX) error_exit("bad projid %lu", TT.p);
329*cf5a6c84SAndroid Build Coastguard Worker   if (TT.v < 0 || TT.v > UINT_MAX) error_exit("bad version %ld", TT.v);
330*cf5a6c84SAndroid Build Coastguard Worker   if (!*argv) help_exit("no file");
331*cf5a6c84SAndroid Build Coastguard Worker   if (TT.have_set && (TT.add || TT.rm))
332*cf5a6c84SAndroid Build Coastguard Worker     error_exit("no '=' with '-' or '+'");
333*cf5a6c84SAndroid Build Coastguard Worker   if (TT.rm & TT.add) error_exit("set/unset same flag");
334*cf5a6c84SAndroid Build Coastguard Worker   if (!(TT.add || TT.rm || TT.have_set || FLAG(p) || FLAG(v)))
335*cf5a6c84SAndroid Build Coastguard Worker     error_exit("need '-p', '-v', '=', '-', or '+'");
336*cf5a6c84SAndroid Build Coastguard Worker   for (; *argv; argv++) dirtree_read(*argv, update_attr);
337*cf5a6c84SAndroid Build Coastguard Worker }
338