Re: [PATCH] NFSv4: Allow retry of operations that used a returned delegation stateid

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



> On Jul 19, 2016, at 3:19 PM, Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx> wrote:
> 
> 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: Olga Kornievskaia <kolga@xxxxxxxxxx>
> Signed-off-by: Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx>
> ---
> fs/nfs/nfs4_fs.h  |  1 +
> fs/nfs/nfs4proc.c | 79 ++++++++++++++++++++++++++++++++-----------------------
> 2 files changed, 47 insertions(+), 33 deletions(-)
> 
> diff --git a/fs/nfs/nfs4_fs.h b/fs/nfs/nfs4_fs.h
> index 768456fa1b17..4be567a54958 100644
> --- a/fs/nfs/nfs4_fs.h
> +++ b/fs/nfs/nfs4_fs.h
> @@ -185,6 +185,7 @@ struct nfs4_state {
> struct nfs4_exception {
> 	struct nfs4_state *state;
> 	struct inode *inode;
> +	nfs4_stateid *stateid;
> 	long timeout;
> 	unsigned char delay : 1,
> 		      recovering : 1,
> diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
> index 6191b7e46913..519368b98762 100644
> --- a/fs/nfs/nfs4proc.c
> +++ b/fs/nfs/nfs4proc.c
> @@ -363,6 +363,7 @@ static int nfs4_do_handle_exception(struct nfs_server *server,
> {
> 	struct nfs_client *clp = server->nfs_client;
> 	struct nfs4_state *state = exception->state;
> +	const nfs4_stateid *stateid = exception->stateid;
> 	struct inode *inode = exception->inode;
> 	int ret = errorcode;
> 
> @@ -376,9 +377,18 @@ static int nfs4_do_handle_exception(struct nfs_server *server,
> 		case -NFS4ERR_DELEG_REVOKED:
> 		case -NFS4ERR_ADMIN_REVOKED:
> 		case -NFS4ERR_BAD_STATEID:
> -			if (inode && nfs_async_inode_return_delegation(inode,
> -						NULL) == 0)
> -				goto wait_on_recovery;
> +			if (inode) {
> +				int err;
> +
> +				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)
> 				break;
> 			ret = nfs4_schedule_stateid_recovery(server, state);
> @@ -2669,28 +2679,17 @@ static struct nfs4_state *nfs4_do_open(struct inode *dir,
> 	return res;
> }
> 
> -static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred,
> -			    struct nfs_fattr *fattr, struct iattr *sattr,
> -			    struct nfs4_state *state, struct nfs4_label *ilabel,
> -			    struct nfs4_label *olabel)
> +static int _nfs4_do_setattr(struct inode *inode,
> +			    struct nfs_setattrargs *arg,
> +			    struct nfs_setattrres *res,
> +			    struct rpc_cred *cred,
> +			    struct nfs4_state *state)
> {
> 	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 = {
> 		.rpc_proc	= &nfs4_procedures[NFSPROC4_CLNT_SETATTR],
> -		.rpc_argp	= &arg,
> -		.rpc_resp	= &res,
> +		.rpc_argp	= arg,
> +		.rpc_resp	= res,
> 		.rpc_cred	= cred,
>         };
> 	struct rpc_cred *delegation_cred = NULL;
> @@ -2699,17 +2698,13 @@ static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred,
> 	bool truncate;
> 	int status;
> 
> -	arg.bitmask = nfs4_bitmask(server, ilabel);
> -	if (ilabel)
> -		arg.bitmask = nfs4_bitmask(server, olabel);
> -
> -	nfs_fattr_init(fattr);
> +	nfs_fattr_init(res->fattr);
> 
> 	/* 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;
> 
> -	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 */
> 	} else if (truncate && state != NULL) {
> 		struct nfs_lockowner lockowner = {
> @@ -2719,19 +2714,19 @@ static int _nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred,
> 		if (!nfs4_valid_open_stateid(state))
> 			return -EBADF;
> 		if (nfs4_select_rw_stateid(state, FMODE_WRITE, &lockowner,
> -				&arg.stateid, &delegation_cred) == -EIO)
> +				&arg->stateid, &delegation_cred) == -EIO)
> 			return -EBADF;
> 	} else
> -		nfs4_stateid_copy(&arg.stateid, &zero_stateid);
> +		nfs4_stateid_copy(&arg->stateid, &zero_stateid);
> 	if (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);
> 	if (status == 0 && state != NULL)
> 		renew_lease(server, timestamp);
> -	trace_nfs4_setattr(inode, &arg.stateid, status);
> +	trace_nfs4_setattr(inode, &arg->stateid, status);
> 	return status;
> }
> 
> @@ -2741,13 +2736,31 @@ static int nfs4_do_setattr(struct inode *inode, struct rpc_cred *cred,
> 			   struct nfs4_label *olabel)
> {
> 	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 = {
> 		.state = state,
> 		.inode = inode,
> +		.stateid = &arg.stateid,
> 	};
> 	int err;
> +
> +	arg.bitmask = nfs4_bitmask(server, ilabel);
> +	if (ilabel)
> +		arg.bitmask = nfs4_bitmask(server, olabel);
> +
> 	do {
> -		err = _nfs4_do_setattr(inode, cred, fattr, sattr, state, ilabel, olabel);
> +		err = _nfs4_do_setattr(inode, &arg, &res, cred, state);
> 		switch (err) {
> 		case -NFS4ERR_OPENMODE:
> 			if (!(sattr->ia_valid & ATTR_SIZE)) {
> -- 
> 2.7.4
> 

Thanks Trond. That works. 

--
To unsubscribe from this list: send the line "unsubscribe linux-nfs" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Linux Filesystem Development]     [Linux USB Development]     [Linux Media Development]     [Video for Linux]     [Linux NILFS]     [Linux Audio Users]     [Yosemite Info]     [Linux SCSI]

  Powered by Linux