1 /* lsattr.c - List file attributes on a Linux second extended file system.
2 *
3 * Copyright 2013 Ranjan Kumar <ranjankumar.bth@gmail.com>
4 * Copyright 2013 Kyungwan Han <asura321@gmail.com>
5 *
6 * No Standard.
7 *
8 * TODO cleanup
9
10 USE_LSATTR(NEWTOY(lsattr, "vldaR", TOYFLAG_BIN))
11 USE_CHATTR(NEWTOY(chattr, NULL, TOYFLAG_BIN))
12
13 config LSATTR
14 bool "lsattr"
15 default y
16 help
17 usage: lsattr [-Radlv] [Files...]
18
19 List file attributes on a Linux second extended file system.
20 (AacDdijsStu defined in chattr --help)
21
22 -R Recursively list attributes of directories and their contents.
23 -a List all files in directories, including files that start with '.'.
24 -d List directories like other files, rather than listing their contents.
25 -l List long flag names.
26 -v List the file's version/generation number.
27
28 config CHATTR
29 bool "chattr"
30 default y
31 help
32 usage: chattr [-R] [-+=AacDdijsStTu] [-v version] [File...]
33
34 Change file attributes on a Linux second extended file system.
35
36 -R Recurse.
37 -v Set the file's version/generation number.
38
39 Operators:
40 '-' Remove attributes.
41 '+' Add attributes.
42 '=' Set attributes.
43
44 Attributes:
45 A Don't track atime.
46 a Append mode only.
47 c Enable compress.
48 D Write dir contents synchronously.
49 d Don't backup with dump.
50 i Cannot be modified (immutable).
51 j Write all data to journal first.
52 s Zero disk storage when deleted.
53 S Write file contents synchronously.
54 t Disable tail-merging of partial blocks with other files.
55 u Allow file to be undeleted.
56 */
57 #define FOR_lsattr
58 #include "toys.h"
59 #include <linux/fs.h>
60
61 static struct ext2_attr {
62 char *name;
63 unsigned long flag;
64 char opt;
65 } e2attrs[] = {
66 {"Secure_Deletion", FS_SECRM_FL, 's'}, // Secure deletion
67 {"Undelete", FS_UNRM_FL, 'u'}, // Undelete
68 {"Compression_Requested", FS_COMPR_FL, 'c'}, // Compress file
69 {"Synchronous_Updates", FS_SYNC_FL, 'S'}, // Synchronous updates
70 {"Immutable", FS_IMMUTABLE_FL, 'i'}, // Immutable file
71 {"Append_Only", FS_APPEND_FL, 'a'}, // writes to file may only append
72 {"No_Dump", FS_NODUMP_FL, 'd'}, // do not dump file
73 {"No_Atime", FS_NOATIME_FL, 'A'}, // do not update atime
74 {"Indexed_directory", FS_INDEX_FL, 'I'}, // hash-indexed directory
75 {"Journaled_Data", FS_JOURNAL_DATA_FL, 'j'}, // file data should be journaled
76 {"No_Tailmerging", FS_NOTAIL_FL, 't'}, // file tail should not be merged
77 {"Synchronous_Directory_Updates", FS_DIRSYNC_FL, 'D'}, // dirsync behaviour (directories only)
78 {"Top_of_Directory_Hierarchies", FS_TOPDIR_FL, 'T'}, // Top of directory hierarchies
79 {NULL, -1, 0},
80 };
81
82 // Get file flags on a Linux second extended file system.
ext2_getflag(int fd,struct stat * sb,unsigned long * flag)83 static int ext2_getflag(int fd, struct stat *sb, unsigned long *flag)
84 {
85 if(!S_ISREG(sb->st_mode) && !S_ISDIR(sb->st_mode)) {
86 errno = EOPNOTSUPP;
87 return -1;
88 }
89 return (ioctl(fd, FS_IOC_GETFLAGS, (void*)flag));
90 }
91
print_file_attr(char * path)92 static void print_file_attr(char *path)
93 {
94 unsigned long flag = 0, version = 0;
95 int fd;
96 struct stat sb;
97
98 if (!stat(path, &sb) && !S_ISREG(sb.st_mode) && !S_ISDIR(sb.st_mode)) {
99 errno = EOPNOTSUPP;
100 goto LABEL1;
101 }
102 if (-1 == (fd=open(path, O_RDONLY | O_NONBLOCK))) goto LABEL1;
103
104 if (toys.optflags & FLAG_v) {
105 if (ioctl(fd, FS_IOC_GETVERSION, (void*)&version) < 0) goto LABEL2;
106 xprintf("%5lu ", version);
107 }
108
109 if (ext2_getflag(fd, &sb, &flag) < 0) perror_msg("reading flags '%s'", path);
110 else {
111 struct ext2_attr *ptr = e2attrs;
112
113 if (toys.optflags & FLAG_l) {
114 int name_found = 0;
115
116 xprintf("%-50s ", path);
117 for (; ptr->name; ptr++) {
118 if (flag & ptr->flag) {
119 if (name_found) xprintf(", "); //for formatting.
120 xprintf("%s", ptr->name);
121 name_found = 1;
122 }
123 }
124 if (!name_found) xprintf("---");
125 xputc('\n');
126 } else {
127 int index = 0;
128
129 for (; ptr->name; ptr++)
130 toybuf[index++] = (flag & ptr->flag) ? ptr->opt : '-';
131 toybuf[index] = '\0';
132 xprintf("%s %s\n", toybuf, path);
133 }
134 }
135 xclose(fd);
136 return;
137 LABEL2: xclose(fd);
138 LABEL1: perror_msg("reading '%s'", path);
139 }
140
141 // Get directory information.
retell_dir(struct dirtree * root)142 static int retell_dir(struct dirtree *root)
143 {
144 char *fpath = NULL;
145
146 if (root->again) {
147 xputc('\n');
148 return 0;
149 }
150 if (S_ISDIR(root->st.st_mode) && !root->parent)
151 return (DIRTREE_RECURSE | DIRTREE_COMEAGAIN);
152
153 fpath = dirtree_path(root, NULL);
154 //Special case: with '-a' option and '.'/'..' also included in printing list.
155 if ((root->name[0] != '.') || (toys.optflags & FLAG_a)) {
156 print_file_attr(fpath);
157 if (S_ISDIR(root->st.st_mode) && (toys.optflags & FLAG_R)
158 && dirtree_notdotdot(root)) {
159 xprintf("\n%s:\n", fpath);
160 free(fpath);
161 return (DIRTREE_RECURSE | DIRTREE_COMEAGAIN);
162 }
163 }
164 free(fpath);
165 return 0;
166 }
167
lsattr_main(void)168 void lsattr_main(void)
169 {
170 if (!*toys.optargs) dirtree_read(".", retell_dir);
171 else
172 for (; *toys.optargs; toys.optargs++) {
173 struct stat sb;
174
175 if (lstat(*toys.optargs, &sb)) perror_msg("stat '%s'", *toys.optargs);
176 else if (S_ISDIR(sb.st_mode) && !(toys.optflags & FLAG_d))
177 dirtree_read(*toys.optargs, retell_dir);
178 else print_file_attr(*toys.optargs);// to handle "./Filename" or "./Dir"
179 }
180 }
181
182 // Switch gears from lsattr to chattr.
183 #define CLEANUP_lsattr
184 #define FOR_chattr
185 #include "generated/flags.h"
186
187 static struct _chattr {
188 unsigned long add, rm, set, version;
189 unsigned char vflag, recursive;
190 } chattr;
191
192 // Set file flags on a Linux second extended file system.
ext2_setflag(int fd,struct stat * sb,unsigned long flag)193 static inline int ext2_setflag(int fd, struct stat *sb, unsigned long flag)
194 {
195 if (!S_ISREG(sb->st_mode) && !S_ISDIR(sb->st_mode)) {
196 errno = EOPNOTSUPP;
197 return -1;
198 }
199 return (ioctl(fd, FS_IOC_SETFLAGS, (void*)&flag));
200 }
201
get_flag_val(char ch)202 static unsigned long get_flag_val(char ch)
203 {
204 struct ext2_attr *ptr = e2attrs;
205
206 for (; ptr->name; ptr++)
207 if (ptr->opt == ch) return ptr->flag;
208 help_exit("bad '%c'", ch);
209 }
210
211 // Parse command line argument and fill the chattr structure.
parse_cmdline_arg(char *** argv)212 static void parse_cmdline_arg(char ***argv)
213 {
214 char *arg = **argv, *ptr = NULL;
215
216 while (arg) {
217 switch (arg[0]) {
218 case '-':
219 for (ptr = ++arg; *ptr; ptr++) {
220 if (*ptr == 'R') {
221 chattr.recursive = 1;
222 continue;
223 } else if (*ptr == 'v') {// get version from next argv.
224 char *endptr;
225
226 errno = 0;
227 arg = *(*argv += 1);
228 if (!arg) help_exit("bad -v");
229 if (*arg == '-') perror_exit("Invalid Number '%s'", arg);
230 chattr.version = strtoul(arg, &endptr, 0);
231 if (errno || *endptr) perror_exit("bad version '%s'", arg);
232 chattr.vflag = 1;
233 continue;
234 } else chattr.rm |= get_flag_val(*ptr);
235 }
236 break;
237 case '+':
238 for (ptr = ++arg; *ptr; ptr++)
239 chattr.add |= get_flag_val(*ptr);
240 break;
241 case '=':
242 for (ptr = ++arg; *ptr; ptr++)
243 chattr.set |= get_flag_val(*ptr);
244 break;
245 default: return;
246 }
247 arg = *(*argv += 1);
248 }
249 }
250
251 // Update attribute of given file.
update_attr(struct dirtree * root)252 static int update_attr(struct dirtree *root)
253 {
254 unsigned long fval = 0;
255 char *fpath = NULL;
256 int fd;
257
258 if (!dirtree_notdotdot(root)) return 0;
259
260 /*
261 * if file is a link and recursive is set or file is not regular+link+dir
262 * (like fifo or dev file) then escape the file.
263 */
264 if ((S_ISLNK(root->st.st_mode) && chattr.recursive)
265 || (!S_ISREG(root->st.st_mode) && !S_ISLNK(root->st.st_mode)
266 && !S_ISDIR(root->st.st_mode)))
267 return 0;
268
269 fpath = dirtree_path(root, NULL);
270 if (-1 == (fd=open(fpath, O_RDONLY | O_NONBLOCK))) {
271 free(fpath);
272 return DIRTREE_ABORT;
273 }
274 // Get current attr of file.
275 if (ext2_getflag(fd, &(root->st), &fval) < 0) {
276 perror_msg("read flags of '%s'", fpath);
277 free(fpath);
278 xclose(fd);
279 return DIRTREE_ABORT;
280 }
281 if (chattr.set) { // for '=' operator.
282 if (ext2_setflag(fd, &(root->st), chattr.set) < 0)
283 perror_msg("setting flags '%s'", fpath);
284 } else { // for '-' / '+' operator.
285 fval &= ~(chattr.rm);
286 fval |= chattr.add;
287 if (!S_ISDIR(root->st.st_mode)) fval &= ~FS_DIRSYNC_FL;
288 if (ext2_setflag(fd, &(root->st), fval) < 0)
289 perror_msg("setting flags '%s'", fpath);
290 }
291 if (chattr.vflag) { // set file version
292 if (ioctl(fd, FS_IOC_SETVERSION, (void*)&chattr.version) < 0)
293 perror_msg("while setting version on '%s'", fpath);
294 }
295 free(fpath);
296 xclose(fd);
297
298 if (S_ISDIR(root->st.st_mode) && chattr.recursive) return DIRTREE_RECURSE;
299 return 0;
300 }
301
chattr_main(void)302 void chattr_main(void)
303 {
304 char **argv = toys.optargs;
305
306 memset(&chattr, 0, sizeof(struct _chattr));
307 parse_cmdline_arg(&argv);
308 if (!*argv) help_exit("no file");
309 if (chattr.set && (chattr.add || chattr.rm))
310 error_exit("no '=' with '-' or '+'");
311 if (chattr.rm & chattr.add) error_exit("set/unset same flag");
312 if (!(chattr.add || chattr.rm || chattr.set || chattr.vflag))
313 error_exit("need '-v', '=', '-' or '+'");
314 for (; *argv; argv++) dirtree_read(*argv, update_attr);
315 toys.exitval = 0; //always set success at this point.
316 }
317