NFSD: change nfsd_create()/nfsd_symlink() to unlock directory before returning.
[ Upstream commit 927bfc5600cd6333c9ef9f090f19e66b7d4c8ee1 ] nfsd_create() usually returns with the directory still locked. nfsd_symlink() usually returns with it unlocked. This is clumsy. Until recently nfsd_create() needed to keep the directory locked until ACLs and security label had been set. These are now set inside nfsd_create() (in nfsd_setattr()) so this need is gone. So change nfsd_create() and nfsd_symlink() to always unlock, and remove any fh_unlock() calls that follow calls to these functions. Signed-off-by: NeilBrown <neilb@suse.de> [ cel: backported to 5.10.y, prior to idmapped mounts ] Signed-off-by: Chuck Lever <chuck.lever@oracle.com> Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
parent
21505959e5
commit
8b67124316
3 changed files with 21 additions and 21 deletions
|
@ -388,7 +388,6 @@ nfsd3_proc_mkdir(struct svc_rqst *rqstp)
|
|||
fh_init(&resp->fh, NFS3_FHSIZE);
|
||||
resp->status = nfsd_create(rqstp, &resp->dirfh, argp->name, argp->len,
|
||||
&attrs, S_IFDIR, 0, &resp->fh);
|
||||
fh_unlock(&resp->dirfh);
|
||||
return rpc_success;
|
||||
}
|
||||
|
||||
|
@ -469,7 +468,6 @@ nfsd3_proc_mknod(struct svc_rqst *rqstp)
|
|||
type = nfs3_ftypes[argp->ftype];
|
||||
resp->status = nfsd_create(rqstp, &resp->dirfh, argp->name, argp->len,
|
||||
&attrs, type, rdev, &resp->fh);
|
||||
fh_unlock(&resp->dirfh);
|
||||
out:
|
||||
return rpc_success;
|
||||
}
|
||||
|
|
|
@ -823,8 +823,6 @@ nfsd4_create(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
|
|||
create->cr_bmval[2] &= ~FATTR4_WORD2_SECURITY_LABEL;
|
||||
if (attrs.na_aclerr)
|
||||
create->cr_bmval[0] &= ~FATTR4_WORD0_ACL;
|
||||
|
||||
fh_unlock(&cstate->current_fh);
|
||||
set_change_info(&create->cr_cinfo, &cstate->current_fh);
|
||||
fh_dup2(&cstate->current_fh, &resfh);
|
||||
out:
|
||||
|
|
|
@ -1394,8 +1394,10 @@ nfsd_create(struct svc_rqst *rqstp, struct svc_fh *fhp,
|
|||
fh_lock_nested(fhp, I_MUTEX_PARENT);
|
||||
dchild = lookup_one_len(fname, dentry, flen);
|
||||
host_err = PTR_ERR(dchild);
|
||||
if (IS_ERR(dchild))
|
||||
return nfserrno(host_err);
|
||||
if (IS_ERR(dchild)) {
|
||||
err = nfserrno(host_err);
|
||||
goto out_unlock;
|
||||
}
|
||||
err = fh_compose(resfhp, fhp->fh_export, dchild, fhp);
|
||||
/*
|
||||
* We unconditionally drop our ref to dchild as fh_compose will have
|
||||
|
@ -1403,9 +1405,12 @@ nfsd_create(struct svc_rqst *rqstp, struct svc_fh *fhp,
|
|||
*/
|
||||
dput(dchild);
|
||||
if (err)
|
||||
return err;
|
||||
return nfsd_create_locked(rqstp, fhp, fname, flen, attrs, type,
|
||||
goto out_unlock;
|
||||
err = nfsd_create_locked(rqstp, fhp, fname, flen, attrs, type,
|
||||
rdev, resfhp);
|
||||
out_unlock:
|
||||
fh_unlock(fhp);
|
||||
return err;
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -1482,16 +1487,19 @@ nfsd_symlink(struct svc_rqst *rqstp, struct svc_fh *fhp,
|
|||
goto out;
|
||||
|
||||
host_err = fh_want_write(fhp);
|
||||
if (host_err)
|
||||
goto out_nfserr;
|
||||
if (host_err) {
|
||||
err = nfserrno(host_err);
|
||||
goto out;
|
||||
}
|
||||
|
||||
fh_lock(fhp);
|
||||
dentry = fhp->fh_dentry;
|
||||
dnew = lookup_one_len(fname, dentry, flen);
|
||||
host_err = PTR_ERR(dnew);
|
||||
if (IS_ERR(dnew))
|
||||
goto out_nfserr;
|
||||
|
||||
if (IS_ERR(dnew)) {
|
||||
err = nfserrno(PTR_ERR(dnew));
|
||||
fh_unlock(fhp);
|
||||
goto out_drop_write;
|
||||
}
|
||||
host_err = vfs_symlink(d_inode(dentry), dnew, path);
|
||||
err = nfserrno(host_err);
|
||||
cerr = fh_compose(resfhp, fhp->fh_export, dnew, fhp);
|
||||
|
@ -1500,16 +1508,12 @@ nfsd_symlink(struct svc_rqst *rqstp, struct svc_fh *fhp,
|
|||
fh_unlock(fhp);
|
||||
if (!err)
|
||||
err = nfserrno(commit_metadata(fhp));
|
||||
fh_drop_write(fhp);
|
||||
|
||||
dput(dnew);
|
||||
if (err==0) err = cerr;
|
||||
out_drop_write:
|
||||
fh_drop_write(fhp);
|
||||
out:
|
||||
return err;
|
||||
|
||||
out_nfserr:
|
||||
err = nfserrno(host_err);
|
||||
goto out;
|
||||
}
|
||||
|
||||
/*
|
||||
|
|
Loading…
Add table
Reference in a new issue