mirror of https://gitee.com/openkylin/linux.git
nfs: use generic posix ACL infrastructure for v3 Posix ACLs
This causes a small behaviour change in that we don't bother to set ACLs on file creation if the mode bit can express the access permissions fully, and thus behaving identical to local filesystems. Signed-off-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
parent
e01580bf9e
commit
013cdf1088
|
@ -1641,10 +1641,6 @@ struct inode *nfs_alloc_inode(struct super_block *sb)
|
||||||
return NULL;
|
return NULL;
|
||||||
nfsi->flags = 0UL;
|
nfsi->flags = 0UL;
|
||||||
nfsi->cache_validity = 0UL;
|
nfsi->cache_validity = 0UL;
|
||||||
#ifdef CONFIG_NFS_V3_ACL
|
|
||||||
nfsi->acl_access = ERR_PTR(-EAGAIN);
|
|
||||||
nfsi->acl_default = ERR_PTR(-EAGAIN);
|
|
||||||
#endif
|
|
||||||
#if IS_ENABLED(CONFIG_NFS_V4)
|
#if IS_ENABLED(CONFIG_NFS_V4)
|
||||||
nfsi->nfs4_acl = NULL;
|
nfsi->nfs4_acl = NULL;
|
||||||
#endif /* CONFIG_NFS_V4 */
|
#endif /* CONFIG_NFS_V4 */
|
||||||
|
|
289
fs/nfs/nfs3acl.c
289
fs/nfs/nfs3acl.c
|
@ -10,179 +10,7 @@
|
||||||
|
|
||||||
#define NFSDBG_FACILITY NFSDBG_PROC
|
#define NFSDBG_FACILITY NFSDBG_PROC
|
||||||
|
|
||||||
ssize_t nfs3_listxattr(struct dentry *dentry, char *buffer, size_t size)
|
struct posix_acl *nfs3_get_acl(struct inode *inode, int type)
|
||||||
{
|
|
||||||
struct inode *inode = dentry->d_inode;
|
|
||||||
struct posix_acl *acl;
|
|
||||||
int pos=0, len=0;
|
|
||||||
|
|
||||||
# define output(s) do { \
|
|
||||||
if (pos + sizeof(s) <= size) { \
|
|
||||||
memcpy(buffer + pos, s, sizeof(s)); \
|
|
||||||
pos += sizeof(s); \
|
|
||||||
} \
|
|
||||||
len += sizeof(s); \
|
|
||||||
} while(0)
|
|
||||||
|
|
||||||
acl = nfs3_proc_getacl(inode, ACL_TYPE_ACCESS);
|
|
||||||
if (IS_ERR(acl))
|
|
||||||
return PTR_ERR(acl);
|
|
||||||
if (acl) {
|
|
||||||
output("system.posix_acl_access");
|
|
||||||
posix_acl_release(acl);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (S_ISDIR(inode->i_mode)) {
|
|
||||||
acl = nfs3_proc_getacl(inode, ACL_TYPE_DEFAULT);
|
|
||||||
if (IS_ERR(acl))
|
|
||||||
return PTR_ERR(acl);
|
|
||||||
if (acl) {
|
|
||||||
output("system.posix_acl_default");
|
|
||||||
posix_acl_release(acl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
# undef output
|
|
||||||
|
|
||||||
if (!buffer || len <= size)
|
|
||||||
return len;
|
|
||||||
return -ERANGE;
|
|
||||||
}
|
|
||||||
|
|
||||||
ssize_t nfs3_getxattr(struct dentry *dentry, const char *name,
|
|
||||||
void *buffer, size_t size)
|
|
||||||
{
|
|
||||||
struct inode *inode = dentry->d_inode;
|
|
||||||
struct posix_acl *acl;
|
|
||||||
int type, error = 0;
|
|
||||||
|
|
||||||
if (strcmp(name, POSIX_ACL_XATTR_ACCESS) == 0)
|
|
||||||
type = ACL_TYPE_ACCESS;
|
|
||||||
else if (strcmp(name, POSIX_ACL_XATTR_DEFAULT) == 0)
|
|
||||||
type = ACL_TYPE_DEFAULT;
|
|
||||||
else
|
|
||||||
return -EOPNOTSUPP;
|
|
||||||
|
|
||||||
acl = nfs3_proc_getacl(inode, type);
|
|
||||||
if (IS_ERR(acl))
|
|
||||||
return PTR_ERR(acl);
|
|
||||||
else if (acl) {
|
|
||||||
if (type == ACL_TYPE_ACCESS && acl->a_count == 0)
|
|
||||||
error = -ENODATA;
|
|
||||||
else
|
|
||||||
error = posix_acl_to_xattr(&init_user_ns, acl, buffer, size);
|
|
||||||
posix_acl_release(acl);
|
|
||||||
} else
|
|
||||||
error = -ENODATA;
|
|
||||||
|
|
||||||
return error;
|
|
||||||
}
|
|
||||||
|
|
||||||
int nfs3_setxattr(struct dentry *dentry, const char *name,
|
|
||||||
const void *value, size_t size, int flags)
|
|
||||||
{
|
|
||||||
struct inode *inode = dentry->d_inode;
|
|
||||||
struct posix_acl *acl;
|
|
||||||
int type, error;
|
|
||||||
|
|
||||||
if (strcmp(name, POSIX_ACL_XATTR_ACCESS) == 0)
|
|
||||||
type = ACL_TYPE_ACCESS;
|
|
||||||
else if (strcmp(name, POSIX_ACL_XATTR_DEFAULT) == 0)
|
|
||||||
type = ACL_TYPE_DEFAULT;
|
|
||||||
else
|
|
||||||
return -EOPNOTSUPP;
|
|
||||||
|
|
||||||
acl = posix_acl_from_xattr(&init_user_ns, value, size);
|
|
||||||
if (IS_ERR(acl))
|
|
||||||
return PTR_ERR(acl);
|
|
||||||
error = nfs3_proc_setacl(inode, type, acl);
|
|
||||||
posix_acl_release(acl);
|
|
||||||
|
|
||||||
return error;
|
|
||||||
}
|
|
||||||
|
|
||||||
int nfs3_removexattr(struct dentry *dentry, const char *name)
|
|
||||||
{
|
|
||||||
struct inode *inode = dentry->d_inode;
|
|
||||||
int type;
|
|
||||||
|
|
||||||
if (strcmp(name, POSIX_ACL_XATTR_ACCESS) == 0)
|
|
||||||
type = ACL_TYPE_ACCESS;
|
|
||||||
else if (strcmp(name, POSIX_ACL_XATTR_DEFAULT) == 0)
|
|
||||||
type = ACL_TYPE_DEFAULT;
|
|
||||||
else
|
|
||||||
return -EOPNOTSUPP;
|
|
||||||
|
|
||||||
return nfs3_proc_setacl(inode, type, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void __nfs3_forget_cached_acls(struct nfs_inode *nfsi)
|
|
||||||
{
|
|
||||||
if (!IS_ERR(nfsi->acl_access)) {
|
|
||||||
posix_acl_release(nfsi->acl_access);
|
|
||||||
nfsi->acl_access = ERR_PTR(-EAGAIN);
|
|
||||||
}
|
|
||||||
if (!IS_ERR(nfsi->acl_default)) {
|
|
||||||
posix_acl_release(nfsi->acl_default);
|
|
||||||
nfsi->acl_default = ERR_PTR(-EAGAIN);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void nfs3_forget_cached_acls(struct inode *inode)
|
|
||||||
{
|
|
||||||
dprintk("NFS: nfs3_forget_cached_acls(%s/%ld)\n", inode->i_sb->s_id,
|
|
||||||
inode->i_ino);
|
|
||||||
spin_lock(&inode->i_lock);
|
|
||||||
__nfs3_forget_cached_acls(NFS_I(inode));
|
|
||||||
spin_unlock(&inode->i_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct posix_acl *nfs3_get_cached_acl(struct inode *inode, int type)
|
|
||||||
{
|
|
||||||
struct nfs_inode *nfsi = NFS_I(inode);
|
|
||||||
struct posix_acl *acl = ERR_PTR(-EINVAL);
|
|
||||||
|
|
||||||
spin_lock(&inode->i_lock);
|
|
||||||
switch(type) {
|
|
||||||
case ACL_TYPE_ACCESS:
|
|
||||||
acl = nfsi->acl_access;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case ACL_TYPE_DEFAULT:
|
|
||||||
acl = nfsi->acl_default;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
if (IS_ERR(acl))
|
|
||||||
acl = ERR_PTR(-EAGAIN);
|
|
||||||
else
|
|
||||||
acl = posix_acl_dup(acl);
|
|
||||||
out:
|
|
||||||
spin_unlock(&inode->i_lock);
|
|
||||||
dprintk("NFS: nfs3_get_cached_acl(%s/%ld, %d) = %p\n", inode->i_sb->s_id,
|
|
||||||
inode->i_ino, type, acl);
|
|
||||||
return acl;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void nfs3_cache_acls(struct inode *inode, struct posix_acl *acl,
|
|
||||||
struct posix_acl *dfacl)
|
|
||||||
{
|
|
||||||
struct nfs_inode *nfsi = NFS_I(inode);
|
|
||||||
|
|
||||||
dprintk("nfs3_cache_acls(%s/%ld, %p, %p)\n", inode->i_sb->s_id,
|
|
||||||
inode->i_ino, acl, dfacl);
|
|
||||||
spin_lock(&inode->i_lock);
|
|
||||||
__nfs3_forget_cached_acls(NFS_I(inode));
|
|
||||||
if (!IS_ERR(acl))
|
|
||||||
nfsi->acl_access = posix_acl_dup(acl);
|
|
||||||
if (!IS_ERR(dfacl))
|
|
||||||
nfsi->acl_default = posix_acl_dup(dfacl);
|
|
||||||
spin_unlock(&inode->i_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
struct posix_acl *nfs3_proc_getacl(struct inode *inode, int type)
|
|
||||||
{
|
{
|
||||||
struct nfs_server *server = NFS_SERVER(inode);
|
struct nfs_server *server = NFS_SERVER(inode);
|
||||||
struct page *pages[NFSACL_MAXPAGES] = { };
|
struct page *pages[NFSACL_MAXPAGES] = { };
|
||||||
|
@ -198,7 +26,6 @@ struct posix_acl *nfs3_proc_getacl(struct inode *inode, int type)
|
||||||
.rpc_argp = &args,
|
.rpc_argp = &args,
|
||||||
.rpc_resp = &res,
|
.rpc_resp = &res,
|
||||||
};
|
};
|
||||||
struct posix_acl *acl;
|
|
||||||
int status, count;
|
int status, count;
|
||||||
|
|
||||||
if (!nfs_server_capable(inode, NFS_CAP_ACLS))
|
if (!nfs_server_capable(inode, NFS_CAP_ACLS))
|
||||||
|
@ -207,10 +34,6 @@ struct posix_acl *nfs3_proc_getacl(struct inode *inode, int type)
|
||||||
status = nfs_revalidate_inode(server, inode);
|
status = nfs_revalidate_inode(server, inode);
|
||||||
if (status < 0)
|
if (status < 0)
|
||||||
return ERR_PTR(status);
|
return ERR_PTR(status);
|
||||||
acl = nfs3_get_cached_acl(inode, type);
|
|
||||||
if (acl != ERR_PTR(-EAGAIN))
|
|
||||||
return acl;
|
|
||||||
acl = NULL;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Only get the access acl when explicitly requested: We don't
|
* Only get the access acl when explicitly requested: We don't
|
||||||
|
@ -257,40 +80,41 @@ struct posix_acl *nfs3_proc_getacl(struct inode *inode, int type)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (res.acl_access != NULL) {
|
if (res.acl_access != NULL) {
|
||||||
if (posix_acl_equiv_mode(res.acl_access, NULL) == 0) {
|
if (posix_acl_equiv_mode(res.acl_access, NULL) ||
|
||||||
|
res.acl_access->a_count == 0) {
|
||||||
posix_acl_release(res.acl_access);
|
posix_acl_release(res.acl_access);
|
||||||
res.acl_access = NULL;
|
res.acl_access = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
nfs3_cache_acls(inode,
|
|
||||||
(res.mask & NFS_ACL) ? res.acl_access : ERR_PTR(-EINVAL),
|
|
||||||
(res.mask & NFS_DFACL) ? res.acl_default : ERR_PTR(-EINVAL));
|
|
||||||
|
|
||||||
switch(type) {
|
if (res.mask & NFS_ACL)
|
||||||
case ACL_TYPE_ACCESS:
|
set_cached_acl(inode, ACL_TYPE_ACCESS, res.acl_access);
|
||||||
acl = res.acl_access;
|
else
|
||||||
res.acl_access = NULL;
|
forget_cached_acl(inode, ACL_TYPE_ACCESS);
|
||||||
break;
|
|
||||||
|
|
||||||
case ACL_TYPE_DEFAULT:
|
if (res.mask & NFS_DFACL)
|
||||||
acl = res.acl_default;
|
set_cached_acl(inode, ACL_TYPE_DEFAULT, res.acl_default);
|
||||||
res.acl_default = NULL;
|
else
|
||||||
|
forget_cached_acl(inode, ACL_TYPE_DEFAULT);
|
||||||
|
|
||||||
|
nfs_free_fattr(res.fattr);
|
||||||
|
if (type == ACL_TYPE_ACCESS) {
|
||||||
|
posix_acl_release(res.acl_default);
|
||||||
|
return res.acl_access;
|
||||||
|
} else {
|
||||||
|
posix_acl_release(res.acl_access);
|
||||||
|
return res.acl_default;
|
||||||
}
|
}
|
||||||
|
|
||||||
getout:
|
getout:
|
||||||
posix_acl_release(res.acl_access);
|
posix_acl_release(res.acl_access);
|
||||||
posix_acl_release(res.acl_default);
|
posix_acl_release(res.acl_default);
|
||||||
nfs_free_fattr(res.fattr);
|
nfs_free_fattr(res.fattr);
|
||||||
|
return ERR_PTR(status);
|
||||||
if (status != 0) {
|
|
||||||
posix_acl_release(acl);
|
|
||||||
acl = ERR_PTR(status);
|
|
||||||
}
|
|
||||||
return acl;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nfs3_proc_setacls(struct inode *inode, struct posix_acl *acl,
|
int nfs3_proc_setacls(struct inode *inode, struct posix_acl *acl,
|
||||||
struct posix_acl *dfacl)
|
struct posix_acl *dfacl)
|
||||||
{
|
{
|
||||||
struct nfs_server *server = NFS_SERVER(inode);
|
struct nfs_server *server = NFS_SERVER(inode);
|
||||||
struct nfs_fattr *fattr;
|
struct nfs_fattr *fattr;
|
||||||
|
@ -353,7 +177,8 @@ static int nfs3_proc_setacls(struct inode *inode, struct posix_acl *acl,
|
||||||
switch (status) {
|
switch (status) {
|
||||||
case 0:
|
case 0:
|
||||||
status = nfs_refresh_inode(inode, fattr);
|
status = nfs_refresh_inode(inode, fattr);
|
||||||
nfs3_cache_acls(inode, acl, dfacl);
|
set_cached_acl(inode, ACL_TYPE_ACCESS, acl);
|
||||||
|
set_cached_acl(inode, ACL_TYPE_DEFAULT, dfacl);
|
||||||
break;
|
break;
|
||||||
case -EPFNOSUPPORT:
|
case -EPFNOSUPPORT:
|
||||||
case -EPROTONOSUPPORT:
|
case -EPROTONOSUPPORT:
|
||||||
|
@ -373,33 +198,27 @@ static int nfs3_proc_setacls(struct inode *inode, struct posix_acl *acl,
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
int nfs3_proc_setacl(struct inode *inode, int type, struct posix_acl *acl)
|
int nfs3_set_acl(struct inode *inode, struct posix_acl *acl, int type)
|
||||||
{
|
{
|
||||||
struct posix_acl *alloc = NULL, *dfacl = NULL;
|
struct posix_acl *alloc = NULL, *dfacl = NULL;
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
if (S_ISDIR(inode->i_mode)) {
|
if (S_ISDIR(inode->i_mode)) {
|
||||||
switch(type) {
|
switch(type) {
|
||||||
case ACL_TYPE_ACCESS:
|
case ACL_TYPE_ACCESS:
|
||||||
alloc = dfacl = nfs3_proc_getacl(inode,
|
alloc = dfacl = get_acl(inode, ACL_TYPE_DEFAULT);
|
||||||
ACL_TYPE_DEFAULT);
|
if (IS_ERR(alloc))
|
||||||
if (IS_ERR(alloc))
|
goto fail;
|
||||||
goto fail;
|
break;
|
||||||
break;
|
|
||||||
|
|
||||||
case ACL_TYPE_DEFAULT:
|
case ACL_TYPE_DEFAULT:
|
||||||
dfacl = acl;
|
dfacl = acl;
|
||||||
alloc = acl = nfs3_proc_getacl(inode,
|
alloc = acl = get_acl(inode, ACL_TYPE_ACCESS);
|
||||||
ACL_TYPE_ACCESS);
|
if (IS_ERR(alloc))
|
||||||
if (IS_ERR(alloc))
|
goto fail;
|
||||||
goto fail;
|
break;
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
return -EINVAL;
|
|
||||||
}
|
}
|
||||||
} else if (type != ACL_TYPE_ACCESS)
|
}
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
if (acl == NULL) {
|
if (acl == NULL) {
|
||||||
alloc = acl = posix_acl_from_mode(inode->i_mode, GFP_KERNEL);
|
alloc = acl = posix_acl_from_mode(inode->i_mode, GFP_KERNEL);
|
||||||
|
@ -417,24 +236,24 @@ int nfs3_proc_setacl(struct inode *inode, int type, struct posix_acl *acl)
|
||||||
int nfs3_proc_set_default_acl(struct inode *dir, struct inode *inode,
|
int nfs3_proc_set_default_acl(struct inode *dir, struct inode *inode,
|
||||||
umode_t mode)
|
umode_t mode)
|
||||||
{
|
{
|
||||||
struct posix_acl *dfacl, *acl;
|
struct posix_acl *default_acl, *acl;
|
||||||
int error = 0;
|
int error;
|
||||||
|
|
||||||
dfacl = nfs3_proc_getacl(dir, ACL_TYPE_DEFAULT);
|
error = posix_acl_create(dir, &mode, &default_acl, &acl);
|
||||||
if (IS_ERR(dfacl)) {
|
if (error)
|
||||||
error = PTR_ERR(dfacl);
|
|
||||||
return (error == -EOPNOTSUPP) ? 0 : error;
|
return (error == -EOPNOTSUPP) ? 0 : error;
|
||||||
}
|
|
||||||
if (!dfacl)
|
error = nfs3_proc_setacls(inode, acl, default_acl);
|
||||||
return 0;
|
|
||||||
acl = posix_acl_dup(dfacl);
|
if (acl)
|
||||||
error = __posix_acl_create(&acl, GFP_KERNEL, &mode);
|
posix_acl_release(acl);
|
||||||
if (error < 0)
|
if (default_acl)
|
||||||
goto out_release_dfacl;
|
posix_acl_release(default_acl);
|
||||||
error = nfs3_proc_setacls(inode, acl, S_ISDIR(inode->i_mode) ?
|
|
||||||
dfacl : NULL);
|
|
||||||
posix_acl_release(acl);
|
|
||||||
out_release_dfacl:
|
|
||||||
posix_acl_release(dfacl);
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const struct xattr_handler *nfs3_xattr_handlers[] = {
|
||||||
|
&posix_acl_access_xattr_handler,
|
||||||
|
&posix_acl_default_xattr_handler,
|
||||||
|
NULL,
|
||||||
|
};
|
||||||
|
|
|
@ -317,8 +317,8 @@ static int
|
||||||
nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
||||||
int flags)
|
int flags)
|
||||||
{
|
{
|
||||||
|
struct posix_acl *default_acl, *acl;
|
||||||
struct nfs3_createdata *data;
|
struct nfs3_createdata *data;
|
||||||
umode_t mode = sattr->ia_mode;
|
|
||||||
int status = -ENOMEM;
|
int status = -ENOMEM;
|
||||||
|
|
||||||
dprintk("NFS call create %pd\n", dentry);
|
dprintk("NFS call create %pd\n", dentry);
|
||||||
|
@ -340,7 +340,9 @@ nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
||||||
data->arg.create.verifier[1] = cpu_to_be32(current->pid);
|
data->arg.create.verifier[1] = cpu_to_be32(current->pid);
|
||||||
}
|
}
|
||||||
|
|
||||||
sattr->ia_mode &= ~current_umask();
|
status = posix_acl_create(dir, &sattr->ia_mode, &default_acl, &acl);
|
||||||
|
if (status)
|
||||||
|
goto out;
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
status = nfs3_do_create(dir, dentry, data);
|
status = nfs3_do_create(dir, dentry, data);
|
||||||
|
@ -366,7 +368,7 @@ nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
||||||
}
|
}
|
||||||
|
|
||||||
if (status != 0)
|
if (status != 0)
|
||||||
goto out;
|
goto out_release_acls;
|
||||||
|
|
||||||
/* When we created the file with exclusive semantics, make
|
/* When we created the file with exclusive semantics, make
|
||||||
* sure we set the attributes afterwards. */
|
* sure we set the attributes afterwards. */
|
||||||
|
@ -385,9 +387,14 @@ nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
||||||
nfs_post_op_update_inode(dentry->d_inode, data->res.fattr);
|
nfs_post_op_update_inode(dentry->d_inode, data->res.fattr);
|
||||||
dprintk("NFS reply setattr (post-create): %d\n", status);
|
dprintk("NFS reply setattr (post-create): %d\n", status);
|
||||||
if (status != 0)
|
if (status != 0)
|
||||||
goto out;
|
goto out_release_acls;
|
||||||
}
|
}
|
||||||
status = nfs3_proc_set_default_acl(dir, dentry->d_inode, mode);
|
|
||||||
|
status = nfs3_proc_setacls(dentry->d_inode, acl, default_acl);
|
||||||
|
|
||||||
|
out_release_acls:
|
||||||
|
posix_acl_release(acl);
|
||||||
|
posix_acl_release(default_acl);
|
||||||
out:
|
out:
|
||||||
nfs3_free_createdata(data);
|
nfs3_free_createdata(data);
|
||||||
dprintk("NFS reply create: %d\n", status);
|
dprintk("NFS reply create: %d\n", status);
|
||||||
|
@ -572,18 +579,20 @@ nfs3_proc_symlink(struct inode *dir, struct dentry *dentry, struct page *page,
|
||||||
static int
|
static int
|
||||||
nfs3_proc_mkdir(struct inode *dir, struct dentry *dentry, struct iattr *sattr)
|
nfs3_proc_mkdir(struct inode *dir, struct dentry *dentry, struct iattr *sattr)
|
||||||
{
|
{
|
||||||
|
struct posix_acl *default_acl, *acl;
|
||||||
struct nfs3_createdata *data;
|
struct nfs3_createdata *data;
|
||||||
umode_t mode = sattr->ia_mode;
|
|
||||||
int status = -ENOMEM;
|
int status = -ENOMEM;
|
||||||
|
|
||||||
dprintk("NFS call mkdir %pd\n", dentry);
|
dprintk("NFS call mkdir %pd\n", dentry);
|
||||||
|
|
||||||
sattr->ia_mode &= ~current_umask();
|
|
||||||
|
|
||||||
data = nfs3_alloc_createdata();
|
data = nfs3_alloc_createdata();
|
||||||
if (data == NULL)
|
if (data == NULL)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
status = posix_acl_create(dir, &sattr->ia_mode, &default_acl, &acl);
|
||||||
|
if (status)
|
||||||
|
goto out;
|
||||||
|
|
||||||
data->msg.rpc_proc = &nfs3_procedures[NFS3PROC_MKDIR];
|
data->msg.rpc_proc = &nfs3_procedures[NFS3PROC_MKDIR];
|
||||||
data->arg.mkdir.fh = NFS_FH(dir);
|
data->arg.mkdir.fh = NFS_FH(dir);
|
||||||
data->arg.mkdir.name = dentry->d_name.name;
|
data->arg.mkdir.name = dentry->d_name.name;
|
||||||
|
@ -592,9 +601,13 @@ nfs3_proc_mkdir(struct inode *dir, struct dentry *dentry, struct iattr *sattr)
|
||||||
|
|
||||||
status = nfs3_do_create(dir, dentry, data);
|
status = nfs3_do_create(dir, dentry, data);
|
||||||
if (status != 0)
|
if (status != 0)
|
||||||
goto out;
|
goto out_release_acls;
|
||||||
|
|
||||||
status = nfs3_proc_set_default_acl(dir, dentry->d_inode, mode);
|
status = nfs3_proc_setacls(dentry->d_inode, acl, default_acl);
|
||||||
|
|
||||||
|
out_release_acls:
|
||||||
|
posix_acl_release(acl);
|
||||||
|
posix_acl_release(default_acl);
|
||||||
out:
|
out:
|
||||||
nfs3_free_createdata(data);
|
nfs3_free_createdata(data);
|
||||||
dprintk("NFS reply mkdir: %d\n", status);
|
dprintk("NFS reply mkdir: %d\n", status);
|
||||||
|
@ -691,19 +704,21 @@ static int
|
||||||
nfs3_proc_mknod(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
nfs3_proc_mknod(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
||||||
dev_t rdev)
|
dev_t rdev)
|
||||||
{
|
{
|
||||||
|
struct posix_acl *default_acl, *acl;
|
||||||
struct nfs3_createdata *data;
|
struct nfs3_createdata *data;
|
||||||
umode_t mode = sattr->ia_mode;
|
|
||||||
int status = -ENOMEM;
|
int status = -ENOMEM;
|
||||||
|
|
||||||
dprintk("NFS call mknod %pd %u:%u\n", dentry,
|
dprintk("NFS call mknod %pd %u:%u\n", dentry,
|
||||||
MAJOR(rdev), MINOR(rdev));
|
MAJOR(rdev), MINOR(rdev));
|
||||||
|
|
||||||
sattr->ia_mode &= ~current_umask();
|
|
||||||
|
|
||||||
data = nfs3_alloc_createdata();
|
data = nfs3_alloc_createdata();
|
||||||
if (data == NULL)
|
if (data == NULL)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
status = posix_acl_create(dir, &sattr->ia_mode, &default_acl, &acl);
|
||||||
|
if (status)
|
||||||
|
goto out;
|
||||||
|
|
||||||
data->msg.rpc_proc = &nfs3_procedures[NFS3PROC_MKNOD];
|
data->msg.rpc_proc = &nfs3_procedures[NFS3PROC_MKNOD];
|
||||||
data->arg.mknod.fh = NFS_FH(dir);
|
data->arg.mknod.fh = NFS_FH(dir);
|
||||||
data->arg.mknod.name = dentry->d_name.name;
|
data->arg.mknod.name = dentry->d_name.name;
|
||||||
|
@ -731,8 +746,13 @@ nfs3_proc_mknod(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
|
||||||
|
|
||||||
status = nfs3_do_create(dir, dentry, data);
|
status = nfs3_do_create(dir, dentry, data);
|
||||||
if (status != 0)
|
if (status != 0)
|
||||||
goto out;
|
goto out_release_acls;
|
||||||
status = nfs3_proc_set_default_acl(dir, dentry->d_inode, mode);
|
|
||||||
|
status = nfs3_proc_setacls(dentry->d_inode, acl, default_acl);
|
||||||
|
|
||||||
|
out_release_acls:
|
||||||
|
posix_acl_release(acl);
|
||||||
|
posix_acl_release(default_acl);
|
||||||
out:
|
out:
|
||||||
nfs3_free_createdata(data);
|
nfs3_free_createdata(data);
|
||||||
dprintk("NFS reply mknod: %d\n", status);
|
dprintk("NFS reply mknod: %d\n", status);
|
||||||
|
@ -904,20 +924,28 @@ static const struct inode_operations nfs3_dir_inode_operations = {
|
||||||
.permission = nfs_permission,
|
.permission = nfs_permission,
|
||||||
.getattr = nfs_getattr,
|
.getattr = nfs_getattr,
|
||||||
.setattr = nfs_setattr,
|
.setattr = nfs_setattr,
|
||||||
.listxattr = nfs3_listxattr,
|
.listxattr = generic_listxattr,
|
||||||
.getxattr = nfs3_getxattr,
|
.getxattr = generic_getxattr,
|
||||||
.setxattr = nfs3_setxattr,
|
.setxattr = generic_setxattr,
|
||||||
.removexattr = nfs3_removexattr,
|
.removexattr = generic_removexattr,
|
||||||
|
#ifdef CONFIG_NFS_V3_ACL
|
||||||
|
.get_acl = nfs3_get_acl,
|
||||||
|
.set_acl = nfs3_set_acl,
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct inode_operations nfs3_file_inode_operations = {
|
static const struct inode_operations nfs3_file_inode_operations = {
|
||||||
.permission = nfs_permission,
|
.permission = nfs_permission,
|
||||||
.getattr = nfs_getattr,
|
.getattr = nfs_getattr,
|
||||||
.setattr = nfs_setattr,
|
.setattr = nfs_setattr,
|
||||||
.listxattr = nfs3_listxattr,
|
.listxattr = generic_listxattr,
|
||||||
.getxattr = nfs3_getxattr,
|
.getxattr = generic_getxattr,
|
||||||
.setxattr = nfs3_setxattr,
|
.setxattr = generic_setxattr,
|
||||||
.removexattr = nfs3_removexattr,
|
.removexattr = generic_removexattr,
|
||||||
|
#ifdef CONFIG_NFS_V3_ACL
|
||||||
|
.get_acl = nfs3_get_acl,
|
||||||
|
.set_acl = nfs3_set_acl,
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
const struct nfs_rpc_ops nfs_v3_clientops = {
|
const struct nfs_rpc_ops nfs_v3_clientops = {
|
||||||
|
@ -965,7 +993,7 @@ const struct nfs_rpc_ops nfs_v3_clientops = {
|
||||||
.commit_rpc_prepare = nfs3_proc_commit_rpc_prepare,
|
.commit_rpc_prepare = nfs3_proc_commit_rpc_prepare,
|
||||||
.commit_done = nfs3_commit_done,
|
.commit_done = nfs3_commit_done,
|
||||||
.lock = nfs3_proc_lock,
|
.lock = nfs3_proc_lock,
|
||||||
.clear_acl_cache = nfs3_forget_cached_acls,
|
.clear_acl_cache = forget_all_cached_acls,
|
||||||
.close_context = nfs_close_context,
|
.close_context = nfs_close_context,
|
||||||
.have_delegation = nfs3_have_delegation,
|
.have_delegation = nfs3_have_delegation,
|
||||||
.return_delegation = nfs3_return_delegation,
|
.return_delegation = nfs3_return_delegation,
|
||||||
|
|
|
@ -12,6 +12,9 @@ static struct nfs_subversion nfs_v3 = {
|
||||||
.rpc_vers = &nfs_version3,
|
.rpc_vers = &nfs_version3,
|
||||||
.rpc_ops = &nfs_v3_clientops,
|
.rpc_ops = &nfs_v3_clientops,
|
||||||
.sops = &nfs_sops,
|
.sops = &nfs_sops,
|
||||||
|
#ifdef CONFIG_NFS_V3_ACL
|
||||||
|
.xattr = nfs3_xattr_handlers,
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
static int __init init_nfs_v3(void)
|
static int __init init_nfs_v3(void)
|
||||||
|
|
|
@ -154,10 +154,6 @@ struct nfs_inode {
|
||||||
struct rb_root access_cache;
|
struct rb_root access_cache;
|
||||||
struct list_head access_cache_entry_lru;
|
struct list_head access_cache_entry_lru;
|
||||||
struct list_head access_cache_inode_lru;
|
struct list_head access_cache_inode_lru;
|
||||||
#ifdef CONFIG_NFS_V3_ACL
|
|
||||||
struct posix_acl *acl_access;
|
|
||||||
struct posix_acl *acl_default;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This is the cookie verifier used for NFSv3 readdir
|
* This is the cookie verifier used for NFSv3 readdir
|
||||||
|
@ -564,23 +560,17 @@ extern int nfs_readpage_async(struct nfs_open_context *, struct inode *,
|
||||||
* linux/fs/nfs3proc.c
|
* linux/fs/nfs3proc.c
|
||||||
*/
|
*/
|
||||||
#ifdef CONFIG_NFS_V3_ACL
|
#ifdef CONFIG_NFS_V3_ACL
|
||||||
extern struct posix_acl *nfs3_proc_getacl(struct inode *inode, int type);
|
extern struct posix_acl *nfs3_get_acl(struct inode *inode, int type);
|
||||||
extern int nfs3_proc_setacl(struct inode *inode, int type,
|
extern int nfs3_set_acl(struct inode *inode, struct posix_acl *acl, int type);
|
||||||
struct posix_acl *acl);
|
extern int nfs3_proc_setacls(struct inode *inode, struct posix_acl *acl,
|
||||||
extern int nfs3_proc_set_default_acl(struct inode *dir, struct inode *inode,
|
struct posix_acl *dfacl);
|
||||||
umode_t mode);
|
extern const struct xattr_handler *nfs3_xattr_handlers[];
|
||||||
extern void nfs3_forget_cached_acls(struct inode *inode);
|
|
||||||
#else
|
#else
|
||||||
static inline int nfs3_proc_set_default_acl(struct inode *dir,
|
static inline int nfs3_proc_setacls(struct inode *inode, struct posix_acl *acl,
|
||||||
struct inode *inode,
|
struct posix_acl *dfacl)
|
||||||
umode_t mode)
|
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void nfs3_forget_cached_acls(struct inode *inode)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
#endif /* CONFIG_NFS_V3_ACL */
|
#endif /* CONFIG_NFS_V3_ACL */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -191,6 +191,10 @@ static inline int posix_acl_create(struct inode *inode, umode_t *mode,
|
||||||
*default_acl = *acl = NULL;
|
*default_acl = *acl = NULL;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void forget_all_cached_acls(struct inode *inode)
|
||||||
|
{
|
||||||
|
}
|
||||||
#endif /* CONFIG_FS_POSIX_ACL */
|
#endif /* CONFIG_FS_POSIX_ACL */
|
||||||
|
|
||||||
struct posix_acl *get_acl(struct inode *inode, int type);
|
struct posix_acl *get_acl(struct inode *inode, int type);
|
||||||
|
|
Loading…
Reference in New Issue