Contributors: 21
Author |
Tokens |
Token Proportion |
Commits |
Commit Proportion |
Josef Bacik |
275 |
50.27% |
3 |
6.38% |
Yan Zheng |
64 |
11.70% |
2 |
4.26% |
Chris Mason |
42 |
7.68% |
10 |
21.28% |
Christoph Hellwig |
29 |
5.30% |
3 |
6.38% |
Ernesto A. Fernández |
22 |
4.02% |
1 |
2.13% |
Jan Kara |
20 |
3.66% |
3 |
6.38% |
Filipe David Borba Manana |
18 |
3.29% |
1 |
2.13% |
Anand Jain |
10 |
1.83% |
1 |
2.13% |
Christian Brauner |
10 |
1.83% |
5 |
10.64% |
Al Viro |
9 |
1.65% |
3 |
6.38% |
Miklos Szeredi |
9 |
1.65% |
1 |
2.13% |
Chengguang Xu |
9 |
1.65% |
3 |
6.38% |
David Sterba |
6 |
1.10% |
3 |
6.38% |
Eric W. Biedermann |
6 |
1.10% |
1 |
2.13% |
Zheng Yan |
4 |
0.73% |
1 |
2.13% |
Andreas Gruenbacher |
4 |
0.73% |
1 |
2.13% |
Dan J Williams |
3 |
0.55% |
1 |
2.13% |
Josef Whiter |
3 |
0.55% |
1 |
2.13% |
Linus Torvalds (pre-git) |
2 |
0.37% |
1 |
2.13% |
Linus Torvalds |
1 |
0.18% |
1 |
2.13% |
Tsutomu Itoh |
1 |
0.18% |
1 |
2.13% |
Total |
547 |
|
47 |
|
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (C) 2007 Red Hat. All rights reserved.
*/
#include <linux/fs.h>
#include <linux/string.h>
#include <linux/xattr.h>
#include <linux/posix_acl_xattr.h>
#include <linux/posix_acl.h>
#include <linux/sched.h>
#include <linux/sched/mm.h>
#include <linux/slab.h>
#include "ctree.h"
#include "btrfs_inode.h"
#include "xattr.h"
#include "acl.h"
struct posix_acl *btrfs_get_acl(struct inode *inode, int type, bool rcu)
{
int size;
const char *name;
char *value = NULL;
struct posix_acl *acl;
if (rcu)
return ERR_PTR(-ECHILD);
switch (type) {
case ACL_TYPE_ACCESS:
name = XATTR_NAME_POSIX_ACL_ACCESS;
break;
case ACL_TYPE_DEFAULT:
name = XATTR_NAME_POSIX_ACL_DEFAULT;
break;
default:
return ERR_PTR(-EINVAL);
}
size = btrfs_getxattr(inode, name, NULL, 0);
if (size > 0) {
value = kzalloc(size, GFP_KERNEL);
if (!value)
return ERR_PTR(-ENOMEM);
size = btrfs_getxattr(inode, name, value, size);
}
if (size > 0)
acl = posix_acl_from_xattr(&init_user_ns, value, size);
else if (size == -ENODATA || size == 0)
acl = NULL;
else
acl = ERR_PTR(size);
kfree(value);
return acl;
}
int __btrfs_set_acl(struct btrfs_trans_handle *trans, struct inode *inode,
struct posix_acl *acl, int type)
{
int ret, size = 0;
const char *name;
char *value = NULL;
switch (type) {
case ACL_TYPE_ACCESS:
name = XATTR_NAME_POSIX_ACL_ACCESS;
break;
case ACL_TYPE_DEFAULT:
if (!S_ISDIR(inode->i_mode))
return acl ? -EINVAL : 0;
name = XATTR_NAME_POSIX_ACL_DEFAULT;
break;
default:
return -EINVAL;
}
if (acl) {
unsigned int nofs_flag;
size = posix_acl_xattr_size(acl->a_count);
/*
* We're holding a transaction handle, so use a NOFS memory
* allocation context to avoid deadlock if reclaim happens.
*/
nofs_flag = memalloc_nofs_save();
value = kmalloc(size, GFP_KERNEL);
memalloc_nofs_restore(nofs_flag);
if (!value) {
ret = -ENOMEM;
goto out;
}
ret = posix_acl_to_xattr(&init_user_ns, acl, value, size);
if (ret < 0)
goto out;
}
if (trans)
ret = btrfs_setxattr(trans, inode, name, value, size, 0);
else
ret = btrfs_setxattr_trans(inode, name, value, size, 0);
out:
kfree(value);
if (!ret)
set_cached_acl(inode, type, acl);
return ret;
}
int btrfs_set_acl(struct mnt_idmap *idmap, struct dentry *dentry,
struct posix_acl *acl, int type)
{
int ret;
struct inode *inode = d_inode(dentry);
umode_t old_mode = inode->i_mode;
if (type == ACL_TYPE_ACCESS && acl) {
ret = posix_acl_update_mode(idmap, inode,
&inode->i_mode, &acl);
if (ret)
return ret;
}
ret = __btrfs_set_acl(NULL, inode, acl, type);
if (ret)
inode->i_mode = old_mode;
return ret;
}