diff options
author | Markus Lidel <Markus.Lidel@shadowconnect.com> | 2005-06-24 01:02:19 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@ppc970.osdl.org> | 2005-06-24 03:05:28 -0400 |
commit | b2aaee33fbb354a2f08121aa1c1be55841102761 (patch) | |
tree | 7567ca61aaf5eed8bb1acd01cd87aa235b854fd4 /drivers/message/i2o/i2o_scsi.c | |
parent | f10378fff658f61307496e0ae00095041725cf07 (diff) |
[PATCH] I2O: Adaptec specific SG_IO access, firmware access through sysfs and 2400A workaround
Changes:
- Provide SG_IO access to BLOCK and EXECUTIVE class on Adaptec
controllers
- Use PRIVATE messages in SCSI-OSM because on some controllers normal
SCSI class commands like READ or READ CAPACITY cause errors
- Use new DMA and SG list creation function
- Added workaround to limit sectors per request for Adaptec 2400A
controllers
Signed-off-by: Markus Lidel <Markus.Lidel@shadowconnect.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'drivers/message/i2o/i2o_scsi.c')
-rw-r--r-- | drivers/message/i2o/i2o_scsi.c | 263 |
1 files changed, 158 insertions, 105 deletions
diff --git a/drivers/message/i2o/i2o_scsi.c b/drivers/message/i2o/i2o_scsi.c index c3b0c29ac02d..fef53b509a61 100644 --- a/drivers/message/i2o/i2o_scsi.c +++ b/drivers/message/i2o/i2o_scsi.c | |||
@@ -55,6 +55,7 @@ | |||
55 | #include <linux/pci.h> | 55 | #include <linux/pci.h> |
56 | #include <linux/blkdev.h> | 56 | #include <linux/blkdev.h> |
57 | #include <linux/i2o.h> | 57 | #include <linux/i2o.h> |
58 | #include <linux/scatterlist.h> | ||
58 | 59 | ||
59 | #include <asm/dma.h> | 60 | #include <asm/dma.h> |
60 | #include <asm/system.h> | 61 | #include <asm/system.h> |
@@ -65,19 +66,23 @@ | |||
65 | #include <scsi/scsi_host.h> | 66 | #include <scsi/scsi_host.h> |
66 | #include <scsi/scsi_device.h> | 67 | #include <scsi/scsi_device.h> |
67 | #include <scsi/scsi_cmnd.h> | 68 | #include <scsi/scsi_cmnd.h> |
69 | #include <scsi/scsi_request.h> | ||
70 | #include <scsi/sg.h> | ||
71 | #include <scsi/sg_request.h> | ||
68 | 72 | ||
69 | #define OSM_NAME "scsi-osm" | 73 | #define OSM_NAME "scsi-osm" |
70 | #define OSM_VERSION "$Rev$" | 74 | #define OSM_VERSION "1.282" |
71 | #define OSM_DESCRIPTION "I2O SCSI Peripheral OSM" | 75 | #define OSM_DESCRIPTION "I2O SCSI Peripheral OSM" |
72 | 76 | ||
73 | static struct i2o_driver i2o_scsi_driver; | 77 | static struct i2o_driver i2o_scsi_driver; |
74 | 78 | ||
75 | static int i2o_scsi_max_id = 16; | 79 | static unsigned int i2o_scsi_max_id = 16; |
76 | static int i2o_scsi_max_lun = 8; | 80 | static unsigned int i2o_scsi_max_lun = 255; |
77 | 81 | ||
78 | struct i2o_scsi_host { | 82 | struct i2o_scsi_host { |
79 | struct Scsi_Host *scsi_host; /* pointer to the SCSI host */ | 83 | struct Scsi_Host *scsi_host; /* pointer to the SCSI host */ |
80 | struct i2o_controller *iop; /* pointer to the I2O controller */ | 84 | struct i2o_controller *iop; /* pointer to the I2O controller */ |
85 | unsigned int lun; /* lun's used for block devices */ | ||
81 | struct i2o_device *channel[0]; /* channel->i2o_dev mapping table */ | 86 | struct i2o_device *channel[0]; /* channel->i2o_dev mapping table */ |
82 | }; | 87 | }; |
83 | 88 | ||
@@ -100,12 +105,17 @@ static struct i2o_scsi_host *i2o_scsi_host_alloc(struct i2o_controller *c) | |||
100 | u8 type; | 105 | u8 type; |
101 | int i; | 106 | int i; |
102 | size_t size; | 107 | size_t size; |
103 | i2o_status_block *sb; | 108 | u16 body_size = 6; |
109 | |||
110 | #ifdef CONFIG_I2O_EXT_ADAPTEC | ||
111 | if (c->adaptec) | ||
112 | body_size = 8; | ||
113 | #endif | ||
104 | 114 | ||
105 | list_for_each_entry(i2o_dev, &c->devices, list) | 115 | list_for_each_entry(i2o_dev, &c->devices, list) |
106 | if (i2o_dev->lct_data.class_id == I2O_CLASS_BUS_ADAPTER) { | 116 | if (i2o_dev->lct_data.class_id == I2O_CLASS_BUS_ADAPTER) { |
107 | if (i2o_parm_field_get(i2o_dev, 0x0000, 0, &type, 1) | 117 | if (i2o_parm_field_get(i2o_dev, 0x0000, 0, &type, 1) |
108 | && (type == 0x01)) /* SCSI bus */ | 118 | && (type == 0x01)) /* SCSI bus */ |
109 | max_channel++; | 119 | max_channel++; |
110 | } | 120 | } |
111 | 121 | ||
@@ -127,20 +137,18 @@ static struct i2o_scsi_host *i2o_scsi_host_alloc(struct i2o_controller *c) | |||
127 | scsi_host->max_id = i2o_scsi_max_id; | 137 | scsi_host->max_id = i2o_scsi_max_id; |
128 | scsi_host->max_lun = i2o_scsi_max_lun; | 138 | scsi_host->max_lun = i2o_scsi_max_lun; |
129 | scsi_host->this_id = c->unit; | 139 | scsi_host->this_id = c->unit; |
130 | 140 | scsi_host->sg_tablesize = i2o_sg_tablesize(c, body_size); | |
131 | sb = c->status_block.virt; | ||
132 | |||
133 | scsi_host->sg_tablesize = (sb->inbound_frame_size - | ||
134 | sizeof(struct i2o_message) / 4 - 6) / 2; | ||
135 | 141 | ||
136 | i2o_shost = (struct i2o_scsi_host *)scsi_host->hostdata; | 142 | i2o_shost = (struct i2o_scsi_host *)scsi_host->hostdata; |
137 | i2o_shost->scsi_host = scsi_host; | 143 | i2o_shost->scsi_host = scsi_host; |
138 | i2o_shost->iop = c; | 144 | i2o_shost->iop = c; |
145 | i2o_shost->lun = 1; | ||
139 | 146 | ||
140 | i = 0; | 147 | i = 0; |
141 | list_for_each_entry(i2o_dev, &c->devices, list) | 148 | list_for_each_entry(i2o_dev, &c->devices, list) |
142 | if (i2o_dev->lct_data.class_id == I2O_CLASS_BUS_ADAPTER) { | 149 | if (i2o_dev->lct_data.class_id == I2O_CLASS_BUS_ADAPTER) { |
143 | if (i2o_parm_field_get(i2o_dev, 0x0000, 0, &type, 1) || (type == 1)) /* only SCSI bus */ | 150 | if (i2o_parm_field_get(i2o_dev, 0x0000, 0, &type, 1) |
151 | && (type == 0x01)) /* only SCSI bus */ | ||
144 | i2o_shost->channel[i++] = i2o_dev; | 152 | i2o_shost->channel[i++] = i2o_dev; |
145 | 153 | ||
146 | if (i >= max_channel) | 154 | if (i >= max_channel) |
@@ -212,8 +220,8 @@ static int i2o_scsi_probe(struct device *dev) | |||
212 | struct Scsi_Host *scsi_host; | 220 | struct Scsi_Host *scsi_host; |
213 | struct i2o_device *parent; | 221 | struct i2o_device *parent; |
214 | struct scsi_device *scsi_dev; | 222 | struct scsi_device *scsi_dev; |
215 | u32 id; | 223 | u32 id = -1; |
216 | u64 lun; | 224 | u64 lun = -1; |
217 | int channel = -1; | 225 | int channel = -1; |
218 | int i; | 226 | int i; |
219 | 227 | ||
@@ -223,8 +231,56 @@ static int i2o_scsi_probe(struct device *dev) | |||
223 | 231 | ||
224 | scsi_host = i2o_shost->scsi_host; | 232 | scsi_host = i2o_shost->scsi_host; |
225 | 233 | ||
226 | if (i2o_parm_field_get(i2o_dev, 0, 3, &id, 4) < 0) | 234 | switch (i2o_dev->lct_data.class_id) { |
235 | case I2O_CLASS_RANDOM_BLOCK_STORAGE: | ||
236 | case I2O_CLASS_EXECUTIVE: | ||
237 | #ifdef CONFIG_I2O_EXT_ADAPTEC | ||
238 | if (c->adaptec) { | ||
239 | u8 type; | ||
240 | struct i2o_device *d = i2o_shost->channel[0]; | ||
241 | |||
242 | if (i2o_parm_field_get(d, 0x0000, 0, &type, 1) | ||
243 | && (type == 0x01)) /* SCSI bus */ | ||
244 | if (i2o_parm_field_get(d, 0x0200, 4, &id, 4)) { | ||
245 | channel = 0; | ||
246 | if (i2o_dev->lct_data.class_id == | ||
247 | I2O_CLASS_RANDOM_BLOCK_STORAGE) | ||
248 | lun = i2o_shost->lun++; | ||
249 | else | ||
250 | lun = 0; | ||
251 | } | ||
252 | } | ||
253 | #endif | ||
254 | break; | ||
255 | |||
256 | case I2O_CLASS_SCSI_PERIPHERAL: | ||
257 | if (i2o_parm_field_get(i2o_dev, 0x0000, 3, &id, 4) < 0) | ||
258 | return -EFAULT; | ||
259 | |||
260 | if (i2o_parm_field_get(i2o_dev, 0x0000, 4, &lun, 8) < 0) | ||
261 | return -EFAULT; | ||
262 | |||
263 | parent = i2o_iop_find_device(c, i2o_dev->lct_data.parent_tid); | ||
264 | if (!parent) { | ||
265 | osm_warn("can not find parent of device %03x\n", | ||
266 | i2o_dev->lct_data.tid); | ||
267 | return -EFAULT; | ||
268 | } | ||
269 | |||
270 | for (i = 0; i <= i2o_shost->scsi_host->max_channel; i++) | ||
271 | if (i2o_shost->channel[i] == parent) | ||
272 | channel = i; | ||
273 | break; | ||
274 | |||
275 | default: | ||
276 | return -EFAULT; | ||
277 | } | ||
278 | |||
279 | if (channel == -1) { | ||
280 | osm_warn("can not find channel of device %03x\n", | ||
281 | i2o_dev->lct_data.tid); | ||
227 | return -EFAULT; | 282 | return -EFAULT; |
283 | } | ||
228 | 284 | ||
229 | if (id >= scsi_host->max_id) { | 285 | if (id >= scsi_host->max_id) { |
230 | osm_warn("SCSI device id (%d) >= max_id of I2O host (%d)", id, | 286 | osm_warn("SCSI device id (%d) >= max_id of I2O host (%d)", id, |
@@ -232,31 +288,12 @@ static int i2o_scsi_probe(struct device *dev) | |||
232 | return -EFAULT; | 288 | return -EFAULT; |
233 | } | 289 | } |
234 | 290 | ||
235 | if (i2o_parm_field_get(i2o_dev, 0, 4, &lun, 8) < 0) | ||
236 | return -EFAULT; | ||
237 | if (lun >= scsi_host->max_lun) { | 291 | if (lun >= scsi_host->max_lun) { |
238 | osm_warn("SCSI device id (%d) >= max_lun of I2O host (%d)", | 292 | osm_warn("SCSI device id (%d) >= max_lun of I2O host (%d)", |
239 | (unsigned int)lun, scsi_host->max_lun); | 293 | (unsigned int)lun, scsi_host->max_lun); |
240 | return -EFAULT; | 294 | return -EFAULT; |
241 | } | 295 | } |
242 | 296 | ||
243 | parent = i2o_iop_find_device(c, i2o_dev->lct_data.parent_tid); | ||
244 | if (!parent) { | ||
245 | osm_warn("can not find parent of device %03x\n", | ||
246 | i2o_dev->lct_data.tid); | ||
247 | return -EFAULT; | ||
248 | } | ||
249 | |||
250 | for (i = 0; i <= i2o_shost->scsi_host->max_channel; i++) | ||
251 | if (i2o_shost->channel[i] == parent) | ||
252 | channel = i; | ||
253 | |||
254 | if (channel == -1) { | ||
255 | osm_warn("can not find channel of device %03x\n", | ||
256 | i2o_dev->lct_data.tid); | ||
257 | return -EFAULT; | ||
258 | } | ||
259 | |||
260 | scsi_dev = | 297 | scsi_dev = |
261 | __scsi_add_device(i2o_shost->scsi_host, channel, id, lun, i2o_dev); | 298 | __scsi_add_device(i2o_shost->scsi_host, channel, id, lun, i2o_dev); |
262 | 299 | ||
@@ -266,7 +303,8 @@ static int i2o_scsi_probe(struct device *dev) | |||
266 | return PTR_ERR(scsi_dev); | 303 | return PTR_ERR(scsi_dev); |
267 | } | 304 | } |
268 | 305 | ||
269 | sysfs_create_link(&i2o_dev->device.kobj, &scsi_dev->sdev_gendev.kobj, "scsi"); | 306 | sysfs_create_link(&i2o_dev->device.kobj, &scsi_dev->sdev_gendev.kobj, |
307 | "scsi"); | ||
270 | 308 | ||
271 | osm_info("device added (TID: %03x) channel: %d, id: %d, lun: %d\n", | 309 | osm_info("device added (TID: %03x) channel: %d, id: %d, lun: %d\n", |
272 | i2o_dev->lct_data.tid, channel, id, (unsigned int)lun); | 310 | i2o_dev->lct_data.tid, channel, id, (unsigned int)lun); |
@@ -542,9 +580,7 @@ static int i2o_scsi_queuecommand(struct scsi_cmnd *SCpnt, | |||
542 | void (*done) (struct scsi_cmnd *)) | 580 | void (*done) (struct scsi_cmnd *)) |
543 | { | 581 | { |
544 | struct i2o_controller *c; | 582 | struct i2o_controller *c; |
545 | struct Scsi_Host *host; | ||
546 | struct i2o_device *i2o_dev; | 583 | struct i2o_device *i2o_dev; |
547 | struct device *dev; | ||
548 | int tid; | 584 | int tid; |
549 | struct i2o_message __iomem *msg; | 585 | struct i2o_message __iomem *msg; |
550 | u32 m; | 586 | u32 m; |
@@ -554,20 +590,16 @@ static int i2o_scsi_queuecommand(struct scsi_cmnd *SCpnt, | |||
554 | * RETURN_SENSE_DATA_IN_REPLY_MESSAGE_FRAME | 590 | * RETURN_SENSE_DATA_IN_REPLY_MESSAGE_FRAME |
555 | */ | 591 | */ |
556 | u32 scsi_flags = 0x20a00000; | 592 | u32 scsi_flags = 0x20a00000; |
557 | u32 sg_flags; | 593 | u32 sgl_offset; |
558 | u32 __iomem *mptr; | 594 | u32 __iomem *mptr; |
559 | u32 __iomem *lenptr; | 595 | u32 cmd = I2O_CMD_SCSI_EXEC << 24; |
560 | u32 len; | 596 | int rc = 0; |
561 | int i; | ||
562 | 597 | ||
563 | /* | 598 | /* |
564 | * Do the incoming paperwork | 599 | * Do the incoming paperwork |
565 | */ | 600 | */ |
566 | |||
567 | i2o_dev = SCpnt->device->hostdata; | 601 | i2o_dev = SCpnt->device->hostdata; |
568 | host = SCpnt->device->host; | ||
569 | c = i2o_dev->iop; | 602 | c = i2o_dev->iop; |
570 | dev = &c->pdev->dev; | ||
571 | 603 | ||
572 | SCpnt->scsi_done = done; | 604 | SCpnt->scsi_done = done; |
573 | 605 | ||
@@ -575,7 +607,7 @@ static int i2o_scsi_queuecommand(struct scsi_cmnd *SCpnt, | |||
575 | osm_warn("no I2O device in request\n"); | 607 | osm_warn("no I2O device in request\n"); |
576 | SCpnt->result = DID_NO_CONNECT << 16; | 608 | SCpnt->result = DID_NO_CONNECT << 16; |
577 | done(SCpnt); | 609 | done(SCpnt); |
578 | return 0; | 610 | goto exit; |
579 | } | 611 | } |
580 | 612 | ||
581 | tid = i2o_dev->lct_data.tid; | 613 | tid = i2o_dev->lct_data.tid; |
@@ -584,46 +616,85 @@ static int i2o_scsi_queuecommand(struct scsi_cmnd *SCpnt, | |||
584 | osm_debug("Real scsi messages.\n"); | 616 | osm_debug("Real scsi messages.\n"); |
585 | 617 | ||
586 | /* | 618 | /* |
587 | * Obtain an I2O message. If there are none free then | ||
588 | * throw it back to the scsi layer | ||
589 | */ | ||
590 | |||
591 | m = i2o_msg_get_wait(c, &msg, I2O_TIMEOUT_MESSAGE_GET); | ||
592 | if (m == I2O_QUEUE_EMPTY) | ||
593 | return SCSI_MLQUEUE_HOST_BUSY; | ||
594 | |||
595 | mptr = &msg->body[0]; | ||
596 | |||
597 | /* | ||
598 | * Put together a scsi execscb message | 619 | * Put together a scsi execscb message |
599 | */ | 620 | */ |
600 | |||
601 | switch (SCpnt->sc_data_direction) { | 621 | switch (SCpnt->sc_data_direction) { |
602 | case PCI_DMA_NONE: | 622 | case PCI_DMA_NONE: |
603 | /* DATA NO XFER */ | 623 | /* DATA NO XFER */ |
604 | sg_flags = 0x00000000; | 624 | sgl_offset = SGL_OFFSET_0; |
605 | break; | 625 | break; |
606 | 626 | ||
607 | case PCI_DMA_TODEVICE: | 627 | case PCI_DMA_TODEVICE: |
608 | /* DATA OUT (iop-->dev) */ | 628 | /* DATA OUT (iop-->dev) */ |
609 | scsi_flags |= 0x80000000; | 629 | scsi_flags |= 0x80000000; |
610 | sg_flags = 0x14000000; | 630 | sgl_offset = SGL_OFFSET_10; |
611 | break; | 631 | break; |
612 | 632 | ||
613 | case PCI_DMA_FROMDEVICE: | 633 | case PCI_DMA_FROMDEVICE: |
614 | /* DATA IN (iop<--dev) */ | 634 | /* DATA IN (iop<--dev) */ |
615 | scsi_flags |= 0x40000000; | 635 | scsi_flags |= 0x40000000; |
616 | sg_flags = 0x10000000; | 636 | sgl_offset = SGL_OFFSET_10; |
617 | break; | 637 | break; |
618 | 638 | ||
619 | default: | 639 | default: |
620 | /* Unknown - kill the command */ | 640 | /* Unknown - kill the command */ |
621 | SCpnt->result = DID_NO_CONNECT << 16; | 641 | SCpnt->result = DID_NO_CONNECT << 16; |
622 | done(SCpnt); | 642 | done(SCpnt); |
623 | return 0; | 643 | goto exit; |
624 | } | 644 | } |
625 | 645 | ||
626 | writel(I2O_CMD_SCSI_EXEC << 24 | HOST_TID << 12 | tid, &msg->u.head[1]); | 646 | /* |
647 | * Obtain an I2O message. If there are none free then | ||
648 | * throw it back to the scsi layer | ||
649 | */ | ||
650 | |||
651 | m = i2o_msg_get_wait(c, &msg, I2O_TIMEOUT_MESSAGE_GET); | ||
652 | if (m == I2O_QUEUE_EMPTY) { | ||
653 | rc = SCSI_MLQUEUE_HOST_BUSY; | ||
654 | goto exit; | ||
655 | } | ||
656 | |||
657 | mptr = &msg->body[0]; | ||
658 | |||
659 | #ifdef CONFIG_I2O_EXT_ADAPTEC | ||
660 | if (c->adaptec) { | ||
661 | u32 adpt_flags = 0; | ||
662 | |||
663 | if (SCpnt->sc_request && SCpnt->sc_request->upper_private_data) { | ||
664 | i2o_sg_io_hdr_t __user *usr_ptr = | ||
665 | ((Sg_request *) (SCpnt->sc_request-> | ||
666 | upper_private_data))->header. | ||
667 | usr_ptr; | ||
668 | |||
669 | if (usr_ptr) | ||
670 | get_user(adpt_flags, &usr_ptr->flags); | ||
671 | } | ||
672 | |||
673 | switch (i2o_dev->lct_data.class_id) { | ||
674 | case I2O_CLASS_EXECUTIVE: | ||
675 | case I2O_CLASS_RANDOM_BLOCK_STORAGE: | ||
676 | /* interpret flag has to be set for executive */ | ||
677 | adpt_flags ^= I2O_DPT_SG_FLAG_INTERPRET; | ||
678 | break; | ||
679 | |||
680 | default: | ||
681 | break; | ||
682 | } | ||
683 | |||
684 | /* | ||
685 | * for Adaptec controllers we use the PRIVATE command, because | ||
686 | * the normal SCSI EXEC doesn't support all SCSI commands on | ||
687 | * all controllers (for example READ CAPACITY). | ||
688 | */ | ||
689 | if (sgl_offset == SGL_OFFSET_10) | ||
690 | sgl_offset = SGL_OFFSET_12; | ||
691 | cmd = I2O_CMD_PRIVATE << 24; | ||
692 | writel(I2O_VENDOR_DPT << 16 | I2O_CMD_SCSI_EXEC, mptr++); | ||
693 | writel(adpt_flags | tid, mptr++); | ||
694 | } | ||
695 | #endif | ||
696 | |||
697 | writel(cmd | HOST_TID << 12 | tid, &msg->u.head[1]); | ||
627 | writel(i2o_scsi_driver.context, &msg->u.s.icntxt); | 698 | writel(i2o_scsi_driver.context, &msg->u.s.icntxt); |
628 | 699 | ||
629 | /* We want the SCSI control block back */ | 700 | /* We want the SCSI control block back */ |
@@ -655,55 +726,30 @@ static int i2o_scsi_queuecommand(struct scsi_cmnd *SCpnt, | |||
655 | /* Write SCSI command into the message - always 16 byte block */ | 726 | /* Write SCSI command into the message - always 16 byte block */ |
656 | memcpy_toio(mptr, SCpnt->cmnd, 16); | 727 | memcpy_toio(mptr, SCpnt->cmnd, 16); |
657 | mptr += 4; | 728 | mptr += 4; |
658 | lenptr = mptr++; /* Remember me - fill in when we know */ | ||
659 | |||
660 | /* Now fill in the SGList and command */ | ||
661 | if (SCpnt->use_sg) { | ||
662 | struct scatterlist *sg; | ||
663 | int sg_count; | ||
664 | |||
665 | sg = SCpnt->request_buffer; | ||
666 | len = 0; | ||
667 | 729 | ||
668 | sg_count = dma_map_sg(dev, sg, SCpnt->use_sg, | 730 | if (sgl_offset != SGL_OFFSET_0) { |
669 | SCpnt->sc_data_direction); | 731 | /* write size of data addressed by SGL */ |
670 | 732 | writel(SCpnt->request_bufflen, mptr++); | |
671 | if (unlikely(sg_count <= 0)) | 733 | |
672 | return -ENOMEM; | 734 | /* Now fill in the SGList and command */ |
673 | 735 | if (SCpnt->use_sg) { | |
674 | for (i = SCpnt->use_sg; i > 0; i--) { | 736 | if (!i2o_dma_map_sg(c, SCpnt->request_buffer, |
675 | if (i == 1) | 737 | SCpnt->use_sg, |
676 | sg_flags |= 0xC0000000; | 738 | SCpnt->sc_data_direction, &mptr)) |
677 | writel(sg_flags | sg_dma_len(sg), mptr++); | 739 | goto nomem; |
678 | writel(sg_dma_address(sg), mptr++); | 740 | } else { |
679 | len += sg_dma_len(sg); | 741 | SCpnt->SCp.dma_handle = |
680 | sg++; | 742 | i2o_dma_map_single(c, SCpnt->request_buffer, |
681 | } | 743 | SCpnt->request_bufflen, |
682 | 744 | SCpnt->sc_data_direction, &mptr); | |
683 | writel(len, lenptr); | 745 | if (dma_mapping_error(SCpnt->SCp.dma_handle)) |
684 | } else { | 746 | goto nomem; |
685 | len = SCpnt->request_bufflen; | ||
686 | |||
687 | writel(len, lenptr); | ||
688 | |||
689 | if (len > 0) { | ||
690 | dma_addr_t dma_addr; | ||
691 | |||
692 | dma_addr = dma_map_single(dev, SCpnt->request_buffer, | ||
693 | SCpnt->request_bufflen, | ||
694 | SCpnt->sc_data_direction); | ||
695 | if (!dma_addr) | ||
696 | return -ENOMEM; | ||
697 | |||
698 | SCpnt->SCp.ptr = (void *)(unsigned long)dma_addr; | ||
699 | sg_flags |= 0xC0000000; | ||
700 | writel(sg_flags | SCpnt->request_bufflen, mptr++); | ||
701 | writel(dma_addr, mptr++); | ||
702 | } | 747 | } |
703 | } | 748 | } |
704 | 749 | ||
705 | /* Stick the headers on */ | 750 | /* Stick the headers on */ |
706 | writel((mptr - &msg->u.head[0]) << 16 | SGL_OFFSET_10, &msg->u.head[0]); | 751 | writel(I2O_MESSAGE_SIZE(mptr - &msg->u.head[0]) | sgl_offset, |
752 | &msg->u.head[0]); | ||
707 | 753 | ||
708 | /* Queue the message */ | 754 | /* Queue the message */ |
709 | i2o_msg_post(c, m); | 755 | i2o_msg_post(c, m); |
@@ -711,6 +757,13 @@ static int i2o_scsi_queuecommand(struct scsi_cmnd *SCpnt, | |||
711 | osm_debug("Issued %ld\n", SCpnt->serial_number); | 757 | osm_debug("Issued %ld\n", SCpnt->serial_number); |
712 | 758 | ||
713 | return 0; | 759 | return 0; |
760 | |||
761 | nomem: | ||
762 | rc = -ENOMEM; | ||
763 | i2o_msg_nop(c, m); | ||
764 | |||
765 | exit: | ||
766 | return rc; | ||
714 | }; | 767 | }; |
715 | 768 | ||
716 | /** | 769 | /** |