diff options
author | Chuck Lever <chuck.lever@oracle.com> | 2014-07-29 17:23:25 -0400 |
---|---|---|
committer | Anna Schumaker <Anna.Schumaker@Netapp.com> | 2014-07-31 16:22:52 -0400 |
commit | 73806c8832b3438ef0439603dab1f3cfc61cb6cd (patch) | |
tree | 8d583ca98b5c68ab0fc89802295ab81bbaf0e590 | |
parent | 5fc83f470d8ada25927701512cf94a53dab6c4c8 (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.c | 23 | ||||
-rw-r--r-- | net/sunrpc/xprtrdma/xprt_rdma.h | 1 |
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; |
617 | out2: | 618 | out2: |
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) | |||
859 | int | 860 | int |
860 | rpcrdma_ep_connect(struct rpcrdma_ep *ep, struct rpcrdma_ia *ia) | 861 | rpcrdma_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 | */ |
61 | struct rpcrdma_ia { | 61 | struct 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; |