- Finish ScsiPortInitialize() refactoring / improving. Only DMA stuff and 2-interrupts case are missing now.

svn path=/trunk/; revision=26220
This commit is contained in:
Aleksey Bragin 2007-03-31 18:37:07 +00:00
parent c08f45390a
commit fc3d38d1cb
2 changed files with 716 additions and 161 deletions

View file

@ -194,6 +194,13 @@ SpiResourceToConfig(IN PHW_INITIALIZATION_DATA HwInitializationData,
IN PCM_FULL_RESOURCE_DESCRIPTOR ResourceDescriptor, IN PCM_FULL_RESOURCE_DESCRIPTOR ResourceDescriptor,
IN PPORT_CONFIGURATION_INFORMATION PortConfig); IN PPORT_CONFIGURATION_INFORMATION PortConfig);
static PCM_RESOURCE_LIST
SpiConfigToResource(PSCSI_PORT_DEVICE_EXTENSION DeviceExtension,
PPORT_CONFIGURATION_INFORMATION PortConfig);
static VOID
SpiCleanupAfterInit(PSCSI_PORT_DEVICE_EXTENSION DeviceExtension);
/* FUNCTIONS *****************************************************************/ /* FUNCTIONS *****************************************************************/
@ -780,7 +787,7 @@ ScsiPortInitialize(IN PVOID Argument1,
{ {
PDRIVER_OBJECT DriverObject = (PDRIVER_OBJECT)Argument1; PDRIVER_OBJECT DriverObject = (PDRIVER_OBJECT)Argument1;
PUNICODE_STRING RegistryPath = (PUNICODE_STRING)Argument2; PUNICODE_STRING RegistryPath = (PUNICODE_STRING)Argument2;
PSCSI_PORT_DEVICE_EXTENSION DeviceExtension; PSCSI_PORT_DEVICE_EXTENSION DeviceExtension = NULL;
PCONFIGURATION_INFORMATION SystemConfig; PCONFIGURATION_INFORMATION SystemConfig;
PPORT_CONFIGURATION_INFORMATION PortConfig; PPORT_CONFIGURATION_INFORMATION PortConfig;
PORT_CONFIGURATION_INFORMATION InitialPortConfig; PORT_CONFIGURATION_INFORMATION InitialPortConfig;
@ -793,7 +800,7 @@ ScsiPortInitialize(IN PVOID Argument1,
ULONG Result; ULONG Result;
NTSTATUS Status; NTSTATUS Status;
ULONG MaxBus; ULONG MaxBus;
ULONG BusNumber; ULONG BusNumber = 0;
PCI_SLOT_NUMBER SlotNumber; PCI_SLOT_NUMBER SlotNumber;
PDEVICE_OBJECT PortDeviceObject; PDEVICE_OBJECT PortDeviceObject;
@ -806,6 +813,9 @@ ScsiPortInitialize(IN PVOID Argument1,
KIRQL Dirql; KIRQL Dirql;
KAFFINITY Affinity; KAFFINITY Affinity;
PCM_RESOURCE_LIST ResourceList;
BOOLEAN Conflict;
DPRINT ("ScsiPortInitialize() called!\n"); DPRINT ("ScsiPortInitialize() called!\n");
@ -857,10 +867,6 @@ ScsiPortInitialize(IN PVOID Argument1,
MaxBus = (HwInitializationData->AdapterInterfaceType == PCIBus) ? 8 : 1; MaxBus = (HwInitializationData->AdapterInterfaceType == PCIBus) ? 8 : 1;
DPRINT("MaxBus: %lu\n", MaxBus); DPRINT("MaxBus: %lu\n", MaxBus);
PortDeviceObject = NULL;
BusNumber = 0;
SlotNumber.u.AsULONG = 0;
while (TRUE) while (TRUE)
{ {
/* Create a unicode device name */ /* Create a unicode device name */
@ -901,6 +907,7 @@ ScsiPortInitialize(IN PVOID Argument1,
DeviceExtension->PortNumber = SystemConfig->ScsiPortCount; DeviceExtension->PortNumber = SystemConfig->ScsiPortCount;
/* Driver's routines... */ /* Driver's routines... */
DeviceExtension->HwInitialize = HwInitializationData->HwInitialize;
DeviceExtension->HwStartIo = HwInitializationData->HwStartIo; DeviceExtension->HwStartIo = HwInitializationData->HwStartIo;
DeviceExtension->HwInterrupt = HwInitializationData->HwInterrupt; DeviceExtension->HwInterrupt = HwInitializationData->HwInterrupt;
DeviceExtension->HwResetBus = HwInitializationData->HwResetBus; DeviceExtension->HwResetBus = HwInitializationData->HwResetBus;
@ -1022,11 +1029,19 @@ CreatePortConfig:
Again = FALSE; Again = FALSE;
goto CreatePortConfig; goto CreatePortConfig;
} }
if (!PortConfig->BusInterruptLevel)
{
/* Bypass this slot, because no interrupt was assigned */
DeviceExtension->PortConfig = NULL;
ExFreePool(PortConfig);
goto CreatePortConfig;
}
}
else
{
DPRINT("Non-pci bus\n");
} }
/* Note: HwFindAdapter is called once for each bus */ /* Note: HwFindAdapter is called once for each bus */
Again = FALSE; Again = FALSE;
@ -1034,15 +1049,256 @@ CreatePortConfig:
Result = (HwInitializationData->HwFindAdapter)(&DeviceExtension->MiniPortDeviceExtension, Result = (HwInitializationData->HwFindAdapter)(&DeviceExtension->MiniPortDeviceExtension,
HwContext, HwContext,
0, /* BusInformation */ 0, /* BusInformation */
"", /* ArgumentString */ ConfigInfo.Parameter, /* ArgumentString */
PortConfig, PortConfig,
&Again); &Again);
DPRINT("HwFindAdapter() Result: %lu Again: %s\n", DPRINT("HwFindAdapter() Result: %lu Again: %s\n",
Result, (Again) ? "True" : "False"); Result, (Again) ? "True" : "False");
if (Result == SP_RETURN_FOUND) /* Free MapRegisterBase, it's not needed anymore */
if (DeviceExtension->MapRegisterBase != NULL)
{ {
DPRINT("ScsiPortInitialize(): Found HBA! (%x)\n", PortConfig->BusInterruptVector); ExFreePool(DeviceExtension->MapRegisterBase);
DeviceExtension->MapRegisterBase = NULL;
}
/* If result is nothing good... */
if (Result != SP_RETURN_FOUND)
{
DPRINT("HwFindAdapter() Result: %lu\n", Result);
if (Result == SP_RETURN_NOT_FOUND)
{
/* We can continue on the next bus */
ConfigInfo.BusNumber++;
Again = FALSE;
DeviceExtension->PortConfig = NULL;
ExFreePool(PortConfig);
goto CreatePortConfig;
}
/* Otherwise, break */
Status = STATUS_INTERNAL_ERROR;
break;
}
DPRINT("ScsiPortInitialize(): Found HBA! (%x), adapter Id %d\n",
PortConfig->BusInterruptVector, PortConfig->InitiatorBusId[0]);
/* If the SRB extension size was updated */
if (!DeviceExtension->NonCachedExtension &&
(PortConfig->SrbExtensionSize != DeviceExtension->SrbExtensionSize))
{
/* Set it (rounding to LONGLONG again) */
DeviceExtension->SrbExtensionSize =
(PortConfig->SrbExtensionSize +
sizeof(LONGLONG)) & ~(sizeof(LONGLONG) - 1);
}
/* The same with LUN extension size */
if (PortConfig->SpecificLuExtensionSize != DeviceExtension->LunExtensionSize)
DeviceExtension->LunExtensionSize = PortConfig->SpecificLuExtensionSize;
if (!((HwInitializationData->AdapterInterfaceType == PCIBus) &&
(HwInitializationData->VendorIdLength > 0) &&
(HwInitializationData->VendorId != NULL) &&
(HwInitializationData->DeviceIdLength > 0) &&
(HwInitializationData->DeviceId != NULL)))
{
/* Construct a resource list */
ResourceList = SpiConfigToResource(DeviceExtension,
PortConfig);
if (ResourceList)
{
UNICODE_STRING UnicodeString;
RtlInitUnicodeString(&UnicodeString, L"ScsiAdapter");
DPRINT("Reporting resources\n");
Status = IoReportResourceUsage(&UnicodeString,
DriverObject,
NULL,
0,
PortDeviceObject,
ResourceList,
FIELD_OFFSET(CM_RESOURCE_LIST,
List[0].PartialResourceList.PartialDescriptors) +
ResourceList->List[0].PartialResourceList.Count
* sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR),
FALSE,
&Conflict);
ExFreePool(ResourceList);
/* In case of a failure or a conflict, break */
if (Conflict || (!NT_SUCCESS(Status)))
{
if (Conflict)
Status = STATUS_CONFLICTING_ADDRESSES;
break;
}
}
}
/* Reset the Conflict var */
Conflict = FALSE;
/* Copy all stuff which we ever need from PortConfig to the DeviceExtension */
if (PortConfig->MaximumNumberOfTargets > SCSI_MAXIMUM_TARGETS_PER_BUS)
DeviceExtension->MaxTargedIds = SCSI_MAXIMUM_TARGETS_PER_BUS;
else
DeviceExtension->MaxTargedIds = PortConfig->MaximumNumberOfTargets;
DeviceExtension->BusNum = PortConfig->NumberOfBuses;
DeviceExtension->CachesData = PortConfig->CachesData;
DeviceExtension->ReceiveEvent = PortConfig->ReceiveEvent;
DeviceExtension->SupportsTaggedQueuing = PortConfig->TaggedQueuing;
DeviceExtension->MultipleReqsPerLun = PortConfig->MultipleRequestPerLu;
/* If something was disabled via registry - apply it */
if (ConfigInfo.DisableMultipleLun)
DeviceExtension->MultipleReqsPerLun = PortConfig->MultipleRequestPerLu = FALSE;
if (ConfigInfo.DisableTaggedQueueing)
DeviceExtension->SupportsTaggedQueuing = PortConfig->MultipleRequestPerLu = FALSE;
/* Check if we need to alloc SRB data */
if (DeviceExtension->SupportsTaggedQueuing ||
DeviceExtension->MultipleReqsPerLun)
{
DeviceExtension->NeedSrbDataAlloc = TRUE;
}
else
{
DeviceExtension->NeedSrbDataAlloc = FALSE;
}
/* Get a pointer to the port capabilities */
PortCapabilities = &DeviceExtension->PortCapabilities;
/* Copy one field there */
DeviceExtension->MapBuffers = PortConfig->MapBuffers;
PortCapabilities->AdapterUsesPio = PortConfig->MapBuffers;
if (DeviceExtension->AdapterObject == NULL &&
(PortConfig->DmaChannel != SP_UNINITIALIZED_VALUE || PortConfig->Master))
{
DPRINT1("DMA is not supported yet\n");
ASSERT(FALSE);
}
if (DeviceExtension->SrbExtensionBuffer == NULL &&
(DeviceExtension->SrbExtensionSize != 0 ||
PortConfig->AutoRequestSense))
{
DeviceExtension->SupportsAutoSense = PortConfig->AutoRequestSense;
DeviceExtension->NeedSrbExtensionAlloc = TRUE;
//Status = STATUS_UNSUCCESFUL;
/* TODO: Allocate common buffer */
ASSERT(FALSE);
/* Check for failure */
if (!NT_SUCCESS(Status))
break;
}
/* Allocate SrbData, if needed */
if (DeviceExtension->NeedSrbDataAlloc)
{
ULONG Count;
PSCSI_REQUEST_BLOCK_INFO SrbData;
if (DeviceExtension->SrbDataCount != 0)
Count = DeviceExtension->SrbDataCount;
else
Count = DeviceExtension->RequestsNumber * 2;
/* Allocate the data */
SrbData = ExAllocatePool(NonPagedPool, Count * sizeof(SCSI_REQUEST_BLOCK_INFO));
if (SrbData == NULL)
return STATUS_INSUFFICIENT_RESOURCES;
RtlZeroMemory(SrbData, Count * sizeof(SCSI_REQUEST_BLOCK_INFO));
DeviceExtension->SrbInfo = SrbData;
DeviceExtension->FreeSrbInfo = SrbData;
DeviceExtension->SrbDataCount = Count;
/* Link it to the list */
while (Count > 0)
{
SrbData->Requests.Flink = (PLIST_ENTRY)(SrbData + 1);
SrbData++;
Count--;
}
/* Mark the last entry of the list */
SrbData--;
SrbData->Requests.Flink = NULL;
}
/* Initialize port capabilities */
PortCapabilities = &DeviceExtension->PortCapabilities;
PortCapabilities->Length = sizeof(IO_SCSI_CAPABILITIES);
PortCapabilities->MaximumTransferLength = PortConfig->MaximumTransferLength;
if (PortConfig->ReceiveEvent)
PortCapabilities->SupportedAsynchronousEvents |= SRBEV_SCSI_ASYNC_NOTIFICATION;
PortCapabilities->TaggedQueuing = DeviceExtension->SupportsTaggedQueuing;
PortCapabilities->AdapterScansDown = PortConfig->AdapterScansDown;
if (PortConfig->AlignmentMask > PortDeviceObject->AlignmentRequirement)
PortDeviceObject->AlignmentRequirement = PortConfig->AlignmentMask;
PortCapabilities->AlignmentMask = PortDeviceObject->AlignmentRequirement;
if (PortCapabilities->MaximumPhysicalPages == 0)
{
PortCapabilities->MaximumPhysicalPages =
BYTES_TO_PAGES(PortCapabilities->MaximumTransferLength);
/* Apply miniport's limits */
if (PortConfig->NumberOfPhysicalBreaks < PortCapabilities->MaximumPhysicalPages)
{
PortCapabilities->MaximumPhysicalPages = PortConfig->NumberOfPhysicalBreaks;
}
}
/* Deal with interrupts */
if (DeviceExtension->HwInterrupt == NULL ||
(PortConfig->BusInterruptLevel == 0 && PortConfig->BusInterruptVector == 0))
{
/* No interrupts */
KeInitializeSpinLock(&DeviceExtension->IrqLock);
/* FIXME: Use synchronization routine */
ASSERT("No interrupts branch requires changes in synchronization\n");
DeviceExtension->Interrupt = (PVOID)DeviceExtension;
DPRINT("No interrupts\n");
}
else
{
/* Are 2 interrupts needed? */
if (DeviceExtension->HwInterrupt != NULL &&
(PortConfig->BusInterruptLevel != 0 || PortConfig->BusInterruptVector != 0) &&
(PortConfig->BusInterruptLevel2 != 0 || PortConfig->BusInterruptVector2 != 0))
{
DPRINT1("2 interrupts requested! Not yet supported\n");
ASSERT(FALSE);
}
else
{
BOOLEAN InterruptShareable;
/* No, only 1 interrupt */
DPRINT("1 interrupt, IRQ is %d\n", PortConfig->BusInterruptLevel);
DeviceExtension->InterruptLevel = PortConfig->BusInterruptLevel;
/* Register an interrupt handler for this device */ /* Register an interrupt handler for this device */
MappedIrq = HalGetInterruptVector(PortConfig->AdapterInterfaceType, MappedIrq = HalGetInterruptVector(PortConfig->AdapterInterfaceType,
@ -1051,22 +1307,48 @@ CreatePortConfig:
PortConfig->BusInterruptVector, PortConfig->BusInterruptVector,
&Dirql, &Dirql,
&Affinity); &Affinity);
/* Determing IRQ sharability as usual */
if (PortConfig->AdapterInterfaceType == MicroChannel ||
PortConfig->InterruptMode == LevelSensitive)
{
InterruptShareable = TRUE;
}
else
{
InterruptShareable = FALSE;
}
Status = IoConnectInterrupt(&DeviceExtension->Interrupt, Status = IoConnectInterrupt(&DeviceExtension->Interrupt,
ScsiPortIsr, (PKSERVICE_ROUTINE)ScsiPortIsr,
DeviceExtension, DeviceExtension,
&DeviceExtension->SpinLock, NULL,
MappedIrq, MappedIrq,
Dirql, Dirql,
Dirql, Dirql,
PortConfig->InterruptMode, PortConfig->InterruptMode,
TRUE, InterruptShareable,
Affinity, Affinity,
FALSE); FALSE);
if (!NT_SUCCESS(Status))
if (!(NT_SUCCESS(Status)))
{ {
DbgPrint("Could not connect interrupt %d\n", DPRINT1("Could not connect interrupt %d\n",
PortConfig->BusInterruptVector); PortConfig->BusInterruptVector);
goto ByeBye; DeviceExtension->Interrupt = NULL;
break;
}
}
}
/* Save IoAddress (from access ranges) */
if (HwInitializationData->NumberOfAccessRanges != 0)
{
DeviceExtension->IoAddress =
((*(PortConfig->AccessRanges))[0]).RangeStart.LowPart;
DPRINT("Io Address %x\n", DeviceExtension->IoAddress);
} }
/* Set flag that it's allowed to disconnect during this command */ /* Set flag that it's allowed to disconnect during this command */
@ -1075,37 +1357,46 @@ CreatePortConfig:
/* Initialize counter of active requests (-1 means there are none) */ /* Initialize counter of active requests (-1 means there are none) */
DeviceExtension->ActiveRequestCounter = -1; DeviceExtension->ActiveRequestCounter = -1;
if (!(HwInitializationData->HwInitialize)(&DeviceExtension->MiniPortDeviceExtension)) /* Analyze what we have about DMA */
if (DeviceExtension->AdapterObject != NULL &&
PortConfig->Master &&
PortConfig->NeedPhysicalAddresses)
{ {
DbgPrint("HwInitialize() failed!"); DeviceExtension->MapRegisters = TRUE;
Status = STATUS_UNSUCCESSFUL; }
goto ByeBye; else
{
DeviceExtension->MapRegisters = FALSE;
} }
/* Initialize port capabilities */ /* Call HwInitialize at DISPATCH_LEVEL */
DeviceExtension->PortCapabilities = ExAllocatePool(NonPagedPool, KeRaiseIrql(DISPATCH_LEVEL, &Dirql);
sizeof(IO_SCSI_CAPABILITIES));
if (DeviceExtension->PortCapabilities == NULL) if (!KeSynchronizeExecution(DeviceExtension->Interrupt,
DeviceExtension->HwInitialize,
DeviceExtension->MiniPortDeviceExtension))
{ {
DbgPrint("Failed to allocate port capabilities!\n"); DPRINT1("HwInitialize() failed!\n");
Status = STATUS_INSUFFICIENT_RESOURCES; KeLowerIrql(Dirql);
goto ByeBye; Status = STATUS_ADAPTER_HARDWARE_ERROR;
break;
} }
PortCapabilities = DeviceExtension->PortCapabilities; /* Check if a notification is needed */
PortCapabilities->Length = sizeof(IO_SCSI_CAPABILITIES); if (DeviceExtension->InterruptData.Flags & SCSI_PORT_NOTIFICATION_NEEDED)
PortCapabilities->MaximumTransferLength = {
PortConfig->MaximumTransferLength; /* Call DPC right away, because we're already at DISPATCH_LEVEL */
PortCapabilities->MaximumPhysicalPages = ScsiPortDpcForIsr(NULL,
PortCapabilities->MaximumTransferLength / PAGE_SIZE; DeviceExtension->DeviceObject,
PortCapabilities->SupportedAsynchronousEvents = 0; /* FIXME */ NULL,
PortCapabilities->AlignmentMask = NULL);
PortConfig->AlignmentMask; }
PortCapabilities->TaggedQueuing =
PortConfig->TaggedQueuing; /* Lower irql back to what it was */
PortCapabilities->AdapterScansDown = KeLowerIrql(Dirql);
PortConfig->AdapterScansDown;
PortCapabilities->AdapterUsesPio = TRUE; /* FIXME */ /* Start our timer */
IoStartTimer(PortDeviceObject);
/* Initialize bus scanning information */ /* Initialize bus scanning information */
DeviceExtension->BusesConfig = ExAllocatePool(PagedPool, DeviceExtension->BusesConfig = ExAllocatePool(PagedPool,
@ -1116,7 +1407,7 @@ CreatePortConfig:
{ {
DPRINT1("Out of resources!\n"); DPRINT1("Out of resources!\n");
Status = STATUS_INSUFFICIENT_RESOURCES; Status = STATUS_INSUFFICIENT_RESOURCES;
goto ByeBye; break;
} }
/* Zero it */ /* Zero it */
@ -1138,66 +1429,42 @@ CreatePortConfig:
swprintf(DosNameBuffer, swprintf(DosNameBuffer,
L"\\??\\Scsi%lu:", L"\\??\\Scsi%lu:",
SystemConfig->ScsiPortCount); SystemConfig->ScsiPortCount);
RtlInitUnicodeString(&DosDeviceName, RtlInitUnicodeString(&DosDeviceName, DosNameBuffer);
DosNameBuffer); IoCreateSymbolicLink(&DosDeviceName, &DeviceName);
IoCreateSymbolicLink(&DosDeviceName,
&DeviceName);
/* Update the system configuration info */
if (PortConfig->AtdiskPrimaryClaimed == TRUE)
SystemConfig->AtDiskPrimaryAddressClaimed = TRUE;
if (PortConfig->AtdiskSecondaryClaimed == TRUE)
SystemConfig->AtDiskSecondaryAddressClaimed = TRUE;
/* Increase the port count */
SystemConfig->ScsiPortCount++; SystemConfig->ScsiPortCount++;
PortDeviceObject = NULL; FirstConfigCall = FALSE;
DeviceFound = TRUE;
}
else
{
DPRINT("HwFindAdapter() Result: %lu\n", Result);
ExFreePool (PortConfig); /* Increase adapter number and bus number respectively */
IoDeleteDevice (PortDeviceObject); ConfigInfo.AdapterNumber++;
PortDeviceObject = NULL;
} if (!Again)
ConfigInfo.BusNumber++;
DPRINT("Bus: %lu MaxBus: %lu\n", BusNumber, MaxBus); DPRINT("Bus: %lu MaxBus: %lu\n", BusNumber, MaxBus);
if (BusNumber >= MaxBus)
{ DeviceFound = TRUE;
DPRINT("Scanned all buses!\n");
Status = STATUS_SUCCESS;
goto ByeBye;
} }
if (Again == FALSE)
{
BusNumber++;
SlotNumber.u.AsULONG = 0;
}
}
ByeBye:
/* Clean up the mess */ /* Clean up the mess */
if (PortDeviceObject != NULL) SpiCleanupAfterInit(DeviceExtension);
{
DPRINT("Delete device: %p\n", PortDeviceObject);
DeviceExtension = PortDeviceObject->DeviceExtension; /* Close registry keys */
if (ConfigInfo.ServiceKey != NULL)
ZwClose(ConfigInfo.ServiceKey);
if (DeviceExtension->PortCapabilities != NULL) if (ConfigInfo.DeviceKey != NULL)
{ ZwClose(ConfigInfo.DeviceKey);
IoDisconnectInterrupt (DeviceExtension->Interrupt);
ExFreePool (DeviceExtension->PortCapabilities);
}
if (DeviceExtension->PortConfig != NULL) if (ConfigInfo.BusKey != NULL)
{ ZwClose(ConfigInfo.BusKey);
ExFreePool (DeviceExtension->PortConfig);
}
IoDeleteDevice (PortDeviceObject); if (ConfigInfo.AccessRanges != NULL)
} ExFreePool(ConfigInfo.AccessRanges);
if (ConfigInfo.Parameter != NULL)
ExFreePool(ConfigInfo.Parameter);
DPRINT("ScsiPortInitialize() done, Status = 0x%08X, DeviceFound = %b!\n", DPRINT("ScsiPortInitialize() done, Status = 0x%08X, DeviceFound = %b!\n",
Status, DeviceFound); Status, DeviceFound);
@ -1205,6 +1472,105 @@ ByeBye:
return (DeviceFound == FALSE) ? Status : STATUS_SUCCESS; return (DeviceFound == FALSE) ? Status : STATUS_SUCCESS;
} }
static VOID
SpiCleanupAfterInit(PSCSI_PORT_DEVICE_EXTENSION DeviceExtension)
{
PSCSI_LUN_INFO LunInfo;
PVOID Ptr;
ULONG Bus, Lun;
/* Check if we have something to clean up */
if (DeviceExtension == NULL)
return;
/* Stop the timer and disconnect the interrupt */
if (DeviceExtension->Interrupt)
{
IoStopTimer(DeviceExtension->DeviceObject);
IoDisconnectInterrupt(DeviceExtension->Interrupt);
}
/* Delete ConfigInfo */
if (DeviceExtension->BusesConfig)
{
for (Bus = 0; Bus < DeviceExtension->BusNum; Bus++)
{
if (!DeviceExtension->BusesConfig->BusScanInfo[Bus])
continue;
LunInfo = DeviceExtension->BusesConfig->BusScanInfo[Bus]->LunInfo;
while (!LunInfo)
{
/* Free current, but save pointer to the next one */
Ptr = LunInfo->Next;
ExFreePool(LunInfo);
LunInfo = Ptr;
}
ExFreePool(DeviceExtension->BusesConfig->BusScanInfo[Bus]);
}
ExFreePool(DeviceExtension->BusesConfig);
}
/* Free PortConfig */
if (DeviceExtension->PortConfig)
ExFreePool(DeviceExtension->PortConfig);
/* Free LUNs*/
for(Lun = 0; Lun < LUS_NUMBER; Lun++)
{
while (DeviceExtension->LunExtensionList[Lun])
{
Ptr = DeviceExtension->LunExtensionList[Lun];
DeviceExtension->LunExtensionList[Lun] = DeviceExtension->LunExtensionList[Lun]->Next;
ExFreePool(Ptr);
}
}
/* Free common buffer (if it exists) */
if (DeviceExtension->SrbExtensionBuffer != NULL &&
DeviceExtension->CommonBufferSize != 0)
{
if (!DeviceExtension->AdapterObject)
{
ExFreePool(DeviceExtension->SrbExtensionBuffer);
}
else
{
#if 0
HalFreeCommonBuffer(DeviceExtension->AdapterObject,
DeviceExtension->CommonBufferSize,
DeviceExtension->PhysicalCommonBuffer,
DeviceExtension->SrbExtensionBuffer,
FALSE);
#endif
}
}
/* Free SRB info */
if (DeviceExtension->SrbInfo != NULL)
ExFreePool(DeviceExtension->SrbInfo);
/* Unmap mapped addresses */
while (DeviceExtension->MappedAddressList != NULL)
{
MmUnmapIoSpace(DeviceExtension->MappedAddressList->MappedAddress,
DeviceExtension->MappedAddressList->NumberOfBytes);
Ptr = DeviceExtension->MappedAddressList;
DeviceExtension->MappedAddressList = DeviceExtension->MappedAddressList->NextMappedAddress;
ExFreePool(Ptr);
}
/* Finally delete the device object */
IoDeleteDevice(DeviceExtension->DeviceObject);
}
/* /*
* @unimplemented * @unimplemented
@ -1491,6 +1857,187 @@ SpiResourceToConfig(IN PHW_INITIALIZATION_DATA HwInitializationData,
} }
} }
static PCM_RESOURCE_LIST
SpiConfigToResource(PSCSI_PORT_DEVICE_EXTENSION DeviceExtension,
PPORT_CONFIGURATION_INFORMATION PortConfig)
{
PCONFIGURATION_INFORMATION ConfigInfo;
PCM_RESOURCE_LIST ResourceList;
PCM_PARTIAL_RESOURCE_DESCRIPTOR ResourceDescriptor;
PACCESS_RANGE AccessRange;
BOOLEAN Dma;
ULONG ListLength = 0, i, FullSize;
ULONG Interrupt;
/* Get current Atdisk usage from the system */
ConfigInfo = IoGetConfigurationInformation();
if (PortConfig->AtdiskPrimaryClaimed)
ConfigInfo->AtDiskPrimaryAddressClaimed = TRUE;
if (PortConfig->AtdiskSecondaryClaimed)
ConfigInfo->AtDiskSecondaryAddressClaimed = TRUE;
/* Do we use DMA? */
if (PortConfig->DmaChannel != SP_UNINITIALIZED_VALUE ||
PortConfig->DmaPort != SP_UNINITIALIZED_VALUE)
{
Dma = TRUE;
ListLength++;
}
else
{
Dma = FALSE;
}
/* How many interrupts to we have? */
if (DeviceExtension->HwInterrupt == NULL ||
(PortConfig->BusInterruptLevel == 0 &&
PortConfig->BusInterruptVector == 0))
{
Interrupt = 0;
}
else
{
Interrupt = 1;
ListLength++;
}
if (DeviceExtension->HwInterrupt != NULL &&
(PortConfig->BusInterruptLevel2 != 0 ||
PortConfig->BusInterruptVector2 != 0))
{
Interrupt++;
ListLength++;
}
/* How many access ranges do we use? */
AccessRange = &((*(PortConfig->AccessRanges))[0]);
for (i = 0; i < PortConfig->NumberOfAccessRanges; i++)
{
if (AccessRange->RangeLength != 0)
ListLength++;
AccessRange++;
}
/* Allocate the resource list, since we know its size now */
FullSize = sizeof(CM_RESOURCE_LIST) + (ListLength - 1) *
sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR);
ResourceList = (PCM_RESOURCE_LIST)ExAllocatePool(PagedPool, FullSize);
if (!ResourceList)
return NULL;
/* Zero it */
RtlZeroMemory(ResourceList, FullSize);
/* Initialize it */
ResourceList->Count = 1;
ResourceList->List[0].InterfaceType = PortConfig->AdapterInterfaceType;
ResourceList->List[0].BusNumber = PortConfig->SystemIoBusNumber;
ResourceList->List[0].PartialResourceList.Count = ListLength;
ResourceDescriptor = ResourceList->List[0].PartialResourceList.PartialDescriptors;
/* Copy access ranges array over */
for (i = 0; i < PortConfig->NumberOfAccessRanges; i++)
{
AccessRange = &((*(PortConfig->AccessRanges))[i]);
/* If the range is empty - skip it */
if (AccessRange->RangeLength == 0)
continue;
if (AccessRange->RangeInMemory)
{
ResourceDescriptor->Type = CmResourceTypeMemory;
ResourceDescriptor->Flags = CM_RESOURCE_MEMORY_READ_WRITE;
}
else
{
ResourceDescriptor->Type = CmResourceTypePort;
ResourceDescriptor->Flags = CM_RESOURCE_PORT_IO;
}
ResourceDescriptor->ShareDisposition = CmResourceShareDeviceExclusive;
ResourceDescriptor->u.Memory.Start = AccessRange->RangeStart;
ResourceDescriptor->u.Memory.Length = AccessRange->RangeLength;
ResourceDescriptor++;
}
/* If we use interrupt(s), copy them */
if (Interrupt)
{
ResourceDescriptor->Type = CmResourceTypeInterrupt;
if (PortConfig->AdapterInterfaceType == MicroChannel ||
PortConfig->InterruptMode == LevelSensitive)
{
ResourceDescriptor->ShareDisposition = CmResourceShareShared;
ResourceDescriptor->Flags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
}
else
{
ResourceDescriptor->ShareDisposition = CmResourceShareDeviceExclusive;
ResourceDescriptor->Flags = CM_RESOURCE_INTERRUPT_LATCHED;
}
ResourceDescriptor->u.Interrupt.Level = PortConfig->BusInterruptLevel;
ResourceDescriptor->u.Interrupt.Vector = PortConfig->BusInterruptVector;
ResourceDescriptor->u.Interrupt.Affinity = 0;
ResourceDescriptor++;
Interrupt--;
}
/* Copy 2nd interrupt
FIXME: Stupid code duplication, remove */
if (Interrupt)
{
ResourceDescriptor->Type = CmResourceTypeInterrupt;
if (PortConfig->AdapterInterfaceType == MicroChannel ||
PortConfig->InterruptMode == LevelSensitive)
{
ResourceDescriptor->ShareDisposition = CmResourceShareShared;
ResourceDescriptor->Flags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
}
else
{
ResourceDescriptor->ShareDisposition = CmResourceShareDeviceExclusive;
ResourceDescriptor->Flags = CM_RESOURCE_INTERRUPT_LATCHED;
}
ResourceDescriptor->u.Interrupt.Level = PortConfig->BusInterruptLevel;
ResourceDescriptor->u.Interrupt.Vector = PortConfig->BusInterruptVector;
ResourceDescriptor->u.Interrupt.Affinity = 0;
ResourceDescriptor++;
}
/* Copy DMA data */
if (Dma)
{
ResourceDescriptor->Type = CmResourceTypeDma;
ResourceDescriptor->ShareDisposition = CmResourceShareDeviceExclusive;
ResourceDescriptor->u.Dma.Channel = PortConfig->DmaChannel;
ResourceDescriptor->u.Dma.Port = PortConfig->DmaPort;
ResourceDescriptor->Flags = 0;
if (PortConfig->DmaChannel == SP_UNINITIALIZED_VALUE)
ResourceDescriptor->u.Dma.Channel = 0;
if (PortConfig->DmaPort == SP_UNINITIALIZED_VALUE)
ResourceDescriptor->u.Dma.Port = 0;
}
return ResourceList;
}
static BOOLEAN static BOOLEAN
SpiGetPciConfigData(IN PDRIVER_OBJECT DriverObject, SpiGetPciConfigData(IN PDRIVER_OBJECT DriverObject,
IN PDEVICE_OBJECT DeviceObject, IN PDEVICE_OBJECT DeviceObject,
@ -1838,7 +2385,7 @@ ScsiPortDeviceControl(IN PDEVICE_OBJECT DeviceObject,
DPRINT(" IOCTL_SCSI_GET_CAPABILITIES\n"); DPRINT(" IOCTL_SCSI_GET_CAPABILITIES\n");
*((PIO_SCSI_CAPABILITIES *)Irp->AssociatedIrp.SystemBuffer) = *((PIO_SCSI_CAPABILITIES *)Irp->AssociatedIrp.SystemBuffer) =
DeviceExtension->PortCapabilities; &DeviceExtension->PortCapabilities;
Irp->IoStatus.Information = sizeof(PIO_SCSI_CAPABILITIES); Irp->IoStatus.Information = sizeof(PIO_SCSI_CAPABILITIES);
} }
@ -3843,7 +4390,7 @@ SpiBuildDeviceMap (PSCSI_PORT_DEVICE_EXTENSION DeviceExtension,
*/ */
/* Set 'DMA Enabled' (REG_DWORD) value */ /* Set 'DMA Enabled' (REG_DWORD) value */
UlongData = (ULONG)!DeviceExtension->PortCapabilities->AdapterUsesPio; UlongData = (ULONG)!DeviceExtension->PortCapabilities.AdapterUsesPio;
DPRINT(" DMA Enabled = %s\n", (UlongData) ? "TRUE" : "FALSE"); DPRINT(" DMA Enabled = %s\n", (UlongData) ? "TRUE" : "FALSE");
RtlInitUnicodeString(&ValueName, RtlInitUnicodeString(&ValueName,
L"DMA Enabled"); L"DMA Enabled");

View file

@ -194,6 +194,7 @@ typedef struct _SCSI_PORT_DEVICE_EXTENSION
ULONG MiniPortExtensionSize; ULONG MiniPortExtensionSize;
PPORT_CONFIGURATION_INFORMATION PortConfig; PPORT_CONFIGURATION_INFORMATION PortConfig;
PBUSES_CONFIGURATION_INFORMATION BusesConfig; PBUSES_CONFIGURATION_INFORMATION BusesConfig;
PVOID NonCachedExtension;
ULONG PortNumber; ULONG PortNumber;
LONG ActiveRequestCounter; LONG ActiveRequestCounter;
@ -223,6 +224,7 @@ typedef struct _SCSI_PORT_DEVICE_EXTENSION
SCSI_PORT_INTERRUPT_DATA InterruptData; SCSI_PORT_INTERRUPT_DATA InterruptData;
ULONG CommonBufferSize;
/* SRB extension stuff*/ /* SRB extension stuff*/
ULONG SrbExtensionSize; ULONG SrbExtensionSize;
PVOID SrbExtensionBuffer; PVOID SrbExtensionBuffer;
@ -231,12 +233,14 @@ typedef struct _SCSI_PORT_DEVICE_EXTENSION
/* SRB information */ /* SRB information */
PSCSI_REQUEST_BLOCK_INFO SrbInfo; PSCSI_REQUEST_BLOCK_INFO SrbInfo;
PSCSI_REQUEST_BLOCK_INFO FreeSrbInfo; PSCSI_REQUEST_BLOCK_INFO FreeSrbInfo;
ULONG SrbDataCount;
PIO_SCSI_CAPABILITIES PortCapabilities; IO_SCSI_CAPABILITIES PortCapabilities;
PDEVICE_OBJECT DeviceObject; PDEVICE_OBJECT DeviceObject;
PCONTROLLER_OBJECT ControllerObject; PCONTROLLER_OBJECT ControllerObject;
PHW_INITIALIZE HwInitialize;
PHW_STARTIO HwStartIo; PHW_STARTIO HwStartIo;
PHW_INTERRUPT HwInterrupt; PHW_INTERRUPT HwInterrupt;
PHW_RESET_BUS HwResetBus; PHW_RESET_BUS HwResetBus;
@ -254,13 +258,17 @@ typedef struct _SCSI_PORT_DEVICE_EXTENSION
PVOID MapRegisterBase; PVOID MapRegisterBase;
/* Features */ /* Features */
BOOLEAN CachesData;
BOOLEAN SupportsTaggedQueuing; BOOLEAN SupportsTaggedQueuing;
BOOLEAN SupportsAutoSense; BOOLEAN SupportsAutoSense;
BOOLEAN MultipleReqsPerLun;
BOOLEAN ReceiveEvent;
PHYSICAL_ADDRESS PhysicalAddress; PHYSICAL_ADDRESS PhysicalAddress;
PVOID VirtualAddress; PVOID VirtualAddress;
ULONG CommonBufferLength; ULONG CommonBufferLength;
ULONG InterruptLevel;
ULONG IoAddress;
BOOLEAN NeedSrbExtensionAlloc; BOOLEAN NeedSrbExtensionAlloc;
BOOLEAN NeedSrbDataAlloc; BOOLEAN NeedSrbDataAlloc;