From e32205997f2dc3e0e74fd0a5571ea707a3bf831c Mon Sep 17 00:00:00 2001 From: Filipe David Borba Manana Date: Wed, 13 Nov 2013 01:22:09 +0000 Subject: [PATCH] Btrfs-progs: add support for the compression property With this property, one can enable compression for individual files without the need to mount the filesystem with the compress or compress-force options, and specify the compression algorithm. When applied against a directory, files created under that directory will inherit the compression property. This requires the corresponding kernel patch, which adds the support for setting and getting properties and implements the compression property. Signed-off-by: Filipe David Borba Manana Signed-off-by: David Sterba Signed-off-by: Chris Mason --- props.c | 78 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 78 insertions(+) diff --git a/props.c b/props.c index 12fdc524..4d0aeeab 100644 --- a/props.c +++ b/props.c @@ -16,6 +16,8 @@ #include #include +#include +#include #include #include @@ -24,6 +26,10 @@ #include "utils.h" #include "props.h" +#define XATTR_BTRFS_PREFIX "btrfs." +#define XATTR_BTRFS_PREFIX_LEN (sizeof(XATTR_BTRFS_PREFIX) - 1) + + static int prop_read_only(enum prop_object_type type, const char *object, const char *name, @@ -102,10 +108,82 @@ static int prop_label(enum prop_object_type type, return ret; } +static int prop_compression(enum prop_object_type type, + const char *object, + const char *name, + const char *value) +{ + int ret; + ssize_t sret; + int fd = -1; + DIR *dirstream = NULL; + char *buf = NULL; + char *xattr_name = NULL; + + fd = open_file_or_dir(object, &dirstream); + if (fd == -1) { + ret = -errno; + fprintf(stderr, "ERROR: open %s failed. %s\n", + object, strerror(-ret)); + goto out; + } + + xattr_name = malloc(XATTR_BTRFS_PREFIX_LEN + strlen(name)); + if (!xattr_name) { + ret = -ENOMEM; + goto out; + } + memcpy(xattr_name, XATTR_BTRFS_PREFIX, XATTR_BTRFS_PREFIX_LEN); + memcpy(xattr_name + XATTR_BTRFS_PREFIX_LEN, name, strlen(name)); + + if (value) + sret = fsetxattr(fd, xattr_name, value, strlen(value), 0); + else + sret = fgetxattr(fd, xattr_name, NULL, 0); + if (sret < 0) { + ret = -errno; + if (ret != -ENODATA) + fprintf(stderr, + "ERROR: failed to %s compression for %s. %s\n", + value ? "set" : "get", object, strerror(-ret)); + goto out; + } + if (!value) { + size_t len = sret; + + buf = malloc(len); + if (!buf) { + ret = -ENOMEM; + goto out; + } + sret = fgetxattr(fd, xattr_name, buf, len); + if (sret < 0) { + ret = -errno; + fprintf(stderr, + "ERROR: failed to get compression for %s. %s\n", + object, strerror(-ret)); + goto out; + } + fprintf(stdout, "compression=%.*s\n", (int)len, buf); + } + + ret = 0; +out: + free(xattr_name); + free(buf); + if (fd >= 0) + close_file_or_dir(fd, dirstream); + + return ret; +} + + const struct prop_handler prop_handlers[] = { {"ro", "Set/get read-only flag of subvolume.", 0, prop_object_subvol, prop_read_only}, {"label", "Set/get label of device.", 0, prop_object_dev | prop_object_root, prop_label}, + {"compression", "Set/get compression for a file or directory", 0, + prop_object_inode, prop_compression}, {0, 0, 0, 0, 0} };