| /* vi: set sw=4 ts=4: */ |
| /* |
| * Display or change file attributes on a fat file system |
| * |
| * Copyright 2005 H. Peter Anvin |
| * Busybox'ed (2014) by Pascal Bellard <pascal.bellard@ads-lu.com> |
| * |
| * This file can be redistributed under the terms of the GNU General |
| * Public License |
| */ |
| //config:config FATATTR |
| //config: bool "fatattr" |
| //config: default y |
| //config: select PLATFORM_LINUX |
| //config: help |
| //config: fatattr lists or changes the file attributes on a fat file system. |
| |
| //applet:IF_FATATTR(APPLET(fatattr, BB_DIR_BIN, BB_SUID_DROP)) |
| //kbuild:lib-$(CONFIG_FATATTR) += fatattr.o |
| |
| //usage:#define fatattr_trivial_usage |
| //usage: "[-+rhsvda] FILE..." |
| //usage:#define fatattr_full_usage "\n\n" |
| //usage: "Change file attributes on FAT filesystem\n" |
| //usage: "\n - Clear attributes" |
| //usage: "\n + Set attributes" |
| //usage: "\n r Read only" |
| //usage: "\n h Hidden" |
| //usage: "\n s System" |
| //usage: "\n v Volume label" |
| //usage: "\n d Directory" |
| //usage: "\n a Archive" |
| |
| #include "libbb.h" |
| /* linux/msdos_fs.h says: */ |
| #ifndef FAT_IOCTL_GET_ATTRIBUTES |
| # define FAT_IOCTL_GET_ATTRIBUTES _IOR('r', 0x10, uint32_t) |
| # define FAT_IOCTL_SET_ATTRIBUTES _IOW('r', 0x11, uint32_t) |
| #endif |
| |
| /* Currently supports only the FAT flags, not the NTFS ones. |
| * Extra space at the end is a hack to print space separator in file listing. |
| * Let's hope no one ever passes space as an option char :) |
| */ |
| static const char bit_to_char[] ALIGN1 = "rhsvda67 "; |
| |
| static inline unsigned long get_flag(char c) |
| { |
| const char *fp = strchr(bit_to_char, c); |
| if (!fp) |
| bb_error_msg_and_die("invalid character '%c'", c); |
| return 1 << (fp - bit_to_char); |
| } |
| |
| static unsigned decode_arg(const char *arg) |
| { |
| unsigned fl = 0; |
| while (*++arg) |
| fl |= get_flag(*arg); |
| return fl; |
| } |
| |
| int fatattr_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; |
| int fatattr_main(int argc UNUSED_PARAM, char **argv) |
| { |
| unsigned set_mask = 0; |
| unsigned clear_mask = 0; |
| |
| for (;;) { |
| unsigned fl; |
| char *arg = *++argv; |
| |
| if (!arg) |
| bb_show_usage(); |
| if (arg[0] != '-' && arg[0] != '+') |
| break; |
| fl = decode_arg(arg); |
| if (arg[0] == '+') |
| set_mask |= fl; |
| else |
| clear_mask |= fl; |
| } |
| |
| do { |
| int fd, i; |
| uint32_t attr; |
| |
| fd = xopen(*argv, O_RDONLY); |
| xioctl(fd, FAT_IOCTL_GET_ATTRIBUTES, &attr); |
| attr = (attr | set_mask) & ~clear_mask; |
| if (set_mask | clear_mask) |
| xioctl(fd, FAT_IOCTL_SET_ATTRIBUTES, &attr); |
| else { |
| for (i = 0; bit_to_char[i]; i++) { |
| bb_putchar((attr & 1) ? bit_to_char[i] : ' '); |
| attr >>= 1; |
| } |
| puts(*argv); |
| } |
| close(fd); |
| } while (*++argv); |
| |
| return EXIT_SUCCESS; |
| } |