Commit 3f29cc82 authored by NeilBrown's avatar NeilBrown Committed by Chuck Lever

nfsd: split sc_status out of sc_type

sc_type identifies the type of a state - open, lock, deleg, layout - and
also the status of a state - closed or revoked.

This is a bit untidy and could get worse when "admin-revoked" states are
added.  So clean it up.

With this patch, the type is now all that is stored in sc_type.  This is
zero when the state is first added to ->cl_stateids (causing it to be
ignored), and is then set appropriately once it is fully initialised.
It is set under ->cl_lock to ensure atomicity w.r.t lookup.  It is now
never cleared.

sc_type is still a bit-set even though at most one bit is set.  This allows
lookup functions to be given a bitmap of acceptable types.

sc_type is now an unsigned short rather than char.  There is no value in
restricting to just 8 bits.

All the constants now start SC_TYPE_ matching the field in which they
are stored.  Keeping the existing names and ensuring clear separation
from non-type flags would have required something like
NFS4_STID_TYPE_CLOSED which is cumbersome.  The "NFS4" prefix is
redundant was they only appear in NFS4 code, so remove that and change
STID to SC to match the field.

The status is stored in a separate unsigned short named "sc_status".  It
has two flags: SC_STATUS_CLOSED and SC_STATUS_REVOKED.
CLOSED combines NFS4_CLOSED_STID, NFS4_CLOSED_DELEG_STID, and is used
for SC_TYPE_LOCK and SC_TYPE_LAYOUT instead of setting the sc_type to zero.
These flags are only ever set, never cleared.
For deleg stateids they are set under the global state_lock.
For open and lock stateids they are set under ->cl_lock.
For layout stateids they are set under ->ls_lock

nfs4_unhash_stid() has been removed, and we never set sc_type = 0.  This
was only used for LOCK and LAYOUT stids and they now use
SC_STATUS_CLOSED.

Also TRACE_DEFINE_NUM() calls for the various STID #define have been
removed because these things are not enums, and so that call is
incorrect.
Reviewed-by: default avatarJeff Layton <jlayton@kernel.org>
Signed-off-by: default avatarNeilBrown <neilb@suse.de>
Signed-off-by: default avatarChuck Lever <chuck.lever@oracle.com>
parent 83e73316
...@@ -236,7 +236,7 @@ nfsd4_alloc_layout_stateid(struct nfsd4_compound_state *cstate, ...@@ -236,7 +236,7 @@ nfsd4_alloc_layout_stateid(struct nfsd4_compound_state *cstate,
nfsd4_init_cb(&ls->ls_recall, clp, &nfsd4_cb_layout_ops, nfsd4_init_cb(&ls->ls_recall, clp, &nfsd4_cb_layout_ops,
NFSPROC4_CLNT_CB_LAYOUT); NFSPROC4_CLNT_CB_LAYOUT);
if (parent->sc_type == NFS4_DELEG_STID) if (parent->sc_type == SC_TYPE_DELEG)
ls->ls_file = nfsd_file_get(fp->fi_deleg_file); ls->ls_file = nfsd_file_get(fp->fi_deleg_file);
else else
ls->ls_file = find_any_file(fp); ls->ls_file = find_any_file(fp);
...@@ -250,7 +250,7 @@ nfsd4_alloc_layout_stateid(struct nfsd4_compound_state *cstate, ...@@ -250,7 +250,7 @@ nfsd4_alloc_layout_stateid(struct nfsd4_compound_state *cstate,
} }
spin_lock(&clp->cl_lock); spin_lock(&clp->cl_lock);
stp->sc_type = NFS4_LAYOUT_STID; stp->sc_type = SC_TYPE_LAYOUT;
list_add(&ls->ls_perclnt, &clp->cl_lo_states); list_add(&ls->ls_perclnt, &clp->cl_lo_states);
spin_unlock(&clp->cl_lock); spin_unlock(&clp->cl_lock);
...@@ -269,13 +269,13 @@ nfsd4_preprocess_layout_stateid(struct svc_rqst *rqstp, ...@@ -269,13 +269,13 @@ nfsd4_preprocess_layout_stateid(struct svc_rqst *rqstp,
{ {
struct nfs4_layout_stateid *ls; struct nfs4_layout_stateid *ls;
struct nfs4_stid *stid; struct nfs4_stid *stid;
unsigned char typemask = NFS4_LAYOUT_STID; unsigned short typemask = SC_TYPE_LAYOUT;
__be32 status; __be32 status;
if (create) if (create)
typemask |= (NFS4_OPEN_STID | NFS4_LOCK_STID | NFS4_DELEG_STID); typemask |= (SC_TYPE_OPEN | SC_TYPE_LOCK | SC_TYPE_DELEG);
status = nfsd4_lookup_stateid(cstate, stateid, typemask, &stid, status = nfsd4_lookup_stateid(cstate, stateid, typemask, 0, &stid,
net_generic(SVC_NET(rqstp), nfsd_net_id)); net_generic(SVC_NET(rqstp), nfsd_net_id));
if (status) if (status)
goto out; goto out;
...@@ -286,7 +286,7 @@ nfsd4_preprocess_layout_stateid(struct svc_rqst *rqstp, ...@@ -286,7 +286,7 @@ nfsd4_preprocess_layout_stateid(struct svc_rqst *rqstp,
goto out_put_stid; goto out_put_stid;
} }
if (stid->sc_type != NFS4_LAYOUT_STID) { if (stid->sc_type != SC_TYPE_LAYOUT) {
ls = nfsd4_alloc_layout_stateid(cstate, stid, layout_type); ls = nfsd4_alloc_layout_stateid(cstate, stid, layout_type);
nfs4_put_stid(stid); nfs4_put_stid(stid);
...@@ -518,7 +518,7 @@ nfsd4_return_file_layouts(struct svc_rqst *rqstp, ...@@ -518,7 +518,7 @@ nfsd4_return_file_layouts(struct svc_rqst *rqstp,
lrp->lrs_present = true; lrp->lrs_present = true;
} else { } else {
trace_nfsd_layoutstate_unhash(&ls->ls_stid.sc_stateid); trace_nfsd_layoutstate_unhash(&ls->ls_stid.sc_stateid);
nfs4_unhash_stid(&ls->ls_stid); ls->ls_stid.sc_status |= SC_STATUS_CLOSED;
lrp->lrs_present = false; lrp->lrs_present = false;
} }
spin_unlock(&ls->ls_lock); spin_unlock(&ls->ls_lock);
......
This diff is collapsed.
...@@ -88,17 +88,33 @@ struct nfsd4_callback_ops { ...@@ -88,17 +88,33 @@ struct nfsd4_callback_ops {
*/ */
struct nfs4_stid { struct nfs4_stid {
refcount_t sc_count; refcount_t sc_count;
#define NFS4_OPEN_STID 1
#define NFS4_LOCK_STID 2 /* A new stateid is added to the cl_stateids idr early before it
#define NFS4_DELEG_STID 4 * is fully initialised. Its sc_type is then zero. After
/* For an open stateid kept around *only* to process close replays: */ * initialisation the sc_type it set under cl_lock, and then
#define NFS4_CLOSED_STID 8 * never changes.
*/
#define SC_TYPE_OPEN BIT(0)
#define SC_TYPE_LOCK BIT(1)
#define SC_TYPE_DELEG BIT(2)
#define SC_TYPE_LAYOUT BIT(3)
unsigned short sc_type;
/* state_lock protects sc_status for delegation stateids.
* ->cl_lock protects sc_status for open and lock stateids.
* ->st_mutex also protect sc_status for open stateids.
* ->ls_lock protects sc_status for layout stateids.
*/
/*
* For an open stateid kept around *only* to process close replays.
* For deleg stateid, kept in idr until last reference is dropped.
*/
#define SC_STATUS_CLOSED BIT(0)
/* For a deleg stateid kept around only to process free_stateid's: */ /* For a deleg stateid kept around only to process free_stateid's: */
#define NFS4_REVOKED_DELEG_STID 16 #define SC_STATUS_REVOKED BIT(1)
#define NFS4_CLOSED_DELEG_STID 32 unsigned short sc_status;
#define NFS4_LAYOUT_STID 64
struct list_head sc_cp_list; struct list_head sc_cp_list;
unsigned char sc_type;
stateid_t sc_stateid; stateid_t sc_stateid;
spinlock_t sc_lock; spinlock_t sc_lock;
struct nfs4_client *sc_client; struct nfs4_client *sc_client;
...@@ -672,15 +688,15 @@ extern __be32 nfs4_preprocess_stateid_op(struct svc_rqst *rqstp, ...@@ -672,15 +688,15 @@ extern __be32 nfs4_preprocess_stateid_op(struct svc_rqst *rqstp,
stateid_t *stateid, int flags, struct nfsd_file **filp, stateid_t *stateid, int flags, struct nfsd_file **filp,
struct nfs4_stid **cstid); struct nfs4_stid **cstid);
__be32 nfsd4_lookup_stateid(struct nfsd4_compound_state *cstate, __be32 nfsd4_lookup_stateid(struct nfsd4_compound_state *cstate,
stateid_t *stateid, unsigned char typemask, stateid_t *stateid, unsigned short typemask,
struct nfs4_stid **s, struct nfsd_net *nn); unsigned short statusmask,
struct nfs4_stid **s, struct nfsd_net *nn);
struct nfs4_stid *nfs4_alloc_stid(struct nfs4_client *cl, struct kmem_cache *slab, struct nfs4_stid *nfs4_alloc_stid(struct nfs4_client *cl, struct kmem_cache *slab,
void (*sc_free)(struct nfs4_stid *)); void (*sc_free)(struct nfs4_stid *));
int nfs4_init_copy_state(struct nfsd_net *nn, struct nfsd4_copy *copy); int nfs4_init_copy_state(struct nfsd_net *nn, struct nfsd4_copy *copy);
void nfs4_free_copy_state(struct nfsd4_copy *copy); void nfs4_free_copy_state(struct nfsd4_copy *copy);
struct nfs4_cpntf_state *nfs4_alloc_init_cpntf_state(struct nfsd_net *nn, struct nfs4_cpntf_state *nfs4_alloc_init_cpntf_state(struct nfsd_net *nn,
struct nfs4_stid *p_stid); struct nfs4_stid *p_stid);
void nfs4_unhash_stid(struct nfs4_stid *s);
void nfs4_put_stid(struct nfs4_stid *s); void nfs4_put_stid(struct nfs4_stid *s);
void nfs4_inc_and_copy_stateid(stateid_t *dst, struct nfs4_stid *stid); void nfs4_inc_and_copy_stateid(stateid_t *dst, struct nfs4_stid *stid);
void nfs4_remove_reclaim_record(struct nfs4_client_reclaim *, struct nfsd_net *); void nfs4_remove_reclaim_record(struct nfs4_client_reclaim *, struct nfsd_net *);
......
...@@ -643,23 +643,17 @@ DEFINE_EVENT(nfsd_stateseqid_class, nfsd_##name, \ ...@@ -643,23 +643,17 @@ DEFINE_EVENT(nfsd_stateseqid_class, nfsd_##name, \
DEFINE_STATESEQID_EVENT(preprocess); DEFINE_STATESEQID_EVENT(preprocess);
DEFINE_STATESEQID_EVENT(open_confirm); DEFINE_STATESEQID_EVENT(open_confirm);
TRACE_DEFINE_ENUM(NFS4_OPEN_STID);
TRACE_DEFINE_ENUM(NFS4_LOCK_STID);
TRACE_DEFINE_ENUM(NFS4_DELEG_STID);
TRACE_DEFINE_ENUM(NFS4_CLOSED_STID);
TRACE_DEFINE_ENUM(NFS4_REVOKED_DELEG_STID);
TRACE_DEFINE_ENUM(NFS4_CLOSED_DELEG_STID);
TRACE_DEFINE_ENUM(NFS4_LAYOUT_STID);
#define show_stid_type(x) \ #define show_stid_type(x) \
__print_flags(x, "|", \ __print_flags(x, "|", \
{ NFS4_OPEN_STID, "OPEN" }, \ { SC_TYPE_OPEN, "OPEN" }, \
{ NFS4_LOCK_STID, "LOCK" }, \ { SC_TYPE_LOCK, "LOCK" }, \
{ NFS4_DELEG_STID, "DELEG" }, \ { SC_TYPE_DELEG, "DELEG" }, \
{ NFS4_CLOSED_STID, "CLOSED" }, \ { SC_TYPE_LAYOUT, "LAYOUT" })
{ NFS4_REVOKED_DELEG_STID, "REVOKED" }, \
{ NFS4_CLOSED_DELEG_STID, "CLOSED_DELEG" }, \ #define show_stid_status(x) \
{ NFS4_LAYOUT_STID, "LAYOUT" }) __print_flags(x, "|", \
{ SC_STATUS_CLOSED, "CLOSED" }, \
{ SC_STATUS_REVOKED, "REVOKED" }) \
DECLARE_EVENT_CLASS(nfsd_stid_class, DECLARE_EVENT_CLASS(nfsd_stid_class,
TP_PROTO( TP_PROTO(
...@@ -668,6 +662,7 @@ DECLARE_EVENT_CLASS(nfsd_stid_class, ...@@ -668,6 +662,7 @@ DECLARE_EVENT_CLASS(nfsd_stid_class,
TP_ARGS(stid), TP_ARGS(stid),
TP_STRUCT__entry( TP_STRUCT__entry(
__field(unsigned long, sc_type) __field(unsigned long, sc_type)
__field(unsigned long, sc_status)
__field(int, sc_count) __field(int, sc_count)
__field(u32, cl_boot) __field(u32, cl_boot)
__field(u32, cl_id) __field(u32, cl_id)
...@@ -678,16 +673,18 @@ DECLARE_EVENT_CLASS(nfsd_stid_class, ...@@ -678,16 +673,18 @@ DECLARE_EVENT_CLASS(nfsd_stid_class,
const stateid_t *stp = &stid->sc_stateid; const stateid_t *stp = &stid->sc_stateid;
__entry->sc_type = stid->sc_type; __entry->sc_type = stid->sc_type;
__entry->sc_status = stid->sc_status;
__entry->sc_count = refcount_read(&stid->sc_count); __entry->sc_count = refcount_read(&stid->sc_count);
__entry->cl_boot = stp->si_opaque.so_clid.cl_boot; __entry->cl_boot = stp->si_opaque.so_clid.cl_boot;
__entry->cl_id = stp->si_opaque.so_clid.cl_id; __entry->cl_id = stp->si_opaque.so_clid.cl_id;
__entry->si_id = stp->si_opaque.so_id; __entry->si_id = stp->si_opaque.so_id;
__entry->si_generation = stp->si_generation; __entry->si_generation = stp->si_generation;
), ),
TP_printk("client %08x:%08x stateid %08x:%08x ref=%d type=%s", TP_printk("client %08x:%08x stateid %08x:%08x ref=%d type=%s state=%s",
__entry->cl_boot, __entry->cl_id, __entry->cl_boot, __entry->cl_id,
__entry->si_id, __entry->si_generation, __entry->si_id, __entry->si_generation,
__entry->sc_count, show_stid_type(__entry->sc_type) __entry->sc_count, show_stid_type(__entry->sc_type),
show_stid_status(__entry->sc_status)
) )
); );
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment