pnfs: change layout state seqlock to a spinlock
This prepares for future changes, where the layout state needs to change atomically with several other variables. In particular, it will need to know if lo->segs is empty, as we test that instead of manipulating the NFS_LAYOUT_STATEID_SET bit. Moreover, the layoutstateid is not really a read-mostly structure, as it is written almost as often as it is read. The behavior of pnfs_get_layout_stateid is also slightly changed, so that it no longer changes the stateid. Its name is changed to +pnfs_choose_layoutget_stateid. Signed-off-by: Fred Isaman <iisaman@netapp.com> Signed-off-by: Benny Halevy <bhalevy@panasas.com> Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
This commit is contained in:
parent
b7edfaa198
commit
fd6002e9b8
|
@ -1798,7 +1798,7 @@ encode_layoutget(struct xdr_stream *xdr,
|
||||||
p = xdr_encode_hyper(p, args->range.offset);
|
p = xdr_encode_hyper(p, args->range.offset);
|
||||||
p = xdr_encode_hyper(p, args->range.length);
|
p = xdr_encode_hyper(p, args->range.length);
|
||||||
p = xdr_encode_hyper(p, args->minlength);
|
p = xdr_encode_hyper(p, args->minlength);
|
||||||
pnfs_get_layout_stateid(&stateid, NFS_I(args->inode)->layout,
|
pnfs_choose_layoutget_stateid(&stateid, NFS_I(args->inode)->layout,
|
||||||
args->ctx->state);
|
args->ctx->state);
|
||||||
p = xdr_encode_opaque_fixed(p, &stateid.data, NFS4_STATEID_SIZE);
|
p = xdr_encode_opaque_fixed(p, &stateid.data, NFS4_STATEID_SIZE);
|
||||||
*p = cpu_to_be32(args->maxcount);
|
*p = cpu_to_be32(args->maxcount);
|
||||||
|
|
|
@ -258,9 +258,6 @@ pnfs_clear_lseg_list(struct pnfs_layout_hdr *lo, struct list_head *tmp_list)
|
||||||
/* List does not take a reference, so no need for put here */
|
/* List does not take a reference, so no need for put here */
|
||||||
list_del_init(&lo->plh_layouts);
|
list_del_init(&lo->plh_layouts);
|
||||||
spin_unlock(&clp->cl_lock);
|
spin_unlock(&clp->cl_lock);
|
||||||
write_seqlock(&lo->plh_seqlock);
|
|
||||||
clear_bit(NFS_LAYOUT_STATEID_SET, &lo->plh_flags);
|
|
||||||
write_sequnlock(&lo->plh_seqlock);
|
|
||||||
|
|
||||||
dprintk("%s:Return\n", __func__);
|
dprintk("%s:Return\n", __func__);
|
||||||
}
|
}
|
||||||
|
@ -319,69 +316,40 @@ pnfs_destroy_all_layouts(struct nfs_client *clp)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* update lo->plh_stateid with new if is more recent
|
/* update lo->plh_stateid with new if is more recent */
|
||||||
*
|
|
||||||
* lo->plh_stateid could be the open stateid, in which case we just use what given.
|
|
||||||
*/
|
|
||||||
static void
|
static void
|
||||||
pnfs_set_layout_stateid(struct pnfs_layout_hdr *lo,
|
pnfs_set_layout_stateid(struct pnfs_layout_hdr *lo,
|
||||||
const nfs4_stateid *new)
|
const nfs4_stateid *new)
|
||||||
{
|
{
|
||||||
nfs4_stateid *old = &lo->plh_stateid;
|
u32 oldseq, newseq;
|
||||||
bool overwrite = false;
|
|
||||||
|
|
||||||
write_seqlock(&lo->plh_seqlock);
|
oldseq = be32_to_cpu(lo->plh_stateid.stateid.seqid);
|
||||||
if (!test_bit(NFS_LAYOUT_STATEID_SET, &lo->plh_flags) ||
|
newseq = be32_to_cpu(new->stateid.seqid);
|
||||||
memcmp(old->stateid.other, new->stateid.other, sizeof(new->stateid.other)))
|
if ((int)(newseq - oldseq) > 0)
|
||||||
overwrite = true;
|
memcpy(&lo->plh_stateid, &new->stateid, sizeof(new->stateid));
|
||||||
else {
|
|
||||||
u32 oldseq, newseq;
|
|
||||||
|
|
||||||
oldseq = be32_to_cpu(old->stateid.seqid);
|
|
||||||
newseq = be32_to_cpu(new->stateid.seqid);
|
|
||||||
if ((int)(newseq - oldseq) > 0)
|
|
||||||
overwrite = true;
|
|
||||||
}
|
|
||||||
if (overwrite)
|
|
||||||
memcpy(&old->stateid, &new->stateid, sizeof(new->stateid));
|
|
||||||
write_sequnlock(&lo->plh_seqlock);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
int
|
||||||
pnfs_layout_from_open_stateid(struct pnfs_layout_hdr *lo,
|
pnfs_choose_layoutget_stateid(nfs4_stateid *dst, struct pnfs_layout_hdr *lo,
|
||||||
struct nfs4_state *state)
|
struct nfs4_state *open_state)
|
||||||
{
|
{
|
||||||
int seq;
|
int status = 0;
|
||||||
|
|
||||||
dprintk("--> %s\n", __func__);
|
dprintk("--> %s\n", __func__);
|
||||||
write_seqlock(&lo->plh_seqlock);
|
spin_lock(&lo->plh_inode->i_lock);
|
||||||
do {
|
if (list_empty(&lo->plh_segs)) {
|
||||||
seq = read_seqbegin(&state->seqlock);
|
int seq;
|
||||||
memcpy(lo->plh_stateid.data, state->stateid.data,
|
|
||||||
sizeof(state->stateid.data));
|
|
||||||
} while (read_seqretry(&state->seqlock, seq));
|
|
||||||
set_bit(NFS_LAYOUT_STATEID_SET, &lo->plh_flags);
|
|
||||||
write_sequnlock(&lo->plh_seqlock);
|
|
||||||
dprintk("<-- %s\n", __func__);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
do {
|
||||||
pnfs_get_layout_stateid(nfs4_stateid *dst, struct pnfs_layout_hdr *lo,
|
seq = read_seqbegin(&open_state->seqlock);
|
||||||
struct nfs4_state *open_state)
|
memcpy(dst->data, open_state->stateid.data,
|
||||||
{
|
sizeof(open_state->stateid.data));
|
||||||
int seq;
|
} while (read_seqretry(&open_state->seqlock, seq));
|
||||||
|
} else
|
||||||
dprintk("--> %s\n", __func__);
|
memcpy(dst->data, lo->plh_stateid.data, sizeof(lo->plh_stateid.data));
|
||||||
do {
|
spin_unlock(&lo->plh_inode->i_lock);
|
||||||
seq = read_seqbegin(&lo->plh_seqlock);
|
|
||||||
if (!test_bit(NFS_LAYOUT_STATEID_SET, &lo->plh_flags)) {
|
|
||||||
/* This will trigger retry of the read */
|
|
||||||
pnfs_layout_from_open_stateid(lo, open_state);
|
|
||||||
} else
|
|
||||||
memcpy(dst->data, lo->plh_stateid.data,
|
|
||||||
sizeof(lo->plh_stateid.data));
|
|
||||||
} while (read_seqretry(&lo->plh_seqlock, seq));
|
|
||||||
dprintk("<-- %s\n", __func__);
|
dprintk("<-- %s\n", __func__);
|
||||||
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -496,7 +464,6 @@ alloc_init_layout_hdr(struct inode *ino)
|
||||||
lo->plh_refcount = 1;
|
lo->plh_refcount = 1;
|
||||||
INIT_LIST_HEAD(&lo->plh_layouts);
|
INIT_LIST_HEAD(&lo->plh_layouts);
|
||||||
INIT_LIST_HEAD(&lo->plh_segs);
|
INIT_LIST_HEAD(&lo->plh_segs);
|
||||||
seqlock_init(&lo->plh_seqlock);
|
|
||||||
lo->plh_inode = ino;
|
lo->plh_inode = ino;
|
||||||
return lo;
|
return lo;
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,7 +44,6 @@ struct pnfs_layout_segment {
|
||||||
enum {
|
enum {
|
||||||
NFS_LAYOUT_RO_FAILED = 0, /* get ro layout failed stop trying */
|
NFS_LAYOUT_RO_FAILED = 0, /* get ro layout failed stop trying */
|
||||||
NFS_LAYOUT_RW_FAILED, /* get rw layout failed stop trying */
|
NFS_LAYOUT_RW_FAILED, /* get rw layout failed stop trying */
|
||||||
NFS_LAYOUT_STATEID_SET, /* have a valid layout stateid */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Per-layout driver specific registration structure */
|
/* Per-layout driver specific registration structure */
|
||||||
|
@ -63,7 +62,6 @@ struct pnfs_layout_hdr {
|
||||||
unsigned long plh_refcount;
|
unsigned long plh_refcount;
|
||||||
struct list_head plh_layouts; /* other client layouts */
|
struct list_head plh_layouts; /* other client layouts */
|
||||||
struct list_head plh_segs; /* layout segments list */
|
struct list_head plh_segs; /* layout segments list */
|
||||||
seqlock_t plh_seqlock; /* Protects the stateid */
|
|
||||||
nfs4_stateid plh_stateid;
|
nfs4_stateid plh_stateid;
|
||||||
unsigned long plh_flags;
|
unsigned long plh_flags;
|
||||||
struct inode *plh_inode;
|
struct inode *plh_inode;
|
||||||
|
@ -143,8 +141,9 @@ int pnfs_layout_process(struct nfs4_layoutget *lgp);
|
||||||
void pnfs_destroy_layout(struct nfs_inode *);
|
void pnfs_destroy_layout(struct nfs_inode *);
|
||||||
void pnfs_destroy_all_layouts(struct nfs_client *);
|
void pnfs_destroy_all_layouts(struct nfs_client *);
|
||||||
void put_layout_hdr(struct inode *inode);
|
void put_layout_hdr(struct inode *inode);
|
||||||
void pnfs_get_layout_stateid(nfs4_stateid *dst, struct pnfs_layout_hdr *lo,
|
int pnfs_choose_layoutget_stateid(nfs4_stateid *dst,
|
||||||
struct nfs4_state *open_state);
|
struct pnfs_layout_hdr *lo,
|
||||||
|
struct nfs4_state *open_state);
|
||||||
|
|
||||||
|
|
||||||
static inline int lo_fail_bit(u32 iomode)
|
static inline int lo_fail_bit(u32 iomode)
|
||||||
|
|
Loading…
Reference in New Issue