Browse Source

IB/srp: Fix srp_map_data() error paths

Ensure that req->nmdesc is set correctly in srp_map_sg() if mapping
fails. Avoid that mapping failure causes a memory descriptor leak.
Report srp_map_sg() failure to the caller.

Signed-off-by: Bart Van Assche <bart.vanassche@sandisk.com>
Cc: Christoph Hellwig <hch@lst.de>
Cc: Sagi Grimberg <sagi@grimberg.me>
Cc: Laurence Oberman <loberman@redhat.com>
Reviewed-by: Sagi Grimberg <sagi@grimberg.me>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Doug Ledford <dledford@redhat.com>
Bart Van Assche 9 years ago
parent
commit
e012f3639c
1 changed files with 13 additions and 11 deletions
  1. 13 11
      drivers/infiniband/ulp/srp/ib_srp.c

+ 13 - 11
drivers/infiniband/ulp/srp/ib_srp.c

@@ -1428,8 +1428,6 @@ static int srp_map_sg_fmr(struct srp_map_state *state, struct srp_rdma_ch *ch,
 	if (ret)
 	if (ret)
 		return ret;
 		return ret;
 
 
-	req->nmdesc = state->nmdesc;
-
 	return 0;
 	return 0;
 }
 }
 
 
@@ -1454,8 +1452,6 @@ static int srp_map_sg_fr(struct srp_map_state *state, struct srp_rdma_ch *ch,
 			state->sg = sg_next(state->sg);
 			state->sg = sg_next(state->sg);
 	}
 	}
 
 
-	req->nmdesc = state->nmdesc;
-
 	return 0;
 	return 0;
 }
 }
 
 
@@ -1475,8 +1471,6 @@ static int srp_map_sg_dma(struct srp_map_state *state, struct srp_rdma_ch *ch,
 			     target->global_mr->rkey);
 			     target->global_mr->rkey);
 	}
 	}
 
 
-	req->nmdesc = state->nmdesc;
-
 	return 0;
 	return 0;
 }
 }
 
 
@@ -1610,11 +1604,14 @@ static int srp_map_data(struct scsi_cmnd *scmnd, struct srp_rdma_ch *ch,
 
 
 	memset(&state, 0, sizeof(state));
 	memset(&state, 0, sizeof(state));
 	if (dev->use_fast_reg)
 	if (dev->use_fast_reg)
-		srp_map_sg_fr(&state, ch, req, scat, count);
+		ret = srp_map_sg_fr(&state, ch, req, scat, count);
 	else if (dev->use_fmr)
 	else if (dev->use_fmr)
-		srp_map_sg_fmr(&state, ch, req, scat, count);
+		ret = srp_map_sg_fmr(&state, ch, req, scat, count);
 	else
 	else
-		srp_map_sg_dma(&state, ch, req, scat, count);
+		ret = srp_map_sg_dma(&state, ch, req, scat, count);
+	req->nmdesc = state.nmdesc;
+	if (ret < 0)
+		goto unmap;
 
 
 	/* We've mapped the request, now pull as much of the indirect
 	/* We've mapped the request, now pull as much of the indirect
 	 * descriptor table as we can into the command buffer. If this
 	 * descriptor table as we can into the command buffer. If this
@@ -1637,7 +1634,8 @@ static int srp_map_data(struct scsi_cmnd *scmnd, struct srp_rdma_ch *ch,
 						!target->allow_ext_sg)) {
 						!target->allow_ext_sg)) {
 		shost_printk(KERN_ERR, target->scsi_host,
 		shost_printk(KERN_ERR, target->scsi_host,
 			     "Could not fit S/G list into SRP_CMD\n");
 			     "Could not fit S/G list into SRP_CMD\n");
-		return -EIO;
+		ret = -EIO;
+		goto unmap;
 	}
 	}
 
 
 	count = min(state.ndesc, target->cmd_sg_cnt);
 	count = min(state.ndesc, target->cmd_sg_cnt);
@@ -1655,7 +1653,7 @@ static int srp_map_data(struct scsi_cmnd *scmnd, struct srp_rdma_ch *ch,
 		ret = srp_map_idb(ch, req, state.gen.next, state.gen.end,
 		ret = srp_map_idb(ch, req, state.gen.next, state.gen.end,
 				  idb_len, &idb_rkey);
 				  idb_len, &idb_rkey);
 		if (ret < 0)
 		if (ret < 0)
-			return ret;
+			goto unmap;
 		req->nmdesc++;
 		req->nmdesc++;
 	} else {
 	} else {
 		idb_rkey = cpu_to_be32(target->global_mr->rkey);
 		idb_rkey = cpu_to_be32(target->global_mr->rkey);
@@ -1681,6 +1679,10 @@ map_complete:
 		cmd->buf_fmt = fmt;
 		cmd->buf_fmt = fmt;
 
 
 	return len;
 	return len;
+
+unmap:
+	srp_unmap_data(scmnd, ch, req);
+	return ret;
 }
 }
 
 
 /*
 /*