aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorChuck Lever <chuck.lever@oracle.com>2014-07-29 17:23:25 -0400
committerAnna Schumaker <Anna.Schumaker@Netapp.com>2014-07-31 16:22:52 -0400
commit73806c8832b3438ef0439603dab1f3cfc61cb6cd (patch)
tree8d583ca98b5c68ab0fc89802295ab81bbaf0e590
parent5fc83f470d8ada25927701512cf94a53dab6c4c8 (diff)
xprtrdma: Protect ia->ri_id when unmapping/invalidating MRs
Ensure ia->ri_id remains valid while invoking dma_unmap_page() or posting LOCAL_INV during a transport reconnect. Otherwise, ia->ri_id->device or ia->ri_id->qp is NULL, which triggers a panic. BugLink: https://bugzilla.linux-nfs.org/show_bug.cgi?id=259 Fixes: ec62f40 'xprtrdma: Ensure ia->ri_id->qp is not NULL when reconnecting' Signed-off-by: Chuck Lever <chuck.lever@oracle.com> Tested-by: Steve Wise <swise@opengridcomputing.com> Tested-by: Shirley Ma <shirley.ma@oracle.com> Tested-by: Devesh Sharma <devesh.sharma@emulex.com> Signed-off-by: Anna Schumaker <Anna.Schumaker@Netapp.com>
-rw-r--r--net/sunrpc/xprtrdma/verbs.c23
-rw-r--r--net/sunrpc/xprtrdma/xprt_rdma.h1
2 files changed, 18 insertions, 6 deletions
diff --git a/net/sunrpc/xprtrdma/verbs.c b/net/sunrpc/xprtrdma/verbs.c
index f337bdaa9939..aa08de89de42 100644
--- a/net/sunrpc/xprtrdma/verbs.c
+++ b/net/sunrpc/xprtrdma/verbs.c
@@ -613,6 +613,7 @@ rpcrdma_ia_open(struct rpcrdma_xprt *xprt, struct sockaddr *addr, int memreg)
613 /* Else will do memory reg/dereg for each chunk */ 613 /* Else will do memory reg/dereg for each chunk */
614 ia->ri_memreg_strategy = memreg; 614 ia->ri_memreg_strategy = memreg;
615 615
616 rwlock_init(&ia->ri_qplock);
616 return 0; 617 return 0;
617out2: 618out2:
618 rdma_destroy_id(ia->ri_id); 619 rdma_destroy_id(ia->ri_id);
@@ -859,7 +860,7 @@ rpcrdma_ep_destroy(struct rpcrdma_ep *ep, struct rpcrdma_ia *ia)
859int 860int
860rpcrdma_ep_connect(struct rpcrdma_ep *ep, struct rpcrdma_ia *ia) 861rpcrdma_ep_connect(struct rpcrdma_ep *ep, struct rpcrdma_ia *ia)
861{ 862{
862 struct rdma_cm_id *id; 863 struct rdma_cm_id *id, *old;
863 int rc = 0; 864 int rc = 0;
864 int retry_count = 0; 865 int retry_count = 0;
865 866
@@ -905,9 +906,14 @@ retry:
905 rc = -ENETUNREACH; 906 rc = -ENETUNREACH;
906 goto out; 907 goto out;
907 } 908 }
908 rdma_destroy_qp(ia->ri_id); 909
909 rdma_destroy_id(ia->ri_id); 910 write_lock(&ia->ri_qplock);
911 old = ia->ri_id;
910 ia->ri_id = id; 912 ia->ri_id = id;
913 write_unlock(&ia->ri_qplock);
914
915 rdma_destroy_qp(old);
916 rdma_destroy_id(old);
911 } else { 917 } else {
912 dprintk("RPC: %s: connecting...\n", __func__); 918 dprintk("RPC: %s: connecting...\n", __func__);
913 rc = rdma_create_qp(ia->ri_id, ia->ri_pd, &ep->rep_attr); 919 rc = rdma_create_qp(ia->ri_id, ia->ri_pd, &ep->rep_attr);
@@ -1590,9 +1596,6 @@ rpcrdma_deregister_frmr_external(struct rpcrdma_mr_seg *seg,
1590 struct ib_send_wr invalidate_wr, *bad_wr; 1596 struct ib_send_wr invalidate_wr, *bad_wr;
1591 int rc; 1597 int rc;
1592 1598
1593 while (seg1->mr_nsegs--)
1594 rpcrdma_unmap_one(ia, seg++);
1595
1596 memset(&invalidate_wr, 0, sizeof invalidate_wr); 1599 memset(&invalidate_wr, 0, sizeof invalidate_wr);
1597 invalidate_wr.wr_id = (unsigned long)(void *)seg1->mr_chunk.rl_mw; 1600 invalidate_wr.wr_id = (unsigned long)(void *)seg1->mr_chunk.rl_mw;
1598 invalidate_wr.opcode = IB_WR_LOCAL_INV; 1601 invalidate_wr.opcode = IB_WR_LOCAL_INV;
@@ -1600,7 +1603,11 @@ rpcrdma_deregister_frmr_external(struct rpcrdma_mr_seg *seg,
1600 invalidate_wr.ex.invalidate_rkey = seg1->mr_chunk.rl_mw->r.frmr.fr_mr->rkey; 1603 invalidate_wr.ex.invalidate_rkey = seg1->mr_chunk.rl_mw->r.frmr.fr_mr->rkey;
1601 DECR_CQCOUNT(&r_xprt->rx_ep); 1604 DECR_CQCOUNT(&r_xprt->rx_ep);
1602 1605
1606 read_lock(&ia->ri_qplock);
1607 while (seg1->mr_nsegs--)
1608 rpcrdma_unmap_one(ia, seg++);
1603 rc = ib_post_send(ia->ri_id->qp, &invalidate_wr, &bad_wr); 1609 rc = ib_post_send(ia->ri_id->qp, &invalidate_wr, &bad_wr);
1610 read_unlock(&ia->ri_qplock);
1604 if (rc) 1611 if (rc)
1605 dprintk("RPC: %s: failed ib_post_send for invalidate," 1612 dprintk("RPC: %s: failed ib_post_send for invalidate,"
1606 " status %i\n", __func__, rc); 1613 " status %i\n", __func__, rc);
@@ -1661,8 +1668,10 @@ rpcrdma_deregister_fmr_external(struct rpcrdma_mr_seg *seg,
1661 1668
1662 list_add(&seg1->mr_chunk.rl_mw->r.fmr->list, &l); 1669 list_add(&seg1->mr_chunk.rl_mw->r.fmr->list, &l);
1663 rc = ib_unmap_fmr(&l); 1670 rc = ib_unmap_fmr(&l);
1671 read_lock(&ia->ri_qplock);
1664 while (seg1->mr_nsegs--) 1672 while (seg1->mr_nsegs--)
1665 rpcrdma_unmap_one(ia, seg++); 1673 rpcrdma_unmap_one(ia, seg++);
1674 read_unlock(&ia->ri_qplock);
1666 if (rc) 1675 if (rc)
1667 dprintk("RPC: %s: failed ib_unmap_fmr," 1676 dprintk("RPC: %s: failed ib_unmap_fmr,"
1668 " status %i\n", __func__, rc); 1677 " status %i\n", __func__, rc);
@@ -1718,7 +1727,9 @@ rpcrdma_deregister_external(struct rpcrdma_mr_seg *seg,
1718 1727
1719#if RPCRDMA_PERSISTENT_REGISTRATION 1728#if RPCRDMA_PERSISTENT_REGISTRATION
1720 case RPCRDMA_ALLPHYSICAL: 1729 case RPCRDMA_ALLPHYSICAL:
1730 read_lock(&ia->ri_qplock);
1721 rpcrdma_unmap_one(ia, seg); 1731 rpcrdma_unmap_one(ia, seg);
1732 read_unlock(&ia->ri_qplock);
1722 break; 1733 break;
1723#endif 1734#endif
1724 1735
diff --git a/net/sunrpc/xprtrdma/xprt_rdma.h b/net/sunrpc/xprtrdma/xprt_rdma.h
index 89e7cd479705..97ca516ec619 100644
--- a/net/sunrpc/xprtrdma/xprt_rdma.h
+++ b/net/sunrpc/xprtrdma/xprt_rdma.h
@@ -59,6 +59,7 @@
59 * Interface Adapter -- one per transport instance 59 * Interface Adapter -- one per transport instance
60 */ 60 */
61struct rpcrdma_ia { 61struct rpcrdma_ia {
62 rwlock_t ri_qplock;
62 struct rdma_cm_id *ri_id; 63 struct rdma_cm_id *ri_id;
63 struct ib_pd *ri_pd; 64 struct ib_pd *ri_pd;
64 struct ib_mr *ri_bind_mem; 65 struct ib_mr *ri_bind_mem;