Merge git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi-misc-2.6

* git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi-misc-2.6:
  [SCSI] aic94xx: fix section mismatch
  [SCSI] u14-34f: Fix 32bit only problem
  [SCSI] dpt_i2o: sysfs code
  [SCSI] dpt_i2o: 64 bit support
  [SCSI] dpt_i2o: move from virt_to_bus/bus_to_virt to dma_alloc_coherent
  [SCSI] dpt_i2o: use standard __init / __exit code
  [SCSI] megaraid_sas: fix suspend/resume sections
  [SCSI] aacraid: Add Power Management support
  [SCSI] aacraid: Fix jbod operations scan issues
  [SCSI] aacraid: Fix warning about macro side-effects
  [SCSI] add support for variable length extended commands
  [SCSI] Let scsi_cmnd->cmnd use request->cmd buffer
  [SCSI] bsg: add large command support
  [SCSI] aacraid: Fix down_interruptible() to check the return value correctly
  [SCSI] megaraid_sas; Update the Version and Changelog
  [SCSI] ibmvscsi: Handle non SCSI error status
  [SCSI] bug fix for free list handling
  [SCSI] ipr: Rename ipr's state scsi host attribute to prevent collisions
  [SCSI] megaraid_mbox: fix Dell CERC firmware problem
This commit is contained in:
Linus Torvalds 2008-05-02 13:52:35 -07:00
commit d626e3bf72
43 changed files with 884 additions and 291 deletions

View file

@ -1,3 +1,25 @@
1 Release Date : Mon. March 10 11:02:31 PDT 2008 -
(emaild-id:megaraidlinux@lsi.com)
Sumant Patro
Bo Yang
2 Current Version : 00.00.03.20-RC1
3 Older Version : 00.00.03.16
1. Rollback the sense info implementation
Sense buffer ptr data type in the ioctl path is reverted back
to u32 * as in previous versions of driver.
2. Fixed the driver frame count.
When Driver sent wrong frame count to firmware. As this
particular command is sent to drive, FW is seeing continuous
chip resets and so the command will timeout.
3. Add the new controller(1078DE) support to the driver
and Increase the max_wait to 60 from 10 in the controller
operational status. With this max_wait increase, driver will
make sure the FW will finish the pending cmd for KDUMP case.
1 Release Date : Thur. Nov. 07 16:30:43 PST 2007 - 1 Release Date : Thur. Nov. 07 16:30:43 PST 2007 -
(emaild-id:megaraidlinux@lsi.com) (emaild-id:megaraidlinux@lsi.com)
Sumant Patro Sumant Patro

View file

@ -174,7 +174,11 @@ static int bsg_io_schedule(struct bsg_device *bd)
static int blk_fill_sgv4_hdr_rq(struct request_queue *q, struct request *rq, static int blk_fill_sgv4_hdr_rq(struct request_queue *q, struct request *rq,
struct sg_io_v4 *hdr, int has_write_perm) struct sg_io_v4 *hdr, int has_write_perm)
{ {
memset(rq->cmd, 0, BLK_MAX_CDB); /* ATAPI hates garbage after CDB */ if (hdr->request_len > BLK_MAX_CDB) {
rq->cmd = kzalloc(hdr->request_len, GFP_KERNEL);
if (!rq->cmd)
return -ENOMEM;
}
if (copy_from_user(rq->cmd, (void *)(unsigned long)hdr->request, if (copy_from_user(rq->cmd, (void *)(unsigned long)hdr->request,
hdr->request_len)) hdr->request_len))
@ -211,8 +215,6 @@ bsg_validate_sgv4_hdr(struct request_queue *q, struct sg_io_v4 *hdr, int *rw)
if (hdr->guard != 'Q') if (hdr->guard != 'Q')
return -EINVAL; return -EINVAL;
if (hdr->request_len > BLK_MAX_CDB)
return -EINVAL;
if (hdr->dout_xfer_len > (q->max_sectors << 9) || if (hdr->dout_xfer_len > (q->max_sectors << 9) ||
hdr->din_xfer_len > (q->max_sectors << 9)) hdr->din_xfer_len > (q->max_sectors << 9))
return -EIO; return -EIO;
@ -302,6 +304,8 @@ bsg_map_hdr(struct bsg_device *bd, struct sg_io_v4 *hdr)
} }
return rq; return rq;
out: out:
if (rq->cmd != rq->__cmd)
kfree(rq->cmd);
blk_put_request(rq); blk_put_request(rq);
if (next_rq) { if (next_rq) {
blk_rq_unmap_user(next_rq->bio); blk_rq_unmap_user(next_rq->bio);
@ -455,6 +459,8 @@ static int blk_complete_sgv4_hdr_rq(struct request *rq, struct sg_io_v4 *hdr,
ret = rq->errors; ret = rq->errors;
blk_rq_unmap_user(bio); blk_rq_unmap_user(bio);
if (rq->cmd != rq->__cmd)
kfree(rq->cmd);
blk_put_request(rq); blk_put_request(rq);
return ret; return ret;

View file

@ -33,13 +33,12 @@
#include <scsi/scsi_cmnd.h> #include <scsi/scsi_cmnd.h>
/* Command group 3 is reserved and should never be used. */ /* Command group 3 is reserved and should never be used. */
const unsigned char scsi_command_size[8] = const unsigned char scsi_command_size_tbl[8] =
{ {
6, 10, 10, 12, 6, 10, 10, 12,
16, 12, 10, 10 16, 12, 10, 10
}; };
EXPORT_SYMBOL(scsi_command_size_tbl);
EXPORT_SYMBOL(scsi_command_size);
#include <scsi/sg.h> #include <scsi/sg.h>

View file

@ -1487,7 +1487,7 @@ static int sbp2_scsi_queuecommand(struct scsi_cmnd *cmd, scsi_done_fn_t done)
if (scsi_sg_count(cmd) && sbp2_map_scatterlist(orb, device, lu) < 0) if (scsi_sg_count(cmd) && sbp2_map_scatterlist(orb, device, lu) < 0)
goto out; goto out;
memcpy(orb->request.command_block, cmd->cmnd, COMMAND_SIZE(*cmd->cmnd)); memcpy(orb->request.command_block, cmd->cmnd, cmd->cmd_len);
orb->base.callback = complete_command_orb; orb->base.callback = complete_command_orb;
orb->base.request_bus = orb->base.request_bus =

View file

@ -1055,7 +1055,7 @@ static void zfcp_scsi_dbf_event(const char *tag, const char *tag2, int level,
rec->scsi_result = scsi_cmnd->result; rec->scsi_result = scsi_cmnd->result;
rec->scsi_cmnd = (unsigned long)scsi_cmnd; rec->scsi_cmnd = (unsigned long)scsi_cmnd;
rec->scsi_serial = scsi_cmnd->serial_number; rec->scsi_serial = scsi_cmnd->serial_number;
memcpy(rec->scsi_opcode, &scsi_cmnd->cmnd, memcpy(rec->scsi_opcode, scsi_cmnd->cmnd,
min((int)scsi_cmnd->cmd_len, min((int)scsi_cmnd->cmd_len,
ZFCP_DBF_SCSI_OPCODE)); ZFCP_DBF_SCSI_OPCODE));
rec->scsi_retries = scsi_cmnd->retries; rec->scsi_retries = scsi_cmnd->retries;

View file

@ -4014,7 +4014,7 @@ zfcp_fsf_send_fcp_command_task_handler(struct zfcp_fsf_req *fsf_req)
ZFCP_LOG_TRACE("scpnt->result =0x%x, command was:\n", ZFCP_LOG_TRACE("scpnt->result =0x%x, command was:\n",
scpnt->result); scpnt->result);
ZFCP_HEX_DUMP(ZFCP_LOG_LEVEL_TRACE, ZFCP_HEX_DUMP(ZFCP_LOG_LEVEL_TRACE,
(void *) &scpnt->cmnd, scpnt->cmd_len); scpnt->cmnd, scpnt->cmd_len);
ZFCP_LOG_TRACE("%i bytes sense data provided by FCP\n", ZFCP_LOG_TRACE("%i bytes sense data provided by FCP\n",
fcp_rsp_iu->fcp_sns_len); fcp_rsp_iu->fcp_sns_len);

View file

@ -599,7 +599,7 @@ NCR_700_scsi_done(struct NCR_700_Host_Parameters *hostdata,
(struct NCR_700_command_slot *)SCp->host_scribble; (struct NCR_700_command_slot *)SCp->host_scribble;
dma_unmap_single(hostdata->dev, slot->pCmd, dma_unmap_single(hostdata->dev, slot->pCmd,
sizeof(SCp->cmnd), DMA_TO_DEVICE); MAX_COMMAND_SIZE, DMA_TO_DEVICE);
if (slot->flags == NCR_700_FLAG_AUTOSENSE) { if (slot->flags == NCR_700_FLAG_AUTOSENSE) {
char *cmnd = NCR_700_get_sense_cmnd(SCp->device); char *cmnd = NCR_700_get_sense_cmnd(SCp->device);
#ifdef NCR_700_DEBUG #ifdef NCR_700_DEBUG
@ -1004,7 +1004,7 @@ process_script_interrupt(__u32 dsps, __u32 dsp, struct scsi_cmnd *SCp,
* here */ * here */
NCR_700_unmap(hostdata, SCp, slot); NCR_700_unmap(hostdata, SCp, slot);
dma_unmap_single(hostdata->dev, slot->pCmd, dma_unmap_single(hostdata->dev, slot->pCmd,
sizeof(SCp->cmnd), MAX_COMMAND_SIZE,
DMA_TO_DEVICE); DMA_TO_DEVICE);
cmnd[0] = REQUEST_SENSE; cmnd[0] = REQUEST_SENSE;
@ -1901,7 +1901,7 @@ NCR_700_queuecommand(struct scsi_cmnd *SCp, void (*done)(struct scsi_cmnd *))
} }
slot->resume_offset = 0; slot->resume_offset = 0;
slot->pCmd = dma_map_single(hostdata->dev, SCp->cmnd, slot->pCmd = dma_map_single(hostdata->dev, SCp->cmnd,
sizeof(SCp->cmnd), DMA_TO_DEVICE); MAX_COMMAND_SIZE, DMA_TO_DEVICE);
NCR_700_start_command(SCp); NCR_700_start_command(SCp);
return 0; return 0;
} }

View file

@ -504,10 +504,9 @@ config SCSI_AIC7XXX_OLD
source "drivers/scsi/aic7xxx/Kconfig.aic79xx" source "drivers/scsi/aic7xxx/Kconfig.aic79xx"
source "drivers/scsi/aic94xx/Kconfig" source "drivers/scsi/aic94xx/Kconfig"
# All the I2O code and drivers do not seem to be 64bit safe.
config SCSI_DPT_I2O config SCSI_DPT_I2O
tristate "Adaptec I2O RAID support " tristate "Adaptec I2O RAID support "
depends on !64BIT && SCSI && PCI && VIRT_TO_BUS depends on SCSI && PCI && VIRT_TO_BUS
help help
This driver supports all of Adaptec's I2O based RAID controllers as This driver supports all of Adaptec's I2O based RAID controllers as
well as the DPT SmartRaid V cards. This is an Adaptec maintained well as the DPT SmartRaid V cards. This is an Adaptec maintained

View file

@ -895,7 +895,7 @@ static void inia100_build_scb(struct orc_host * host, struct orc_scb * scb, stru
} else { } else {
scb->tag_msg = 0; /* No tag support */ scb->tag_msg = 0; /* No tag support */
} }
memcpy(&scb->cdb[0], &cmd->cmnd, scb->cdb_len); memcpy(scb->cdb, cmd->cmnd, scb->cdb_len);
} }
/** /**

View file

@ -498,6 +498,11 @@ static void _aac_probe_container2(void * context, struct fib * fibptr)
(le32_to_cpu(dresp->mnt[0].vol) != CT_NONE) && (le32_to_cpu(dresp->mnt[0].vol) != CT_NONE) &&
(le32_to_cpu(dresp->mnt[0].state) != FSCS_HIDDEN)) { (le32_to_cpu(dresp->mnt[0].state) != FSCS_HIDDEN)) {
fsa_dev_ptr->valid = 1; fsa_dev_ptr->valid = 1;
/* sense_key holds the current state of the spin-up */
if (dresp->mnt[0].state & cpu_to_le32(FSCS_NOT_READY))
fsa_dev_ptr->sense_data.sense_key = NOT_READY;
else if (fsa_dev_ptr->sense_data.sense_key == NOT_READY)
fsa_dev_ptr->sense_data.sense_key = NO_SENSE;
fsa_dev_ptr->type = le32_to_cpu(dresp->mnt[0].vol); fsa_dev_ptr->type = le32_to_cpu(dresp->mnt[0].vol);
fsa_dev_ptr->size fsa_dev_ptr->size
= ((u64)le32_to_cpu(dresp->mnt[0].capacity)) + = ((u64)le32_to_cpu(dresp->mnt[0].capacity)) +
@ -1509,20 +1514,35 @@ static void io_callback(void *context, struct fib * fibptr)
scsi_dma_unmap(scsicmd); scsi_dma_unmap(scsicmd);
readreply = (struct aac_read_reply *)fib_data(fibptr); readreply = (struct aac_read_reply *)fib_data(fibptr);
if (le32_to_cpu(readreply->status) == ST_OK) switch (le32_to_cpu(readreply->status)) {
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD; case ST_OK:
else { scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 |
SAM_STAT_GOOD;
dev->fsa_dev[cid].sense_data.sense_key = NO_SENSE;
break;
case ST_NOT_READY:
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 |
SAM_STAT_CHECK_CONDITION;
set_sense(&dev->fsa_dev[cid].sense_data, NOT_READY,
SENCODE_BECOMING_READY, ASENCODE_BECOMING_READY, 0, 0);
memcpy(scsicmd->sense_buffer, &dev->fsa_dev[cid].sense_data,
min_t(size_t, sizeof(dev->fsa_dev[cid].sense_data),
SCSI_SENSE_BUFFERSIZE));
break;
default:
#ifdef AAC_DETAILED_STATUS_INFO #ifdef AAC_DETAILED_STATUS_INFO
printk(KERN_WARNING "io_callback: io failed, status = %d\n", printk(KERN_WARNING "io_callback: io failed, status = %d\n",
le32_to_cpu(readreply->status)); le32_to_cpu(readreply->status));
#endif #endif
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_CHECK_CONDITION; scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 |
SAM_STAT_CHECK_CONDITION;
set_sense(&dev->fsa_dev[cid].sense_data, set_sense(&dev->fsa_dev[cid].sense_data,
HARDWARE_ERROR, SENCODE_INTERNAL_TARGET_FAILURE, HARDWARE_ERROR, SENCODE_INTERNAL_TARGET_FAILURE,
ASENCODE_INTERNAL_TARGET_FAILURE, 0, 0); ASENCODE_INTERNAL_TARGET_FAILURE, 0, 0);
memcpy(scsicmd->sense_buffer, &dev->fsa_dev[cid].sense_data, memcpy(scsicmd->sense_buffer, &dev->fsa_dev[cid].sense_data,
min_t(size_t, sizeof(dev->fsa_dev[cid].sense_data), min_t(size_t, sizeof(dev->fsa_dev[cid].sense_data),
SCSI_SENSE_BUFFERSIZE)); SCSI_SENSE_BUFFERSIZE));
break;
} }
aac_fib_complete(fibptr); aac_fib_complete(fibptr);
aac_fib_free(fibptr); aac_fib_free(fibptr);
@ -1863,6 +1883,84 @@ static int aac_synchronize(struct scsi_cmnd *scsicmd)
return SCSI_MLQUEUE_HOST_BUSY; return SCSI_MLQUEUE_HOST_BUSY;
} }
static void aac_start_stop_callback(void *context, struct fib *fibptr)
{
struct scsi_cmnd *scsicmd = context;
if (!aac_valid_context(scsicmd, fibptr))
return;
BUG_ON(fibptr == NULL);
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD;
aac_fib_complete(fibptr);
aac_fib_free(fibptr);
scsicmd->scsi_done(scsicmd);
}
static int aac_start_stop(struct scsi_cmnd *scsicmd)
{
int status;
struct fib *cmd_fibcontext;
struct aac_power_management *pmcmd;
struct scsi_device *sdev = scsicmd->device;
struct aac_dev *aac = (struct aac_dev *)sdev->host->hostdata;
if (!(aac->supplement_adapter_info.SupportedOptions2 &
AAC_OPTION_POWER_MANAGEMENT)) {
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 |
SAM_STAT_GOOD;
scsicmd->scsi_done(scsicmd);
return 0;
}
if (aac->in_reset)
return SCSI_MLQUEUE_HOST_BUSY;
/*
* Allocate and initialize a Fib
*/
cmd_fibcontext = aac_fib_alloc(aac);
if (!cmd_fibcontext)
return SCSI_MLQUEUE_HOST_BUSY;
aac_fib_init(cmd_fibcontext);
pmcmd = fib_data(cmd_fibcontext);
pmcmd->command = cpu_to_le32(VM_ContainerConfig);
pmcmd->type = cpu_to_le32(CT_POWER_MANAGEMENT);
/* Eject bit ignored, not relevant */
pmcmd->sub = (scsicmd->cmnd[4] & 1) ?
cpu_to_le32(CT_PM_START_UNIT) : cpu_to_le32(CT_PM_STOP_UNIT);
pmcmd->cid = cpu_to_le32(sdev_id(sdev));
pmcmd->parm = (scsicmd->cmnd[1] & 1) ?
cpu_to_le32(CT_PM_UNIT_IMMEDIATE) : 0;
/*
* Now send the Fib to the adapter
*/
status = aac_fib_send(ContainerCommand,
cmd_fibcontext,
sizeof(struct aac_power_management),
FsaNormal,
0, 1,
(fib_callback)aac_start_stop_callback,
(void *)scsicmd);
/*
* Check that the command queued to the controller
*/
if (status == -EINPROGRESS) {
scsicmd->SCp.phase = AAC_OWNER_FIRMWARE;
return 0;
}
aac_fib_complete(cmd_fibcontext);
aac_fib_free(cmd_fibcontext);
return SCSI_MLQUEUE_HOST_BUSY;
}
/** /**
* aac_scsi_cmd() - Process SCSI command * aac_scsi_cmd() - Process SCSI command
* @scsicmd: SCSI command block * @scsicmd: SCSI command block
@ -1899,7 +1997,9 @@ int aac_scsi_cmd(struct scsi_cmnd * scsicmd)
* If the target container doesn't exist, it may have * If the target container doesn't exist, it may have
* been newly created * been newly created
*/ */
if ((fsa_dev_ptr[cid].valid & 1) == 0) { if (((fsa_dev_ptr[cid].valid & 1) == 0) ||
(fsa_dev_ptr[cid].sense_data.sense_key ==
NOT_READY)) {
switch (scsicmd->cmnd[0]) { switch (scsicmd->cmnd[0]) {
case SERVICE_ACTION_IN: case SERVICE_ACTION_IN:
if (!(dev->raw_io_interface) || if (!(dev->raw_io_interface) ||
@ -2091,8 +2191,8 @@ int aac_scsi_cmd(struct scsi_cmnd * scsicmd)
scsi_sg_copy_from_buffer(scsicmd, cp, sizeof(cp)); scsi_sg_copy_from_buffer(scsicmd, cp, sizeof(cp));
/* Do not cache partition table for arrays */ /* Do not cache partition table for arrays */
scsicmd->device->removable = 1; scsicmd->device->removable = 1;
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 |
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD; SAM_STAT_GOOD;
scsicmd->scsi_done(scsicmd); scsicmd->scsi_done(scsicmd);
return 0; return 0;
@ -2187,15 +2287,32 @@ int aac_scsi_cmd(struct scsi_cmnd * scsicmd)
* These commands are all No-Ops * These commands are all No-Ops
*/ */
case TEST_UNIT_READY: case TEST_UNIT_READY:
if (fsa_dev_ptr[cid].sense_data.sense_key == NOT_READY) {
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 |
SAM_STAT_CHECK_CONDITION;
set_sense(&dev->fsa_dev[cid].sense_data,
NOT_READY, SENCODE_BECOMING_READY,
ASENCODE_BECOMING_READY, 0, 0);
memcpy(scsicmd->sense_buffer,
&dev->fsa_dev[cid].sense_data,
min_t(size_t,
sizeof(dev->fsa_dev[cid].sense_data),
SCSI_SENSE_BUFFERSIZE));
scsicmd->scsi_done(scsicmd);
return 0;
}
/* FALLTHRU */
case RESERVE: case RESERVE:
case RELEASE: case RELEASE:
case REZERO_UNIT: case REZERO_UNIT:
case REASSIGN_BLOCKS: case REASSIGN_BLOCKS:
case SEEK_10: case SEEK_10:
case START_STOP:
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD; scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD;
scsicmd->scsi_done(scsicmd); scsicmd->scsi_done(scsicmd);
return 0; return 0;
case START_STOP:
return aac_start_stop(scsicmd);
} }
switch (scsicmd->cmnd[0]) switch (scsicmd->cmnd[0])

View file

@ -12,7 +12,7 @@
*----------------------------------------------------------------------------*/ *----------------------------------------------------------------------------*/
#ifndef AAC_DRIVER_BUILD #ifndef AAC_DRIVER_BUILD
# define AAC_DRIVER_BUILD 2455 # define AAC_DRIVER_BUILD 2456
# define AAC_DRIVER_BRANCH "-ms" # define AAC_DRIVER_BRANCH "-ms"
#endif #endif
#define MAXIMUM_NUM_CONTAINERS 32 #define MAXIMUM_NUM_CONTAINERS 32
@ -34,8 +34,8 @@
#define CONTAINER_TO_ID(cont) (cont) #define CONTAINER_TO_ID(cont) (cont)
#define CONTAINER_TO_LUN(cont) (0) #define CONTAINER_TO_LUN(cont) (0)
#define aac_phys_to_logical(x) (x+1) #define aac_phys_to_logical(x) ((x)+1)
#define aac_logical_to_phys(x) (x?x-1:0) #define aac_logical_to_phys(x) ((x)?(x)-1:0)
/* #define AAC_DETAILED_STATUS_INFO */ /* #define AAC_DETAILED_STATUS_INFO */
@ -424,6 +424,8 @@ struct aac_init
*/ */
__le32 InitFlags; /* flags for supported features */ __le32 InitFlags; /* flags for supported features */
#define INITFLAGS_NEW_COMM_SUPPORTED 0x00000001 #define INITFLAGS_NEW_COMM_SUPPORTED 0x00000001
#define INITFLAGS_DRIVER_USES_UTC_TIME 0x00000010
#define INITFLAGS_DRIVER_SUPPORTS_PM 0x00000020
__le32 MaxIoCommands; /* max outstanding commands */ __le32 MaxIoCommands; /* max outstanding commands */
__le32 MaxIoSize; /* largest I/O command */ __le32 MaxIoSize; /* largest I/O command */
__le32 MaxFibSize; /* largest FIB to adapter */ __le32 MaxFibSize; /* largest FIB to adapter */
@ -867,8 +869,10 @@ struct aac_supplement_adapter_info
}; };
#define AAC_FEATURE_FALCON cpu_to_le32(0x00000010) #define AAC_FEATURE_FALCON cpu_to_le32(0x00000010)
#define AAC_FEATURE_JBOD cpu_to_le32(0x08000000) #define AAC_FEATURE_JBOD cpu_to_le32(0x08000000)
/* SupportedOptions2 */
#define AAC_OPTION_MU_RESET cpu_to_le32(0x00000001) #define AAC_OPTION_MU_RESET cpu_to_le32(0x00000001)
#define AAC_OPTION_IGNORE_RESET cpu_to_le32(0x00000002) #define AAC_OPTION_IGNORE_RESET cpu_to_le32(0x00000002)
#define AAC_OPTION_POWER_MANAGEMENT cpu_to_le32(0x00000004)
#define AAC_SIS_VERSION_V3 3 #define AAC_SIS_VERSION_V3 3
#define AAC_SIS_SLOT_UNKNOWN 0xFF #define AAC_SIS_SLOT_UNKNOWN 0xFF
@ -1148,6 +1152,7 @@ struct aac_dev
#define ST_DQUOT 69 #define ST_DQUOT 69
#define ST_STALE 70 #define ST_STALE 70
#define ST_REMOTE 71 #define ST_REMOTE 71
#define ST_NOT_READY 72
#define ST_BADHANDLE 10001 #define ST_BADHANDLE 10001
#define ST_NOT_SYNC 10002 #define ST_NOT_SYNC 10002
#define ST_BAD_COOKIE 10003 #define ST_BAD_COOKIE 10003
@ -1269,6 +1274,18 @@ struct aac_synchronize_reply {
u8 data[16]; u8 data[16];
}; };
#define CT_POWER_MANAGEMENT 245
#define CT_PM_START_UNIT 2
#define CT_PM_STOP_UNIT 3
#define CT_PM_UNIT_IMMEDIATE 1
struct aac_power_management {
__le32 command; /* VM_ContainerConfig */
__le32 type; /* CT_POWER_MANAGEMENT */
__le32 sub; /* CT_PM_* */
__le32 cid;
__le32 parm; /* CT_PM_sub_* */
};
#define CT_PAUSE_IO 65 #define CT_PAUSE_IO 65
#define CT_RELEASE_IO 66 #define CT_RELEASE_IO 66
struct aac_pause { struct aac_pause {
@ -1536,6 +1553,7 @@ struct aac_mntent {
#define FSCS_NOTCLEAN 0x0001 /* fsck is necessary before mounting */ #define FSCS_NOTCLEAN 0x0001 /* fsck is necessary before mounting */
#define FSCS_READONLY 0x0002 /* possible result of broken mirror */ #define FSCS_READONLY 0x0002 /* possible result of broken mirror */
#define FSCS_HIDDEN 0x0004 /* should be ignored - set during a clear */ #define FSCS_HIDDEN 0x0004 /* should be ignored - set during a clear */
#define FSCS_NOT_READY 0x0008 /* Array spinning up to fulfil request */
struct aac_query_mount { struct aac_query_mount {
__le32 command; __le32 command;

View file

@ -97,6 +97,8 @@ static int aac_alloc_comm(struct aac_dev *dev, void **commaddr, unsigned long co
init->InitFlags = cpu_to_le32(INITFLAGS_NEW_COMM_SUPPORTED); init->InitFlags = cpu_to_le32(INITFLAGS_NEW_COMM_SUPPORTED);
dprintk((KERN_WARNING"aacraid: New Comm Interface enabled\n")); dprintk((KERN_WARNING"aacraid: New Comm Interface enabled\n"));
} }
init->InitFlags |= cpu_to_le32(INITFLAGS_DRIVER_USES_UTC_TIME |
INITFLAGS_DRIVER_SUPPORTS_PM);
init->MaxIoCommands = cpu_to_le32(dev->scsi_host_ptr->can_queue + AAC_NUM_MGT_FIB); init->MaxIoCommands = cpu_to_le32(dev->scsi_host_ptr->can_queue + AAC_NUM_MGT_FIB);
init->MaxIoSize = cpu_to_le32(dev->scsi_host_ptr->max_sectors << 9); init->MaxIoSize = cpu_to_le32(dev->scsi_host_ptr->max_sectors << 9);
init->MaxFibSize = cpu_to_le32(dev->max_fib_size); init->MaxFibSize = cpu_to_le32(dev->max_fib_size);

View file

@ -515,7 +515,7 @@ int aac_fib_send(u16 command, struct fib *fibptr, unsigned long size,
} }
udelay(5); udelay(5);
} }
} else if (down_interruptible(&fibptr->event_wait) == 0) { } else if (down_interruptible(&fibptr->event_wait)) {
fibptr->done = 2; fibptr->done = 2;
up(&fibptr->event_wait); up(&fibptr->event_wait);
} }
@ -906,15 +906,22 @@ static void aac_handle_aif(struct aac_dev * dev, struct fib * fibptr)
case AifEnAddJBOD: case AifEnAddJBOD:
case AifEnDeleteJBOD: case AifEnDeleteJBOD:
container = le32_to_cpu(((__le32 *)aifcmd->data)[1]); container = le32_to_cpu(((__le32 *)aifcmd->data)[1]);
if ((container >> 28)) if ((container >> 28)) {
container = (u32)-1;
break; break;
}
channel = (container >> 24) & 0xF; channel = (container >> 24) & 0xF;
if (channel >= dev->maximum_num_channels) if (channel >= dev->maximum_num_channels) {
container = (u32)-1;
break; break;
}
id = container & 0xFFFF; id = container & 0xFFFF;
if (id >= dev->maximum_num_physicals) if (id >= dev->maximum_num_physicals) {
container = (u32)-1;
break; break;
}
lun = (container >> 16) & 0xFF; lun = (container >> 16) & 0xFF;
container = (u32)-1;
channel = aac_phys_to_logical(channel); channel = aac_phys_to_logical(channel);
device_config_needed = device_config_needed =
(((__le32 *)aifcmd->data)[0] == (((__le32 *)aifcmd->data)[0] ==
@ -933,13 +940,18 @@ static void aac_handle_aif(struct aac_dev * dev, struct fib * fibptr)
case EM_DRIVE_REMOVAL: case EM_DRIVE_REMOVAL:
container = le32_to_cpu( container = le32_to_cpu(
((__le32 *)aifcmd->data)[2]); ((__le32 *)aifcmd->data)[2]);
if ((container >> 28)) if ((container >> 28)) {
container = (u32)-1;
break; break;
}
channel = (container >> 24) & 0xF; channel = (container >> 24) & 0xF;
if (channel >= dev->maximum_num_channels) if (channel >= dev->maximum_num_channels) {
container = (u32)-1;
break; break;
}
id = container & 0xFFFF; id = container & 0xFFFF;
lun = (container >> 16) & 0xFF; lun = (container >> 16) & 0xFF;
container = (u32)-1;
if (id >= dev->maximum_num_physicals) { if (id >= dev->maximum_num_physicals) {
/* legacy dev_t ? */ /* legacy dev_t ? */
if ((0x2000 <= id) || lun || channel || if ((0x2000 <= id) || lun || channel ||
@ -1025,9 +1037,10 @@ static void aac_handle_aif(struct aac_dev * dev, struct fib * fibptr)
break; break;
} }
container = 0;
retry_next:
if (device_config_needed == NOTHING) if (device_config_needed == NOTHING)
for (container = 0; container < dev->maximum_num_containers; for (; container < dev->maximum_num_containers; ++container) {
++container) {
if ((dev->fsa_dev[container].config_waiting_on == 0) && if ((dev->fsa_dev[container].config_waiting_on == 0) &&
(dev->fsa_dev[container].config_needed != NOTHING) && (dev->fsa_dev[container].config_needed != NOTHING) &&
time_before(jiffies, dev->fsa_dev[container].config_waiting_stamp + AIF_SNIFF_TIMEOUT)) { time_before(jiffies, dev->fsa_dev[container].config_waiting_stamp + AIF_SNIFF_TIMEOUT)) {
@ -1110,6 +1123,11 @@ static void aac_handle_aif(struct aac_dev * dev, struct fib * fibptr)
} }
if (device_config_needed == ADD) if (device_config_needed == ADD)
scsi_add_device(dev->scsi_host_ptr, channel, id, lun); scsi_add_device(dev->scsi_host_ptr, channel, id, lun);
if (channel == CONTAINER_CHANNEL) {
container++;
device_config_needed = NOTHING;
goto retry_next;
}
} }
static int _aac_reset_adapter(struct aac_dev *aac, int forced) static int _aac_reset_adapter(struct aac_dev *aac, int forced)

View file

@ -401,6 +401,8 @@ static int aac_biosparm(struct scsi_device *sdev, struct block_device *bdev,
static int aac_slave_configure(struct scsi_device *sdev) static int aac_slave_configure(struct scsi_device *sdev)
{ {
struct aac_dev *aac = (struct aac_dev *)sdev->host->hostdata; struct aac_dev *aac = (struct aac_dev *)sdev->host->hostdata;
if (aac->jbod && (sdev->type == TYPE_DISK))
sdev->removable = 1;
if ((sdev->type == TYPE_DISK) && if ((sdev->type == TYPE_DISK) &&
(sdev_channel(sdev) != CONTAINER_CHANNEL) && (sdev_channel(sdev) != CONTAINER_CHANNEL) &&
(!aac->jbod || sdev->inq_periph_qual) && (!aac->jbod || sdev->inq_periph_qual) &&
@ -809,6 +811,12 @@ static ssize_t aac_show_flags(struct device *cdev,
"SAI_READ_CAPACITY_16\n"); "SAI_READ_CAPACITY_16\n");
if (dev->jbod) if (dev->jbod)
len += snprintf(buf + len, PAGE_SIZE - len, "SUPPORTED_JBOD\n"); len += snprintf(buf + len, PAGE_SIZE - len, "SUPPORTED_JBOD\n");
if (dev->supplement_adapter_info.SupportedOptions2 &
AAC_OPTION_POWER_MANAGEMENT)
len += snprintf(buf + len, PAGE_SIZE - len,
"SUPPORTED_POWER_MANAGEMENT\n");
if (dev->msi)
len += snprintf(buf + len, PAGE_SIZE - len, "PCI_HAS_MSI\n");
return len; return len;
} }

View file

@ -529,10 +529,10 @@ static void asd_remove_dev_attrs(struct asd_ha_struct *asd_ha)
/* The first entry, 0, is used for dynamic ids, the rest for devices /* The first entry, 0, is used for dynamic ids, the rest for devices
* we know about. * we know about.
*/ */
static struct asd_pcidev_struct { static const struct asd_pcidev_struct {
const char * name; const char * name;
int (*setup)(struct asd_ha_struct *asd_ha); int (*setup)(struct asd_ha_struct *asd_ha);
} asd_pcidev_data[] = { } asd_pcidev_data[] __devinitconst = {
/* Id 0 is used for dynamic ids. */ /* Id 0 is used for dynamic ids. */
{ .name = "Adaptec AIC-94xx SAS/SATA Host Adapter", { .name = "Adaptec AIC-94xx SAS/SATA Host Adapter",
.setup = asd_aic9410_setup .setup = asd_aic9410_setup
@ -735,7 +735,7 @@ static int asd_unregister_sas_ha(struct asd_ha_struct *asd_ha)
static int __devinit asd_pci_probe(struct pci_dev *dev, static int __devinit asd_pci_probe(struct pci_dev *dev,
const struct pci_device_id *id) const struct pci_device_id *id)
{ {
struct asd_pcidev_struct *asd_dev; const struct asd_pcidev_struct *asd_dev;
unsigned asd_id = (unsigned) id->driver_data; unsigned asd_id = (unsigned) id->driver_data;
struct asd_ha_struct *asd_ha; struct asd_ha_struct *asd_ha;
struct Scsi_Host *shost; struct Scsi_Host *shost;

View file

@ -28,7 +28,6 @@
#define SERVICE_ACTION_OUT_12 0xa9 #define SERVICE_ACTION_OUT_12 0xa9
#define SERVICE_ACTION_IN_16 0x9e #define SERVICE_ACTION_IN_16 0x9e
#define SERVICE_ACTION_OUT_16 0x9f #define SERVICE_ACTION_OUT_16 0x9f
#define VARIABLE_LENGTH_CMD 0x7f
@ -210,7 +209,7 @@ static void print_opcode_name(unsigned char * cdbp, int cdb_len)
cdb0 = cdbp[0]; cdb0 = cdbp[0];
switch(cdb0) { switch(cdb0) {
case VARIABLE_LENGTH_CMD: case VARIABLE_LENGTH_CMD:
len = cdbp[7] + 8; len = scsi_varlen_cdb_length(cdbp);
if (len < 10) { if (len < 10) {
printk("short variable length command, " printk("short variable length command, "
"len=%d ext_len=%d", len, cdb_len); "len=%d ext_len=%d", len, cdb_len);
@ -300,7 +299,7 @@ static void print_opcode_name(unsigned char * cdbp, int cdb_len)
cdb0 = cdbp[0]; cdb0 = cdbp[0];
switch(cdb0) { switch(cdb0) {
case VARIABLE_LENGTH_CMD: case VARIABLE_LENGTH_CMD:
len = cdbp[7] + 8; len = scsi_varlen_cdb_length(cdbp);
if (len < 10) { if (len < 10) {
printk("short opcode=0x%x command, len=%d " printk("short opcode=0x%x command, len=%d "
"ext_len=%d", cdb0, len, cdb_len); "ext_len=%d", cdb0, len, cdb_len);
@ -335,10 +334,7 @@ void __scsi_print_command(unsigned char *cdb)
int k, len; int k, len;
print_opcode_name(cdb, 0); print_opcode_name(cdb, 0);
if (VARIABLE_LENGTH_CMD == cdb[0]) len = scsi_command_size(cdb);
len = cdb[7] + 8;
else
len = COMMAND_SIZE(cdb[0]);
/* print out all bytes in cdb */ /* print out all bytes in cdb */
for (k = 0; k < len; ++k) for (k = 0; k < len; ++k)
printk(" %02x", cdb[k]); printk(" %02x", cdb[k]);

View file

@ -89,7 +89,7 @@ typedef struct {
int njobs; /* # of jobs sent to HA */ int njobs; /* # of jobs sent to HA */
int qdepth; /* Controller queue depth. */ int qdepth; /* Controller queue depth. */
int wakebase; /* mpx wakeup base index. */ int wakebase; /* mpx wakeup base index. */
uLONG SGsize; /* Scatter/Gather list size. */ uINT SGsize; /* Scatter/Gather list size. */
unsigned heads; /* heads for drives on cntlr. */ unsigned heads; /* heads for drives on cntlr. */
unsigned sectors; /* sectors for drives on cntlr. */ unsigned sectors; /* sectors for drives on cntlr. */
uCHAR do_drive32; /* Flag for Above 16 MB Ability */ uCHAR do_drive32; /* Flag for Above 16 MB Ability */
@ -97,8 +97,8 @@ typedef struct {
char idPAL[4]; /* 4 Bytes Of The ID Pal */ char idPAL[4]; /* 4 Bytes Of The ID Pal */
uCHAR primary; /* 1 For Primary, 0 For Secondary */ uCHAR primary; /* 1 For Primary, 0 For Secondary */
uCHAR eataVersion; /* EATA Version */ uCHAR eataVersion; /* EATA Version */
uLONG cpLength; /* EATA Command Packet Length */ uINT cpLength; /* EATA Command Packet Length */
uLONG spLength; /* EATA Status Packet Length */ uINT spLength; /* EATA Status Packet Length */
uCHAR drqNum; /* DRQ Index (0,5,6,7) */ uCHAR drqNum; /* DRQ Index (0,5,6,7) */
uCHAR flag1; /* EATA Flags 1 (Byte 9) */ uCHAR flag1; /* EATA Flags 1 (Byte 9) */
uCHAR flag2; /* EATA Flags 2 (Byte 30) */ uCHAR flag2; /* EATA Flags 2 (Byte 30) */
@ -107,23 +107,23 @@ typedef struct {
typedef struct { typedef struct {
uSHORT length; // Remaining length of this uSHORT length; // Remaining length of this
uSHORT drvrHBAnum; // Relative HBA # used by the driver uSHORT drvrHBAnum; // Relative HBA # used by the driver
uLONG baseAddr; // Base I/O address uINT baseAddr; // Base I/O address
uSHORT blinkState; // Blink LED state (0=Not in blink LED) uSHORT blinkState; // Blink LED state (0=Not in blink LED)
uCHAR pciBusNum; // PCI Bus # (Optional) uCHAR pciBusNum; // PCI Bus # (Optional)
uCHAR pciDeviceNum; // PCI Device # (Optional) uCHAR pciDeviceNum; // PCI Device # (Optional)
uSHORT hbaFlags; // Miscellaneous HBA flags uSHORT hbaFlags; // Miscellaneous HBA flags
uSHORT Interrupt; // Interrupt set for this device. uSHORT Interrupt; // Interrupt set for this device.
# if (defined(_DPT_ARC)) # if (defined(_DPT_ARC))
uLONG baseLength; uINT baseLength;
ADAPTER_OBJECT *AdapterObject; ADAPTER_OBJECT *AdapterObject;
LARGE_INTEGER DmaLogicalAddress; LARGE_INTEGER DmaLogicalAddress;
PVOID DmaVirtualAddress; PVOID DmaVirtualAddress;
LARGE_INTEGER ReplyLogicalAddress; LARGE_INTEGER ReplyLogicalAddress;
PVOID ReplyVirtualAddress; PVOID ReplyVirtualAddress;
# else # else
uLONG reserved1; // Reserved for future expansion uINT reserved1; // Reserved for future expansion
uLONG reserved2; // Reserved for future expansion uINT reserved2; // Reserved for future expansion
uLONG reserved3; // Reserved for future expansion uINT reserved3; // Reserved for future expansion
# endif # endif
} drvrHBAinfo_S; } drvrHBAinfo_S;

View file

@ -33,11 +33,7 @@
/* to make sure we are talking the same size under all OS's */ /* to make sure we are talking the same size under all OS's */
typedef unsigned char sigBYTE; typedef unsigned char sigBYTE;
typedef unsigned short sigWORD; typedef unsigned short sigWORD;
#if (defined(_MULTI_DATAMODEL) && defined(sun) && !defined(_ILP32)) typedef unsigned int sigINT;
typedef uint32_t sigLONG;
#else
typedef unsigned long sigLONG;
#endif
/* /*
* use sigWORDLittleEndian for: * use sigWORDLittleEndian for:
@ -300,7 +296,7 @@ typedef struct dpt_sig {
sigBYTE dsFiletype; /* type of file */ sigBYTE dsFiletype; /* type of file */
sigBYTE dsFiletypeFlags; /* flags to specify load type, etc. */ sigBYTE dsFiletypeFlags; /* flags to specify load type, etc. */
sigBYTE dsOEM; /* OEM file was created for */ sigBYTE dsOEM; /* OEM file was created for */
sigLONG dsOS; /* which Operating systems */ sigINT dsOS; /* which Operating systems */
sigWORD dsCapabilities; /* RAID levels, etc. */ sigWORD dsCapabilities; /* RAID levels, etc. */
sigWORD dsDeviceSupp; /* Types of SCSI devices supported */ sigWORD dsDeviceSupp; /* Types of SCSI devices supported */
sigWORD dsAdapterSupp; /* DPT adapter families supported */ sigWORD dsAdapterSupp; /* DPT adapter families supported */

View file

@ -145,8 +145,8 @@
uCHAR smartROMRevision; uCHAR smartROMRevision;
uSHORT flags; /* See bit definitions above */ uSHORT flags; /* See bit definitions above */
uSHORT conventionalMemSize; /* in KB */ uSHORT conventionalMemSize; /* in KB */
uLONG extendedMemSize; /* in KB */ uINT extendedMemSize; /* in KB */
uLONG osType; /* Same as DPTSIG's definition */ uINT osType; /* Same as DPTSIG's definition */
uCHAR osMajorVersion; uCHAR osMajorVersion;
uCHAR osMinorVersion; /* The OS version */ uCHAR osMinorVersion; /* The OS version */
uCHAR osRevision; uCHAR osRevision;

File diff suppressed because it is too large Load diff

View file

@ -84,7 +84,6 @@ static int adpt_device_reset(struct scsi_cmnd* cmd);
#define PCI_DPT_DEVICE_ID (0xA501) // DPT PCI I2O Device ID #define PCI_DPT_DEVICE_ID (0xA501) // DPT PCI I2O Device ID
#define PCI_DPT_RAPTOR_DEVICE_ID (0xA511) #define PCI_DPT_RAPTOR_DEVICE_ID (0xA511)
//#define REBOOT_NOTIFIER 1
/* Debugging macro from Linux Device Drivers - Rubini */ /* Debugging macro from Linux Device Drivers - Rubini */
#undef PDEBUG #undef PDEBUG
#ifdef DEBUG #ifdef DEBUG
@ -229,14 +228,19 @@ typedef struct _adpt_hba {
u32 post_fifo_size; u32 post_fifo_size;
u32 reply_fifo_size; u32 reply_fifo_size;
u32* reply_pool; u32* reply_pool;
dma_addr_t reply_pool_pa;
u32 sg_tablesize; // Scatter/Gather List Size. u32 sg_tablesize; // Scatter/Gather List Size.
u8 top_scsi_channel; u8 top_scsi_channel;
u8 top_scsi_id; u8 top_scsi_id;
u8 top_scsi_lun; u8 top_scsi_lun;
u8 dma64;
i2o_status_block* status_block; i2o_status_block* status_block;
dma_addr_t status_block_pa;
i2o_hrt* hrt; i2o_hrt* hrt;
dma_addr_t hrt_pa;
i2o_lct* lct; i2o_lct* lct;
dma_addr_t lct_pa;
uint lct_size; uint lct_size;
struct i2o_device* devices; struct i2o_device* devices;
struct adpt_channel channel[MAX_CHANNEL]; struct adpt_channel channel[MAX_CHANNEL];
@ -249,6 +253,7 @@ typedef struct _adpt_hba {
void __iomem *FwDebugBLEDflag_P;// Virtual Addr Of FW Debug BLED void __iomem *FwDebugBLEDflag_P;// Virtual Addr Of FW Debug BLED
void __iomem *FwDebugBLEDvalue_P;// Virtual Addr Of FW Debug BLED void __iomem *FwDebugBLEDvalue_P;// Virtual Addr Of FW Debug BLED
u32 FwDebugFlags; u32 FwDebugFlags;
u32 *ioctl_reply_context[4];
} adpt_hba; } adpt_hba;
struct sg_simple_element { struct sg_simple_element {
@ -264,9 +269,6 @@ static void adpt_i2o_sys_shutdown(void);
static int adpt_init(void); static int adpt_init(void);
static int adpt_i2o_build_sys_table(void); static int adpt_i2o_build_sys_table(void);
static irqreturn_t adpt_isr(int irq, void *dev_id); static irqreturn_t adpt_isr(int irq, void *dev_id);
#ifdef REBOOT_NOTIFIER
static int adpt_reboot_event(struct notifier_block *n, ulong code, void *p);
#endif
static void adpt_i2o_report_hba_unit(adpt_hba* pHba, struct i2o_device *d); static void adpt_i2o_report_hba_unit(adpt_hba* pHba, struct i2o_device *d);
static int adpt_i2o_query_scalar(adpt_hba* pHba, int tid, static int adpt_i2o_query_scalar(adpt_hba* pHba, int tid,
@ -275,7 +277,8 @@ static int adpt_i2o_query_scalar(adpt_hba* pHba, int tid,
static const char *adpt_i2o_get_class_name(int class); static const char *adpt_i2o_get_class_name(int class);
#endif #endif
static int adpt_i2o_issue_params(int cmd, adpt_hba* pHba, int tid, static int adpt_i2o_issue_params(int cmd, adpt_hba* pHba, int tid,
void *opblk, int oplen, void *resblk, int reslen); void *opblk, dma_addr_t opblk_pa, int oplen,
void *resblk, dma_addr_t resblk_pa, int reslen);
static int adpt_i2o_post_wait(adpt_hba* pHba, u32* msg, int len, int timeout); static int adpt_i2o_post_wait(adpt_hba* pHba, u32* msg, int len, int timeout);
static int adpt_i2o_lct_get(adpt_hba* pHba); static int adpt_i2o_lct_get(adpt_hba* pHba);
static int adpt_i2o_parse_lct(adpt_hba* pHba); static int adpt_i2o_parse_lct(adpt_hba* pHba);
@ -289,7 +292,7 @@ static s32 adpt_i2o_init_outbound_q(adpt_hba* pHba);
static s32 adpt_i2o_hrt_get(adpt_hba* pHba); static s32 adpt_i2o_hrt_get(adpt_hba* pHba);
static s32 adpt_scsi_to_i2o(adpt_hba* pHba, struct scsi_cmnd* cmd, struct adpt_device* dptdevice); static s32 adpt_scsi_to_i2o(adpt_hba* pHba, struct scsi_cmnd* cmd, struct adpt_device* dptdevice);
static s32 adpt_i2o_to_scsi(void __iomem *reply, struct scsi_cmnd* cmd); static s32 adpt_i2o_to_scsi(void __iomem *reply, struct scsi_cmnd* cmd);
static s32 adpt_scsi_register(adpt_hba* pHba,struct scsi_host_template * sht); static s32 adpt_scsi_host_alloc(adpt_hba* pHba,struct scsi_host_template * sht);
static s32 adpt_hba_reset(adpt_hba* pHba); static s32 adpt_hba_reset(adpt_hba* pHba);
static s32 adpt_i2o_reset_hba(adpt_hba* pHba); static s32 adpt_i2o_reset_hba(adpt_hba* pHba);
static s32 adpt_rescan(adpt_hba* pHba); static s32 adpt_rescan(adpt_hba* pHba);

View file

@ -465,7 +465,7 @@ int __gdth_execute(struct scsi_device *sdev, gdth_cmd_str *gdtcmd, char *cmnd,
scp->request = (struct request *)&wait; scp->request = (struct request *)&wait;
scp->timeout_per_command = timeout*HZ; scp->timeout_per_command = timeout*HZ;
scp->cmd_len = 12; scp->cmd_len = 12;
memcpy(scp->cmnd, cmnd, 12); scp->cmnd = cmnd;
cmndinfo.priority = IOCTL_PRI; cmndinfo.priority = IOCTL_PRI;
cmndinfo.internal_cmd_str = gdtcmd; cmndinfo.internal_cmd_str = gdtcmd;
cmndinfo.internal_command = 1; cmndinfo.internal_command = 1;

View file

@ -763,9 +763,9 @@ static int hptiop_queuecommand(struct scsi_cmnd *scp,
scp, scp,
host->host_no, scp->device->channel, host->host_no, scp->device->channel,
scp->device->id, scp->device->lun, scp->device->id, scp->device->lun,
*((u32 *)&scp->cmnd), ((u32 *)scp->cmnd)[0],
*((u32 *)&scp->cmnd + 1), ((u32 *)scp->cmnd)[1],
*((u32 *)&scp->cmnd + 2), ((u32 *)scp->cmnd)[2],
_req->index, _req->req_virt); _req->index, _req->req_virt);
scp->result = 0; scp->result = 0;

View file

@ -686,7 +686,7 @@ static void handle_cmd_rsp(struct srp_event_struct *evt_struct)
} }
if (cmnd) { if (cmnd) {
cmnd->result = rsp->status; cmnd->result |= rsp->status;
if (((cmnd->result >> 1) & 0x1f) == CHECK_CONDITION) if (((cmnd->result >> 1) & 0x1f) == CHECK_CONDITION)
memcpy(cmnd->sense_buffer, memcpy(cmnd->sense_buffer,
rsp->data, rsp->data,
@ -730,6 +730,7 @@ static int ibmvscsi_queuecommand(struct scsi_cmnd *cmnd,
u16 lun = lun_from_dev(cmnd->device); u16 lun = lun_from_dev(cmnd->device);
u8 out_fmt, in_fmt; u8 out_fmt, in_fmt;
cmnd->result = (DID_OK << 16);
evt_struct = get_event_struct(&hostdata->pool); evt_struct = get_event_struct(&hostdata->pool);
if (!evt_struct) if (!evt_struct)
return SCSI_MLQUEUE_HOST_BUSY; return SCSI_MLQUEUE_HOST_BUSY;
@ -738,7 +739,7 @@ static int ibmvscsi_queuecommand(struct scsi_cmnd *cmnd,
srp_cmd = &evt_struct->iu.srp.cmd; srp_cmd = &evt_struct->iu.srp.cmd;
memset(srp_cmd, 0x00, SRP_MAX_IU_LEN); memset(srp_cmd, 0x00, SRP_MAX_IU_LEN);
srp_cmd->opcode = SRP_CMD; srp_cmd->opcode = SRP_CMD;
memcpy(srp_cmd->cdb, cmnd->cmnd, sizeof(cmnd->cmnd)); memcpy(srp_cmd->cdb, cmnd->cmnd, sizeof(srp_cmd->cdb));
srp_cmd->lun = ((u64) lun) << 48; srp_cmd->lun = ((u64) lun) << 48;
if (!map_data_for_srp_cmd(cmnd, evt_struct, srp_cmd, hostdata->dev)) { if (!map_data_for_srp_cmd(cmnd, evt_struct, srp_cmd, hostdata->dev)) {
@ -1347,6 +1348,8 @@ void ibmvscsi_handle_crq(struct viosrp_crq *crq,
del_timer(&evt_struct->timer); del_timer(&evt_struct->timer);
if (crq->status != VIOSRP_OK && evt_struct->cmnd)
evt_struct->cmnd->result = DID_ERROR << 16;
if (evt_struct->done) if (evt_struct->done)
evt_struct->done(evt_struct); evt_struct->done(evt_struct);
else else

View file

@ -59,6 +59,15 @@ enum viosrp_crq_formats {
VIOSRP_INLINE_FORMAT = 0x07 VIOSRP_INLINE_FORMAT = 0x07
}; };
enum viosrp_crq_status {
VIOSRP_OK = 0x0,
VIOSRP_NONRECOVERABLE_ERR = 0x1,
VIOSRP_VIOLATES_MAX_XFER = 0x2,
VIOSRP_PARTNER_PANIC = 0x3,
VIOSRP_DEVICE_BUSY = 0x8,
VIOSRP_ADAPTER_FAIL = 0x10
};
struct viosrp_crq { struct viosrp_crq {
u8 valid; /* used by RPA */ u8 valid; /* used by RPA */
u8 format; /* SCSI vs out-of-band */ u8 format; /* SCSI vs out-of-band */

View file

@ -2590,7 +2590,7 @@ static void initio_build_scb(struct initio_host * host, struct scsi_ctrl_blk * c
cblk->hastat = 0; cblk->hastat = 0;
cblk->tastat = 0; cblk->tastat = 0;
/* Command the command */ /* Command the command */
memcpy(&cblk->cdb[0], &cmnd->cmnd, cmnd->cmd_len); memcpy(cblk->cdb, cmnd->cmnd, cmnd->cmd_len);
/* Set up tags */ /* Set up tags */
if (cmnd->device->tagged_supported) { /* Tag Support */ if (cmnd->device->tagged_supported) { /* Tag Support */

View file

@ -2791,7 +2791,7 @@ static ssize_t ipr_store_adapter_state(struct device *dev,
static struct device_attribute ipr_ioa_state_attr = { static struct device_attribute ipr_ioa_state_attr = {
.attr = { .attr = {
.name = "state", .name = "online_state",
.mode = S_IRUGO | S_IWUSR, .mode = S_IRUGO | S_IWUSR,
}, },
.show = ipr_show_adapter_state, .show = ipr_show_adapter_state,

View file

@ -3168,6 +3168,23 @@ megaraid_mbox_support_random_del(adapter_t *adapter)
uint8_t raw_mbox[sizeof(mbox_t)]; uint8_t raw_mbox[sizeof(mbox_t)];
int rval; int rval;
/*
* Newer firmware on Dell CERC expect a different
* random deletion handling, so disable it.
*/
if (adapter->pdev->vendor == PCI_VENDOR_ID_AMI &&
adapter->pdev->device == PCI_DEVICE_ID_AMI_MEGARAID3 &&
adapter->pdev->subsystem_vendor == PCI_VENDOR_ID_DELL &&
adapter->pdev->subsystem_device == PCI_SUBSYS_ID_CERC_ATA100_4CH &&
(adapter->fw_version[0] > '6' ||
(adapter->fw_version[0] == '6' &&
adapter->fw_version[2] > '6') ||
(adapter->fw_version[0] == '6'
&& adapter->fw_version[2] == '6'
&& adapter->fw_version[3] > '1'))) {
con_log(CL_DLEVEL1, ("megaraid: disable random deletion\n"));
return 0;
}
mbox = (mbox_t *)raw_mbox; mbox = (mbox_t *)raw_mbox;

View file

@ -88,6 +88,7 @@
#define PCI_SUBSYS_ID_PERC3_QC 0x0471 #define PCI_SUBSYS_ID_PERC3_QC 0x0471
#define PCI_SUBSYS_ID_PERC3_DC 0x0493 #define PCI_SUBSYS_ID_PERC3_DC 0x0493
#define PCI_SUBSYS_ID_PERC3_SC 0x0475 #define PCI_SUBSYS_ID_PERC3_SC 0x0475
#define PCI_SUBSYS_ID_CERC_ATA100_4CH 0x0511
#define MBOX_MAX_SCSI_CMDS 128 // number of cmds reserved for kernel #define MBOX_MAX_SCSI_CMDS 128 // number of cmds reserved for kernel

View file

@ -10,7 +10,7 @@
* 2 of the License, or (at your option) any later version. * 2 of the License, or (at your option) any later version.
* *
* FILE : megaraid_sas.c * FILE : megaraid_sas.c
* Version : v00.00.03.16-rc1 * Version : v00.00.03.20-rc1
* *
* Authors: * Authors:
* (email-id : megaraidlinux@lsi.com) * (email-id : megaraidlinux@lsi.com)
@ -2650,12 +2650,13 @@ static void megasas_shutdown_controller(struct megasas_instance *instance,
return; return;
} }
#ifdef CONFIG_PM
/** /**
* megasas_suspend - driver suspend entry point * megasas_suspend - driver suspend entry point
* @pdev: PCI device structure * @pdev: PCI device structure
* @state: PCI power state to suspend routine * @state: PCI power state to suspend routine
*/ */
static int __devinit static int
megasas_suspend(struct pci_dev *pdev, pm_message_t state) megasas_suspend(struct pci_dev *pdev, pm_message_t state)
{ {
struct Scsi_Host *host; struct Scsi_Host *host;
@ -2687,7 +2688,7 @@ megasas_suspend(struct pci_dev *pdev, pm_message_t state)
* megasas_resume- driver resume entry point * megasas_resume- driver resume entry point
* @pdev: PCI device structure * @pdev: PCI device structure
*/ */
static int __devinit static int
megasas_resume(struct pci_dev *pdev) megasas_resume(struct pci_dev *pdev)
{ {
int rval; int rval;
@ -2782,12 +2783,16 @@ megasas_resume(struct pci_dev *pdev)
return -ENODEV; return -ENODEV;
} }
#else
#define megasas_suspend NULL
#define megasas_resume NULL
#endif
/** /**
* megasas_detach_one - PCI hot"un"plug entry point * megasas_detach_one - PCI hot"un"plug entry point
* @pdev: PCI device structure * @pdev: PCI device structure
*/ */
static void megasas_detach_one(struct pci_dev *pdev) static void __devexit megasas_detach_one(struct pci_dev *pdev)
{ {
int i; int i;
struct Scsi_Host *host; struct Scsi_Host *host;

View file

@ -18,9 +18,9 @@
/* /*
* MegaRAID SAS Driver meta data * MegaRAID SAS Driver meta data
*/ */
#define MEGASAS_VERSION "00.00.03.16-rc1" #define MEGASAS_VERSION "00.00.03.20-rc1"
#define MEGASAS_RELDATE "Nov. 07, 2007" #define MEGASAS_RELDATE "March 10, 2008"
#define MEGASAS_EXT_VERSION "Thu. Nov. 07 10:09:32 PDT 2007" #define MEGASAS_EXT_VERSION "Mon. March 10 11:02:31 PDT 2008"
/* /*
* Device IDs * Device IDs

View file

@ -2858,7 +2858,7 @@ qla1280_64bit_start_scsi(struct scsi_qla_host *ha, struct srb * sp)
/* Load SCSI command packet. */ /* Load SCSI command packet. */
pkt->cdb_len = cpu_to_le16(CMD_CDBLEN(cmd)); pkt->cdb_len = cpu_to_le16(CMD_CDBLEN(cmd));
memcpy(pkt->scsi_cdb, &(CMD_CDBP(cmd)), CMD_CDBLEN(cmd)); memcpy(pkt->scsi_cdb, CMD_CDBP(cmd), CMD_CDBLEN(cmd));
/* dprintk(1, "Build packet for command[0]=0x%x\n",pkt->scsi_cdb[0]); */ /* dprintk(1, "Build packet for command[0]=0x%x\n",pkt->scsi_cdb[0]); */
/* Set transfer direction. */ /* Set transfer direction. */
@ -3127,7 +3127,7 @@ qla1280_32bit_start_scsi(struct scsi_qla_host *ha, struct srb * sp)
/* Load SCSI command packet. */ /* Load SCSI command packet. */
pkt->cdb_len = cpu_to_le16(CMD_CDBLEN(cmd)); pkt->cdb_len = cpu_to_le16(CMD_CDBLEN(cmd));
memcpy(pkt->scsi_cdb, &(CMD_CDBP(cmd)), CMD_CDBLEN(cmd)); memcpy(pkt->scsi_cdb, CMD_CDBP(cmd), CMD_CDBLEN(cmd));
/*dprintk(1, "Build packet for command[0]=0x%x\n",pkt->scsi_cdb[0]); */ /*dprintk(1, "Build packet for command[0]=0x%x\n",pkt->scsi_cdb[0]); */
/* Set transfer direction. */ /* Set transfer direction. */

View file

@ -78,15 +78,6 @@ static void scsi_done(struct scsi_cmnd *cmd);
/* Do not call reset on error if we just did a reset within 15 sec. */ /* Do not call reset on error if we just did a reset within 15 sec. */
#define MIN_RESET_PERIOD (15*HZ) #define MIN_RESET_PERIOD (15*HZ)
/*
* Macro to determine the size of SCSI command. This macro takes vendor
* unique commands into account. SCSI commands in groups 6 and 7 are
* vendor unique and we will depend upon the command length being
* supplied correctly in cmd_len.
*/
#define CDB_SIZE(cmd) (((((cmd)->cmnd[0] >> 5) & 7) < 6) ? \
COMMAND_SIZE((cmd)->cmnd[0]) : (cmd)->cmd_len)
/* /*
* Note - the initial logging level can be set here to log events at boot time. * Note - the initial logging level can be set here to log events at boot time.
* After the system is up, you may enable logging via the /proc interface. * After the system is up, you may enable logging via the /proc interface.
@ -469,6 +460,7 @@ int scsi_setup_command_freelist(struct Scsi_Host *shost)
cmd = scsi_pool_alloc_command(shost->cmd_pool, gfp_mask); cmd = scsi_pool_alloc_command(shost->cmd_pool, gfp_mask);
if (!cmd) { if (!cmd) {
scsi_put_host_cmd_pool(gfp_mask); scsi_put_host_cmd_pool(gfp_mask);
shost->cmd_pool = NULL;
return -ENOMEM; return -ENOMEM;
} }
list_add(&cmd->list, &shost->free_list); list_add(&cmd->list, &shost->free_list);
@ -481,6 +473,13 @@ int scsi_setup_command_freelist(struct Scsi_Host *shost)
*/ */
void scsi_destroy_command_freelist(struct Scsi_Host *shost) void scsi_destroy_command_freelist(struct Scsi_Host *shost)
{ {
/*
* If cmd_pool is NULL the free list was not initialized, so
* do not attempt to release resources.
*/
if (!shost->cmd_pool)
return;
while (!list_empty(&shost->free_list)) { while (!list_empty(&shost->free_list)) {
struct scsi_cmnd *cmd; struct scsi_cmnd *cmd;
@ -701,9 +700,11 @@ int scsi_dispatch_cmd(struct scsi_cmnd *cmd)
* Before we queue this command, check if the command * Before we queue this command, check if the command
* length exceeds what the host adapter can handle. * length exceeds what the host adapter can handle.
*/ */
if (CDB_SIZE(cmd) > cmd->device->host->max_cmd_len) { if (cmd->cmd_len > cmd->device->host->max_cmd_len) {
SCSI_LOG_MLQUEUE(3, SCSI_LOG_MLQUEUE(3,
printk("queuecommand : command too long.\n")); printk("queuecommand : command too long. "
"cdb_size=%d host->max_cmd_len=%d\n",
cmd->cmd_len, cmd->device->host->max_cmd_len));
cmd->result = (DID_ABORT << 16); cmd->result = (DID_ABORT << 16);
scsi_done(cmd); scsi_done(cmd);

View file

@ -626,7 +626,7 @@ static void scsi_abort_eh_cmnd(struct scsi_cmnd *scmd)
* @scmd: SCSI command structure to hijack * @scmd: SCSI command structure to hijack
* @ses: structure to save restore information * @ses: structure to save restore information
* @cmnd: CDB to send. Can be NULL if no new cmnd is needed * @cmnd: CDB to send. Can be NULL if no new cmnd is needed
* @cmnd_size: size in bytes of @cmnd * @cmnd_size: size in bytes of @cmnd (must be <= BLK_MAX_CDB)
* @sense_bytes: size of sense data to copy. or 0 (if != 0 @cmnd is ignored) * @sense_bytes: size of sense data to copy. or 0 (if != 0 @cmnd is ignored)
* *
* This function is used to save a scsi command information before re-execution * This function is used to save a scsi command information before re-execution
@ -648,12 +648,14 @@ void scsi_eh_prep_cmnd(struct scsi_cmnd *scmd, struct scsi_eh_save *ses,
* command. * command.
*/ */
ses->cmd_len = scmd->cmd_len; ses->cmd_len = scmd->cmd_len;
memcpy(ses->cmnd, scmd->cmnd, sizeof(scmd->cmnd)); ses->cmnd = scmd->cmnd;
ses->data_direction = scmd->sc_data_direction; ses->data_direction = scmd->sc_data_direction;
ses->sdb = scmd->sdb; ses->sdb = scmd->sdb;
ses->next_rq = scmd->request->next_rq; ses->next_rq = scmd->request->next_rq;
ses->result = scmd->result; ses->result = scmd->result;
scmd->cmnd = ses->eh_cmnd;
memset(scmd->cmnd, 0, BLK_MAX_CDB);
memset(&scmd->sdb, 0, sizeof(scmd->sdb)); memset(&scmd->sdb, 0, sizeof(scmd->sdb));
scmd->request->next_rq = NULL; scmd->request->next_rq = NULL;
@ -665,14 +667,13 @@ void scsi_eh_prep_cmnd(struct scsi_cmnd *scmd, struct scsi_eh_save *ses,
scmd->sdb.table.sgl = &ses->sense_sgl; scmd->sdb.table.sgl = &ses->sense_sgl;
scmd->sc_data_direction = DMA_FROM_DEVICE; scmd->sc_data_direction = DMA_FROM_DEVICE;
scmd->sdb.table.nents = 1; scmd->sdb.table.nents = 1;
memset(scmd->cmnd, 0, sizeof(scmd->cmnd));
scmd->cmnd[0] = REQUEST_SENSE; scmd->cmnd[0] = REQUEST_SENSE;
scmd->cmnd[4] = scmd->sdb.length; scmd->cmnd[4] = scmd->sdb.length;
scmd->cmd_len = COMMAND_SIZE(scmd->cmnd[0]); scmd->cmd_len = COMMAND_SIZE(scmd->cmnd[0]);
} else { } else {
scmd->sc_data_direction = DMA_NONE; scmd->sc_data_direction = DMA_NONE;
if (cmnd) { if (cmnd) {
memset(scmd->cmnd, 0, sizeof(scmd->cmnd)); BUG_ON(cmnd_size > BLK_MAX_CDB);
memcpy(scmd->cmnd, cmnd, cmnd_size); memcpy(scmd->cmnd, cmnd, cmnd_size);
scmd->cmd_len = COMMAND_SIZE(scmd->cmnd[0]); scmd->cmd_len = COMMAND_SIZE(scmd->cmnd[0]);
} }
@ -705,7 +706,7 @@ void scsi_eh_restore_cmnd(struct scsi_cmnd* scmd, struct scsi_eh_save *ses)
* Restore original data * Restore original data
*/ */
scmd->cmd_len = ses->cmd_len; scmd->cmd_len = ses->cmd_len;
memcpy(scmd->cmnd, ses->cmnd, sizeof(scmd->cmnd)); scmd->cmnd = ses->cmnd;
scmd->sc_data_direction = ses->data_direction; scmd->sc_data_direction = ses->data_direction;
scmd->sdb = ses->sdb; scmd->sdb = ses->sdb;
scmd->request->next_rq = ses->next_rq; scmd->request->next_rq = ses->next_rq;
@ -1775,7 +1776,7 @@ scsi_reset_provider(struct scsi_device *dev, int flag)
scmd->request = &req; scmd->request = &req;
memset(&scmd->eh_timeout, 0, sizeof(scmd->eh_timeout)); memset(&scmd->eh_timeout, 0, sizeof(scmd->eh_timeout));
memset(&scmd->cmnd, '\0', sizeof(scmd->cmnd)); scmd->cmnd = req.cmd;
scmd->scsi_done = scsi_reset_provider_done_command; scmd->scsi_done = scsi_reset_provider_done_command;
memset(&scmd->sdb, 0, sizeof(scmd->sdb)); memset(&scmd->sdb, 0, sizeof(scmd->sdb));

View file

@ -445,7 +445,7 @@ static void scsi_init_cmd_errh(struct scsi_cmnd *cmd)
scsi_set_resid(cmd, 0); scsi_set_resid(cmd, 0);
memset(cmd->sense_buffer, 0, SCSI_SENSE_BUFFERSIZE); memset(cmd->sense_buffer, 0, SCSI_SENSE_BUFFERSIZE);
if (cmd->cmd_len == 0) if (cmd->cmd_len == 0)
cmd->cmd_len = COMMAND_SIZE(cmd->cmnd[0]); cmd->cmd_len = scsi_command_size(cmd->cmnd);
} }
void scsi_device_unbusy(struct scsi_device *sdev) void scsi_device_unbusy(struct scsi_device *sdev)
@ -1094,6 +1094,8 @@ static struct scsi_cmnd *scsi_get_cmd_from_req(struct scsi_device *sdev,
cmd->tag = req->tag; cmd->tag = req->tag;
cmd->request = req; cmd->request = req;
cmd->cmnd = req->cmd;
return cmd; return cmd;
} }
@ -1131,8 +1133,6 @@ int scsi_setup_blk_pc_cmnd(struct scsi_device *sdev, struct request *req)
req->buffer = NULL; req->buffer = NULL;
} }
BUILD_BUG_ON(sizeof(req->cmd) > sizeof(cmd->cmnd));
memcpy(cmd->cmnd, req->cmd, sizeof(cmd->cmnd));
cmd->cmd_len = req->cmd_len; cmd->cmd_len = req->cmd_len;
if (!req->data_len) if (!req->data_len)
cmd->sc_data_direction = DMA_NONE; cmd->sc_data_direction = DMA_NONE;
@ -1169,6 +1169,7 @@ int scsi_setup_fs_cmnd(struct scsi_device *sdev, struct request *req)
if (unlikely(!cmd)) if (unlikely(!cmd))
return BLKPREP_DEFER; return BLKPREP_DEFER;
memset(cmd->cmnd, 0, BLK_MAX_CDB);
return scsi_init_io(cmd, GFP_ATOMIC); return scsi_init_io(cmd, GFP_ATOMIC);
} }
EXPORT_SYMBOL(scsi_setup_fs_cmnd); EXPORT_SYMBOL(scsi_setup_fs_cmnd);

View file

@ -107,6 +107,8 @@ struct scsi_cmnd *scsi_host_get_command(struct Scsi_Host *shost,
cmd->jiffies_at_alloc = jiffies; cmd->jiffies_at_alloc = jiffies;
cmd->request = rq; cmd->request = rq;
cmd->cmnd = rq->cmd;
rq->special = cmd; rq->special = cmd;
rq->cmd_type = REQ_TYPE_SPECIAL; rq->cmd_type = REQ_TYPE_SPECIAL;
rq->cmd_flags |= REQ_TYPE_BLOCK_PC; rq->cmd_flags |= REQ_TYPE_BLOCK_PC;

View file

@ -744,7 +744,8 @@ static int wait_on_busy(unsigned long iobase, unsigned int loop) {
static int board_inquiry(unsigned int j) { static int board_inquiry(unsigned int j) {
struct mscp *cpp; struct mscp *cpp;
dma_addr_t id_dma_addr; dma_addr_t id_dma_addr;
unsigned int time, limit = 0; unsigned int limit = 0;
unsigned long time;
id_dma_addr = pci_map_single(HD(j)->pdev, HD(j)->board_id, id_dma_addr = pci_map_single(HD(j)->pdev, HD(j)->board_id,
sizeof(HD(j)->board_id), PCI_DMA_BIDIRECTIONAL); sizeof(HD(j)->board_id), PCI_DMA_BIDIRECTIONAL);
@ -1392,7 +1393,8 @@ static int u14_34f_eh_abort(struct scsi_cmnd *SCarg) {
} }
static int u14_34f_eh_host_reset(struct scsi_cmnd *SCarg) { static int u14_34f_eh_host_reset(struct scsi_cmnd *SCarg) {
unsigned int i, j, time, k, c, limit = 0; unsigned int i, j, k, c, limit = 0;
unsigned long time;
int arg_done = FALSE; int arg_done = FALSE;
struct scsi_cmnd *SCpnt; struct scsi_cmnd *SCpnt;

View file

@ -46,7 +46,7 @@ void cypress_atacb_passthrough(struct scsi_cmnd *srb, struct us_data *us)
} }
memcpy(save_cmnd, srb->cmnd, sizeof(save_cmnd)); memcpy(save_cmnd, srb->cmnd, sizeof(save_cmnd));
memset(srb->cmnd, 0, sizeof(srb->cmnd)); memset(srb->cmnd, 0, MAX_COMMAND_SIZE);
/* check if we support the command */ /* check if we support the command */
if (save_cmnd[1] >> 5) /* MULTIPLE_COUNT */ if (save_cmnd[1] >> 5) /* MULTIPLE_COUNT */

View file

@ -292,6 +292,7 @@ struct isd200_info {
/* maximum number of LUNs supported */ /* maximum number of LUNs supported */
unsigned char MaxLUNs; unsigned char MaxLUNs;
unsigned char cmnd[BLK_MAX_CDB];
struct scsi_cmnd srb; struct scsi_cmnd srb;
struct scatterlist sg; struct scatterlist sg;
}; };
@ -450,6 +451,7 @@ static int isd200_action( struct us_data *us, int action,
memset(&ata, 0, sizeof(ata)); memset(&ata, 0, sizeof(ata));
memset(&srb_dev, 0, sizeof(srb_dev)); memset(&srb_dev, 0, sizeof(srb_dev));
srb->cmnd = info->cmnd;
srb->device = &srb_dev; srb->device = &srb_dev;
++srb->serial_number; ++srb->serial_number;

View file

@ -29,13 +29,6 @@
#define SCSI_MAX_SG_CHAIN_SEGMENTS SCSI_MAX_SG_SEGMENTS #define SCSI_MAX_SG_CHAIN_SEGMENTS SCSI_MAX_SG_SEGMENTS
#endif #endif
/*
* SCSI command lengths
*/
extern const unsigned char scsi_command_size[8];
#define COMMAND_SIZE(opcode) scsi_command_size[((opcode) >> 5) & 7]
/* /*
* Special value for scanning to specify scanning or rescanning of all * Special value for scanning to specify scanning or rescanning of all
* possible channels, (target) ids, or luns on a given shost. * possible channels, (target) ids, or luns on a given shost.
@ -109,6 +102,7 @@ extern const unsigned char scsi_command_size[8];
#define MODE_SENSE_10 0x5a #define MODE_SENSE_10 0x5a
#define PERSISTENT_RESERVE_IN 0x5e #define PERSISTENT_RESERVE_IN 0x5e
#define PERSISTENT_RESERVE_OUT 0x5f #define PERSISTENT_RESERVE_OUT 0x5f
#define VARIABLE_LENGTH_CMD 0x7f
#define REPORT_LUNS 0xa0 #define REPORT_LUNS 0xa0
#define MAINTENANCE_IN 0xa3 #define MAINTENANCE_IN 0xa3
#define MOVE_MEDIUM 0xa5 #define MOVE_MEDIUM 0xa5
@ -135,6 +129,38 @@ extern const unsigned char scsi_command_size[8];
#define ATA_16 0x85 /* 16-byte pass-thru */ #define ATA_16 0x85 /* 16-byte pass-thru */
#define ATA_12 0xa1 /* 12-byte pass-thru */ #define ATA_12 0xa1 /* 12-byte pass-thru */
/*
* SCSI command lengths
*/
#define SCSI_MAX_VARLEN_CDB_SIZE 260
/* defined in T10 SCSI Primary Commands-2 (SPC2) */
struct scsi_varlen_cdb_hdr {
u8 opcode; /* opcode always == VARIABLE_LENGTH_CMD */
u8 control;
u8 misc[5];
u8 additional_cdb_length; /* total cdb length - 8 */
__be16 service_action;
/* service specific data follows */
};
static inline unsigned
scsi_varlen_cdb_length(const void *hdr)
{
return ((struct scsi_varlen_cdb_hdr *)hdr)->additional_cdb_length + 8;
}
extern const unsigned char scsi_command_size_tbl[8];
#define COMMAND_SIZE(opcode) scsi_command_size_tbl[((opcode) >> 5) & 7]
static inline unsigned
scsi_command_size(const unsigned char *cmnd)
{
return (cmnd[0] == VARIABLE_LENGTH_CMD) ?
scsi_varlen_cdb_length(cmnd) : COMMAND_SIZE(cmnd[0]);
}
/* /*
* SCSI Architecture Model (SAM) Status codes. Taken from SAM-3 draft * SCSI Architecture Model (SAM) Status codes. Taken from SAM-3 draft
* T10/1561-D Revision 4 Draft dated 7th November 2002. * T10/1561-D Revision 4 Draft dated 7th November 2002.

View file

@ -7,10 +7,28 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/timer.h> #include <linux/timer.h>
#include <linux/scatterlist.h> #include <linux/scatterlist.h>
#include <linux/blkdev.h>
struct Scsi_Host; struct Scsi_Host;
struct scsi_device; struct scsi_device;
/*
* MAX_COMMAND_SIZE is:
* The longest fixed-length SCSI CDB as per the SCSI standard.
* fixed-length means: commands that their size can be determined
* by their opcode and the CDB does not carry a length specifier, (unlike
* the VARIABLE_LENGTH_CMD(0x7f) command). This is actually not exactly
* true and the SCSI standard also defines extended commands and
* vendor specific commands that can be bigger than 16 bytes. The kernel
* will support these using the same infrastructure used for VARLEN CDB's.
* So in effect MAX_COMMAND_SIZE means the maximum size command scsi-ml
* supports without specifying a cmd_len by ULD's
*/
#define MAX_COMMAND_SIZE 16
#if (MAX_COMMAND_SIZE > BLK_MAX_CDB)
# error MAX_COMMAND_SIZE can not be bigger than BLK_MAX_CDB
#endif
struct scsi_data_buffer { struct scsi_data_buffer {
struct sg_table table; struct sg_table table;
unsigned length; unsigned length;
@ -60,12 +78,11 @@ struct scsi_cmnd {
int allowed; int allowed;
int timeout_per_command; int timeout_per_command;
unsigned char cmd_len; unsigned short cmd_len;
enum dma_data_direction sc_data_direction; enum dma_data_direction sc_data_direction;
/* These elements define the operation we are about to perform */ /* These elements define the operation we are about to perform */
#define MAX_COMMAND_SIZE 16 unsigned char *cmnd;
unsigned char cmnd[MAX_COMMAND_SIZE];
struct timer_list eh_timeout; /* Used to time out the command. */ struct timer_list eh_timeout; /* Used to time out the command. */

View file

@ -75,11 +75,11 @@ struct scsi_eh_save {
int result; int result;
enum dma_data_direction data_direction; enum dma_data_direction data_direction;
unsigned char cmd_len; unsigned char cmd_len;
unsigned char cmnd[MAX_COMMAND_SIZE]; unsigned char *cmnd;
struct scsi_data_buffer sdb; struct scsi_data_buffer sdb;
struct request *next_rq; struct request *next_rq;
/* new command support */ /* new command support */
unsigned char eh_cmnd[BLK_MAX_CDB];
struct scatterlist sense_sgl; struct scatterlist sense_sgl;
}; };

View file

@ -573,13 +573,11 @@ struct Scsi_Host {
/* /*
* The maximum length of SCSI commands that this host can accept. * The maximum length of SCSI commands that this host can accept.
* Probably 12 for most host adapters, but could be 16 for others. * Probably 12 for most host adapters, but could be 16 for others.
* or 260 if the driver supports variable length cdbs.
* For drivers that don't set this field, a value of 12 is * For drivers that don't set this field, a value of 12 is
* assumed. I am leaving this as a number rather than a bit * assumed.
* because you never know what subsequent SCSI standards might do
* (i.e. could there be a 20 byte or a 24-byte command a few years
* down the road?).
*/ */
unsigned char max_cmd_len; unsigned short max_cmd_len;
int this_id; int this_id;
int can_queue; int can_queue;