ovl: use posix acl api

Now that posix acls have a proper api us it to copy them.

All filesystems that can serve as lower or upper layers for overlayfs
have gained support for the new posix acl api in previous patches.
So switch all internal overlayfs codepaths for copying posix acls to the
new posix acl api.

Acked-by: Miklos Szeredi <miklos@szeredi.hu>
Signed-off-by: Christian Brauner (Microsoft) <brauner@kernel.org>
This commit is contained in:
Christian Brauner
2022-09-22 17:17:21 +02:00
committed by Christian Brauner (Microsoft)
parent 0e64185732
commit 31acceb975
7 changed files with 58 additions and 30 deletions

View File

@@ -44,6 +44,35 @@ static bool ovl_must_copy_xattr(const char *name)
!strncmp(name, XATTR_SECURITY_PREFIX, XATTR_SECURITY_PREFIX_LEN); !strncmp(name, XATTR_SECURITY_PREFIX, XATTR_SECURITY_PREFIX_LEN);
} }
static int ovl_copy_acl(struct ovl_fs *ofs, const struct path *path,
struct dentry *dentry, const char *acl_name)
{
int err;
struct posix_acl *clone, *real_acl = NULL;
real_acl = ovl_get_acl_path(path, acl_name, false);
if (!real_acl)
return 0;
if (IS_ERR(real_acl)) {
err = PTR_ERR(real_acl);
if (err == -ENODATA || err == -EOPNOTSUPP)
return 0;
return err;
}
clone = posix_acl_clone(real_acl, GFP_KERNEL);
posix_acl_release(real_acl); /* release original acl */
if (!clone)
return -ENOMEM;
err = ovl_do_set_acl(ofs, dentry, acl_name, clone);
/* release cloned acl */
posix_acl_release(clone);
return err;
}
int ovl_copy_xattr(struct super_block *sb, const struct path *oldpath, struct dentry *new) int ovl_copy_xattr(struct super_block *sb, const struct path *oldpath, struct dentry *new)
{ {
struct dentry *old = oldpath->dentry; struct dentry *old = oldpath->dentry;
@@ -93,6 +122,15 @@ int ovl_copy_xattr(struct super_block *sb, const struct path *oldpath, struct de
error = 0; error = 0;
continue; /* Discard */ continue; /* Discard */
} }
if (is_posix_acl_xattr(name)) {
error = ovl_copy_acl(OVL_FS(sb), oldpath, new, name);
if (!error)
continue;
/* POSIX ACLs must be copied. */
break;
}
retry: retry:
size = ovl_do_getxattr(oldpath, name, value, value_size); size = ovl_do_getxattr(oldpath, name, value, value_size);
if (size == -ERANGE) if (size == -ERANGE)

View File

@@ -435,28 +435,12 @@ out:
} }
static int ovl_set_upper_acl(struct ovl_fs *ofs, struct dentry *upperdentry, static int ovl_set_upper_acl(struct ovl_fs *ofs, struct dentry *upperdentry,
const char *name, const struct posix_acl *acl) const char *acl_name, struct posix_acl *acl)
{ {
void *buffer;
size_t size;
int err;
if (!IS_ENABLED(CONFIG_FS_POSIX_ACL) || !acl) if (!IS_ENABLED(CONFIG_FS_POSIX_ACL) || !acl)
return 0; return 0;
size = posix_acl_xattr_size(acl->a_count); return ovl_do_set_acl(ofs, upperdentry, acl_name, acl);
buffer = kmalloc(size, GFP_KERNEL);
if (!buffer)
return -ENOMEM;
err = posix_acl_to_xattr(&init_user_ns, acl, buffer, size);
if (err < 0)
goto out_free;
err = ovl_do_setxattr(ofs, upperdentry, name, buffer, size, XATTR_CREATE);
out_free:
kfree(buffer);
return err;
} }
static int ovl_create_over_whiteout(struct dentry *dentry, struct inode *inode, static int ovl_create_over_whiteout(struct dentry *dentry, struct inode *inode,

View File

@@ -510,8 +510,8 @@ static void ovl_idmap_posix_acl(const struct inode *realinode,
* Until we have made a decision allow this helper to take the @noperm * Until we have made a decision allow this helper to take the @noperm
* argument. We should hopefully be able to remove it soon. * argument. We should hopefully be able to remove it soon.
*/ */
static struct posix_acl *ovl_get_acl_path(const struct path *path, struct posix_acl *ovl_get_acl_path(const struct path *path,
const char *acl_name, bool noperm) const char *acl_name, bool noperm)
{ {
struct posix_acl *real_acl, *clone; struct posix_acl *real_acl, *clone;
struct user_namespace *mnt_userns; struct user_namespace *mnt_userns;

View File

@@ -623,10 +623,18 @@ static inline struct posix_acl *ovl_get_acl(struct user_namespace *mnt_userns,
} }
int ovl_set_acl(struct user_namespace *mnt_userns, struct dentry *dentry, int ovl_set_acl(struct user_namespace *mnt_userns, struct dentry *dentry,
struct posix_acl *acl, int type); struct posix_acl *acl, int type);
struct posix_acl *ovl_get_acl_path(const struct path *path,
const char *acl_name, bool noperm);
#else #else
#define ovl_get_inode_acl NULL #define ovl_get_inode_acl NULL
#define ovl_get_acl NULL #define ovl_get_acl NULL
#define ovl_set_acl NULL #define ovl_set_acl NULL
static inline struct posix_acl *ovl_get_acl_path(const struct path *path,
const char *acl_name,
bool noperm)
{
return NULL;
}
#endif #endif
int ovl_update_time(struct inode *inode, struct timespec64 *ts, int flags); int ovl_update_time(struct inode *inode, struct timespec64 *ts, int flags);

View File

@@ -813,13 +813,11 @@ retry:
* allowed as upper are limited to "normal" ones, where checking * allowed as upper are limited to "normal" ones, where checking
* for the above two errors is sufficient. * for the above two errors is sufficient.
*/ */
err = ovl_do_removexattr(ofs, work, err = ovl_do_remove_acl(ofs, work, XATTR_NAME_POSIX_ACL_DEFAULT);
XATTR_NAME_POSIX_ACL_DEFAULT);
if (err && err != -ENODATA && err != -EOPNOTSUPP) if (err && err != -ENODATA && err != -EOPNOTSUPP)
goto out_dput; goto out_dput;
err = ovl_do_removexattr(ofs, work, err = ovl_do_remove_acl(ofs, work, XATTR_NAME_POSIX_ACL_ACCESS);
XATTR_NAME_POSIX_ACL_ACCESS);
if (err && err != -ENODATA && err != -EOPNOTSUPP) if (err && err != -ENODATA && err != -EOPNOTSUPP)
goto out_dput; goto out_dput;

View File

@@ -299,12 +299,6 @@ out:
} }
EXPORT_SYMBOL_GPL(__vfs_setxattr_locked); EXPORT_SYMBOL_GPL(__vfs_setxattr_locked);
static inline bool is_posix_acl_xattr(const char *name)
{
return (strcmp(name, XATTR_NAME_POSIX_ACL_ACCESS) == 0) ||
(strcmp(name, XATTR_NAME_POSIX_ACL_DEFAULT) == 0);
}
int int
vfs_setxattr(struct user_namespace *mnt_userns, struct dentry *dentry, vfs_setxattr(struct user_namespace *mnt_userns, struct dentry *dentry,
const char *name, const void *value, size_t size, int flags) const char *name, const void *value, size_t size, int flags)

View File

@@ -22,6 +22,12 @@
struct inode; struct inode;
struct dentry; struct dentry;
static inline bool is_posix_acl_xattr(const char *name)
{
return (strcmp(name, XATTR_NAME_POSIX_ACL_ACCESS) == 0) ||
(strcmp(name, XATTR_NAME_POSIX_ACL_DEFAULT) == 0);
}
/* /*
* struct xattr_handler: When @name is set, match attributes with exactly that * struct xattr_handler: When @name is set, match attributes with exactly that
* name. When @prefix is set instead, match attributes with that prefix and * name. When @prefix is set instead, match attributes with that prefix and