[Btrfs-devel] [PATCH] Add btrfs acl checking
Yan Zheng
yanzheng at 21cn.com
Mon Jan 21 04:30:37 PST 2008
2008/1/19, only flyer <onlyflyer at gmail.com>:
> Hi:
> This patch add btrfs acl checking. It makes btrfs's acl checking
> really work if you mount btrfs by using "acl" option, e.g. mount -t
> btrfs -o acl /dev/hda1 /mnt/
>
> Thanks
> LiuHui
>
> diff -r 6c243ad8dddf acl.c
> --- a/acl.c Fri Jan 18 10:54:22 2008 -0500
> +++ b/acl.c Sat Jan 19 21:38:02 2008 +0800
> @@ -26,6 +26,35 @@
> #define is_owner_or_cap(inode) \
> ((current->fsuid == (inode)->i_uid) || capable(CAP_FOWNER))
> #endif
> +
> +struct posix_acl *btrfs_get_posix_acl(struct inode *inode, int type)
> +{
> + int ret, len;
> + char *value = NULL;
> + struct posix_acl *acl;
> +
> + len = btrfs_xattr_get(inode, type, "", NULL, 0);
> + if (len > 0) {
> + value = kmalloc(len, GFP_KERNEL);
> + if (!value) {
> + return ERR_PTR(-ENOMEM);
> + }
> +
> + } else {
> + return NULL;
> + }
> + ret = btrfs_xattr_get(inode, type, "", value, len);
> + if (ret > 0) {
> + acl = posix_acl_from_xattr(value, len);
> + } else if (-ENODATA == ret) {
> + acl = NULL;
> + } else {
> + acl = ERR_PTR(ret);
> + }
> +
> + kfree(value);
> + return acl;
> +}
>
> static int btrfs_xattr_set_acl(struct inode *inode, int type,
> const void *value, size_t size)
> diff -r 6c243ad8dddf ctree.h
> --- a/ctree.h Fri Jan 18 10:54:22 2008 -0500
> +++ b/ctree.h Sat Jan 19 21:38:02 2008 +0800
> @@ -446,6 +446,7 @@ struct btrfs_root {
> #define BTRFS_MOUNT_NODATACOW (1 << 1)
> #define BTRFS_MOUNT_NOBARRIER (1 << 2)
> #define BTRFS_MOUNT_SSD (1 << 3)
> +#define BTRFS_MOUNT_POSIX_ACL (1 << 4)
>
> #define btrfs_clear_opt(o, opt) ((o) &= ~BTRFS_MOUNT_##opt)
> #define btrfs_set_opt(o, opt) ((o) |= BTRFS_MOUNT_##opt)
> diff -r 6c243ad8dddf inode.c
> --- a/inode.c Fri Jan 18 10:54:22 2008 -0500
> +++ b/inode.c Sat Jan 19 21:38:02 2008 +0800
> @@ -33,12 +33,16 @@
> #include <linux/bit_spinlock.h>
> #include <linux/version.h>
> #include <linux/xattr.h>
> +#include <linux/posix_acl.h>
> #include "ctree.h"
> #include "disk-io.h"
> #include "transaction.h"
> #include "btrfs_inode.h"
> #include "ioctl.h"
> #include "print-tree.h"
> +#include "xattr.h"
> +
> +extern struct posix_acl *btrfs_get_posix_acl(struct inode *inode, int type);
>
> struct btrfs_iget_args {
> u64 ino;
> @@ -2886,12 +2890,29 @@ out_fail:
> btrfs_throttle(root);
> return err;
> }
> +
> +static int btrfs_check_acl(struct inode *inode, int mask)
> +{
> + struct posix_acl *acl;
> + acl = btrfs_get_posix_acl(inode, BTRFS_XATTR_INDEX_POSIX_ACL_ACCESS);
> +
> + if (IS_ERR(acl))
> + return PTR_ERR(acl);
> + if (acl) {
> + int error = posix_acl_permission(inode, acl, mask);
> + posix_acl_release(acl);
> + return error;
> + }
> +
> + return -EAGAIN;
> +}
> +
I feel this function should be moved into acl.c, so that btrfs can be
compiled on kernel without acl support.
> static int btrfs_permission(struct inode *inode, int mask,
> struct nameidata *nd)
> {
> if (btrfs_test_flag(inode, READONLY) && (mask & MAY_WRITE))
> return -EACCES;
> - return generic_permission(inode, mask, NULL);
> + return generic_permission(inode, mask, btrfs_check_acl);
> }
>
> static struct inode_operations btrfs_dir_inode_operations = {
> diff -r 6c243ad8dddf super.c
> --- a/super.c Fri Jan 18 10:54:22 2008 -0500
> +++ b/super.c Sat Jan 19 21:38:02 2008 +0800
> @@ -64,7 +64,7 @@ static void btrfs_put_super (struct supe
>
> enum {
> Opt_subvol, Opt_nodatasum, Opt_nodatacow, Opt_max_extent,
> - Opt_alloc_start, Opt_nobarrier, Opt_ssd, Opt_err,
> + Opt_alloc_start, Opt_nobarrier, Opt_ssd, Opt_posix_acl, Opt_err,
> };
>
> static match_table_t tokens = {
> @@ -75,6 +75,7 @@ static match_table_t tokens = {
> {Opt_max_extent, "max_extent=%s"},
> {Opt_alloc_start, "alloc_start=%s"},
> {Opt_ssd, "ssd"},
> + {Opt_posix_acl, "acl"},
> {Opt_err, NULL}
> };
>
> @@ -162,6 +163,14 @@ static int parse_options (char * options
> btrfs_set_opt(info->mount_opt, NOBARRIER);
> }
> break;
> +
> + case Opt_posix_acl:
> + if (info) {
> + printk("btrfs: enabling posix acl\n");
> + btrfs_set_opt(info->mount_opt, POSIX_ACL);
> + }
> + break;
> +
> case Opt_max_extent:
> if (info) {
> char *num = match_strdup(&args[0]);
> @@ -247,6 +256,12 @@ static int btrfs_fill_super(struct super
> }
>
> parse_options((char *)data, tree_root, NULL);
> +
> + if (tree_root->fs_info->mount_opt & BTRFS_MOUNT_POSIX_ACL) {
You can use 'btrfs_test_opt' to do this test.
> + sb->s_flags |= MS_POSIXACL;
> + } else {
> + sb->s_flags = sb->s_flags & ~MS_POSIXACL;
> + }
>
> /* this does the super kobj at the same time */
> err = btrfs_sysfs_add_super(tree_root->fs_info);
>
Regards
YZ
More information about the Btrfs-devel
mailing list