Commit 8487c479 authored by Trond Myklebust's avatar Trond Myklebust

NFSv4: Allow retry of operations that used a returned delegation stateid

Fix up nfs4_do_handle_exception() so that it can check if the operation
that received the NFS4ERR_BAD_STATEID was using a defunct delegation.
Apply that to the case of SETATTR, which will currently return EIO
in some cases where this happens.
Reported-by: default avatarOlga Kornievskaia <kolga@netapp.com>
Signed-off-by: default avatarTrond Myklebust <trond.myklebust@primarydata.com>
parent ca857cc1
...@@ -185,6 +185,7 @@ struct nfs4_state { ...@@ -185,6 +185,7 @@ struct nfs4_state {
struct nfs4_exception { struct nfs4_exception {
struct nfs4_state *state; struct nfs4_state *state;
struct inode *inode; struct inode *inode;
nfs4_stateid *stateid;
long timeout; long timeout;
unsigned char delay : 1, unsigned char delay : 1,
recovering : 1, recovering : 1,
......
...@@ -363,6 +363,7 @@ static int nfs4_do_handle_exception(struct nfs_server *server, ...@@ -363,6 +363,7 @@ static int nfs4_do_handle_exception(struct nfs_server *server,
{ {
struct nfs_client *clp = server->nfs_client; struct nfs_client *clp = server->nfs_client;
struct nfs4_state *state = exception->state; struct nfs4_state *state = exception->state;
const nfs4_stateid *stateid = exception->stateid;
struct inode *inode = exception->inode; struct inode *inode = exception->inode;
int ret = errorcode; int ret = errorcode;
...@@ -376,9 +377,18 @@ static int nfs4_do_handle_exception(struct nfs_server *server, ...@@ -376,9 +377,18 @@ static int nfs4_do_handle_exception(struct nfs_server *server,
case -NFS4ERR_DELEG_REVOKED: case -NFS4ERR_DELEG_REVOKED:
case -NFS4ERR_ADMIN_REVOKED: case -NFS4ERR_ADMIN_REVOKED:
case -NFS4ERR_BAD_STATEID: case -NFS4ERR_BAD_STATEID:
if (inode && nfs_async_inode_return_delegation(inode, if (inode) {
NULL) == 0) int err;
goto wait_on_recovery;
err = nfs_async_inode_return_delegation(inode,
stateid);
if (err == 0)
goto wait_on_recovery;
if (stateid != NULL && stateid->type == NFS4_DELEGATION_STATEID_TYPE) {
exception->retry = 1;
break;
}
}
if (state == NULL) if (state == NULL)
break; break;
ret = nfs4_schedule_stateid_recovery(server, state); ret = nfs4_schedule_stateid_recovery(server, state);
...@@ -2669,28 +2679,17 @@ static struct nfs4_state *nfs4_do_open(struct inode *dir, ...@@ -2669,28 +2679,17 @@ static struct nfs4_state *nfs4_do_open(struct inode *dir,
return res; return res;
} }
static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred, static int _nfs4_do_setattr(struct inode *inode,
struct nfs_fattr *fattr, struct iattr *sattr, struct nfs_setattrargs *arg,
struct nfs4_state *state, struct nfs4_label *ilabel, struct nfs_setattrres *res,
struct nfs4_label *olabel) struct rpc_cred *cred,
struct nfs4_state *state)
{ {
struct nfs_server *server = NFS_SERVER(inode); struct nfs_server *server = NFS_SERVER(inode);
struct nfs_setattrargs arg = {
.fh = NFS_FH(inode),
.iap = sattr,
.server = server,
.bitmask = server->attr_bitmask,
.label = ilabel,
};
struct nfs_setattrres res = {
.fattr = fattr,
.label = olabel,
.server = server,
};
struct rpc_message msg = { struct rpc_message msg = {
.rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_SETATTR], .rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_SETATTR],
.rpc_argp = &arg, .rpc_argp = arg,
.rpc_resp = &res, .rpc_resp = res,
.rpc_cred = cred, .rpc_cred = cred,
}; };
struct rpc_cred *delegation_cred = NULL; struct rpc_cred *delegation_cred = NULL;
...@@ -2699,17 +2698,13 @@ static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred, ...@@ -2699,17 +2698,13 @@ static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred,
bool truncate; bool truncate;
int status; int status;
arg.bitmask = nfs4_bitmask(server, ilabel); nfs_fattr_init(res->fattr);
if (ilabel)
arg.bitmask = nfs4_bitmask(server, olabel);
nfs_fattr_init(fattr);
/* Servers should only apply open mode checks for file size changes */ /* Servers should only apply open mode checks for file size changes */
truncate = (sattr->ia_valid & ATTR_SIZE) ? true : false; truncate = (arg->iap->ia_valid & ATTR_SIZE) ? true : false;
fmode = truncate ? FMODE_WRITE : FMODE_READ; fmode = truncate ? FMODE_WRITE : FMODE_READ;
if (nfs4_copy_delegation_stateid(inode, fmode, &arg.stateid, &delegation_cred)) { if (nfs4_copy_delegation_stateid(inode, fmode, &arg->stateid, &delegation_cred)) {
/* Use that stateid */ /* Use that stateid */
} else if (truncate && state != NULL) { } else if (truncate && state != NULL) {
struct nfs_lockowner lockowner = { struct nfs_lockowner lockowner = {
...@@ -2719,19 +2714,19 @@ static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred, ...@@ -2719,19 +2714,19 @@ static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred,
if (!nfs4_valid_open_stateid(state)) if (!nfs4_valid_open_stateid(state))
return -EBADF; return -EBADF;
if (nfs4_select_rw_stateid(state, FMODE_WRITE, &lockowner, if (nfs4_select_rw_stateid(state, FMODE_WRITE, &lockowner,
&arg.stateid, &delegation_cred) == -EIO) &arg->stateid, &delegation_cred) == -EIO)
return -EBADF; return -EBADF;
} else } else
nfs4_stateid_copy(&arg.stateid, &zero_stateid); nfs4_stateid_copy(&arg->stateid, &zero_stateid);
if (delegation_cred) if (delegation_cred)
msg.rpc_cred = delegation_cred; msg.rpc_cred = delegation_cred;
status = nfs4_call_sync(server->client, server, &msg, &arg.seq_args, &res.seq_res, 1); status = nfs4_call_sync(server->client, server, &msg, &arg->seq_args, &res->seq_res, 1);
put_rpccred(delegation_cred); put_rpccred(delegation_cred);
if (status == 0 && state != NULL) if (status == 0 && state != NULL)
renew_lease(server, timestamp); renew_lease(server, timestamp);
trace_nfs4_setattr(inode, &arg.stateid, status); trace_nfs4_setattr(inode, &arg->stateid, status);
return status; return status;
} }
...@@ -2741,13 +2736,31 @@ static int nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred, ...@@ -2741,13 +2736,31 @@ static int nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred,
struct nfs4_label *olabel) struct nfs4_label *olabel)
{ {
struct nfs_server *server = NFS_SERVER(inode); struct nfs_server *server = NFS_SERVER(inode);
struct nfs_setattrargs arg = {
.fh = NFS_FH(inode),
.iap = sattr,
.server = server,
.bitmask = server->attr_bitmask,
.label = ilabel,
};
struct nfs_setattrres res = {
.fattr = fattr,
.label = olabel,
.server = server,
};
struct nfs4_exception exception = { struct nfs4_exception exception = {
.state = state, .state = state,
.inode = inode, .inode = inode,
.stateid = &arg.stateid,
}; };
int err; int err;
arg.bitmask = nfs4_bitmask(server, ilabel);
if (ilabel)
arg.bitmask = nfs4_bitmask(server, olabel);
do { do {
err = _nfs4_do_setattr(inode, cred, fattr, sattr, state, ilabel, olabel); err = _nfs4_do_setattr(inode, &arg, &res, cred, state);
switch (err) { switch (err) {
case -NFS4ERR_OPENMODE: case -NFS4ERR_OPENMODE:
if (!(sattr->ia_valid & ATTR_SIZE)) { if (!(sattr->ia_valid & ATTR_SIZE)) {
......
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