On Thu, Apr 23, 2009 at 12:42:43PM -0400, andros@xxxxxxxxxx wrote: > From: Andy Adamson <andros@xxxxxxxxxx> > > CREATE_SESSION can be preceeded by a SEQUENCE operation (an embedded > CREATE_SESSION) and the create session single slot cache must be maintained. > Always replay a create session from nfsd4_replay_create_session(). Leave > nfsd4_store_cache_entry() for session (sequence operation) replays. > > Do not set cstate->slot, so that nfs4svc_encode_compoundres session replay > processing is not called. > > Since create sesssion caching and replay all occur in nfsd4_create_session, > no need to set the slot inuse boolean. > > Set cstate->status to a new internal error, nfserr_replay_clientid_cache, and > change the logic in nfsd4_proc_compound so that CREATE_SESSION replays replace > encode_operation. > The new internal error is needed to differentiate between an embedded > CREATE_SESSION replay and a SEQUENCE replay. That seems OK, thanks. --b. > > Signed-off-by: Andy Adamson <andros@xxxxxxxxxx> > --- > fs/nfsd/nfs4proc.c | 9 +++++++-- > fs/nfsd/nfs4state.c | 9 ++------- > fs/nfsd/nfs4xdr.c | 18 ++++++++++++++++++ > include/linux/nfsd/nfsd.h | 2 ++ > include/linux/nfsd/xdr4.h | 2 ++ > 5 files changed, 31 insertions(+), 9 deletions(-) > > diff --git a/fs/nfsd/nfs4proc.c b/fs/nfsd/nfs4proc.c > index b2883e9..32d5866 100644 > --- a/fs/nfsd/nfs4proc.c > +++ b/fs/nfsd/nfs4proc.c > @@ -996,7 +996,7 @@ nfsd4_proc_compound(struct svc_rqst *rqstp, > BUG_ON(op->status == nfs_ok); > > encode_op: > - /* Only from SEQUENCE or CREATE_SESSION */ > + /* Only from SEQUENCE */ > if (resp->cstate.status == nfserr_replay_cache) { > dprintk("%s NFS4.1 replay from cache\n", __func__); > if (nfsd4_not_cached(resp)) > @@ -1005,7 +1005,12 @@ encode_op: > status = op->status; > goto out; > } > - if (op->status == nfserr_replay_me) { > + /* Only from CREATE_SESSION */ > + if (resp->cstate.status == nfserr_replay_clientid_cache) { > + dprintk("%s NFS4.1 replay from clientid cache\n", > + __func__); > + status = op->status; > + } else if (op->status == nfserr_replay_me) { > op->replay = &cstate->replay_owner->so_replay; > nfsd4_encode_replay(resp, op); > status = op->status = op->replay->rp_status; > diff --git a/fs/nfsd/nfs4state.c b/fs/nfsd/nfs4state.c > index 20485c5..a585a58 100644 > --- a/fs/nfsd/nfs4state.c > +++ b/fs/nfsd/nfs4state.c > @@ -1360,10 +1360,9 @@ nfsd4_create_session(struct svc_rqst *rqstp, > if (status == nfserr_replay_cache) { > dprintk("Got a create_session replay! seqid= %d\n", > slot->sl_seqid); > - cstate->slot = slot; > - cstate->status = status; > + cstate->status = nfserr_replay_clientid_cache; > /* Return the cached reply status */ > - status = nfsd4_replay_cache_entry(resp, NULL); > + status = nfsd4_replay_create_session(resp, slot); > goto out; > } else if (cr_ses->seqid != conf->cl_slot.sl_seqid + 1) { > status = nfserr_seq_misordered; > @@ -1420,10 +1419,6 @@ nfsd4_create_session(struct svc_rqst *rqstp, > NFS4_MAX_SESSIONID_LEN); > cr_ses->seqid = slot->sl_seqid; > > - slot->sl_inuse = true; > - cstate->slot = slot; > - /* Ensure a page is used for the cache */ > - slot->sl_cache_entry.ce_cachethis = 1; > out_cache: > /* cache solo and embedded create sessions under the state lock */ > nfsd4_cache_create_session(cr_ses, slot, status); > diff --git a/fs/nfsd/nfs4xdr.c b/fs/nfsd/nfs4xdr.c > index 6b34fac..7459900 100644 > --- a/fs/nfsd/nfs4xdr.c > +++ b/fs/nfsd/nfs4xdr.c > @@ -3065,6 +3065,24 @@ nfsd4_cache_create_session(struct nfsd4_create_session *cr_ses, > entry->ce_datav.iov_len = (char *)resp->p - (char *)p; > } > > +__be32 > +nfsd4_replay_create_session(struct nfsd4_compoundres *resp, > + struct nfsd4_slot *slot) > +{ > + struct nfsd4_cache_entry *entry = &slot->sl_cache_entry; > + ENCODE_HEAD; > + > + RESERVE_SPACE(8); > + WRITE32(OP_CREATE_SESSION); > + *p++ = entry->ce_status; /* already in network byte order */ > + ADJUST_ARGS(); > + > + memcpy(resp->p, entry->ce_datav.iov_base, entry->ce_datav.iov_len); > + p += XDR_QUADLEN(entry->ce_datav.iov_len); > + ADJUST_ARGS(); > + return entry->ce_status; > +} > + > static __be32 > nfsd4_encode_destroy_session(struct nfsd4_compoundres *resp, int nfserr, > struct nfsd4_destroy_session *destroy_session) > diff --git a/include/linux/nfsd/nfsd.h b/include/linux/nfsd/nfsd.h > index 2b49d67..9cd6399 100644 > --- a/include/linux/nfsd/nfsd.h > +++ b/include/linux/nfsd/nfsd.h > @@ -301,6 +301,8 @@ void nfsd_lockd_shutdown(void); > #define nfserr_replay_me cpu_to_be32(11001) > /* nfs41 replay detected */ > #define nfserr_replay_cache cpu_to_be32(11002) > +/* nfs41 clientid cache replay detected */ > +#define nfserr_replay_clientid_cache cpu_to_be32(11003) > > /* Check for dir entries '.' and '..' */ > #define isdotent(n, l) (l < 3 && n[0] == '.' && (l == 1 || n[1] == '.')) > diff --git a/include/linux/nfsd/xdr4.h b/include/linux/nfsd/xdr4.h > index cc50ca0..8fd5eb1 100644 > --- a/include/linux/nfsd/xdr4.h > +++ b/include/linux/nfsd/xdr4.h > @@ -519,6 +519,8 @@ extern __be32 nfsd4_replay_cache_entry(struct nfsd4_compoundres *resp, > struct nfsd4_sequence *seq); > extern void nfsd4_cache_create_session(struct nfsd4_create_session *cr_ses, > struct nfsd4_slot *slot, int nfserr); > +extern __be32 nfsd4_replay_create_session(struct nfsd4_compoundres *resp, > + struct nfsd4_slot *slot); > extern __be32 nfsd4_exchange_id(struct svc_rqst *rqstp, > struct nfsd4_compound_state *, > struct nfsd4_exchange_id *); > -- > 1.5.4.3 > -- To unsubscribe from this list: send the line "unsubscribe linux-nfs" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html