| /* lsattr.c - List file attributes on a Linux second extended file system. |
| * |
| * Copyright 2013 Ranjan Kumar <ranjankumar.bth@gmail.com> |
| * Copyright 2013 Kyungwan Han <asura321@gmail.com> |
| * |
| * No Standard. |
| |
| USE_LSATTR(NEWTOY(lsattr, "vldaR", TOYFLAG_BIN)) |
| USE_CHATTR(NEWTOY(chattr, NULL, TOYFLAG_BIN)) |
| |
| config LSATTR |
| bool "lsattr" |
| default y |
| help |
| usage: lsattr [-Radlv] [Files...] |
| |
| List file attributes on a Linux second extended file system. |
| |
| -R Recursively list attributes of directories and their contents. |
| -a List all files in directories, including files that start with '.'. |
| -d List directories like other files, rather than listing their contents. |
| -l List long flag names. |
| -v List the file's version/generation number. |
| |
| config CHATTR |
| bool "chattr" |
| default y |
| help |
| usage: chattr [-R] [-+=AacDdijsStTu] [-v version] [File...] |
| |
| Change file attributes on a Linux second extended file system. |
| |
| Operators: |
| '-' Remove attributes. |
| '+' Add attributes. |
| '=' Set attributes. |
| |
| Attributes: |
| A Don't track atime. |
| a Append mode only. |
| c Enable compress. |
| D Write dir contents synchronously. |
| d Don't backup with dump. |
| i Cannot be modified (immutable). |
| j Write all data to journal first. |
| s Zero disk storage when deleted. |
| S Write file contents synchronously. |
| t Disable tail-merging of partial blocks with other files. |
| u Allow file to be undeleted. |
| -R Recurse. |
| -v Set the file's version/generation number. |
| |
| */ |
| #define FOR_lsattr |
| #include "toys.h" |
| #include <linux/fs.h> |
| |
| static struct ext2_attr { |
| char *name; |
| unsigned long flag; |
| char opt; |
| } e2attrs[] = { |
| {"Secure_Deletion", FS_SECRM_FL, 's'}, // Secure deletion |
| {"Undelete", FS_UNRM_FL, 'u'}, // Undelete |
| {"Compression_Requested", FS_COMPR_FL, 'c'}, // Compress file |
| {"Synchronous_Updates", FS_SYNC_FL, 'S'}, // Synchronous updates |
| {"Immutable", FS_IMMUTABLE_FL, 'i'}, // Immutable file |
| {"Append_Only", FS_APPEND_FL, 'a'}, // writes to file may only append |
| {"No_Dump", FS_NODUMP_FL, 'd'}, // do not dump file |
| {"No_Atime", FS_NOATIME_FL, 'A'}, // do not update atime |
| {"Indexed_directory", FS_INDEX_FL, 'I'}, // hash-indexed directory |
| {"Journaled_Data", FS_JOURNAL_DATA_FL, 'j'}, // file data should be journaled |
| {"No_Tailmerging", FS_NOTAIL_FL, 't'}, // file tail should not be merged |
| {"Synchronous_Directory_Updates", FS_DIRSYNC_FL, 'D'}, // dirsync behaviour (directories only) |
| {"Top_of_Directory_Hierarchies", FS_TOPDIR_FL, 'T'}, // Top of directory hierarchies |
| {NULL, -1, 0}, |
| }; |
| |
| // Get file flags on a Linux second extended file system. |
| static int ext2_getflag(int fd, struct stat *sb, unsigned long *flag) |
| { |
| if(!S_ISREG(sb->st_mode) && !S_ISDIR(sb->st_mode)) { |
| errno = EOPNOTSUPP; |
| return -1; |
| } |
| return (ioctl(fd, FS_IOC_GETFLAGS, (void*)flag)); |
| } |
| |
| static void print_file_attr(char *path) |
| { |
| unsigned long flag = 0, version = 0; |
| int fd; |
| struct stat sb; |
| |
| if (!stat(path, &sb) && !S_ISREG(sb.st_mode) && !S_ISDIR(sb.st_mode)) { |
| errno = EOPNOTSUPP; |
| goto LABEL1; |
| } |
| if (-1 == (fd=open(path, O_RDONLY | O_NONBLOCK))) goto LABEL1; |
| |
| if (toys.optflags & FLAG_v) { |
| if (ioctl(fd, FS_IOC_GETVERSION, (void*)&version) < 0) goto LABEL2; |
| xprintf("%5lu ", version); |
| } |
| |
| if (ext2_getflag(fd, &sb, &flag) < 0) perror_msg("reading flags '%s'", path); |
| else { |
| struct ext2_attr *ptr = e2attrs; |
| |
| if (toys.optflags & FLAG_l) { |
| int name_found = 0; |
| |
| xprintf("%-50s ", path); |
| for (; ptr->name; ptr++) { |
| if (flag & ptr->flag) { |
| if (name_found) xprintf(", "); //for formatting. |
| xprintf("%s", ptr->name); |
| name_found = 1; |
| } |
| } |
| if (!name_found) xprintf("---"); |
| xputc('\n'); |
| } else { |
| int index = 0; |
| |
| for (; ptr->name; ptr++) |
| toybuf[index++] = (flag & ptr->flag) ? ptr->opt : '-'; |
| toybuf[index] = '\0'; |
| xprintf("%s %s\n", toybuf, path); |
| } |
| } |
| xclose(fd); |
| return; |
| LABEL2: xclose(fd); |
| LABEL1: perror_msg("reading '%s'", path); |
| } |
| |
| // Get directory information. |
| static int retell_dir(struct dirtree *root) |
| { |
| char *fpath = NULL; |
| |
| if (root->data == -1) { |
| xputc('\n'); |
| return 0; |
| } |
| if (S_ISDIR(root->st.st_mode) && !root->parent) |
| return (DIRTREE_RECURSE | DIRTREE_COMEAGAIN); |
| |
| fpath = dirtree_path(root, NULL); |
| //Special case: with '-a' option and '.'/'..' also included in printing list. |
| if ((root->name[0] != '.') || (toys.optflags & FLAG_a)) { |
| print_file_attr(fpath); |
| if (S_ISDIR(root->st.st_mode) && (toys.optflags & FLAG_R) |
| && dirtree_notdotdot(root)) { |
| xprintf("\n%s:\n", fpath); |
| free(fpath); |
| return (DIRTREE_RECURSE | DIRTREE_COMEAGAIN); |
| } |
| } |
| free(fpath); |
| return 0; |
| } |
| |
| void lsattr_main(void) |
| { |
| if (!*toys.optargs) dirtree_read(".", retell_dir); |
| else |
| for (; *toys.optargs; toys.optargs++) { |
| struct stat sb; |
| |
| if (lstat(*toys.optargs, &sb)) perror_msg("stat '%s'", *toys.optargs); |
| else if (S_ISDIR(sb.st_mode) && !(toys.optflags & FLAG_d)) |
| dirtree_read(*toys.optargs, retell_dir); |
| else print_file_attr(*toys.optargs);// to handle "./Filename" or "./Dir" |
| } |
| } |
| |
| // Switch gears from lsattr to chattr. |
| #define CLEANUP_lsattr |
| #define FOR_chattr |
| #include "generated/flags.h" |
| |
| static struct _chattr { |
| unsigned long add, rm, set, version; |
| unsigned char vflag, recursive; |
| } chattr; |
| |
| static inline void chattr_help(void) |
| { |
| toys.exithelp++; |
| error_exit("Invalid Argument"); |
| } |
| |
| // Set file flags on a Linux second extended file system. |
| static inline int ext2_setflag(int fd, struct stat *sb, unsigned long flag) |
| { |
| if (!S_ISREG(sb->st_mode) && !S_ISDIR(sb->st_mode)) { |
| errno = EOPNOTSUPP; |
| return -1; |
| } |
| return (ioctl(fd, FS_IOC_SETFLAGS, (void*)&flag)); |
| } |
| |
| static unsigned long get_flag_val(char ch) |
| { |
| struct ext2_attr *ptr = e2attrs; |
| |
| for (; ptr->name; ptr++) |
| if (ptr->opt == ch) return ptr->flag; |
| chattr_help(); // if no match found then Show help |
| return 0; // silent warning. |
| } |
| |
| // Parse command line argument and fill the chattr structure. |
| static void parse_cmdline_arg(char ***argv) |
| { |
| char *arg = **argv, *ptr = NULL; |
| |
| while (arg) { |
| switch (arg[0]) { |
| case '-': |
| for (ptr = ++arg; *ptr; ptr++) { |
| if (*ptr == 'R') { |
| chattr.recursive = 1; |
| continue; |
| } else if (*ptr == 'v') {// get version from next argv. |
| char *endptr; |
| |
| errno = 0; |
| arg = *(*argv += 1); |
| if (!arg) chattr_help(); |
| if (*arg == '-') perror_exit("Invalid Number '%s'", arg); |
| chattr.version = strtoul(arg, &endptr, 0); |
| if (errno || *endptr) perror_exit("bad version '%s'", arg); |
| chattr.vflag = 1; |
| continue; |
| } else chattr.rm |= get_flag_val(*ptr); |
| } |
| break; |
| case '+': |
| for (ptr = ++arg; *ptr; ptr++) |
| chattr.add |= get_flag_val(*ptr); |
| break; |
| case '=': |
| for (ptr = ++arg; *ptr; ptr++) |
| chattr.set |= get_flag_val(*ptr); |
| break; |
| default: return; |
| } |
| arg = *(*argv += 1); |
| } |
| } |
| |
| // Update attribute of given file. |
| static int update_attr(struct dirtree *root) |
| { |
| unsigned long fval = 0; |
| char *fpath = NULL; |
| int fd; |
| |
| if (!dirtree_notdotdot(root)) return 0; |
| |
| /* |
| * if file is a link and recursive is set or file is not regular+link+dir |
| * (like fifo or dev file) then escape the file. |
| */ |
| if ((S_ISLNK(root->st.st_mode) && chattr.recursive) |
| || (!S_ISREG(root->st.st_mode) && !S_ISLNK(root->st.st_mode) |
| && !S_ISDIR(root->st.st_mode))) |
| return 0; |
| |
| fpath = dirtree_path(root, NULL); |
| if (-1 == (fd=open(fpath, O_RDONLY | O_NONBLOCK))) { |
| free(fpath); |
| return DIRTREE_ABORT; |
| } |
| // Get current attr of file. |
| if (ext2_getflag(fd, &(root->st), &fval) < 0) { |
| perror_msg("read flags of '%s'", fpath); |
| free(fpath); |
| xclose(fd); |
| return DIRTREE_ABORT; |
| } |
| if (chattr.set) { // for '=' operator. |
| if (ext2_setflag(fd, &(root->st), chattr.set) < 0) |
| perror_msg("setting flags '%s'", fpath); |
| } else { // for '-' / '+' operator. |
| fval &= ~(chattr.rm); |
| fval |= chattr.add; |
| if (!S_ISDIR(root->st.st_mode)) fval &= ~FS_DIRSYNC_FL; |
| if (ext2_setflag(fd, &(root->st), fval) < 0) |
| perror_msg("setting flags '%s'", fpath); |
| } |
| if (chattr.vflag) { // set file version |
| if (ioctl(fd, FS_IOC_SETVERSION, (void*)&chattr.version) < 0) |
| perror_msg("while setting version on '%s'", fpath); |
| } |
| free(fpath); |
| xclose(fd); |
| |
| if (S_ISDIR(root->st.st_mode) && chattr.recursive) return DIRTREE_RECURSE; |
| return 0; |
| } |
| |
| void chattr_main(void) |
| { |
| char **argv = toys.optargs; |
| |
| memset(&chattr, 0, sizeof(struct _chattr)); |
| parse_cmdline_arg(&argv); |
| if (!*argv) chattr_help(); |
| if (chattr.set && (chattr.add || chattr.rm)) |
| error_exit("'=' is incompatible with '-' and '+'"); |
| if (chattr.rm & chattr.add) error_exit("Can't set and unset same flag."); |
| if (!(chattr.add || chattr.rm || chattr.set || chattr.vflag)) |
| error_exit(("Must use '-v', '=', '-' or '+'")); |
| for (; *argv; argv++) dirtree_read(*argv, update_attr); |
| toys.exitval = 0; //always set success at this point. |
| } |