Commit 06e0fa2f authored by Trond Myklebust's avatar Trond Myklebust

NFSv4: Convert the NFSv4 close and open_downgrade operations to use

       asynchronous RPC calls.
Signed-off-by: default avatarTrond Myklebust <trond.myklebust@fys.uio.no>
parent f3115798
...@@ -660,115 +660,100 @@ int nfs4_do_setattr(struct nfs_server *server, struct nfs_fattr *fattr, ...@@ -660,115 +660,100 @@ int nfs4_do_setattr(struct nfs_server *server, struct nfs_fattr *fattr,
return err; return err;
} }
/* struct nfs4_closedata {
* It is possible for data to be read/written from a mem-mapped file struct inode *inode;
* after the sys_close call (which hits the vfs layer as a flush). struct nfs4_state *state;
* This means that we can't safely call nfsv4 close on a file until struct nfs_closeargs arg;
* the inode is cleared. This in turn means that we are not good
* NFSv4 citizens - we do not indicate to the server to update the file's
* share state even when we are done with one of the three share
* stateid's in the inode.
*
* NOTE: Caller must be holding the sp->so_owner semaphore!
*/
static int _nfs4_do_close(struct inode *inode, struct nfs4_state *state)
{
struct nfs4_state_owner *sp = state->owner;
int status = 0;
struct nfs_closeargs arg = {
.fh = NFS_FH(inode),
};
struct nfs_closeres res; struct nfs_closeres res;
struct rpc_message msg = { };
.rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_CLOSE],
.rpc_argp = &arg,
.rpc_resp = &res,
.rpc_cred = sp->so_cred,
};
if (test_bit(NFS_DELEGATED_STATE, &state->flags)) static void nfs4_close_done(struct rpc_task *task)
return 0; {
memcpy(&arg.stateid, &state->stateid, sizeof(arg.stateid)); struct nfs4_closedata *calldata = (struct nfs4_closedata *)task->tk_calldata;
/* Serialization for the sequence id */ struct nfs4_state *state = calldata->state;
arg.seqid = sp->so_seqid, struct nfs4_state_owner *sp = state->owner;
status = rpc_call_sync(NFS_SERVER(inode)->client, &msg, RPC_TASK_NOINTR); struct nfs_server *server = NFS_SERVER(calldata->inode);
/* hmm. we are done with the inode, and in the process of freeing /* hmm. we are done with the inode, and in the process of freeing
* the state_owner. we keep this around to process errors * the state_owner. we keep this around to process errors
*/ */
nfs4_increment_seqid(status, sp); nfs4_increment_seqid(task->tk_status, sp);
if (!status) switch (task->tk_status) {
memcpy(&state->stateid, &res.stateid, sizeof(state->stateid)); case 0:
state->state = calldata->arg.open_flags;
return status; memcpy(&state->stateid, &calldata->res.stateid,
} sizeof(state->stateid));
break;
int nfs4_do_close(struct inode *inode, struct nfs4_state *state)
{
struct nfs_server *server = NFS_SERVER(state->inode);
struct nfs4_exception exception = { };
int err;
do {
err = _nfs4_do_close(inode, state);
switch (err) {
case -NFS4ERR_STALE_STATEID: case -NFS4ERR_STALE_STATEID:
case -NFS4ERR_EXPIRED: case -NFS4ERR_EXPIRED:
state->state = calldata->arg.open_flags;
nfs4_schedule_state_recovery(server->nfs4_state); nfs4_schedule_state_recovery(server->nfs4_state);
err = 0; break;
default: default:
state->state = 0; if (nfs4_async_handle_error(task, server) == -EAGAIN) {
rpc_restart_call(task);
return;
} }
err = nfs4_handle_exception(server, err, &exception); }
} while (exception.retry); nfs4_put_open_state(state);
return err; up(&sp->so_sema);
nfs4_put_state_owner(sp);
up_read(&server->nfs4_state->cl_sem);
kfree(calldata);
} }
static int _nfs4_do_downgrade(struct inode *inode, struct nfs4_state *state, mode_t mode) static inline int nfs4_close_call(struct rpc_clnt *clnt, struct nfs4_closedata *calldata)
{ {
struct nfs4_state_owner *sp = state->owner;
int status = 0;
struct nfs_closeargs arg = {
.fh = NFS_FH(inode),
.seqid = sp->so_seqid,
.open_flags = mode,
};
struct nfs_closeres res;
struct rpc_message msg = { struct rpc_message msg = {
.rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_OPEN_DOWNGRADE], .rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_CLOSE],
.rpc_argp = &arg, .rpc_argp = &calldata->arg,
.rpc_resp = &res, .rpc_resp = &calldata->res,
.rpc_cred = sp->so_cred, .rpc_cred = calldata->state->owner->so_cred,
}; };
if (calldata->arg.open_flags != 0)
if (test_bit(NFS_DELEGATED_STATE, &state->flags)) msg.rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_OPEN_DOWNGRADE];
return 0; return rpc_call_async(clnt, &msg, 0, nfs4_close_done, calldata);
memcpy(&arg.stateid, &state->stateid, sizeof(arg.stateid));
status = rpc_call_sync(NFS_SERVER(inode)->client, &msg, RPC_TASK_NOINTR);
nfs4_increment_seqid(status, sp);
if (!status)
memcpy(&state->stateid, &res.stateid, sizeof(state->stateid));
return status;
} }
int nfs4_do_downgrade(struct inode *inode, struct nfs4_state *state, mode_t mode) /*
* It is possible for data to be read/written from a mem-mapped file
* after the sys_close call (which hits the vfs layer as a flush).
* This means that we can't safely call nfsv4 close on a file until
* the inode is cleared. This in turn means that we are not good
* NFSv4 citizens - we do not indicate to the server to update the file's
* share state even when we are done with one of the three share
* stateid's in the inode.
*
* NOTE: Caller must be holding the sp->so_owner semaphore!
*/
int nfs4_do_close(struct inode *inode, struct nfs4_state *state, mode_t mode)
{ {
struct nfs_server *server = NFS_SERVER(state->inode); struct nfs4_closedata *calldata;
struct nfs4_exception exception = { }; int status;
int err;
do { /* Tell caller we're done */
err = _nfs4_do_downgrade(inode, state, mode); if (test_bit(NFS_DELEGATED_STATE, &state->flags)) {
switch (err) {
case -NFS4ERR_STALE_STATEID:
case -NFS4ERR_EXPIRED:
nfs4_schedule_state_recovery(server->nfs4_state);
err = 0;
default:
state->state = mode; state->state = mode;
return 0;
} }
err = nfs4_handle_exception(server, err, &exception); calldata = (struct nfs4_closedata *)kmalloc(sizeof(*calldata), GFP_KERNEL);
} while (exception.retry); if (calldata == NULL)
return err; return -ENOMEM;
calldata->inode = inode;
calldata->state = state;
calldata->arg.fh = NFS_FH(inode);
/* Serialization for the sequence id */
calldata->arg.seqid = state->owner->so_seqid;
calldata->arg.open_flags = mode;
memcpy(&calldata->arg.stateid, &state->stateid,
sizeof(calldata->arg.stateid));
status = nfs4_close_call(NFS_SERVER(inode)->client, calldata);
/*
* Return -EINPROGRESS on success in order to indicate to the
* caller that an asynchronous RPC call has been launched, and
* that it will release the semaphores on completion.
*/
return (status == 0) ? -EINPROGRESS : status;
} }
struct inode * struct inode *
......
...@@ -445,7 +445,7 @@ nfs4_get_open_state(struct inode *inode, struct nfs4_state_owner *owner) ...@@ -445,7 +445,7 @@ nfs4_get_open_state(struct inode *inode, struct nfs4_state_owner *owner)
state->owner = owner; state->owner = owner;
atomic_inc(&owner->so_count); atomic_inc(&owner->so_count);
list_add(&state->inode_states, &nfsi->open_states); list_add(&state->inode_states, &nfsi->open_states);
state->inode = inode; state->inode = igrab(inode);
spin_unlock(&inode->i_lock); spin_unlock(&inode->i_lock);
} else { } else {
spin_unlock(&inode->i_lock); spin_unlock(&inode->i_lock);
...@@ -471,6 +471,7 @@ void nfs4_put_open_state(struct nfs4_state *state) ...@@ -471,6 +471,7 @@ void nfs4_put_open_state(struct nfs4_state *state)
list_del(&state->inode_states); list_del(&state->inode_states);
spin_unlock(&inode->i_lock); spin_unlock(&inode->i_lock);
list_del(&state->open_states); list_del(&state->open_states);
iput(inode);
BUG_ON (state->state != 0); BUG_ON (state->state != 0);
nfs4_free_open_state(state); nfs4_free_open_state(state);
nfs4_put_state_owner(owner); nfs4_put_state_owner(owner);
...@@ -486,7 +487,6 @@ void nfs4_close_state(struct nfs4_state *state, mode_t mode) ...@@ -486,7 +487,6 @@ void nfs4_close_state(struct nfs4_state *state, mode_t mode)
struct nfs4_state_owner *owner = state->owner; struct nfs4_state_owner *owner = state->owner;
struct nfs4_client *clp = owner->so_client; struct nfs4_client *clp = owner->so_client;
int newstate; int newstate;
int status = 0;
atomic_inc(&owner->so_count); atomic_inc(&owner->so_count);
down_read(&clp->cl_sem); down_read(&clp->cl_sem);
...@@ -508,10 +508,8 @@ void nfs4_close_state(struct nfs4_state *state, mode_t mode) ...@@ -508,10 +508,8 @@ void nfs4_close_state(struct nfs4_state *state, mode_t mode)
newstate |= FMODE_WRITE; newstate |= FMODE_WRITE;
if (state->state == newstate) if (state->state == newstate)
goto out; goto out;
if (newstate != 0) if (nfs4_do_close(inode, state, newstate) == -EINPROGRESS)
status = nfs4_do_downgrade(inode, state, newstate); return;
else
status = nfs4_do_close(inode, state);
} }
out: out:
nfs4_put_open_state(state); nfs4_put_open_state(state);
......
...@@ -651,8 +651,7 @@ extern int nfs4_proc_setclientid_confirm(struct nfs4_client *); ...@@ -651,8 +651,7 @@ extern int nfs4_proc_setclientid_confirm(struct nfs4_client *);
extern int nfs4_open_reclaim(struct nfs4_state_owner *, struct nfs4_state *); extern int nfs4_open_reclaim(struct nfs4_state_owner *, struct nfs4_state *);
extern int nfs4_proc_async_renew(struct nfs4_client *); extern int nfs4_proc_async_renew(struct nfs4_client *);
extern int nfs4_proc_renew(struct nfs4_client *); extern int nfs4_proc_renew(struct nfs4_client *);
extern int nfs4_do_close(struct inode *, struct nfs4_state *); extern int nfs4_do_close(struct inode *inode, struct nfs4_state *state, mode_t mode);
extern int nfs4_do_downgrade(struct inode *inode, struct nfs4_state *state, mode_t mode);
extern int nfs4_wait_clnt_recover(struct rpc_clnt *, struct nfs4_client *); extern int nfs4_wait_clnt_recover(struct rpc_clnt *, struct nfs4_client *);
extern struct inode *nfs4_atomic_open(struct inode *, struct dentry *, struct nameidata *); extern struct inode *nfs4_atomic_open(struct inode *, struct dentry *, struct nameidata *);
extern int nfs4_open_revalidate(struct inode *, struct dentry *, int); extern int nfs4_open_revalidate(struct inode *, struct dentry *, int);
......
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