nfsd: nfsd4_process_open2() must reference the delegation stateid

Ensure that nfsd4_process_open2() keeps a reference to the delegation
stateid until it is done working with it. Necessary step toward
client_mutex removal.

Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
This commit is contained in:
Trond Myklebust 2014-07-29 21:34:18 -04:00 committed by J. Bruce Fields
parent 67cb1279be
commit dcd94cc2e7
1 changed files with 13 additions and 5 deletions

View File

@ -3325,6 +3325,8 @@ static struct nfs4_delegation *find_deleg_stateid(struct nfs4_client *cl, statei
ret = find_stateid_by_type(cl, s, NFS4_DELEG_STID); ret = find_stateid_by_type(cl, s, NFS4_DELEG_STID);
if (!ret) if (!ret)
return NULL; return NULL;
/* FIXME: move into find_stateid_by_type */
atomic_inc(&ret->sc_count);
return delegstateid(ret); return delegstateid(ret);
} }
@ -3340,14 +3342,18 @@ nfs4_check_deleg(struct nfs4_client *cl, struct nfsd4_open *open,
{ {
int flags; int flags;
__be32 status = nfserr_bad_stateid; __be32 status = nfserr_bad_stateid;
struct nfs4_delegation *deleg;
*dp = find_deleg_stateid(cl, &open->op_delegate_stateid); deleg = find_deleg_stateid(cl, &open->op_delegate_stateid);
if (*dp == NULL) if (deleg == NULL)
goto out; goto out;
flags = share_access_to_flags(open->op_share_access); flags = share_access_to_flags(open->op_share_access);
status = nfs4_check_delegmode(*dp, flags); status = nfs4_check_delegmode(deleg, flags);
if (status) if (status) {
*dp = NULL; nfs4_put_stid(&deleg->dl_stid);
goto out;
}
*dp = deleg;
out: out:
if (!nfsd4_is_deleg_cur(open)) if (!nfsd4_is_deleg_cur(open))
return nfs_ok; return nfs_ok;
@ -3828,6 +3834,8 @@ out:
if (!(open->op_openowner->oo_flags & NFS4_OO_CONFIRMED) && if (!(open->op_openowner->oo_flags & NFS4_OO_CONFIRMED) &&
!nfsd4_has_session(&resp->cstate)) !nfsd4_has_session(&resp->cstate))
open->op_rflags |= NFS4_OPEN_RESULT_CONFIRM; open->op_rflags |= NFS4_OPEN_RESULT_CONFIRM;
if (dp)
nfs4_put_stid(&dp->dl_stid);
return status; return status;
} }