Commit 82c5ff1b authored by Jeff Layton's avatar Jeff Layton Committed by J. Bruce Fields

nfsd: wrap accesses to st_access_bmap

Currently, we do this for the most part with "bare" bitops, but
eventually we'll need to expand the share mode code to handle access
and deny modes on other nodes.

In order to facilitate that code in the future, move to some generic
accessor functions. For now, these are mostly static inlines, but
eventually we'll want to move these to "real" functions that are
able to handle multi-node configurations or have a way to "swap in"
new operations to be done in lieu of or in conjunction with these
atomic bitops.
Signed-off-by: default avatarJeff Layton <jlayton@redhat.com>
Signed-off-by: default avatarJ. Bruce Fields <bfields@redhat.com>
parent 3a328614
...@@ -471,6 +471,27 @@ test_share(struct nfs4_ol_stateid *stp, struct nfsd4_open *open) { ...@@ -471,6 +471,27 @@ test_share(struct nfs4_ol_stateid *stp, struct nfsd4_open *open) {
return true; return true;
} }
/* set share access for a given stateid */
static inline void
set_access(u32 access, struct nfs4_ol_stateid *stp)
{
__set_bit(access, &stp->st_access_bmap);
}
/* clear share access for a given stateid */
static inline void
clear_access(u32 access, struct nfs4_ol_stateid *stp)
{
__clear_bit(access, &stp->st_access_bmap);
}
/* test whether a given stateid has access */
static inline bool
test_access(u32 access, struct nfs4_ol_stateid *stp)
{
return test_bit(access, &stp->st_access_bmap);
}
static int nfs4_access_to_omode(u32 access) static int nfs4_access_to_omode(u32 access)
{ {
switch (access & NFS4_SHARE_ACCESS_BOTH) { switch (access & NFS4_SHARE_ACCESS_BOTH) {
...@@ -484,6 +505,20 @@ static int nfs4_access_to_omode(u32 access) ...@@ -484,6 +505,20 @@ static int nfs4_access_to_omode(u32 access)
BUG(); BUG();
} }
/* release all access and file references for a given stateid */
static void
release_all_access(struct nfs4_ol_stateid *stp)
{
int i;
for (i = 1; i < 4; i++) {
if (test_access(i, stp))
nfs4_file_put_access(stp->st_file,
nfs4_access_to_omode(i));
clear_access(i, stp);
}
}
static void unhash_generic_stateid(struct nfs4_ol_stateid *stp) static void unhash_generic_stateid(struct nfs4_ol_stateid *stp)
{ {
list_del(&stp->st_perfile); list_del(&stp->st_perfile);
...@@ -492,16 +527,7 @@ static void unhash_generic_stateid(struct nfs4_ol_stateid *stp) ...@@ -492,16 +527,7 @@ static void unhash_generic_stateid(struct nfs4_ol_stateid *stp)
static void close_generic_stateid(struct nfs4_ol_stateid *stp) static void close_generic_stateid(struct nfs4_ol_stateid *stp)
{ {
int i; release_all_access(stp);
if (stp->st_access_bmap) {
for (i = 1; i < 4; i++) {
if (test_bit(i, &stp->st_access_bmap))
nfs4_file_put_access(stp->st_file,
nfs4_access_to_omode(i));
__clear_bit(i, &stp->st_access_bmap);
}
}
put_nfs4_file(stp->st_file); put_nfs4_file(stp->st_file);
stp->st_file = NULL; stp->st_file = NULL;
} }
...@@ -2435,7 +2461,7 @@ static void init_open_stateid(struct nfs4_ol_stateid *stp, struct nfs4_file *fp, ...@@ -2435,7 +2461,7 @@ static void init_open_stateid(struct nfs4_ol_stateid *stp, struct nfs4_file *fp,
stp->st_file = fp; stp->st_file = fp;
stp->st_access_bmap = 0; stp->st_access_bmap = 0;
stp->st_deny_bmap = 0; stp->st_deny_bmap = 0;
__set_bit(open->op_share_access, &stp->st_access_bmap); set_access(open->op_share_access, stp);
__set_bit(open->op_share_deny, &stp->st_deny_bmap); __set_bit(open->op_share_deny, &stp->st_deny_bmap);
stp->st_openstp = NULL; stp->st_openstp = NULL;
} }
...@@ -2772,7 +2798,7 @@ nfs4_upgrade_open(struct svc_rqst *rqstp, struct nfs4_file *fp, struct svc_fh *c ...@@ -2772,7 +2798,7 @@ nfs4_upgrade_open(struct svc_rqst *rqstp, struct nfs4_file *fp, struct svc_fh *c
bool new_access; bool new_access;
__be32 status; __be32 status;
new_access = !test_bit(op_share_access, &stp->st_access_bmap); new_access = !test_access(op_share_access, stp);
if (new_access) { if (new_access) {
status = nfs4_get_vfs_file(rqstp, fp, cur_fh, open); status = nfs4_get_vfs_file(rqstp, fp, cur_fh, open);
if (status) if (status)
...@@ -2787,7 +2813,7 @@ nfs4_upgrade_open(struct svc_rqst *rqstp, struct nfs4_file *fp, struct svc_fh *c ...@@ -2787,7 +2813,7 @@ nfs4_upgrade_open(struct svc_rqst *rqstp, struct nfs4_file *fp, struct svc_fh *c
return status; return status;
} }
/* remember the open */ /* remember the open */
__set_bit(op_share_access, &stp->st_access_bmap); set_access(op_share_access, stp);
__set_bit(open->op_share_deny, &stp->st_deny_bmap); __set_bit(open->op_share_deny, &stp->st_deny_bmap);
return nfs_ok; return nfs_ok;
...@@ -3263,18 +3289,18 @@ STALE_STATEID(stateid_t *stateid) ...@@ -3263,18 +3289,18 @@ STALE_STATEID(stateid_t *stateid)
} }
static inline int static inline int
access_permit_read(unsigned long access_bmap) access_permit_read(struct nfs4_ol_stateid *stp)
{ {
return test_bit(NFS4_SHARE_ACCESS_READ, &access_bmap) || return test_access(NFS4_SHARE_ACCESS_READ, stp) ||
test_bit(NFS4_SHARE_ACCESS_BOTH, &access_bmap) || test_access(NFS4_SHARE_ACCESS_BOTH, stp) ||
test_bit(NFS4_SHARE_ACCESS_WRITE, &access_bmap); test_access(NFS4_SHARE_ACCESS_WRITE, stp);
} }
static inline int static inline int
access_permit_write(unsigned long access_bmap) access_permit_write(struct nfs4_ol_stateid *stp)
{ {
return test_bit(NFS4_SHARE_ACCESS_WRITE, &access_bmap) || return test_access(NFS4_SHARE_ACCESS_WRITE, stp) ||
test_bit(NFS4_SHARE_ACCESS_BOTH, &access_bmap); test_access(NFS4_SHARE_ACCESS_BOTH, stp);
} }
static static
...@@ -3285,9 +3311,9 @@ __be32 nfs4_check_openmode(struct nfs4_ol_stateid *stp, int flags) ...@@ -3285,9 +3311,9 @@ __be32 nfs4_check_openmode(struct nfs4_ol_stateid *stp, int flags)
/* For lock stateid's, we test the parent open, not the lock: */ /* For lock stateid's, we test the parent open, not the lock: */
if (stp->st_openstp) if (stp->st_openstp)
stp = stp->st_openstp; stp = stp->st_openstp;
if ((flags & WR_STATE) && (!access_permit_write(stp->st_access_bmap))) if ((flags & WR_STATE) && !access_permit_write(stp))
goto out; goto out;
if ((flags & RD_STATE) && (!access_permit_read(stp->st_access_bmap))) if ((flags & RD_STATE) && !access_permit_read(stp))
goto out; goto out;
status = nfs_ok; status = nfs_ok;
out: out:
...@@ -3636,10 +3662,10 @@ nfsd4_open_confirm(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate, ...@@ -3636,10 +3662,10 @@ nfsd4_open_confirm(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
static inline void nfs4_stateid_downgrade_bit(struct nfs4_ol_stateid *stp, u32 access) static inline void nfs4_stateid_downgrade_bit(struct nfs4_ol_stateid *stp, u32 access)
{ {
if (!test_bit(access, &stp->st_access_bmap)) if (!test_access(access, stp))
return; return;
nfs4_file_put_access(stp->st_file, nfs4_access_to_omode(access)); nfs4_file_put_access(stp->st_file, nfs4_access_to_omode(access));
__clear_bit(access, &stp->st_access_bmap); clear_access(access, stp);
} }
static inline void nfs4_stateid_downgrade(struct nfs4_ol_stateid *stp, u32 to_access) static inline void nfs4_stateid_downgrade(struct nfs4_ol_stateid *stp, u32 to_access)
...@@ -3693,8 +3719,8 @@ nfsd4_open_downgrade(struct svc_rqst *rqstp, ...@@ -3693,8 +3719,8 @@ nfsd4_open_downgrade(struct svc_rqst *rqstp,
if (status) if (status)
goto out; goto out;
status = nfserr_inval; status = nfserr_inval;
if (!test_bit(od->od_share_access, &stp->st_access_bmap)) { if (!test_access(od->od_share_access, stp)) {
dprintk("NFSD:access not a subset current bitmap: 0x%lx, input access=%08x\n", dprintk("NFSD: access not a subset current bitmap: 0x%lx, input access=%08x\n",
stp->st_access_bmap, od->od_share_access); stp->st_access_bmap, od->od_share_access);
goto out; goto out;
} }
...@@ -3995,10 +4021,10 @@ static void get_lock_access(struct nfs4_ol_stateid *lock_stp, u32 access) ...@@ -3995,10 +4021,10 @@ static void get_lock_access(struct nfs4_ol_stateid *lock_stp, u32 access)
struct nfs4_file *fp = lock_stp->st_file; struct nfs4_file *fp = lock_stp->st_file;
int oflag = nfs4_access_to_omode(access); int oflag = nfs4_access_to_omode(access);
if (test_bit(access, &lock_stp->st_access_bmap)) if (test_access(access, lock_stp))
return; return;
nfs4_file_get_access(fp, oflag); nfs4_file_get_access(fp, oflag);
__set_bit(access, &lock_stp->st_access_bmap); set_access(access, lock_stp);
} }
static __be32 lookup_or_create_lock_state(struct nfsd4_compound_state *cstate, struct nfs4_ol_stateid *ost, struct nfsd4_lock *lock, struct nfs4_ol_stateid **lst, bool *new) static __be32 lookup_or_create_lock_state(struct nfsd4_compound_state *cstate, struct nfs4_ol_stateid *ost, struct nfsd4_lock *lock, struct nfs4_ol_stateid **lst, bool *new)
......
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