Commit 85e81b27 authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] patch to DAC960 driver for error retry

Patch from Dave Olien <dmo@osdl.org>

The following patch implements retry on media errors for the DAC960 driver.
On such media errors, the DAC960 apparently doesn't report how much of the
transfer may have been successful before the error was encountered.

This type of error should be rare on healthy hardware, especially if the
disks are stripped or mirrored.  But, when large transfers are submitted to
the controller, it's especially bad to have to fail the entire transfer
because one disk sector may have been bad.
parent d0863ebd
...@@ -267,10 +267,12 @@ static boolean DAC960_CreateAuxiliaryStructures(DAC960_Controller_T *Controller) ...@@ -267,10 +267,12 @@ static boolean DAC960_CreateAuxiliaryStructures(DAC960_Controller_T *Controller)
} }
} }
if (Controller->FirmwareType == DAC960_V1_Controller) { if (Controller->FirmwareType == DAC960_V1_Controller) {
Command->cmd_sglist = Command->V1.ScatterList;
Command->V1.ScatterGatherList = Command->V1.ScatterGatherList =
(DAC960_V1_ScatterGatherSegment_T *)ScatterGatherCPU; (DAC960_V1_ScatterGatherSegment_T *)ScatterGatherCPU;
Command->V1.ScatterGatherListDMA = ScatterGatherDMA; Command->V1.ScatterGatherListDMA = ScatterGatherDMA;
} else { } else {
Command->cmd_sglist = Command->V2.ScatterList;
Command->V2.ScatterGatherList = Command->V2.ScatterGatherList =
(DAC960_V2_ScatterGatherSegment_T *)ScatterGatherCPU; (DAC960_V2_ScatterGatherSegment_T *)ScatterGatherCPU;
Command->V2.ScatterGatherListDMA = ScatterGatherDMA; Command->V2.ScatterGatherListDMA = ScatterGatherDMA;
...@@ -3046,25 +3048,12 @@ static void DAC960_V1_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3046,25 +3048,12 @@ static void DAC960_V1_QueueReadWriteCommand(DAC960_Command_T *Command)
DAC960_V1_ScatterGatherSegment_T *ScatterGatherList = DAC960_V1_ScatterGatherSegment_T *ScatterGatherList =
Command->V1.ScatterGatherList; Command->V1.ScatterGatherList;
struct scatterlist *ScatterList = Command->V1.ScatterList; struct scatterlist *ScatterList = Command->V1.ScatterList;
int DmaDirection, SegCount;
DAC960_V1_ClearCommand(Command); DAC960_V1_ClearCommand(Command);
if (Command->CommandType == DAC960_ReadCommand) if (Command->SegmentCount == 1)
DmaDirection = PCI_DMA_FROMDEVICE;
else
DmaDirection = PCI_DMA_TODEVICE;
SegCount = blk_rq_map_sg(&Controller->RequestQueue, Command->Request,
ScatterList);
/* pci_map_sg MAY change the value of SegCount */
SegCount = pci_map_sg(Command->PciDevice, ScatterList, SegCount,
DmaDirection);
Command->SegmentCount = SegCount;
if (SegCount == 1)
{ {
if (Command->CommandType == DAC960_ReadCommand) if (Command->DmaDirection == PCI_DMA_FROMDEVICE)
CommandMailbox->Type5.CommandOpcode = DAC960_V1_Read; CommandMailbox->Type5.CommandOpcode = DAC960_V1_Read;
else else
CommandMailbox->Type5.CommandOpcode = DAC960_V1_Write; CommandMailbox->Type5.CommandOpcode = DAC960_V1_Write;
...@@ -3079,7 +3068,7 @@ static void DAC960_V1_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3079,7 +3068,7 @@ static void DAC960_V1_QueueReadWriteCommand(DAC960_Command_T *Command)
{ {
int i; int i;
if (Command->CommandType == DAC960_ReadCommand) if (Command->DmaDirection == PCI_DMA_FROMDEVICE)
CommandMailbox->Type5.CommandOpcode = DAC960_V1_ReadWithScatterGather; CommandMailbox->Type5.CommandOpcode = DAC960_V1_ReadWithScatterGather;
else else
CommandMailbox->Type5.CommandOpcode = DAC960_V1_WriteWithScatterGather; CommandMailbox->Type5.CommandOpcode = DAC960_V1_WriteWithScatterGather;
...@@ -3089,9 +3078,9 @@ static void DAC960_V1_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3089,9 +3078,9 @@ static void DAC960_V1_QueueReadWriteCommand(DAC960_Command_T *Command)
CommandMailbox->Type5.LogicalBlockAddress = Command->BlockNumber; CommandMailbox->Type5.LogicalBlockAddress = Command->BlockNumber;
CommandMailbox->Type5.BusAddress = Command->V1.ScatterGatherListDMA; CommandMailbox->Type5.BusAddress = Command->V1.ScatterGatherListDMA;
CommandMailbox->Type5.ScatterGatherCount = SegCount; CommandMailbox->Type5.ScatterGatherCount = Command->SegmentCount;
for (i = 0; i < SegCount; i++, ScatterList++, ScatterGatherList++) { for (i = 0; i < Command->SegmentCount; i++, ScatterList++, ScatterGatherList++) {
ScatterGatherList->SegmentDataPointer = ScatterGatherList->SegmentDataPointer =
(DAC960_BusAddress32_T)sg_dma_address(ScatterList); (DAC960_BusAddress32_T)sg_dma_address(ScatterList);
ScatterGatherList->SegmentByteCount = ScatterGatherList->SegmentByteCount =
...@@ -3112,25 +3101,12 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3112,25 +3101,12 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command)
DAC960_Controller_T *Controller = Command->Controller; DAC960_Controller_T *Controller = Command->Controller;
DAC960_V2_CommandMailbox_T *CommandMailbox = &Command->V2.CommandMailbox; DAC960_V2_CommandMailbox_T *CommandMailbox = &Command->V2.CommandMailbox;
struct scatterlist *ScatterList = Command->V2.ScatterList; struct scatterlist *ScatterList = Command->V2.ScatterList;
int DmaDirection, SegCount;
DAC960_V2_ClearCommand(Command); DAC960_V2_ClearCommand(Command);
if (Command->CommandType == DAC960_ReadCommand)
DmaDirection = PCI_DMA_FROMDEVICE;
else
DmaDirection = PCI_DMA_TODEVICE;
SegCount = blk_rq_map_sg(&Controller->RequestQueue, Command->Request,
ScatterList);
/* pci_map_sg MAY change the value of SegCount */
SegCount = pci_map_sg(Command->PciDevice, ScatterList, SegCount,
DmaDirection);
Command->SegmentCount = SegCount;
CommandMailbox->SCSI_10.CommandOpcode = DAC960_V2_SCSI_10; CommandMailbox->SCSI_10.CommandOpcode = DAC960_V2_SCSI_10;
CommandMailbox->SCSI_10.CommandControlBits.DataTransferControllerToHost = CommandMailbox->SCSI_10.CommandControlBits.DataTransferControllerToHost =
(Command->CommandType == DAC960_ReadCommand); (Command->DmaDirection == PCI_DMA_FROMDEVICE);
CommandMailbox->SCSI_10.DataTransferSize = CommandMailbox->SCSI_10.DataTransferSize =
Command->BlockCount << DAC960_BlockSizeBits; Command->BlockCount << DAC960_BlockSizeBits;
CommandMailbox->SCSI_10.RequestSenseBusAddress = Command->V2.RequestSenseDMA; CommandMailbox->SCSI_10.RequestSenseBusAddress = Command->V2.RequestSenseDMA;
...@@ -3139,7 +3115,7 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3139,7 +3115,7 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command)
CommandMailbox->SCSI_10.RequestSenseSize = sizeof(DAC960_SCSI_RequestSense_T); CommandMailbox->SCSI_10.RequestSenseSize = sizeof(DAC960_SCSI_RequestSense_T);
CommandMailbox->SCSI_10.CDBLength = 10; CommandMailbox->SCSI_10.CDBLength = 10;
CommandMailbox->SCSI_10.SCSI_CDB[0] = CommandMailbox->SCSI_10.SCSI_CDB[0] =
(Command->CommandType == DAC960_ReadCommand ? 0x28 : 0x2A); (Command->DmaDirection == PCI_DMA_FROMDEVICE ? 0x28 : 0x2A);
CommandMailbox->SCSI_10.SCSI_CDB[2] = Command->BlockNumber >> 24; CommandMailbox->SCSI_10.SCSI_CDB[2] = Command->BlockNumber >> 24;
CommandMailbox->SCSI_10.SCSI_CDB[3] = Command->BlockNumber >> 16; CommandMailbox->SCSI_10.SCSI_CDB[3] = Command->BlockNumber >> 16;
CommandMailbox->SCSI_10.SCSI_CDB[4] = Command->BlockNumber >> 8; CommandMailbox->SCSI_10.SCSI_CDB[4] = Command->BlockNumber >> 8;
...@@ -3147,7 +3123,7 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3147,7 +3123,7 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command)
CommandMailbox->SCSI_10.SCSI_CDB[7] = Command->BlockCount >> 8; CommandMailbox->SCSI_10.SCSI_CDB[7] = Command->BlockCount >> 8;
CommandMailbox->SCSI_10.SCSI_CDB[8] = Command->BlockCount; CommandMailbox->SCSI_10.SCSI_CDB[8] = Command->BlockCount;
if (SegCount == 1) if (Command->SegmentCount == 1)
{ {
CommandMailbox->SCSI_10.DataTransferMemoryAddress CommandMailbox->SCSI_10.DataTransferMemoryAddress
.ScatterGatherSegments[0] .ScatterGatherSegments[0]
...@@ -3163,13 +3139,13 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3163,13 +3139,13 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command)
DAC960_V2_ScatterGatherSegment_T *ScatterGatherList; DAC960_V2_ScatterGatherSegment_T *ScatterGatherList;
int i; int i;
if (SegCount > 2) if (Command->SegmentCount > 2)
{ {
ScatterGatherList = Command->V2.ScatterGatherList; ScatterGatherList = Command->V2.ScatterGatherList;
CommandMailbox->SCSI_10.CommandControlBits CommandMailbox->SCSI_10.CommandControlBits
.AdditionalScatterGatherListMemory = true; .AdditionalScatterGatherListMemory = true;
CommandMailbox->SCSI_10.DataTransferMemoryAddress CommandMailbox->SCSI_10.DataTransferMemoryAddress
.ExtendedScatterGather.ScatterGatherList0Length = SegCount; .ExtendedScatterGather.ScatterGatherList0Length = Command->SegmentCount;
CommandMailbox->SCSI_10.DataTransferMemoryAddress CommandMailbox->SCSI_10.DataTransferMemoryAddress
.ExtendedScatterGather.ScatterGatherList0Address = .ExtendedScatterGather.ScatterGatherList0Address =
Command->V2.ScatterGatherListDMA; Command->V2.ScatterGatherListDMA;
...@@ -3178,7 +3154,7 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command) ...@@ -3178,7 +3154,7 @@ static void DAC960_V2_QueueReadWriteCommand(DAC960_Command_T *Command)
ScatterGatherList = CommandMailbox->SCSI_10.DataTransferMemoryAddress ScatterGatherList = CommandMailbox->SCSI_10.DataTransferMemoryAddress
.ScatterGatherSegments; .ScatterGatherSegments;
for (i = 0; i < SegCount; i++, ScatterList++, ScatterGatherList++) { for (i = 0; i < Command->SegmentCount; i++, ScatterList++, ScatterGatherList++) {
ScatterGatherList->SegmentDataPointer = ScatterGatherList->SegmentDataPointer =
(DAC960_BusAddress64_T)sg_dma_address(ScatterList); (DAC960_BusAddress64_T)sg_dma_address(ScatterList);
ScatterGatherList->SegmentByteCount = ScatterGatherList->SegmentByteCount =
...@@ -3220,21 +3196,74 @@ static boolean DAC960_ProcessRequest(DAC960_Controller_T *Controller, ...@@ -3220,21 +3196,74 @@ static boolean DAC960_ProcessRequest(DAC960_Controller_T *Controller,
DAC960_WaitForCommand(Controller); DAC960_WaitForCommand(Controller);
} }
if (rq_data_dir(Request) == READ) if (rq_data_dir(Request) == READ) {
Command->DmaDirection = PCI_DMA_FROMDEVICE;
Command->CommandType = DAC960_ReadCommand; Command->CommandType = DAC960_ReadCommand;
else } else {
Command->DmaDirection = PCI_DMA_TODEVICE;
Command->CommandType = DAC960_WriteCommand; Command->CommandType = DAC960_WriteCommand;
}
Command->Completion = Request->waiting; Command->Completion = Request->waiting;
Command->LogicalDriveNumber = (int)Request->rq_disk->private_data; Command->LogicalDriveNumber = (int)Request->rq_disk->private_data;
Command->BlockNumber = Request->sector; Command->BlockNumber = Request->sector;
Command->BlockCount = Request->nr_sectors; Command->BlockCount = Request->nr_sectors;
Command->Request = Request; Command->Request = Request;
blkdev_dequeue_request(Request); blkdev_dequeue_request(Request);
Command->SegmentCount = blk_rq_map_sg(&Controller->RequestQueue,
Command->Request, Command->cmd_sglist);
/* pci_map_sg MAY change the value of SegCount */
Command->SegmentCount = pci_map_sg(Command->PciDevice, Command->cmd_sglist,
Command->SegmentCount, Command->DmaDirection);
DAC960_QueueReadWriteCommand(Command); DAC960_QueueReadWriteCommand(Command);
return true; return true;
} }
/*
DAC960_queue_partial_rw extracts one bio from the request already
associated with argument command, and construct a new command block to retry I/O
only on that bio. Queue that command to the controller.
This function re-uses a previously-allocated Command,
there is no failure mode from trying to allocate a command.
*/
static void DAC960_queue_partial_rw(DAC960_Command_T *Command)
{
DAC960_Controller_T *Controller = Command->Controller;
IO_Request_T *Request = Command->Request;
if (Command->DmaDirection == PCI_DMA_FROMDEVICE)
Command->CommandType = DAC960_ReadRetryCommand;
else
Command->CommandType = DAC960_WriteRetryCommand;
/*
* We could be more efficient with these mapping requests
* and map only the portions that we need. But since this
* code should almost never be called, just go with a
* simple coding.
*/
(void)blk_rq_map_sg(&Controller->RequestQueue, Command->Request,
Command->cmd_sglist);
(void)pci_map_sg(Command->PciDevice, Command->cmd_sglist, 1,
Command->DmaDirection);
/*
* Resubmitting the request sector at a time is really tedious.
* But, this should almost never happen. So, we're willing to pay
* this price so that in the end, as much of the transfer is completed
* successfully as possible.
*/
Command->SegmentCount = 1;
Command->BlockNumber = Request->sector;
Command->BlockCount = 1;
DAC960_QueueReadWriteCommand(Command);
return;
}
/* /*
DAC960_ProcessRequests attempts to remove as many I/O Requests as possible DAC960_ProcessRequests attempts to remove as many I/O Requests as possible
from Controller's I/O Request Queue and queue them to the Controller. from Controller's I/O Request Queue and queue them to the Controller.
...@@ -3276,43 +3305,30 @@ static void DAC960_RequestFunction(RequestQueue_T *RequestQueue) ...@@ -3276,43 +3305,30 @@ static void DAC960_RequestFunction(RequestQueue_T *RequestQueue)
individual Buffer. individual Buffer.
*/ */
static inline void DAC960_ProcessCompletedRequest(DAC960_Command_T *Command, static inline boolean DAC960_ProcessCompletedRequest(DAC960_Command_T *Command,
boolean SuccessfulIO) boolean SuccessfulIO)
{ {
DAC960_CommandType_T CommandType = Command->CommandType;
IO_Request_T *Request = Command->Request; IO_Request_T *Request = Command->Request;
int DmaDirection, UpToDate; int UpToDate;
UpToDate = 0; UpToDate = 0;
if (SuccessfulIO) if (SuccessfulIO)
UpToDate = 1; UpToDate = 1;
/* pci_unmap_sg(Command->PciDevice, Command->cmd_sglist,
* We could save DmaDirection in the command structure Command->SegmentCount, Command->DmaDirection);
* and just reuse that information here.
*/ if (!end_that_request_first(Request, UpToDate, Command->BlockCount)) {
if (CommandType == DAC960_ReadCommand ||
CommandType == DAC960_ReadRetryCommand)
DmaDirection = PCI_DMA_FROMDEVICE;
else
DmaDirection = PCI_DMA_TODEVICE;
pci_unmap_sg(Command->PciDevice, Command->V1.ScatterList,
Command->SegmentCount, DmaDirection);
/*
* BlockCount is redundant with nr_sectors in the request
* structure. Consider eliminating BlockCount from the
* command structure now that Command includes a pointer to
* the request.
*/
while (end_that_request_first(Request, UpToDate, Command->BlockCount))
;
end_that_request_last(Request); end_that_request_last(Request);
if (Command->Completion) { if (Command->Completion) {
complete(Command->Completion); complete(Command->Completion);
Command->Completion = NULL; Command->Completion = NULL;
} }
return true;
}
return false;
} }
/* /*
...@@ -3384,46 +3400,65 @@ static void DAC960_V1_ProcessCompletedCommand(DAC960_Command_T *Command) ...@@ -3384,46 +3400,65 @@ static void DAC960_V1_ProcessCompletedCommand(DAC960_Command_T *Command)
if (CommandType == DAC960_ReadCommand || if (CommandType == DAC960_ReadCommand ||
CommandType == DAC960_WriteCommand) CommandType == DAC960_WriteCommand)
{ {
if (CommandStatus == DAC960_V1_NormalCompletion)
DAC960_ProcessCompletedRequest(Command, true); #ifdef FORCE_RETRY_DEBUG
CommandStatus = DAC960_V1_IrrecoverableDataError;
#endif
if (CommandStatus == DAC960_V1_NormalCompletion) {
else if (CommandStatus == DAC960_V1_IrrecoverableDataError || if (!DAC960_ProcessCompletedRequest(Command, true))
BUG();
} else if (CommandStatus == DAC960_V1_IrrecoverableDataError ||
CommandStatus == DAC960_V1_BadDataEncountered) CommandStatus == DAC960_V1_BadDataEncountered)
{ {
/* /*
* Finish this later. * break the command down into pieces and resubmit each
* * piece, hoping that some of them will succeed.
* We should call "complete_that_request_first()"
* to remove the first part of the request. Then, if there
* is still more I/O to be done, resubmit the request.
*
* We want to recalculate scatter/gather list,
* and requeue the command.
*
* For now, print a message on the console, and clone
* the code for "normal" completion.
*/ */
printk("V1_ProcessCompletedCommand: I/O error on read/write\n"); DAC960_queue_partial_rw(Command);
return;
DAC960_ProcessCompletedRequest(Command, false);
} }
else else
{ {
if (CommandStatus != DAC960_V1_LogicalDriveNonexistentOrOffline) if (CommandStatus != DAC960_V1_LogicalDriveNonexistentOrOffline)
DAC960_V1_ReadWriteError(Command); DAC960_V1_ReadWriteError(Command);
DAC960_ProcessCompletedRequest(Command, false); if (!DAC960_ProcessCompletedRequest(Command, false))
BUG();
} }
} }
else if (CommandType == DAC960_ReadRetryCommand || else if (CommandType == DAC960_ReadRetryCommand ||
CommandType == DAC960_WriteRetryCommand) CommandType == DAC960_WriteRetryCommand)
{ {
boolean normal_completion;
#ifdef FORCE_RETRY_FAILURE_DEBUG
static int retry_count = 1;
#endif
/* /*
* We're not doing retry commands yet. Perform completion processing for the portion that was
retried, and submit the next portion, if any.
*/ */
printk("DAC960_ProcessCompletedCommand: RetryCommand not done yet\n"); normal_completion = true;
if (CommandStatus != DAC960_V1_NormalCompletion) {
normal_completion = false;
if (CommandStatus != DAC960_V1_LogicalDriveNonexistentOrOffline)
DAC960_V1_ReadWriteError(Command);
}
#ifdef FORCE_RETRY_FAILURE_DEBUG
if (!(++retry_count % 10000)) {
printk("V1 error retry failure test\n");
normal_completion = false;
DAC960_V1_ReadWriteError(Command);
}
#endif
if (!DAC960_ProcessCompletedRequest(Command, normal_completion)) {
DAC960_queue_partial_rw(Command);
return;
}
} }
else if (CommandType == DAC960_MonitoringCommand) else if (CommandType == DAC960_MonitoringCommand)
...@@ -4451,23 +4486,25 @@ static void DAC960_V2_ProcessCompletedCommand(DAC960_Command_T *Command) ...@@ -4451,23 +4486,25 @@ static void DAC960_V2_ProcessCompletedCommand(DAC960_Command_T *Command)
if (CommandType == DAC960_ReadCommand || if (CommandType == DAC960_ReadCommand ||
CommandType == DAC960_WriteCommand) CommandType == DAC960_WriteCommand)
{ {
if (CommandStatus == DAC960_V2_NormalCompletion)
DAC960_ProcessCompletedRequest(Command, true); #ifdef FORCE_RETRY_DEBUG
CommandStatus = DAC960_V2_AbormalCompletion;
#endif
Command->V2.RequestSense->SenseKey = DAC960_SenseKey_MediumError;
else if (Command->V2.RequestSense->SenseKey == DAC960_SenseKey_MediumError) if (CommandStatus == DAC960_V2_NormalCompletion) {
{
if (!DAC960_ProcessCompletedRequest(Command, true))
BUG();
} else if (Command->V2.RequestSense->SenseKey == DAC960_SenseKey_MediumError)
{
/* /*
* Don't know yet how to handle this case. * break the command down into pieces and resubmit each
* See comments in DAC960_V1_ProcessCompletedCommand() * piece, hoping that some of them will succeed.
*
* For now, print a message on the console, and clone
* the code for "normal" completion.
*/ */
printk("V1_ProcessCompletedCommand: I/O error on read/write\n"); DAC960_queue_partial_rw(Command);
return;
DAC960_ProcessCompletedRequest(Command, false);
} }
else else
{ {
...@@ -4476,13 +4513,40 @@ static void DAC960_V2_ProcessCompletedCommand(DAC960_Command_T *Command) ...@@ -4476,13 +4513,40 @@ static void DAC960_V2_ProcessCompletedCommand(DAC960_Command_T *Command)
/* /*
Perform completion processing for all buffers in this I/O Request. Perform completion processing for all buffers in this I/O Request.
*/ */
DAC960_ProcessCompletedRequest(Command, false); (void)DAC960_ProcessCompletedRequest(Command, false);
} }
} }
else if (CommandType == DAC960_ReadRetryCommand || else if (CommandType == DAC960_ReadRetryCommand ||
CommandType == DAC960_WriteRetryCommand) CommandType == DAC960_WriteRetryCommand)
{ {
printk("DAC960_V2_ProcessCompletedCommand: retries not coded yet\n"); boolean normal_completion;
#ifdef FORCE_RETRY_FAILURE_DEBUG
static int retry_count = 1;
#endif
/*
Perform completion processing for the portion that was
retried, and submit the next portion, if any.
*/
normal_completion = true;
if (CommandStatus != DAC960_V2_NormalCompletion) {
normal_completion = false;
if (Command->V2.RequestSense->SenseKey != DAC960_SenseKey_NotReady)
DAC960_V2_ReadWriteError(Command);
}
#ifdef FORCE_RETRY_FAILURE_DEBUG
if (!(++retry_count % 10000)) {
printk("V2 error retry failure test\n");
normal_completion = false;
DAC960_V2_ReadWriteError(Command);
}
#endif
if (!DAC960_ProcessCompletedRequest(Command, normal_completion)) {
DAC960_queue_partial_rw(Command);
return;
}
} }
else if (CommandType == DAC960_MonitoringCommand) else if (CommandType == DAC960_MonitoringCommand)
{ {
......
...@@ -2305,6 +2305,8 @@ typedef struct DAC960_Command ...@@ -2305,6 +2305,8 @@ typedef struct DAC960_Command
unsigned int BlockNumber; unsigned int BlockNumber;
unsigned int BlockCount; unsigned int BlockCount;
unsigned int SegmentCount; unsigned int SegmentCount;
int DmaDirection;
struct scatterlist *cmd_sglist;
IO_Request_T *Request; IO_Request_T *Request;
struct pci_dev *PciDevice; struct pci_dev *PciDevice;
union { union {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment