Re: [PATCH] SUNRPC: Fix locking around callback channel reply receive

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

 



On Wed, 12 Nov 2014 18:04:04 -0500
Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx> wrote:

> Both xprt_lookup_rqst() and xprt_complete_rqst() require that you
> take the transport lock in order to avoid races with xprt_transmit().
> 
> Signed-off-by: Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx>
> Cc: Bruce Fields <bfields@xxxxxxxxxxxx>
> ---
>  net/sunrpc/svcsock.c | 27 ++++++++++++++++-----------
>  1 file changed, 16 insertions(+), 11 deletions(-)
> 
> diff --git a/net/sunrpc/svcsock.c b/net/sunrpc/svcsock.c
> index 3f959c681885..f9c052d508f0 100644
> --- a/net/sunrpc/svcsock.c
> +++ b/net/sunrpc/svcsock.c
> @@ -1019,17 +1019,12 @@ static int receive_cb_reply(struct svc_sock *svsk, struct svc_rqst *rqstp)
>  	xid = *p++;
>  	calldir = *p;
>  
> -	if (bc_xprt)
> -		req = xprt_lookup_rqst(bc_xprt, xid);
> -
> -	if (!req) {
> -		printk(KERN_NOTICE
> -			"%s: Got unrecognized reply: "
> -			"calldir 0x%x xpt_bc_xprt %p xid %08x\n",
> -			__func__, ntohl(calldir),
> -			bc_xprt, ntohl(xid));
> +	if (!bc_xprt)
>  		return -EAGAIN;
> -	}
> +	spin_lock_bh(&bc_xprt->transport_lock);
> +	req = xprt_lookup_rqst(bc_xprt, xid);
> +	if (!req)
> +		goto unlock_notfound;
>  
>  	memcpy(&req->rq_private_buf, &req->rq_rcv_buf, sizeof(struct xdr_buf));
>  	/*
> @@ -1040,11 +1035,21 @@ static int receive_cb_reply(struct svc_sock *svsk, struct svc_rqst *rqstp)
>  	dst = &req->rq_private_buf.head[0];
>  	src = &rqstp->rq_arg.head[0];
>  	if (dst->iov_len < src->iov_len)
> -		return -EAGAIN; /* whatever; just giving up. */
> +		goto unlock_eagain; /* whatever; just giving up. */
>  	memcpy(dst->iov_base, src->iov_base, src->iov_len);
>  	xprt_complete_rqst(req->rq_task, rqstp->rq_arg.len);
>  	rqstp->rq_arg.len = 0;
> +	spin_unlock_bh(&bc_xprt->transport_lock);
>  	return 0;
> +unlock_notfound:
> +	printk(KERN_NOTICE
> +		"%s: Got unrecognized reply: "
> +		"calldir 0x%x xpt_bc_xprt %p xid %08x\n",
> +		__func__, ntohl(calldir),
> +		bc_xprt, ntohl(xid));
> +unlock_eagain:
> +	spin_unlock_bh(&bc_xprt->transport_lock);
> +	return -EAGAIN;
>  }
>  
>  static int copy_pages_to_kvecs(struct kvec *vec, struct page **pages, int len)

Nice catch. It would also be good to pair this with a
lockdep_assert_held() call in xprt_lookup_rqst, but that could be added
in a separate patch.

Reviewed-by: Jeff Layton <jlayton@xxxxxxxxxxxxxxx>
--
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