Jiri Slaby 9aec45
From: Chuck Lever <chuck.lever@oracle.com>
Jiri Slaby 9aec45
Date: Tue, 2 May 2023 14:59:10 -0400
Jiri Slaby 1ef714
Subject: SUNRPC: Fix encoding of accepted but unsuccessful RPC replies
Jiri Slaby 1ef714
Git-commit: 29cd2927fb914cc53b5ba4f67d2b74695c994ba4
Jiri Slaby 1ef714
Patch-mainline: 6.4-rc3
Jiri Slaby 9aec45
References: bsc#1210995
Jiri Slaby 9aec45
Jiri Slaby 9aec45
Jiri Slaby says:
Jiri Slaby 9aec45
> I bisected to this ... as it breaks nfs3-only servers in 6.3.
Jiri Slaby 9aec45
> I.e. /etc/nfs.conf containing:
Jiri Slaby 9aec45
> [nfsd]
Jiri Slaby 9aec45
> vers4=no
Jiri Slaby 9aec45
>
Jiri Slaby 9aec45
> The client sees:
Jiri Slaby 9aec45
>  mount("10.0.2.15:/tmp", "/mnt", "nfs", 0, "vers=4.2,addr=10.0.2.15,clientad"...) = -1 EIO (Input/output error)
Jiri Slaby 9aec45
>  write(2, "mount.nfs: mount system call fai"..., 45
Jiri Slaby 9aec45
>  mount.nfs: mount system call failed for /mnt
Jiri Slaby 9aec45
>
Jiri Slaby 9aec45
> And the kernel says:
Jiri Slaby 9aec45
>  nfs4_discover_server_trunking unhandled error -5. Exiting with error EIO
Jiri Slaby 9aec45
Jiri Slaby 9aec45
Reported-by: Jiri Slaby <jirislaby@kernel.org>
Jiri Slaby 9aec45
Link: https://bugzilla.suse.com/show_bug.cgi?id=1210995
Jiri Slaby 9aec45
Fixes: 4bcf0343e8a6 ("SUNRPC: Set rq_accept_statp inside ->accept methods")
Jiri Slaby 1ef714
Tested-by: Jiri Slaby <jirislaby@kernel.org>
Jiri Slaby 9aec45
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
Jiri Slaby 9aec45
Signed-off-by: Jiri Slaby <jslaby@suse.cz>
Jiri Slaby 9aec45
---
Jiri Slaby 9aec45
 net/sunrpc/svc.c |   17 +++++++++++------
Jiri Slaby 9aec45
 1 file changed, 11 insertions(+), 6 deletions(-)
Jiri Slaby 9aec45
Jiri Slaby 9aec45
--- a/net/sunrpc/svc.c
Jiri Slaby 9aec45
+++ b/net/sunrpc/svc.c
Jiri Slaby 9aec45
@@ -1382,7 +1382,7 @@ err_bad_rpc:
Jiri Slaby 9aec45
 	/* Only RPCv2 supported */
Jiri Slaby 9aec45
 	xdr_stream_encode_u32(xdr, RPC_VERSION);
Jiri Slaby 9aec45
 	xdr_stream_encode_u32(xdr, RPC_VERSION);
Jiri Slaby 9aec45
-	goto sendit;
Jiri Slaby 9aec45
+	return 1;	/* don't wrap */
Jiri Slaby 9aec45
 
Jiri Slaby 9aec45
 err_bad_auth:
Jiri Slaby 9aec45
 	dprintk("svc: authentication failed (%d)\n",
Jiri Slaby 9aec45
@@ -1398,7 +1398,7 @@ err_bad_auth:
Jiri Slaby 9aec45
 err_bad_prog:
Jiri Slaby 9aec45
 	dprintk("svc: unknown program %d\n", rqstp->rq_prog);
Jiri Slaby 9aec45
 	serv->sv_stats->rpcbadfmt++;
Jiri Slaby 9aec45
-	xdr_stream_encode_u32(xdr, RPC_PROG_UNAVAIL);
Jiri Slaby 9aec45
+	*rqstp->rq_accept_statp = rpc_prog_unavail;
Jiri Slaby 9aec45
 	goto sendit;
Jiri Slaby 9aec45
 
Jiri Slaby 9aec45
 err_bad_vers:
Jiri Slaby 9aec45
@@ -1406,7 +1406,12 @@ err_bad_vers:
Jiri Slaby 9aec45
 		       rqstp->rq_vers, rqstp->rq_prog, progp->pg_name);
Jiri Slaby 9aec45
 
Jiri Slaby 9aec45
 	serv->sv_stats->rpcbadfmt++;
Jiri Slaby 9aec45
-	xdr_stream_encode_u32(xdr, RPC_PROG_MISMATCH);
Jiri Slaby 9aec45
+	*rqstp->rq_accept_statp = rpc_prog_mismatch;
Jiri Slaby 9aec45
+
Jiri Slaby 9aec45
+	/*
Jiri Slaby 9aec45
+	 * svc_authenticate() has already added the verifier and
Jiri Slaby 9aec45
+	 * advanced the stream just past rq_accept_statp.
Jiri Slaby 9aec45
+	 */
Jiri Slaby 9aec45
 	xdr_stream_encode_u32(xdr, process.mismatch.lovers);
Jiri Slaby 9aec45
 	xdr_stream_encode_u32(xdr, process.mismatch.hivers);
Jiri Slaby 9aec45
 	goto sendit;
Jiri Slaby 9aec45
@@ -1415,19 +1420,19 @@ err_bad_proc:
Jiri Slaby 9aec45
 	svc_printk(rqstp, "unknown procedure (%d)\n", rqstp->rq_proc);
Jiri Slaby 9aec45
 
Jiri Slaby 9aec45
 	serv->sv_stats->rpcbadfmt++;
Jiri Slaby 9aec45
-	xdr_stream_encode_u32(xdr, RPC_PROC_UNAVAIL);
Jiri Slaby 9aec45
+	*rqstp->rq_accept_statp = rpc_proc_unavail;
Jiri Slaby 9aec45
 	goto sendit;
Jiri Slaby 9aec45
 
Jiri Slaby 9aec45
 err_garbage_args:
Jiri Slaby 9aec45
 	svc_printk(rqstp, "failed to decode RPC header\n");
Jiri Slaby 9aec45
 
Jiri Slaby 9aec45
 	serv->sv_stats->rpcbadfmt++;
Jiri Slaby 9aec45
-	xdr_stream_encode_u32(xdr, RPC_GARBAGE_ARGS);
Jiri Slaby 9aec45
+	*rqstp->rq_accept_statp = rpc_garbage_args;
Jiri Slaby 9aec45
 	goto sendit;
Jiri Slaby 9aec45
 
Jiri Slaby 9aec45
 err_system_err:
Jiri Slaby 9aec45
 	serv->sv_stats->rpcbadfmt++;
Jiri Slaby 9aec45
-	xdr_stream_encode_u32(xdr, RPC_SYSTEM_ERR);
Jiri Slaby 9aec45
+	*rqstp->rq_accept_statp = rpc_system_err;
Jiri Slaby 9aec45
 	goto sendit;
Jiri Slaby 9aec45
 }
Jiri Slaby 9aec45