mirror of https://gitee.com/openkylin/linux.git
xprtrdma: Fix panic in rpcrdma_register_frmr_external()
seg1->mr_nsegs is not yet initialized when it is used to unmap
segments during an error exit. Use the same unmapping logic for
all error exits.
"if (frmr_wr.wr.fast_reg.length < len) {" used to be a BUG_ON check.
The broken code will never be executed under normal operation.
Fixes: c977dea
(xprtrdma: Remove BUG_ON() call sites)
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>
This commit is contained in:
parent
bf858ab0ad
commit
5fc83f470d
|
@ -1548,9 +1548,8 @@ rpcrdma_register_frmr_external(struct rpcrdma_mr_seg *seg,
|
||||||
frmr_wr.wr.fast_reg.page_shift = PAGE_SHIFT;
|
frmr_wr.wr.fast_reg.page_shift = PAGE_SHIFT;
|
||||||
frmr_wr.wr.fast_reg.length = page_no << PAGE_SHIFT;
|
frmr_wr.wr.fast_reg.length = page_no << PAGE_SHIFT;
|
||||||
if (frmr_wr.wr.fast_reg.length < len) {
|
if (frmr_wr.wr.fast_reg.length < len) {
|
||||||
while (seg1->mr_nsegs--)
|
rc = -EIO;
|
||||||
rpcrdma_unmap_one(ia, seg++);
|
goto out_err;
|
||||||
return -EIO;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Bump the key */
|
/* Bump the key */
|
||||||
|
@ -1568,8 +1567,7 @@ rpcrdma_register_frmr_external(struct rpcrdma_mr_seg *seg,
|
||||||
if (rc) {
|
if (rc) {
|
||||||
dprintk("RPC: %s: failed ib_post_send for register,"
|
dprintk("RPC: %s: failed ib_post_send for register,"
|
||||||
" status %i\n", __func__, rc);
|
" status %i\n", __func__, rc);
|
||||||
while (i--)
|
goto out_err;
|
||||||
rpcrdma_unmap_one(ia, --seg);
|
|
||||||
} else {
|
} else {
|
||||||
seg1->mr_rkey = seg1->mr_chunk.rl_mw->r.frmr.fr_mr->rkey;
|
seg1->mr_rkey = seg1->mr_chunk.rl_mw->r.frmr.fr_mr->rkey;
|
||||||
seg1->mr_base = seg1->mr_dma + pageoff;
|
seg1->mr_base = seg1->mr_dma + pageoff;
|
||||||
|
@ -1577,6 +1575,10 @@ rpcrdma_register_frmr_external(struct rpcrdma_mr_seg *seg,
|
||||||
seg1->mr_len = len;
|
seg1->mr_len = len;
|
||||||
}
|
}
|
||||||
*nsegs = i;
|
*nsegs = i;
|
||||||
|
return 0;
|
||||||
|
out_err:
|
||||||
|
while (i--)
|
||||||
|
rpcrdma_unmap_one(ia, --seg);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue