lsattr,chattr: support -p
function old new delta fgetsetprojid - 107 +107 list_attributes 169 222 +53 change_attributes 277 326 +49 chattr_main 272 307 +35 close_silently - 22 +22 .rodata 103378 103393 +15 packed_usage 33658 33666 +8 fgetsetversion 88 74 -14 fgetsetflags 162 148 -14 ------------------------------------------------------------------------------ (add/remove: 2/0 grow/shrink: 5/2 up/down: 289/-28) Total: 261 bytes Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
parent
9c291f2cc0
commit
526b834790
@ -20,12 +20,12 @@
|
|||||||
//kbuild:lib-$(CONFIG_CHATTR) += chattr.o e2fs_lib.o
|
//kbuild:lib-$(CONFIG_CHATTR) += chattr.o e2fs_lib.o
|
||||||
|
|
||||||
//usage:#define chattr_trivial_usage
|
//usage:#define chattr_trivial_usage
|
||||||
//usage: "[-R] [-v VERSION] [-+=AacDdijsStTu] FILE..."
|
//usage: "[-R] [-v VERSION] [-p PROJID] [-+=AacDdijsStTu] FILE..."
|
||||||
//usage:#define chattr_full_usage "\n\n"
|
//usage:#define chattr_full_usage "\n\n"
|
||||||
//usage: "Change ext2 file attributes\n"
|
//usage: "Change ext2 file attributes\n"
|
||||||
//usage: "\n -R Recurse"
|
//usage: "\n -R Recurse"
|
||||||
//TODD? "\n -p NUM Set project number"
|
|
||||||
//usage: "\n -v NUM Set version/generation number"
|
//usage: "\n -v NUM Set version/generation number"
|
||||||
|
//usage: "\n -p NUM Set project number"
|
||||||
//-V, -f accepted but ignored
|
//-V, -f accepted but ignored
|
||||||
//usage: "\nModifiers:"
|
//usage: "\nModifiers:"
|
||||||
//usage: "\n -,+,= Remove/add/set attributes"
|
//usage: "\n -,+,= Remove/add/set attributes"
|
||||||
@ -45,16 +45,18 @@
|
|||||||
#include "libbb.h"
|
#include "libbb.h"
|
||||||
#include "e2fs_lib.h"
|
#include "e2fs_lib.h"
|
||||||
|
|
||||||
#define OPT_ADD 1
|
#define OPT_ADD (1 << 0)
|
||||||
#define OPT_REM 2
|
#define OPT_REM (1 << 1)
|
||||||
#define OPT_SET 4
|
#define OPT_SET (1 << 2)
|
||||||
#define OPT_SET_VER 8
|
#define OPT_SET_VER (1 << 3)
|
||||||
|
#define OPT_SET_PROJ (1 << 4)
|
||||||
|
|
||||||
struct globals {
|
struct globals {
|
||||||
unsigned long version;
|
unsigned long version;
|
||||||
unsigned long af;
|
unsigned long af;
|
||||||
unsigned long rf;
|
unsigned long rf;
|
||||||
int flags;
|
int flags;
|
||||||
|
uint32_t projid;
|
||||||
smallint recursive;
|
smallint recursive;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -108,7 +110,13 @@ static char** decode_arg(char **argv, struct globals *gp)
|
|||||||
gp->flags |= OPT_SET_VER;
|
gp->flags |= OPT_SET_VER;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
//TODO: "-p PROJECT_NUM" ?
|
if (*arg == 'p') {
|
||||||
|
if (!*++argv)
|
||||||
|
bb_show_usage();
|
||||||
|
gp->projid = xatou32(*argv);
|
||||||
|
gp->flags |= OPT_SET_PROJ;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
/* not a known option, try as an attribute */
|
/* not a known option, try as an attribute */
|
||||||
}
|
}
|
||||||
*fl |= get_flag(*arg);
|
*fl |= get_flag(*arg);
|
||||||
@ -151,7 +159,11 @@ static void change_attributes(const char *name, struct globals *gp)
|
|||||||
|
|
||||||
if (gp->flags & OPT_SET_VER)
|
if (gp->flags & OPT_SET_VER)
|
||||||
if (fsetversion(name, gp->version) != 0)
|
if (fsetversion(name, gp->version) != 0)
|
||||||
bb_perror_msg("setting version on %s", name);
|
bb_perror_msg("setting %s on %s", "version", name);
|
||||||
|
|
||||||
|
if (gp->flags & OPT_SET_PROJ)
|
||||||
|
if (fsetprojid(name, gp->projid) != 0)
|
||||||
|
bb_perror_msg("setting %s on %s", "project ID", name);
|
||||||
|
|
||||||
if (gp->flags & OPT_SET) {
|
if (gp->flags & OPT_SET) {
|
||||||
fsflags = gp->af;
|
fsflags = gp->af;
|
||||||
|
@ -51,14 +51,14 @@ int fgetsetversion(const char *name, unsigned long *get_version, unsigned long s
|
|||||||
{
|
{
|
||||||
#if HAVE_EXT2_IOCTLS
|
#if HAVE_EXT2_IOCTLS
|
||||||
int fd, r;
|
int fd, r;
|
||||||
IF_LONG_IS_WIDER(int ver;)
|
IF_LONG_IS_WIDER(unsigned ver;)
|
||||||
|
|
||||||
fd = open(name, O_RDONLY | O_NONBLOCK);
|
fd = open(name, O_RDONLY | O_NONBLOCK);
|
||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
return -1;
|
return -1;
|
||||||
if (!get_version) {
|
if (!get_version) {
|
||||||
IF_LONG_IS_WIDER(
|
IF_LONG_IS_WIDER(
|
||||||
ver = (int) set_version;
|
ver = (unsigned) set_version;
|
||||||
r = ioctl(fd, EXT2_IOC_SETVERSION, &ver);
|
r = ioctl(fd, EXT2_IOC_SETVERSION, &ver);
|
||||||
)
|
)
|
||||||
IF_LONG_IS_SAME(
|
IF_LONG_IS_SAME(
|
||||||
@ -81,6 +81,32 @@ int fgetsetversion(const char *name, unsigned long *get_version, unsigned long s
|
|||||||
#endif /* ! HAVE_EXT2_IOCTLS */
|
#endif /* ! HAVE_EXT2_IOCTLS */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int fgetsetprojid(const char *name, uint32_t *get, uint32_t set)
|
||||||
|
{
|
||||||
|
#if HAVE_EXT2_IOCTLS
|
||||||
|
struct ext2_fsxattr fsxattr;
|
||||||
|
int fd, r;
|
||||||
|
|
||||||
|
fd = open(name, O_RDONLY | O_NONBLOCK);
|
||||||
|
if (fd == -1)
|
||||||
|
return -1;
|
||||||
|
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) {
|
||||||
|
if (get) {
|
||||||
|
*get = fsxattr.fsx_projid;
|
||||||
|
} else {
|
||||||
|
fsxattr.fsx_projid = set;
|
||||||
|
r = ioctl(fd, EXT2_IOC_FSSETXATTR, &fsxattr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
close_silently(fd);
|
||||||
|
return r;
|
||||||
|
#else /* ! HAVE_EXT2_IOCTLS */
|
||||||
|
errno = EOPNOTSUPP;
|
||||||
|
return -1;
|
||||||
|
#endif /* ! HAVE_EXT2_IOCTLS */
|
||||||
|
}
|
||||||
|
|
||||||
/* Get/set a file flags on an ext2 file system */
|
/* Get/set a file flags on an ext2 file system */
|
||||||
int fgetsetflags(const char *name, unsigned long *get_flags, unsigned long set_flags)
|
int fgetsetflags(const char *name, unsigned long *get_flags, unsigned long set_flags)
|
||||||
@ -88,7 +114,7 @@ int fgetsetflags(const char *name, unsigned long *get_flags, unsigned long set_f
|
|||||||
#if HAVE_EXT2_IOCTLS
|
#if HAVE_EXT2_IOCTLS
|
||||||
struct stat buf;
|
struct stat buf;
|
||||||
int fd, r;
|
int fd, r;
|
||||||
IF_LONG_IS_WIDER(int f;)
|
IF_LONG_IS_WIDER(unsigned f;)
|
||||||
|
|
||||||
if (stat(name, &buf) == 0 /* stat is ok */
|
if (stat(name, &buf) == 0 /* stat is ok */
|
||||||
&& !S_ISREG(buf.st_mode) && !S_ISDIR(buf.st_mode)
|
&& !S_ISREG(buf.st_mode) && !S_ISDIR(buf.st_mode)
|
||||||
@ -101,7 +127,7 @@ int fgetsetflags(const char *name, unsigned long *get_flags, unsigned long set_f
|
|||||||
|
|
||||||
if (!get_flags) {
|
if (!get_flags) {
|
||||||
IF_LONG_IS_WIDER(
|
IF_LONG_IS_WIDER(
|
||||||
f = (int) set_flags;
|
f = (unsigned) set_flags;
|
||||||
r = ioctl(fd, EXT2_IOC_SETFLAGS, &f);
|
r = ioctl(fd, EXT2_IOC_SETFLAGS, &f);
|
||||||
)
|
)
|
||||||
IF_LONG_IS_SAME(
|
IF_LONG_IS_SAME(
|
||||||
|
@ -21,6 +21,11 @@ int fgetsetversion(const char *name, unsigned long *get_version, unsigned long s
|
|||||||
#define fgetversion(name, version) fgetsetversion(name, version, 0)
|
#define fgetversion(name, version) fgetsetversion(name, version, 0)
|
||||||
#define fsetversion(name, version) fgetsetversion(name, NULL, version)
|
#define fsetversion(name, version) fgetsetversion(name, NULL, version)
|
||||||
|
|
||||||
|
/* Get/set a file project ID on an ext2 file system */
|
||||||
|
int fgetsetprojid(const char *name, uint32_t *get, uint32_t set);
|
||||||
|
#define fgetprojid(name, projid) fgetsetprojid(name, projid, 0)
|
||||||
|
#define fsetprojid(name, projid) fgetsetprojid(name, NULL, projid)
|
||||||
|
|
||||||
/* Get/set a file flags on an ext2 file system */
|
/* Get/set a file flags on an ext2 file system */
|
||||||
int fgetsetflags(const char *name, unsigned long *get_flags, unsigned long set_flags);
|
int fgetsetflags(const char *name, unsigned long *get_flags, unsigned long set_flags);
|
||||||
#define fgetflags(name, flags) fgetsetflags(name, flags, 0)
|
#define fgetflags(name, flags) fgetsetflags(name, flags, 0)
|
||||||
|
@ -21,38 +21,48 @@
|
|||||||
//kbuild:lib-$(CONFIG_LSATTR) += lsattr.o e2fs_lib.o
|
//kbuild:lib-$(CONFIG_LSATTR) += lsattr.o e2fs_lib.o
|
||||||
|
|
||||||
//usage:#define lsattr_trivial_usage
|
//usage:#define lsattr_trivial_usage
|
||||||
//usage: "[-Radlv] [FILE]..."
|
//usage: "[-Radlpv] [FILE]..."
|
||||||
//usage:#define lsattr_full_usage "\n\n"
|
//usage:#define lsattr_full_usage "\n\n"
|
||||||
//usage: "List ext2 file attributes\n"
|
//usage: "List ext2 file attributes\n"
|
||||||
//usage: "\n -R Recurse"
|
//usage: "\n -R Recurse"
|
||||||
//usage: "\n -a Don't hide entries starting with ."
|
//usage: "\n -a Include names starting with ."
|
||||||
//usage: "\n -d List directory entries instead of contents"
|
//usage: "\n -d List directory names, not contents"
|
||||||
|
// -a,-d text should match ls --help
|
||||||
//usage: "\n -l List long flag names"
|
//usage: "\n -l List long flag names"
|
||||||
|
//usage: "\n -p List project ID"
|
||||||
//usage: "\n -v List version/generation number"
|
//usage: "\n -v List version/generation number"
|
||||||
|
|
||||||
#include "libbb.h"
|
#include "libbb.h"
|
||||||
#include "e2fs_lib.h"
|
#include "e2fs_lib.h"
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
OPT_RECUR = 0x1,
|
OPT_RECUR = 1 << 0,
|
||||||
OPT_ALL = 0x2,
|
OPT_ALL = 1 << 1,
|
||||||
OPT_DIRS_OPT = 0x4,
|
OPT_DIRS_OPT = 1 << 2,
|
||||||
OPT_PF_LONG = 0x8,
|
OPT_PF_LONG = 1 << 3,
|
||||||
OPT_GENERATION = 0x10,
|
OPT_GENERATION = 1 << 4,
|
||||||
|
OPT_PROJID = 1 << 5,
|
||||||
};
|
};
|
||||||
|
|
||||||
static void list_attributes(const char *name)
|
static void list_attributes(const char *name)
|
||||||
{
|
{
|
||||||
unsigned long fsflags;
|
unsigned long fsflags;
|
||||||
unsigned long generation;
|
|
||||||
|
|
||||||
if (fgetflags(name, &fsflags) != 0)
|
if (fgetflags(name, &fsflags) != 0)
|
||||||
goto read_err;
|
goto read_err;
|
||||||
|
|
||||||
|
if (option_mask32 & OPT_PROJID) {
|
||||||
|
uint32_t p;
|
||||||
|
if (fgetprojid(name, &p) != 0)
|
||||||
|
goto read_err;
|
||||||
|
printf("%5lu ", (unsigned long)p);
|
||||||
|
}
|
||||||
|
|
||||||
if (option_mask32 & OPT_GENERATION) {
|
if (option_mask32 & OPT_GENERATION) {
|
||||||
|
unsigned long generation;
|
||||||
if (fgetversion(name, &generation) != 0)
|
if (fgetversion(name, &generation) != 0)
|
||||||
goto read_err;
|
goto read_err;
|
||||||
printf("%5lu ", generation);
|
printf("%-10lu ", generation);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (option_mask32 & OPT_PF_LONG) {
|
if (option_mask32 & OPT_PF_LONG) {
|
||||||
@ -111,7 +121,7 @@ static void lsattr_args(const char *name)
|
|||||||
int lsattr_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
int lsattr_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
||||||
int lsattr_main(int argc UNUSED_PARAM, char **argv)
|
int lsattr_main(int argc UNUSED_PARAM, char **argv)
|
||||||
{
|
{
|
||||||
getopt32(argv, "Radlv");
|
getopt32(argv, "Radlvp");
|
||||||
argv += optind;
|
argv += optind;
|
||||||
|
|
||||||
if (!*argv)
|
if (!*argv)
|
||||||
|
@ -195,6 +195,18 @@ struct ext2_dx_countlimit {
|
|||||||
#define EXT2_IOC_SETFLAGS _IOW('f', 2, long)
|
#define EXT2_IOC_SETFLAGS _IOW('f', 2, long)
|
||||||
#define EXT2_IOC_GETVERSION _IOR('v', 1, long)
|
#define EXT2_IOC_GETVERSION _IOR('v', 1, long)
|
||||||
#define EXT2_IOC_SETVERSION _IOW('v', 2, long)
|
#define EXT2_IOC_SETVERSION _IOW('v', 2, long)
|
||||||
|
//NB: despite "long" in defs above, these ioctls use an _int_!
|
||||||
|
//passing them a pointer to long will read/write only int-sized data!
|
||||||
|
struct ext2_fsxattr {
|
||||||
|
uint32_t fsx_xflags; /* xflags field value (get/set) */
|
||||||
|
uint32_t fsx_extsize; /* extsize field value (get/set)*/
|
||||||
|
uint32_t fsx_nextents; /* nextents field value (get) */
|
||||||
|
uint32_t fsx_projid; /* project identifier (get/set) */
|
||||||
|
uint32_t fsx_cowextsize; /* CoW extsize field value (get/set)*/
|
||||||
|
unsigned char fsx_pad[8];
|
||||||
|
};
|
||||||
|
#define EXT2_IOC_FSGETXATTR _IOR('X', 31, struct ext2_fsxattr)
|
||||||
|
#define EXT2_IOC_FSSETXATTR _IOW('X', 32, struct ext2_fsxattr)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Structure of an inode on the disk
|
* Structure of an inode on the disk
|
||||||
|
Loading…
x
Reference in New Issue
Block a user