/* vi: set sw=4 ts=4: */ /* * lsattr.c - List file attributes on an ext2 file system * * Copyright (C) 1993, 1994 Remy Card * Laboratoire MASI, Institut Blaise Pascal * Universite Pierre et Marie Curie (Paris VI) * * This file can be redistributed under the terms of the GNU General * Public License */ //config:config LSATTR //config: bool "lsattr (5.5 kb)" //config: default y //config: help //config: lsattr lists the file attributes on a second extended file system. //applet:IF_LSATTR(APPLET_NOEXEC(lsattr, lsattr, BB_DIR_BIN, BB_SUID_DROP, lsattr)) /* ls is NOEXEC, so we should be too! ;) */ //kbuild:lib-$(CONFIG_LSATTR) += lsattr.o e2fs_lib.o //usage:#define lsattr_trivial_usage //usage: "[-Radlpv] [FILE]..." //usage:#define lsattr_full_usage "\n\n" //usage: "List ext2 file attributes\n" //usage: "\n -R Recurse" //usage: "\n -a Include names starting with ." //usage: "\n -d List directory names, not contents" // -a,-d text should match ls --help //usage: "\n -l List long flag names" //usage: "\n -p List project ID" //usage: "\n -v List version/generation number" #include "libbb.h" #include "e2fs_lib.h" enum { OPT_RECUR = 1 << 0, OPT_ALL = 1 << 1, OPT_DIRS_OPT = 1 << 2, OPT_PF_LONG = 1 << 3, OPT_GENERATION = 1 << 4, OPT_PROJID = 1 << 5, }; static void list_attributes(const char *name) { unsigned fsflags; int fd, r; /* There is no way to run needed ioctls on a symlink. * open(O_PATH | O_NOFOLLOW) _can_ be used to get a fd referring to the symlink, * but ioctls fail on such a fd (tried on 4.12.0 kernel). * e2fsprogs-1.46.2 uses open(O_NOFOLLOW), it fails on symlinks. */ fd = open_or_warn(name, O_RDONLY | O_NONBLOCK | O_NOCTTY | O_NOFOLLOW); if (fd < 0) return; if (option_mask32 & OPT_PROJID) { struct ext2_fsxattr fsxattr; r = ioctl(fd, EXT2_IOC_FSGETXATTR, &fsxattr); /* note: ^^^ may fail in 32-bit userspace on 64-bit kernel (seen on 4.12.0) */ if (r != 0) goto read_err; printf("%5u ", (unsigned)fsxattr.fsx_projid); } if (option_mask32 & OPT_GENERATION) { unsigned generation; r = ioctl(fd, EXT2_IOC_GETVERSION, &generation); if (r != 0) goto read_err; printf("%-10u ", generation); } r = ioctl(fd, EXT2_IOC_GETFLAGS, &fsflags); if (r != 0) goto read_err; close(fd); if (option_mask32 & OPT_PF_LONG) { printf("%-28s ", name); print_e2flags_long(fsflags); bb_putchar('\n'); } else { print_e2flags(fsflags); printf(" %s\n", name); } return; read_err: bb_perror_msg("reading %s", name); close(fd); } static int FAST_FUNC lsattr_dir_proc(const char *dir_name, struct dirent *de, void *private UNUSED_PARAM) { struct stat st; char *path; path = concat_path_file(dir_name, de->d_name); if (lstat(path, &st) != 0) bb_perror_msg("can't stat '%s'", path); else if (de->d_name[0] != '.' || (option_mask32 & OPT_ALL)) { /* Don't try to open device files, fifos etc */ if (S_ISREG(st.st_mode) || S_ISLNK(st.st_mode) || S_ISDIR(st.st_mode)) list_attributes(path); if (S_ISDIR(st.st_mode) && (option_mask32 & OPT_RECUR) && !DOT_OR_DOTDOT(de->d_name) ) { printf("\n%s:\n", path); iterate_on_dir(path, lsattr_dir_proc, NULL); bb_putchar('\n'); } } free(path); return 0; } static void lsattr_args(const char *name) { struct stat st; if (lstat(name, &st) == -1) { bb_perror_msg("can't stat '%s'", name); } else if (S_ISDIR(st.st_mode) && !(option_mask32 & OPT_DIRS_OPT)) { iterate_on_dir(name, lsattr_dir_proc, NULL); } else { list_attributes(name); } } int lsattr_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; int lsattr_main(int argc UNUSED_PARAM, char **argv) { getopt32(argv, "Radlvp"); argv += optind; if (!*argv) *--argv = (char*)"."; do lsattr_args(*argv++); while (*argv); return EXIT_SUCCESS; }