mirror of
https://github.com/torvalds/linux.git
synced 2025-01-01 15:51:46 +00:00
nfsd: initialize i_private before d_add
A process could race in an open and attempt to read one of these files before i_private is initialized, and get a spurious error. Reported-by: Al Viro <viro@zeniv.linux.org.uk> Signed-off-by: J. Bruce Fields <bfields@redhat.com>
This commit is contained in:
parent
dc46bba709
commit
bebd699716
@ -1171,13 +1171,17 @@ static struct inode *nfsd_get_inode(struct super_block *sb, umode_t mode)
|
|||||||
return inode;
|
return inode;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int __nfsd_mkdir(struct inode *dir, struct dentry *dentry, umode_t mode)
|
static int __nfsd_mkdir(struct inode *dir, struct dentry *dentry, umode_t mode, struct nfsdfs_client *ncl)
|
||||||
{
|
{
|
||||||
struct inode *inode;
|
struct inode *inode;
|
||||||
|
|
||||||
inode = nfsd_get_inode(dir->i_sb, mode);
|
inode = nfsd_get_inode(dir->i_sb, mode);
|
||||||
if (!inode)
|
if (!inode)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
if (ncl) {
|
||||||
|
inode->i_private = ncl;
|
||||||
|
kref_get(&ncl->cl_ref);
|
||||||
|
}
|
||||||
d_add(dentry, inode);
|
d_add(dentry, inode);
|
||||||
inc_nlink(dir);
|
inc_nlink(dir);
|
||||||
fsnotify_mkdir(dir, dentry);
|
fsnotify_mkdir(dir, dentry);
|
||||||
@ -1194,13 +1198,9 @@ static struct dentry *nfsd_mkdir(struct dentry *parent, struct nfsdfs_client *nc
|
|||||||
dentry = d_alloc_name(parent, name);
|
dentry = d_alloc_name(parent, name);
|
||||||
if (!dentry)
|
if (!dentry)
|
||||||
goto out_err;
|
goto out_err;
|
||||||
ret = __nfsd_mkdir(d_inode(parent), dentry, S_IFDIR | 0600);
|
ret = __nfsd_mkdir(d_inode(parent), dentry, S_IFDIR | 0600, ncl);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out_err;
|
goto out_err;
|
||||||
if (ncl) {
|
|
||||||
d_inode(dentry)->i_private = ncl;
|
|
||||||
kref_get(&ncl->cl_ref);
|
|
||||||
}
|
|
||||||
out:
|
out:
|
||||||
inode_unlock(dir);
|
inode_unlock(dir);
|
||||||
return dentry;
|
return dentry;
|
||||||
|
Loading…
Reference in New Issue
Block a user