aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/block/cciss.c63
1 files changed, 34 insertions, 29 deletions
diff --git a/drivers/block/cciss.c b/drivers/block/cciss.c
index c0d794ce69c6..9e3af307aae1 100644
--- a/drivers/block/cciss.c
+++ b/drivers/block/cciss.c
@@ -301,6 +301,35 @@ clean:
301 return NULL; 301 return NULL;
302} 302}
303 303
304static void cciss_unmap_sg_chain_block(ctlr_info_t *h, CommandList_struct *c)
305{
306 SGDescriptor_struct *chain_sg;
307 u64bit temp64;
308
309 if (c->Header.SGTotal <= h->max_cmd_sgentries)
310 return;
311
312 chain_sg = &c->SG[h->max_cmd_sgentries - 1];
313 temp64.val32.lower = chain_sg->Addr.lower;
314 temp64.val32.upper = chain_sg->Addr.upper;
315 pci_unmap_single(h->pdev, temp64.val, chain_sg->Len, PCI_DMA_TODEVICE);
316}
317
318static void cciss_map_sg_chain_block(ctlr_info_t *h, CommandList_struct *c,
319 SGDescriptor_struct *chain_block, int len)
320{
321 SGDescriptor_struct *chain_sg;
322 u64bit temp64;
323
324 chain_sg = &c->SG[h->max_cmd_sgentries - 1];
325 chain_sg->Ext = CCISS_SG_CHAIN;
326 chain_sg->Len = len;
327 temp64.val = pci_map_single(h->pdev, chain_block, len,
328 PCI_DMA_TODEVICE);
329 chain_sg->Addr.lower = temp64.val32.lower;
330 chain_sg->Addr.upper = temp64.val32.upper;
331}
332
304#include "cciss_scsi.c" /* For SCSI tape support */ 333#include "cciss_scsi.c" /* For SCSI tape support */
305 334
306static const char *raid_label[] = { "0", "4", "1(1+0)", "5", "5+1", "ADG", 335static const char *raid_label[] = { "0", "4", "1(1+0)", "5", "5+1", "ADG",
@@ -1715,10 +1744,7 @@ static void cciss_softirq_done(struct request *rq)
1715 /* unmap the DMA mapping for all the scatter gather elements */ 1744 /* unmap the DMA mapping for all the scatter gather elements */
1716 for (i = 0; i < cmd->Header.SGList; i++) { 1745 for (i = 0; i < cmd->Header.SGList; i++) {
1717 if (curr_sg[sg_index].Ext == CCISS_SG_CHAIN) { 1746 if (curr_sg[sg_index].Ext == CCISS_SG_CHAIN) {
1718 temp64.val32.lower = cmd->SG[i].Addr.lower; 1747 cciss_unmap_sg_chain_block(h, cmd);
1719 temp64.val32.upper = cmd->SG[i].Addr.upper;
1720 pci_unmap_single(h->pdev, temp64.val,
1721 cmd->SG[i].Len, PCI_DMA_TODEVICE);
1722 /* Point to the next block */ 1748 /* Point to the next block */
1723 curr_sg = h->cmd_sg_list[cmd->cmdindex]; 1749 curr_sg = h->cmd_sg_list[cmd->cmdindex];
1724 sg_index = 0; 1750 sg_index = 0;
@@ -3122,7 +3148,6 @@ static void do_cciss_request(struct request_queue *q)
3122 SGDescriptor_struct *curr_sg; 3148 SGDescriptor_struct *curr_sg;
3123 drive_info_struct *drv; 3149 drive_info_struct *drv;
3124 int i, dir; 3150 int i, dir;
3125 int nseg = 0;
3126 int sg_index = 0; 3151 int sg_index = 0;
3127 int chained = 0; 3152 int chained = 0;
3128 3153
@@ -3189,11 +3214,6 @@ static void do_cciss_request(struct request_queue *q)
3189 for (i = 0; i < seg; i++) { 3214 for (i = 0; i < seg; i++) {
3190 if (((sg_index+1) == (h->max_cmd_sgentries)) && 3215 if (((sg_index+1) == (h->max_cmd_sgentries)) &&
3191 !chained && ((seg - i) > 1)) { 3216 !chained && ((seg - i) > 1)) {
3192 nseg = seg - i;
3193 curr_sg[sg_index].Len = (nseg) *
3194 sizeof(SGDescriptor_struct);
3195 curr_sg[sg_index].Ext = CCISS_SG_CHAIN;
3196
3197 /* Point to next chain block. */ 3217 /* Point to next chain block. */
3198 curr_sg = h->cmd_sg_list[c->cmdindex]; 3218 curr_sg = h->cmd_sg_list[c->cmdindex];
3199 sg_index = 0; 3219 sg_index = 0;
@@ -3206,27 +3226,12 @@ static void do_cciss_request(struct request_queue *q)
3206 curr_sg[sg_index].Addr.lower = temp64.val32.lower; 3226 curr_sg[sg_index].Addr.lower = temp64.val32.lower;
3207 curr_sg[sg_index].Addr.upper = temp64.val32.upper; 3227 curr_sg[sg_index].Addr.upper = temp64.val32.upper;
3208 curr_sg[sg_index].Ext = 0; /* we are not chaining */ 3228 curr_sg[sg_index].Ext = 0; /* we are not chaining */
3209
3210 ++sg_index; 3229 ++sg_index;
3211 } 3230 }
3212 3231 if (chained)
3213 if (chained) { 3232 cciss_map_sg_chain_block(h, c, h->cmd_sg_list[c->cmdindex],
3214 int len; 3233 (seg - (h->max_cmd_sgentries - 1)) *
3215 dma_addr_t dma_addr; 3234 sizeof(SGDescriptor_struct));
3216 curr_sg = c->SG;
3217 sg_index = h->max_cmd_sgentries - 1;
3218 len = curr_sg[sg_index].Len;
3219 /* Setup pointer to next chain block.
3220 * Fill out last element in current chain
3221 * block with address of next chain block.
3222 */
3223 temp64.val = pci_map_single(h->pdev,
3224 h->cmd_sg_list[c->cmdindex], len,
3225 PCI_DMA_TODEVICE);
3226 dma_addr = temp64.val;
3227 curr_sg[sg_index].Addr.lower = temp64.val32.lower;
3228 curr_sg[sg_index].Addr.upper = temp64.val32.upper;
3229 }
3230 3235
3231 /* track how many SG entries we are using */ 3236 /* track how many SG entries we are using */
3232 if (seg > h->maxSG) 3237 if (seg > h->maxSG)