[SCSI] 3ware 9000: handle use_sg != 0 for emulated commands
The attached patch updates the driver for the 3ware 9000 series to do the following: - Correctly handle single sgl's with use_sg = 1. This is needed with the latest scsi-block-2.6 merge otherwise the 3w-9xxx driver will not work. I tested the patch James sent a few weeks back to fix this, and it had a bug where the request_buffer was accessed in twa_scsiop_execute_scsi_complete() when it was invalid. This is a corrected variation of that patch. Signed-off-by: Adam Radford <linuxraid@amcc.com> Signed-off-by: James Bottomley <James.Bottomley@SteelEye.com>
This commit is contained in:
parent
c7ebbbce36
commit
d327d08232
|
@ -59,6 +59,7 @@
|
||||||
Fix 'handled=1' ISR usage, remove bogus IRQ check.
|
Fix 'handled=1' ISR usage, remove bogus IRQ check.
|
||||||
Remove un-needed eh_abort handler.
|
Remove un-needed eh_abort handler.
|
||||||
Add support for embedded firmware error strings.
|
Add support for embedded firmware error strings.
|
||||||
|
2.26.02.003 - Correctly handle single sgl's with use_sg=1.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
|
@ -81,7 +82,7 @@
|
||||||
#include "3w-9xxx.h"
|
#include "3w-9xxx.h"
|
||||||
|
|
||||||
/* Globals */
|
/* Globals */
|
||||||
#define TW_DRIVER_VERSION "2.26.02.002"
|
#define TW_DRIVER_VERSION "2.26.02.003"
|
||||||
static TW_Device_Extension *twa_device_extension_list[TW_MAX_SLOT];
|
static TW_Device_Extension *twa_device_extension_list[TW_MAX_SLOT];
|
||||||
static unsigned int twa_device_extension_count;
|
static unsigned int twa_device_extension_count;
|
||||||
static int twa_major = -1;
|
static int twa_major = -1;
|
||||||
|
@ -1805,6 +1806,8 @@ static int twa_scsiop_execute_scsi(TW_Device_Extension *tw_dev, int request_id,
|
||||||
if (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH) {
|
if (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH) {
|
||||||
command_packet->sg_list[0].address = tw_dev->generic_buffer_phys[request_id];
|
command_packet->sg_list[0].address = tw_dev->generic_buffer_phys[request_id];
|
||||||
command_packet->sg_list[0].length = TW_MIN_SGL_LENGTH;
|
command_packet->sg_list[0].length = TW_MIN_SGL_LENGTH;
|
||||||
|
if (tw_dev->srb[request_id]->sc_data_direction == DMA_TO_DEVICE || tw_dev->srb[request_id]->sc_data_direction == DMA_BIDIRECTIONAL)
|
||||||
|
memcpy(tw_dev->generic_buffer_virt[request_id], tw_dev->srb[request_id]->request_buffer, tw_dev->srb[request_id]->request_bufflen);
|
||||||
} else {
|
} else {
|
||||||
buffaddr = twa_map_scsi_single_data(tw_dev, request_id);
|
buffaddr = twa_map_scsi_single_data(tw_dev, request_id);
|
||||||
if (buffaddr == 0)
|
if (buffaddr == 0)
|
||||||
|
@ -1823,6 +1826,12 @@ static int twa_scsiop_execute_scsi(TW_Device_Extension *tw_dev, int request_id,
|
||||||
|
|
||||||
if (tw_dev->srb[request_id]->use_sg > 0) {
|
if (tw_dev->srb[request_id]->use_sg > 0) {
|
||||||
if ((tw_dev->srb[request_id]->use_sg == 1) && (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH)) {
|
if ((tw_dev->srb[request_id]->use_sg == 1) && (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH)) {
|
||||||
|
if (tw_dev->srb[request_id]->sc_data_direction == DMA_TO_DEVICE || tw_dev->srb[request_id]->sc_data_direction == DMA_BIDIRECTIONAL) {
|
||||||
|
struct scatterlist *sg = (struct scatterlist *)tw_dev->srb[request_id]->request_buffer;
|
||||||
|
char *buf = kmap_atomic(sg->page, KM_IRQ0) + sg->offset;
|
||||||
|
memcpy(tw_dev->generic_buffer_virt[request_id], buf, sg->length);
|
||||||
|
kunmap_atomic(buf - sg->offset, KM_IRQ0);
|
||||||
|
}
|
||||||
command_packet->sg_list[0].address = tw_dev->generic_buffer_phys[request_id];
|
command_packet->sg_list[0].address = tw_dev->generic_buffer_phys[request_id];
|
||||||
command_packet->sg_list[0].length = TW_MIN_SGL_LENGTH;
|
command_packet->sg_list[0].length = TW_MIN_SGL_LENGTH;
|
||||||
} else {
|
} else {
|
||||||
|
@ -1888,12 +1897,21 @@ out:
|
||||||
/* This function completes an execute scsi operation */
|
/* This function completes an execute scsi operation */
|
||||||
static void twa_scsiop_execute_scsi_complete(TW_Device_Extension *tw_dev, int request_id)
|
static void twa_scsiop_execute_scsi_complete(TW_Device_Extension *tw_dev, int request_id)
|
||||||
{
|
{
|
||||||
/* Copy the response if too small */
|
if (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH &&
|
||||||
if ((tw_dev->srb[request_id]->request_buffer) && (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH)) {
|
(tw_dev->srb[request_id]->sc_data_direction == DMA_FROM_DEVICE ||
|
||||||
|
tw_dev->srb[request_id]->sc_data_direction == DMA_BIDIRECTIONAL)) {
|
||||||
|
if (tw_dev->srb[request_id]->use_sg == 0) {
|
||||||
memcpy(tw_dev->srb[request_id]->request_buffer,
|
memcpy(tw_dev->srb[request_id]->request_buffer,
|
||||||
tw_dev->generic_buffer_virt[request_id],
|
tw_dev->generic_buffer_virt[request_id],
|
||||||
tw_dev->srb[request_id]->request_bufflen);
|
tw_dev->srb[request_id]->request_bufflen);
|
||||||
}
|
}
|
||||||
|
if (tw_dev->srb[request_id]->use_sg == 1) {
|
||||||
|
struct scatterlist *sg = (struct scatterlist *)tw_dev->srb[request_id]->request_buffer;
|
||||||
|
char *buf = kmap_atomic(sg->page, KM_IRQ0) + sg->offset;
|
||||||
|
memcpy(buf, tw_dev->generic_buffer_virt[request_id], sg->length);
|
||||||
|
kunmap_atomic(buf - sg->offset, KM_IRQ0);
|
||||||
|
}
|
||||||
|
}
|
||||||
} /* End twa_scsiop_execute_scsi_complete() */
|
} /* End twa_scsiop_execute_scsi_complete() */
|
||||||
|
|
||||||
/* This function tells the controller to shut down */
|
/* This function tells the controller to shut down */
|
||||||
|
|
Loading…
Reference in New Issue