mirror of
https://github.com/hardkernel/linux.git
synced 2026-03-25 12:00:22 +09:00
SUNRPC: double free xprt_ctxt while still in use
[ Upstream commiteb8d3a2c80] When an RPC request is deferred, the rq_xprt_ctxt pointer is moved out of the svc_rqst into the svc_deferred_req. When the deferred request is revisited, the pointer is copied into the new svc_rqst - and also remains in the svc_deferred_req. In the (rare?) case that the request is deferred a second time, the old svc_deferred_req is reused - it still has all the correct content. However in that case the rq_xprt_ctxt pointer is NOT cleared so that when xpo_release_xprt is called, the ctxt is freed (UDP) or possible added to a free list (RDMA). When the deferred request is revisited for a second time, it will reference this ctxt which may be invalid, and the free the object a second time which is likely to oops. So change svc_defer() to *always* clear rq_xprt_ctxt, and assert that the value is now stored in the svc_deferred_req. Fixes:773f91b2cf("SUNRPC: Fix NFSD's request deferral on RDMA transports") Signed-off-by: NeilBrown <neilb@suse.de> Reviewed-by: Jeff Layton <jlayton@kernel.org> Signed-off-by: Chuck Lever <chuck.lever@oracle.com> Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
051af3f0b7
commit
7851771789
@@ -1214,13 +1214,14 @@ static struct cache_deferred_req *svc_defer(struct cache_req *req)
|
||||
dr->argslen = rqstp->rq_arg.len >> 2;
|
||||
dr->xprt_hlen = rqstp->rq_xprt_hlen;
|
||||
dr->xprt_ctxt = rqstp->rq_xprt_ctxt;
|
||||
rqstp->rq_xprt_ctxt = NULL;
|
||||
|
||||
/* back up head to the start of the buffer and copy */
|
||||
skip = rqstp->rq_arg.len - rqstp->rq_arg.head[0].iov_len;
|
||||
memcpy(dr->args, rqstp->rq_arg.head[0].iov_base - skip,
|
||||
dr->argslen << 2);
|
||||
}
|
||||
WARN_ON_ONCE(rqstp->rq_xprt_ctxt != dr->xprt_ctxt);
|
||||
rqstp->rq_xprt_ctxt = NULL;
|
||||
trace_svc_defer(rqstp);
|
||||
svc_xprt_get(rqstp->rq_xprt);
|
||||
dr->xprt = rqstp->rq_xprt;
|
||||
|
||||
Reference in New Issue
Block a user