mirror of
https://github.com/reactos/reactos.git
synced 2024-11-05 22:26:39 +00:00
3376 lines
126 KiB
C++
3376 lines
126 KiB
C++
/*++
|
|
|
|
Copyright (c) 2002-2016 Alexandr A. Telyatnikov (Alter)
|
|
|
|
Module Name:
|
|
id_probe.cpp
|
|
|
|
Abstract:
|
|
This module scans PCI and ISA buses for IDE controllers
|
|
and determines their Busmaster DMA capabilities
|
|
|
|
Author:
|
|
Alexander A. Telyatnikov (Alter)
|
|
|
|
Environment:
|
|
kernel mode only
|
|
|
|
Notes:
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
Revision History:
|
|
|
|
Some parts of hardware-specific code were taken from FreeBSD 4.3-6.1 ATA driver by
|
|
Søren Schmidt, Copyright (c) 1998-2007
|
|
|
|
Some parts of device detection code were taken from from standard ATAPI.SYS from NT4 DDK by
|
|
Mike Glass (MGlass)
|
|
Chuck Park (ChuckP)
|
|
|
|
Device search/init algorithm is completly rewritten by
|
|
Alter, Copyright (c) 2002-2004
|
|
|
|
Fixes for Native/Compatible modes of onboard IDE controller by
|
|
Vitaliy Vorobyov, deathsoft@yandex.ru (c) 2004
|
|
|
|
Licence:
|
|
GPLv2
|
|
|
|
--*/
|
|
|
|
#include "stdafx.h"
|
|
|
|
PBUSMASTER_CONTROLLER_INFORMATION BMList = NULL;
|
|
ULONG BMListLen = 0;
|
|
ULONG IsaCount = 0;
|
|
ULONG MCACount = 0;
|
|
|
|
BOOLEAN FirstMasterOk = FALSE;
|
|
// This is our own resource check,
|
|
// ReactOS allows to allocate same I/O range for both PCI and ISA controllers
|
|
BOOLEAN AtdiskPrimaryClaimed = FALSE;
|
|
BOOLEAN AtdiskSecondaryClaimed = FALSE;
|
|
|
|
#ifndef UNIATA_CORE
|
|
|
|
UCHAR pciBuffer[256];
|
|
ULONG maxPciBus = 16;
|
|
|
|
PDRIVER_OBJECT SavedDriverObject = NULL;
|
|
|
|
// local routines
|
|
|
|
ULONG
|
|
NTAPI
|
|
UniataEnumBusMasterController__(
|
|
/* IN PVOID HwDeviceExtension,
|
|
IN PVOID Context,
|
|
IN PVOID BusInformation,
|
|
IN PCHAR ArgumentString,
|
|
IN OUT PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
OUT PBOOLEAN Again*/
|
|
);
|
|
|
|
VOID
|
|
NTAPI
|
|
AtapiDoNothing(VOID)
|
|
{
|
|
return;
|
|
} // end AtapiDoNothing()
|
|
|
|
#endif //UNIATA_CORE
|
|
|
|
USHORT
|
|
NTAPI
|
|
UniataEnableIoPCI(
|
|
IN ULONG busNumber,
|
|
IN ULONG slotNumber,
|
|
IN OUT PPCI_COMMON_CONFIG pciData
|
|
)
|
|
{
|
|
ULONG i;
|
|
ULONG busDataRead;
|
|
USHORT CmdOrig;
|
|
|
|
// Enable Busmastering, IO-space and Mem-space
|
|
// Note: write to CONFIG *may* cause controller to interrupt (not handled yet)
|
|
// even if no bits are updated. Was observed on ICH7
|
|
KdPrint2((PRINT_PREFIX "Enabling Mem/Io spaces and busmastering...\n"));
|
|
KdPrint2((PRINT_PREFIX "Initial pciData.Command = %#x\n", pciData->Command));
|
|
for(i=0; i<3; i++) {
|
|
CmdOrig = pciData->Command;
|
|
switch(i) {
|
|
case 0:
|
|
KdPrint2((PRINT_PREFIX "PCI_ENABLE_IO_SPACE\n"));
|
|
pciData->Command |= PCI_ENABLE_IO_SPACE;
|
|
break;
|
|
case 1:
|
|
KdPrint2((PRINT_PREFIX "PCI_ENABLE_MEMORY_SPACE\n"));
|
|
pciData->Command |= PCI_ENABLE_MEMORY_SPACE;
|
|
break;
|
|
case 2:
|
|
KdPrint2((PRINT_PREFIX "PCI_ENABLE_BUS_MASTER\n"));
|
|
pciData->Command |= PCI_ENABLE_BUS_MASTER;
|
|
break;
|
|
}
|
|
if(CmdOrig == pciData->Command) {
|
|
continue;
|
|
}
|
|
HalSetBusDataByOffset( PCIConfiguration, busNumber, slotNumber,
|
|
&(pciData->Command),
|
|
offsetof(PCI_COMMON_CONFIG, Command),
|
|
sizeof(pciData->Command));
|
|
|
|
// reread config space
|
|
busDataRead = HalGetBusData(PCIConfiguration, busNumber, slotNumber,
|
|
pciData, PCI_COMMON_HDR_LENGTH);
|
|
if(busDataRead < PCI_COMMON_HDR_LENGTH) {
|
|
KdPrint2((PRINT_PREFIX "HalGetBusData() failed %#x\n", busDataRead));
|
|
break;
|
|
}
|
|
KdPrint2((PRINT_PREFIX "New pciData.Command = %#x\n", pciData->Command));
|
|
}
|
|
KdPrint2((PRINT_PREFIX "InterruptLine = %#x\n", pciData->u.type0.InterruptLine));
|
|
KdPrint2((PRINT_PREFIX "Final pciData.Command = %#x\n", pciData->Command));
|
|
return pciData->Command;
|
|
} // end UniataEnableIoPCI()
|
|
|
|
/*
|
|
Get PCI address by ConfigInfo and RID
|
|
*/
|
|
ULONG
|
|
NTAPI
|
|
AtapiGetIoRange(
|
|
IN PVOID HwDeviceExtension,
|
|
IN PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
IN PPCI_COMMON_CONFIG pciData,
|
|
IN ULONG SystemIoBusNumber,
|
|
IN ULONG rid, //range id
|
|
IN ULONG offset,
|
|
IN ULONG length
|
|
)
|
|
{
|
|
ULONGIO_PTR io_start = 0;
|
|
KdPrint2((PRINT_PREFIX " AtapiGetIoRange:\n"));
|
|
|
|
if(ConfigInfo->NumberOfAccessRanges <= rid)
|
|
return 0;
|
|
|
|
KdPrint2((PRINT_PREFIX " AtapiGetIoRange: rid %#x, start %#x, offs %#x, len %#x, mem %#x\n",
|
|
rid,
|
|
ScsiPortConvertPhysicalAddressToUlong((*ConfigInfo->AccessRanges)[rid].RangeStart),
|
|
offset,
|
|
length,
|
|
(*ConfigInfo->AccessRanges)[rid].RangeInMemory
|
|
));
|
|
|
|
if(!(*ConfigInfo->AccessRanges)[rid].RangeInMemory) {
|
|
io_start = (pciData->u.type0.BaseAddresses[rid] & ~0x07/*PCI_ADDRESS_IOMASK*/) + offset;
|
|
// if(pciData->u.type0.BaseAddresses[rid] != 0) ;)
|
|
if(io_start > offset) {
|
|
if(/*(WinVer_Id() <= WinVer_NT) &&*/ offset && rid == 4) {
|
|
// MS atapi.sys does so for BusMaster controllers
|
|
(*ConfigInfo->AccessRanges)[rid+1].RangeStart =
|
|
ScsiPortConvertUlongToPhysicalAddress(io_start);
|
|
(*ConfigInfo->AccessRanges)[rid+1].RangeLength = length;
|
|
} else {
|
|
(*ConfigInfo->AccessRanges)[rid].RangeStart =
|
|
ScsiPortConvertUlongToPhysicalAddress(io_start);
|
|
(*ConfigInfo->AccessRanges)[rid].RangeLength = length;
|
|
}
|
|
if((pciData->u.type0.BaseAddresses[rid] & PCI_ADDRESS_IO_SPACE)) {
|
|
(*ConfigInfo->AccessRanges)[rid].RangeInMemory = FALSE;
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX " AtapiGetIoRange: adjust mem 0 -> 1\n"));
|
|
(*ConfigInfo->AccessRanges)[rid].RangeInMemory = TRUE;
|
|
}
|
|
} else {
|
|
io_start = 0;
|
|
}
|
|
}
|
|
|
|
if((*ConfigInfo->AccessRanges)[rid].RangeInMemory) {
|
|
if(offset) {
|
|
KdPrint2((PRINT_PREFIX " AtapiGetIoRange: can't map memory range with offset\n"));
|
|
return 0;
|
|
}
|
|
io_start =
|
|
// Get the system physical address for this IO range.
|
|
((ULONG_PTR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
PCIBus /*ConfigInfo->AdapterInterfaceType*/,
|
|
SystemIoBusNumber /*ConfigInfo->SystemIoBusNumber*/,
|
|
ScsiPortConvertUlongToPhysicalAddress(
|
|
(ScsiPortConvertPhysicalAddressToUlong((*ConfigInfo->AccessRanges)[rid].RangeStart) &
|
|
~0x07/*PCI_ADDRESS_IOMASK*/) + offset
|
|
),
|
|
length,
|
|
(BOOLEAN)!(*ConfigInfo->AccessRanges)[rid].RangeInMemory)
|
|
);
|
|
|
|
KdPrint2((PRINT_PREFIX " AtapiGetIoRange: %#x\n", io_start));
|
|
// if(io_start > offset) {
|
|
return io_start;
|
|
// }
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX " AtapiGetIoRange: (2) %#x\n", io_start));
|
|
return io_start;
|
|
|
|
} // end AtapiGetIoRange()
|
|
|
|
#ifndef UNIATA_CORE
|
|
|
|
/*
|
|
Do nothing, but build list of supported IDE controllers
|
|
It is a hack, ScsiPort architecture assumes, that DriverEntry
|
|
can support only KNOWN Vendor/Device combinations.
|
|
Thus, we build list here. Later we pretend that always knew
|
|
about found devices.
|
|
|
|
We shall initiate ISA device init, but callback will use
|
|
Hal routines directly in order to scan PCI bus.
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
UniataEnumBusMasterController(
|
|
IN PVOID DriverObject,
|
|
PVOID Argument2
|
|
)
|
|
{
|
|
UniataEnumBusMasterController__();
|
|
|
|
} // end UniataEnumBusMasterController()
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
UniataCheckPCISubclass(
|
|
BOOLEAN known,
|
|
ULONG RaidFlags,
|
|
UCHAR SubClass
|
|
)
|
|
{
|
|
if(known) {
|
|
if((RaidFlags & UNIATA_RAID_CONTROLLER) &&
|
|
SkipRaids) {
|
|
KdPrint2((PRINT_PREFIX "Skip RAID\n"));
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
KdPrint2((PRINT_PREFIX "unknown\n"));
|
|
|
|
switch(SubClass) {
|
|
case PCI_DEV_SUBCLASS_RAID:
|
|
if(SkipRaids) {
|
|
KdPrint2((PRINT_PREFIX "Skip RAID (2)\n"));
|
|
return FALSE;
|
|
}
|
|
break;
|
|
case PCI_DEV_SUBCLASS_IDE:
|
|
case PCI_DEV_SUBCLASS_ATA:
|
|
break;
|
|
case PCI_DEV_SUBCLASS_SATA:
|
|
break;
|
|
default:
|
|
KdPrint2((PRINT_PREFIX "Subclass not supported\n"));
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
} // end UniataCheckPCISubclass()
|
|
|
|
static CONST ULONG StdIsaPorts[] = {IO_WD1, IO_WD1 + ATA_ALTOFFSET, IO_WD2, IO_WD2 + ATA_ALTOFFSET, 0, 0};
|
|
|
|
/*
|
|
Device initializaton callback
|
|
Builds PCI device list using Hal routines (not ScsiPort wrappers)
|
|
*/
|
|
ULONG
|
|
NTAPI
|
|
UniataEnumBusMasterController__(
|
|
)
|
|
{
|
|
// PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
// PVOID HwDeviceExtension;
|
|
PHW_DEVICE_EXTENSION deviceExtension = NULL;
|
|
PCHAR PciDevMap = NULL;
|
|
PCI_SLOT_NUMBER slotData;
|
|
PCI_COMMON_CONFIG pciData;
|
|
ULONG busNumber;
|
|
ULONG slotNumber;
|
|
ULONG funcNumber;
|
|
BOOLEAN no_buses = FALSE;
|
|
BOOLEAN no_ranges = FALSE;
|
|
BOOLEAN non_isa = TRUE;
|
|
ULONG busDataRead;
|
|
// BOOLEAN SimplexOnly;
|
|
|
|
UCHAR vendorString[5];
|
|
UCHAR deviceString[5];
|
|
PUCHAR vendorStrPtr;
|
|
PUCHAR deviceStrPtr;
|
|
|
|
UCHAR BaseClass; // (ro)
|
|
UCHAR SubClass; // (ro)
|
|
ULONG VendorID;
|
|
ULONG DeviceID;
|
|
ULONG dev_id;
|
|
|
|
USHORT SubVendorID;
|
|
USHORT SubSystemID;
|
|
|
|
ULONG i;
|
|
ULONG pass=0;
|
|
|
|
ULONG RaidFlags;
|
|
|
|
BOOLEAN found;
|
|
BOOLEAN known;
|
|
BOOLEAN NeedPciAltInit;
|
|
BOOLEAN NonZeroSubId = 0;
|
|
|
|
UCHAR IrqForCompat = 10;
|
|
|
|
vendorStrPtr = vendorString;
|
|
deviceStrPtr = deviceString;
|
|
slotData.u.AsULONG = 0;
|
|
|
|
KdPrint2((PRINT_PREFIX "UniataEnumBusMasterController__: maxPciBus=%d\n", maxPciBus));
|
|
if(!maxPciBus) {
|
|
return(SP_RETURN_NOT_FOUND);
|
|
}
|
|
/*HwDeviceExtension =*/
|
|
deviceExtension = (PHW_DEVICE_EXTENSION)ExAllocatePool(NonPagedPool, sizeof(HW_DEVICE_EXTENSION));
|
|
if(!deviceExtension) {
|
|
KdPrint2((PRINT_PREFIX "!deviceExtension\n"));
|
|
return(SP_RETURN_NOT_FOUND);
|
|
}
|
|
RtlZeroMemory(deviceExtension, sizeof(HW_DEVICE_EXTENSION));
|
|
PciDevMap = (PCHAR)ExAllocatePool(NonPagedPool, maxPciBus*PCI_MAX_DEVICES);
|
|
if(!PciDevMap) {
|
|
KdPrint2((PRINT_PREFIX "!PciDevMap\n"));
|
|
goto exit;
|
|
}
|
|
RtlZeroMemory(PciDevMap, maxPciBus*PCI_MAX_DEVICES);
|
|
|
|
for(pass=0; pass<3; pass++) {
|
|
KdPrint2((PRINT_PREFIX " pass %d\n", pass));
|
|
no_buses = FALSE;
|
|
for(busNumber=0 ;busNumber<maxPciBus && !no_buses; busNumber++) {
|
|
for(slotNumber=0; slotNumber<PCI_MAX_DEVICES && !no_buses; slotNumber++) {
|
|
NeedPciAltInit = FALSE;
|
|
for(funcNumber=0; funcNumber<PCI_MAX_FUNCTION && !no_buses; funcNumber++) {
|
|
|
|
if(pass) {
|
|
// use cached device presence map from the 1st pass
|
|
if(PciDevMap[busNumber*PCI_MAX_DEVICES + slotNumber] & (1 << funcNumber)) {
|
|
// ok
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
// KdPrint2((PRINT_PREFIX "-- BusID: %#x:%#x:%#x\n",busNumber,slotNumber,funcNumber));
|
|
slotData.u.bits.DeviceNumber = slotNumber;
|
|
slotData.u.bits.FunctionNumber = funcNumber;
|
|
|
|
busDataRead = HalGetBusData
|
|
//ScsiPortGetBusData
|
|
(
|
|
//HwDeviceExtension,
|
|
PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&pciData, PCI_COMMON_HDR_LENGTH);
|
|
// no more buses
|
|
if(!busDataRead) {
|
|
no_buses = TRUE; // break all nested bus scan loops and continue with next pass
|
|
maxPciBus = busNumber;
|
|
break;
|
|
}
|
|
// indicate that system has PCI bus(es)
|
|
hasPCI = TRUE;
|
|
|
|
// no device in this slot
|
|
if(busDataRead == 2) {
|
|
NeedPciAltInit = TRUE;
|
|
continue;
|
|
}
|
|
|
|
if(busDataRead < (ULONG)PCI_COMMON_HDR_LENGTH) {
|
|
NeedPciAltInit = TRUE;
|
|
continue;
|
|
}
|
|
|
|
VendorID = pciData.VendorID;
|
|
DeviceID = pciData.DeviceID;
|
|
BaseClass = pciData.BaseClass;
|
|
SubClass = pciData.SubClass;
|
|
dev_id = VendorID | (DeviceID << 16);
|
|
|
|
SubVendorID = pciData.u.type0.SubVendorID;
|
|
SubSystemID = pciData.u.type0.SubSystemID;
|
|
|
|
if(SubVendorID && SubSystemID) {
|
|
NonZeroSubId = 1;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "DevId = %8.8X Class = %4.4X/%4.4X, SubVen/Sys %4.4x/%4.4x\n", dev_id, BaseClass, SubClass, SubVendorID, SubSystemID));
|
|
|
|
// check for (g_opt_VirtualMachine == VM_AUTO) is performed inside each
|
|
// VM check for debug purposes
|
|
// Do not optimize :)
|
|
if((VendorID == 0x80ee && DeviceID == 0xcafe) ||
|
|
(VendorID == 0x80ee && DeviceID == 0xbeef)) {
|
|
KdPrint2((PRINT_PREFIX "-- BusID: %#x:%#x:%#x - VirtualBox Guest Service\n",busNumber,slotNumber,funcNumber));
|
|
if(g_opt_VirtualMachine == VM_AUTO) {
|
|
g_opt_VirtualMachine = VM_VBOX;
|
|
}
|
|
} else
|
|
if((VendorID == 0x15ad) ||
|
|
(SubVendorID == 0x15ad && SubSystemID == 0x1976)) {
|
|
KdPrint2((PRINT_PREFIX "-- BusID: %#x:%#x:%#x - VMWare\n",busNumber,slotNumber,funcNumber));
|
|
if(g_opt_VirtualMachine == VM_AUTO) {
|
|
g_opt_VirtualMachine = VM_VMWARE;
|
|
}
|
|
} else
|
|
if(SubVendorID == 0x1af4 && SubSystemID == 0x1100) {
|
|
KdPrint2((PRINT_PREFIX "-- BusID: %#x:%#x:%#x - QEmu\n",busNumber,slotNumber,funcNumber));
|
|
if(g_opt_VirtualMachine == VM_AUTO) {
|
|
g_opt_VirtualMachine = VM_QEMU;
|
|
}
|
|
} else
|
|
if(VendorID == 0x1234 && DeviceID == 0x1111) {
|
|
KdPrint2((PRINT_PREFIX "-- BusID: %#x:%#x:%#x - Bochs\n",busNumber,slotNumber,funcNumber));
|
|
if(g_opt_VirtualMachine == VM_AUTO) {
|
|
g_opt_VirtualMachine = VM_BOCHS;
|
|
}
|
|
/* } else
|
|
if(pass>0 && !NonZeroSubId &&
|
|
VendorID == 0x8086 &&
|
|
(DeviceID == 0x7010 ||
|
|
DeviceID == 0x1230)) {
|
|
KdPrint2((PRINT_PREFIX "-- BusID: %#x:%#x:%#x - Bochs PIIX emulation\n",busNumber,slotNumber,funcNumber));
|
|
if(g_opt_VirtualMachine == VM_AUTO) {
|
|
g_opt_VirtualMachine = VM_BOCHS;
|
|
}*/
|
|
}
|
|
|
|
if(BaseClass != PCI_DEV_CLASS_STORAGE) {
|
|
continue;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "-- BusID: %#x:%#x:%#x\n",busNumber,slotNumber,funcNumber));
|
|
KdPrint2((PRINT_PREFIX "Storage Class\n"));
|
|
KdPrint2((PRINT_PREFIX "DevId = %8.8X Class = %4.4X/%4.4X, ProgIf %2.2X\n", dev_id, BaseClass, SubClass, pciData.ProgIf ));
|
|
// look for known chipsets
|
|
found = FALSE;
|
|
known = FALSE;
|
|
|
|
if(pciData.u.type0.InterruptPin == 14 ||
|
|
pciData.u.type0.InterruptPin == 15 ||
|
|
pciData.u.type0.InterruptLine == 14 ||
|
|
pciData.u.type0.InterruptLine == 15) {
|
|
KdPrint2((PRINT_PREFIX "(!) InterruptPin = %#x\n", pciData.u.type0.InterruptPin));
|
|
KdPrint2((PRINT_PREFIX "(!) InterruptLine = %#x\n", pciData.u.type0.InterruptLine));
|
|
}
|
|
|
|
if(deviceExtension) {
|
|
deviceExtension->slotNumber = slotData.u.AsULONG;
|
|
deviceExtension->SystemIoBusNumber = busNumber;
|
|
deviceExtension->DevID = dev_id;
|
|
deviceExtension->RevID = pciData.RevisionID;
|
|
deviceExtension->AdapterInterfaceType = PCIBus;
|
|
}
|
|
|
|
found = (BOOLEAN)AtapiRegCheckDevValue(deviceExtension, CHAN_NOT_SPECIFIED, DEVNUM_NOT_SPECIFIED, L"Include", 0);
|
|
if(!found) {
|
|
KdPrint2((PRINT_PREFIX "No force include, check exclude\n"));
|
|
found = !AtapiRegCheckDevValue(deviceExtension, CHAN_NOT_SPECIFIED, DEVNUM_NOT_SPECIFIED, L"Exclude", 0);
|
|
if(!found) {
|
|
KdPrint2((PRINT_PREFIX "Device excluded\n"));
|
|
continue;
|
|
}
|
|
}
|
|
|
|
//known = UniataChipDetect(HwDeviceExtension, NULL, -1, ConfigInfo, &SimplexOnly);
|
|
i = Ata_is_dev_listed((PBUSMASTER_CONTROLLER_INFORMATION_BASE)&BusMasterAdapters[0], VendorID, DeviceID, 0, NUM_BUSMASTER_ADAPTERS);
|
|
|
|
known = (i != BMLIST_TERMINATOR);
|
|
if(known) {
|
|
deviceExtension->FullDevName = BusMasterAdapters[i].FullDevName;
|
|
RaidFlags = BusMasterAdapters[i].RaidFlags;
|
|
} else {
|
|
deviceExtension->FullDevName = "Unknown Storage";
|
|
RaidFlags = 0;
|
|
}
|
|
found = UniataCheckPCISubclass(known, RaidFlags, SubClass);
|
|
if(!found) {
|
|
KdPrint2((PRINT_PREFIX "Subclass not supported\n"));
|
|
continue;
|
|
}
|
|
|
|
switch(dev_id) {
|
|
/* additional checks for some supported chipsets */
|
|
case 0xc6931080:
|
|
if (SubClass != PCI_DEV_SUBCLASS_IDE)
|
|
found = FALSE;
|
|
break;
|
|
|
|
/* unknown chipsets, try generic DMA if it seems possible */
|
|
default:
|
|
KdPrint2((PRINT_PREFIX "Default device\n"));
|
|
if(Ata_is_supported_dev(&pciData) ||
|
|
Ata_is_ahci_dev(&pciData))
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
|
|
if(!found) {
|
|
continue;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "found, pass %d\n", pass));
|
|
|
|
KdPrint2((PRINT_PREFIX "InterruptPin = %#x\n", pciData.u.type0.InterruptPin));
|
|
KdPrint2((PRINT_PREFIX "InterruptLine = %#x\n", pciData.u.type0.InterruptLine));
|
|
|
|
if(!pass && known) {
|
|
UniataEnableIoPCI(busNumber, slotData.u.AsULONG, &pciData);
|
|
}
|
|
// validate Mem/Io ranges
|
|
no_ranges = TRUE;
|
|
non_isa = TRUE;
|
|
for(i=0; i<PCI_TYPE0_ADDRESSES; i++) {
|
|
if(pciData.u.type0.BaseAddresses[i] & ~0x7) {
|
|
no_ranges = FALSE;
|
|
//break;
|
|
KdPrint2((PRINT_PREFIX "Range %d = %#x\n", i, pciData.u.type0.BaseAddresses[i]));
|
|
if(i<4) {
|
|
if(StdIsaPorts[i] == (pciData.u.type0.BaseAddresses[i] & ~0x7)) {
|
|
non_isa = FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if(no_ranges) {
|
|
KdPrint2((PRINT_PREFIX "No PCI Mem/Io ranges found on device, skip it\n"));
|
|
continue;
|
|
}
|
|
if(!non_isa) {
|
|
KdPrint2((PRINT_PREFIX "standard ISA ranges on PCI, special case ?\n"));
|
|
}
|
|
|
|
if(pass) {
|
|
// fill list of detected devices
|
|
// it'll be used for further init
|
|
KdPrint2((PRINT_PREFIX "found suitable device\n"));
|
|
PBUSMASTER_CONTROLLER_INFORMATION newBMListPtr = BMList+BMListLen;
|
|
|
|
if(pass == 1) {
|
|
if(!IsMasterDev(&pciData)) {
|
|
continue;
|
|
}
|
|
if(AtapiRegCheckDevValue(deviceExtension, CHAN_NOT_SPECIFIED, DEVNUM_NOT_SPECIFIED, L"NativePCIMode", 0)) {
|
|
KdPrint2((PRINT_PREFIX "try switch to native mode\n"));
|
|
|
|
IrqForCompat = (UCHAR)AtapiRegCheckDevValue(deviceExtension, CHAN_NOT_SPECIFIED, DEVNUM_NOT_SPECIFIED, L"NativePCIModeIRQ", 0xff);
|
|
KdPrint2((PRINT_PREFIX "IrqForCompat = %#x\n", IrqForCompat));
|
|
if((IrqForCompat & 0xffffff00) /*||
|
|
(IrqForCompat & 0xff) > 31*/ ||
|
|
(IrqForCompat == 0xff)) {
|
|
IrqForCompat = 0x0b;
|
|
KdPrint2((PRINT_PREFIX "default to IRQ 11\n"));
|
|
}
|
|
|
|
//ChangePciConfig1(0x09, a | PCI_IDE_PROGIF_NATIVE_ALL); // ProgIf
|
|
pciData.ProgIf |= PCI_IDE_PROGIF_NATIVE_ALL;
|
|
HalSetBusDataByOffset( PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&(pciData.ProgIf),
|
|
offsetof(PCI_COMMON_CONFIG, ProgIf),
|
|
sizeof(pciData.ProgIf));
|
|
|
|
// reread config space
|
|
busDataRead = HalGetBusData(PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&pciData, PCI_COMMON_HDR_LENGTH);
|
|
// check if the device have switched to Native Mode
|
|
if(IsMasterDev(&pciData)) {
|
|
KdPrint2((PRINT_PREFIX "Can't switch to native mode\n"));
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "switched to native mode\n"));
|
|
KdPrint2((PRINT_PREFIX "InterruptPin = %#x\n", pciData.u.type0.InterruptPin));
|
|
KdPrint2((PRINT_PREFIX "InterruptLine = %#x\n", pciData.u.type0.InterruptLine));
|
|
// check if IRQ is assigned to device
|
|
if(!(pciData.u.type0.InterruptLine) ||
|
|
(pciData.u.type0.InterruptLine == 0xff)) {
|
|
KdPrint2((PRINT_PREFIX "assign interrupt for device\n"));
|
|
pciData.u.type0.InterruptLine = IrqForCompat;
|
|
HalSetBusDataByOffset( PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&(pciData.u.type0.InterruptLine),
|
|
offsetof(PCI_COMMON_CONFIG, u.type0.InterruptLine),
|
|
sizeof(pciData.u.type0.InterruptLine));
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "Auto-assigned interrupt line %#x\n",
|
|
pciData.u.type0.InterruptLine));
|
|
IrqForCompat = pciData.u.type0.InterruptLine;
|
|
}
|
|
KdPrint2((PRINT_PREFIX "reread config space\n"));
|
|
// reread config space
|
|
busDataRead = HalGetBusData(PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&pciData, PCI_COMMON_HDR_LENGTH);
|
|
KdPrint2((PRINT_PREFIX "busDataRead = %#x\n", busDataRead));
|
|
KdPrint2((PRINT_PREFIX "reread InterruptLine = %#x\n", pciData.u.type0.InterruptLine));
|
|
// check if we have successfully assigned IRQ to device
|
|
if((pciData.u.type0.InterruptLine != IrqForCompat) ||
|
|
(pciData.u.type0.InterruptLine == 0xff) ||
|
|
!pciData.u.type0.InterruptLine) {
|
|
KdPrint2((PRINT_PREFIX "can't assign interrupt for device, revert to compat mode\n"));
|
|
pciData.u.type0.InterruptLine = 0xff;
|
|
KdPrint2((PRINT_PREFIX "set IntrLine to 0xff\n"));
|
|
HalSetBusDataByOffset( PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&(pciData.u.type0.InterruptLine),
|
|
offsetof(PCI_COMMON_CONFIG, u.type0.InterruptLine),
|
|
sizeof(pciData.u.type0.InterruptLine));
|
|
KdPrint2((PRINT_PREFIX "clear PCI_IDE_PROGIF_NATIVE_ALL\n"));
|
|
pciData.ProgIf &= ~PCI_IDE_PROGIF_NATIVE_ALL;
|
|
HalSetBusDataByOffset( PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&(pciData.ProgIf),
|
|
offsetof(PCI_COMMON_CONFIG, ProgIf),
|
|
sizeof(pciData.ProgIf));
|
|
// reread config space
|
|
KdPrint2((PRINT_PREFIX "reread config space on revert\n"));
|
|
busDataRead = HalGetBusData(PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&pciData, PCI_COMMON_HDR_LENGTH);
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "Assigned interrupt %#x for device\n", IrqForCompat));
|
|
KdPrint2((PRINT_PREFIX "continue detection on next round\n"));
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
} else
|
|
if(pass == 2) {
|
|
if(IsMasterDev(&pciData)) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* if(known) {
|
|
RtlCopyMemory(newBMListPtr, (PVOID)&(BusMasterAdapters[i]), sizeof(BUSMASTER_CONTROLLER_INFORMATION));
|
|
} else {*/
|
|
sprintf((PCHAR)vendorStrPtr, "%4.4lx", VendorID);
|
|
sprintf((PCHAR)deviceStrPtr, "%4.4lx", DeviceID);
|
|
|
|
RtlCopyMemory(&(newBMListPtr->VendorIdStr), (PCHAR)vendorStrPtr, 4);
|
|
RtlCopyMemory(&(newBMListPtr->DeviceIdStr), (PCHAR)deviceStrPtr, 4);
|
|
|
|
newBMListPtr->nVendorId = VendorID;
|
|
newBMListPtr->VendorId = (PCHAR)&(newBMListPtr->VendorIdStr);
|
|
newBMListPtr->VendorIdLength = 4;
|
|
newBMListPtr->nDeviceId = DeviceID;
|
|
newBMListPtr->DeviceId = (PCHAR)&(newBMListPtr->DeviceIdStr);
|
|
newBMListPtr->DeviceIdLength = 4;
|
|
|
|
newBMListPtr->RaidFlags = RaidFlags;
|
|
// }
|
|
newBMListPtr->slotNumber = slotData.u.AsULONG;
|
|
newBMListPtr->MasterDev = IsMasterDev(&pciData) ? 1 : 0;
|
|
newBMListPtr->busNumber = busNumber;
|
|
|
|
newBMListPtr->NeedAltInit = NeedPciAltInit;
|
|
newBMListPtr->Known = known;
|
|
|
|
if(!non_isa) {
|
|
KdPrint2((PRINT_PREFIX "* ISA ranges on PCI, special case !\n"));
|
|
// Do not fail init after unseccessfull call of UniataClaimLegacyPCIIDE()
|
|
// some SMP HALs fails to reallocate IO range
|
|
newBMListPtr->ChanInitOk |= 0x40;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "Add to BMList, AltInit %d\n", NeedPciAltInit));
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "count: BMListLen++\n"));
|
|
PciDevMap[busNumber*PCI_MAX_DEVICES + slotNumber] |= (1 << funcNumber);
|
|
}
|
|
|
|
BMListLen++;
|
|
|
|
} // Function
|
|
} // Slot
|
|
if(!hasPCI) {
|
|
break;
|
|
}
|
|
}
|
|
if(!pass) {
|
|
if(!BMListLen)
|
|
break;
|
|
BMList = (PBUSMASTER_CONTROLLER_INFORMATION)ExAllocatePool(NonPagedPool,
|
|
(BMListLen+1)*sizeof(BUSMASTER_CONTROLLER_INFORMATION));
|
|
if(!BMList) {
|
|
BMListLen=0;
|
|
break;
|
|
}
|
|
RtlZeroMemory(BMList, (BMListLen+1)*sizeof(BUSMASTER_CONTROLLER_INFORMATION));
|
|
no_buses = FALSE;
|
|
BMListLen=0;
|
|
}
|
|
}
|
|
exit:
|
|
KdPrint2((PRINT_PREFIX " BMListLen=%x\n", BMListLen));
|
|
if(deviceExtension) {
|
|
ExFreePool(deviceExtension);
|
|
}
|
|
if(PciDevMap) {
|
|
ExFreePool(PciDevMap);
|
|
}
|
|
return(SP_RETURN_NOT_FOUND);
|
|
} // end UniataEnumBusMasterController__()
|
|
|
|
|
|
/*
|
|
Wrapper for read PCI config space
|
|
*/
|
|
ULONG
|
|
NTAPI
|
|
ScsiPortGetBusDataByOffset(
|
|
IN PVOID HwDeviceExtension,
|
|
IN BUS_DATA_TYPE BusDataType,
|
|
IN ULONG BusNumber,
|
|
IN ULONG SlotNumber,
|
|
IN PVOID Buffer,
|
|
IN ULONG Offset,
|
|
IN ULONG Length
|
|
)
|
|
{
|
|
UCHAR tmp[256];
|
|
ULONG busDataRead;
|
|
|
|
if(Offset+Length > 256)
|
|
return 0;
|
|
|
|
busDataRead = HalGetBusData(
|
|
//ScsiPortGetBusData(HwDeviceExtension,
|
|
BusDataType,
|
|
BusNumber,
|
|
SlotNumber,
|
|
&tmp,
|
|
Offset+Length);
|
|
if(busDataRead < Offset+Length) {
|
|
if(busDataRead < Offset)
|
|
return 0;
|
|
return (Offset+Length-busDataRead);
|
|
}
|
|
RtlCopyMemory(Buffer, tmp+Offset, Length);
|
|
return Length;
|
|
} // end ScsiPortGetBusDataByOffset()
|
|
|
|
/*
|
|
Looks for devices from list on specified bus(es)/slot(s)
|
|
returnts its index in list.
|
|
If no matching record found, -1 is returned
|
|
*/
|
|
ULONG
|
|
NTAPI
|
|
AtapiFindListedDev(
|
|
IN PBUSMASTER_CONTROLLER_INFORMATION_BASE BusMasterAdapters,
|
|
IN ULONG lim,
|
|
IN PVOID HwDeviceExtension,
|
|
IN ULONG BusNumber,
|
|
IN ULONG SlotNumber,
|
|
OUT PCI_SLOT_NUMBER* _slotData // optional
|
|
)
|
|
{
|
|
PCI_SLOT_NUMBER slotData;
|
|
PCI_COMMON_CONFIG pciData;
|
|
ULONG busDataRead;
|
|
|
|
ULONG busNumber;
|
|
ULONG slotNumber;
|
|
ULONG funcNumber;
|
|
|
|
ULONG busNumber2;
|
|
ULONG slotNumber2;
|
|
|
|
ULONG i;
|
|
|
|
KdPrint2((PRINT_PREFIX "AtapiFindListedDev: lim=%x, Bus=%x, Slot=%x\n", lim, BusNumber, SlotNumber));
|
|
|
|
// set start/end bus
|
|
if(BusNumber == PCIBUSNUM_NOT_SPECIFIED) {
|
|
busNumber = 0;
|
|
busNumber2 = maxPciBus;
|
|
} else {
|
|
busNumber = BusNumber;
|
|
busNumber2 = BusNumber+1;
|
|
}
|
|
// set start/end slot
|
|
if(SlotNumber == PCISLOTNUM_NOT_SPECIFIED) {
|
|
slotNumber = 0;
|
|
slotNumber2 = PCI_MAX_DEVICES;
|
|
} else {
|
|
slotNumber = SlotNumber;
|
|
slotNumber2 = SlotNumber+1;
|
|
}
|
|
slotData.u.AsULONG = 0;
|
|
|
|
KdPrint2((PRINT_PREFIX " scanning range Bus %x-%x, Slot %x-%x\n", busNumber, busNumber2-1, slotNumber, slotNumber2-1));
|
|
|
|
for( ; busNumber < busNumber2 ; busNumber++ ) {
|
|
for( ; slotNumber < slotNumber2 ; slotNumber++) {
|
|
for(funcNumber=0; funcNumber < PCI_MAX_FUNCTION ; funcNumber++) {
|
|
|
|
slotData.u.bits.DeviceNumber = slotNumber;
|
|
slotData.u.bits.FunctionNumber = funcNumber;
|
|
|
|
busDataRead = HalGetBusData(
|
|
//ScsiPortGetBusData(HwDeviceExtension,
|
|
PCIConfiguration, busNumber, slotData.u.AsULONG,
|
|
&pciData, PCI_COMMON_HDR_LENGTH);
|
|
// no more buses (this should not happen)
|
|
if(!busDataRead) {
|
|
continue;
|
|
}
|
|
// no device in this slot
|
|
if(busDataRead == 2)
|
|
continue;
|
|
|
|
if(busDataRead < (ULONG)PCI_COMMON_HDR_LENGTH)
|
|
continue;
|
|
/*
|
|
KdPrint2((PRINT_PREFIX "AtapiFindListedDev: b:s:f(%x:%x:%x) %4.4x/%4.4x/%2.2x\n",
|
|
busNumber, slotNumber, funcNumber,
|
|
pciData.VendorID, pciData.DeviceID, pciData.RevisionID));
|
|
*/
|
|
i = Ata_is_dev_listed(BusMasterAdapters, pciData.VendorID, pciData.DeviceID, pciData.RevisionID, lim);
|
|
if(i != BMLIST_TERMINATOR) {
|
|
if(_slotData)
|
|
*_slotData = slotData;
|
|
KdPrint2((PRINT_PREFIX "AtapiFindListedDev: found\n"));
|
|
KdPrint2((PRINT_PREFIX "AtapiFindListedDev: b:s:f(%x:%x:%x) %4.4x/%4.4x/%2.2x\n",
|
|
busNumber, slotNumber, funcNumber,
|
|
pciData.VendorID, pciData.DeviceID, pciData.RevisionID));
|
|
return i;
|
|
}
|
|
|
|
}}}
|
|
return -1;
|
|
} // end AtapiFindListedDev()
|
|
|
|
/*
|
|
Looks for device with specified Device/Vendor and Revision
|
|
on specified Bus/Slot
|
|
*/
|
|
ULONG
|
|
NTAPI
|
|
AtapiFindDev(
|
|
IN PVOID HwDeviceExtension,
|
|
IN BUS_DATA_TYPE BusDataType,
|
|
IN ULONG BusNumber,
|
|
IN ULONG SlotNumber,
|
|
IN ULONG dev_id,
|
|
IN ULONG RevID
|
|
)
|
|
{
|
|
PCI_COMMON_CONFIG pciData;
|
|
ULONG funcNumber;
|
|
ULONG busDataRead;
|
|
|
|
ULONG VendorID;
|
|
ULONG DeviceID;
|
|
PCI_SLOT_NUMBER slotData;
|
|
|
|
slotData.u.AsULONG = SlotNumber;
|
|
// walk through all Function Numbers
|
|
for(funcNumber = 0; funcNumber < PCI_MAX_FUNCTION; funcNumber++) {
|
|
|
|
slotData.u.bits.FunctionNumber = funcNumber;
|
|
if(slotData.u.AsULONG == SlotNumber)
|
|
continue;
|
|
|
|
busDataRead = HalGetBusData(
|
|
//busDataRead = ScsiPortGetBusData(HwDeviceExtension,
|
|
PCIConfiguration,
|
|
BusNumber,
|
|
slotData.u.AsULONG,
|
|
&pciData,
|
|
PCI_COMMON_HDR_LENGTH);
|
|
|
|
if (busDataRead < (ULONG)PCI_COMMON_HDR_LENGTH) {
|
|
continue;
|
|
}
|
|
|
|
VendorID = pciData.VendorID;
|
|
DeviceID = pciData.DeviceID;
|
|
|
|
if(dev_id != (VendorID | (DeviceID << 16)) )
|
|
continue;
|
|
if(RevID >= pciData.RevisionID)
|
|
return 1;
|
|
}
|
|
return 0;
|
|
} // end AtapiFindDev()
|
|
|
|
#endif //UNIATA_CORE
|
|
|
|
|
|
ULONG
|
|
NTAPI
|
|
UniataFindCompatBusMasterController1(
|
|
IN PVOID HwDeviceExtension,
|
|
IN PVOID Context,
|
|
IN PVOID BusInformation,
|
|
IN PCHAR ArgumentString,
|
|
IN OUT PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
OUT PBOOLEAN Again
|
|
)
|
|
{
|
|
return UniataFindBusMasterController(
|
|
HwDeviceExtension,
|
|
(PVOID)0x00000000,
|
|
BusInformation,
|
|
ArgumentString,
|
|
ConfigInfo,
|
|
Again
|
|
);
|
|
} // end UniataFindCompatBusMasterController1()
|
|
|
|
ULONG
|
|
NTAPI
|
|
UniataFindCompatBusMasterController2(
|
|
IN PVOID HwDeviceExtension,
|
|
IN PVOID Context,
|
|
IN PVOID BusInformation,
|
|
IN PCHAR ArgumentString,
|
|
IN OUT PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
OUT PBOOLEAN Again
|
|
)
|
|
{
|
|
return UniataFindBusMasterController(
|
|
HwDeviceExtension,
|
|
(PVOID)0x80000000,
|
|
BusInformation,
|
|
ArgumentString,
|
|
ConfigInfo,
|
|
Again
|
|
);
|
|
} // end UniataFindCompatBusMasterController2()
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function is called by the OS-specific port driver after
|
|
the necessary storage has been allocated, to gather information
|
|
about the adapter's configuration.
|
|
|
|
Arguments:
|
|
|
|
HwDeviceExtension - HBA miniport driver's adapter data storage
|
|
Context - Address of adapter count
|
|
BusInformation -
|
|
ArgumentString - Used to determine whether driver is client of ntldr or crash dump utility.
|
|
ConfigInfo - Configuration information structure describing HBA
|
|
Again - Indicates search for adapters to continue
|
|
|
|
Return Value:
|
|
|
|
ULONG
|
|
|
|
--*/
|
|
ULONG
|
|
NTAPI
|
|
UniataFindBusMasterController(
|
|
IN PVOID HwDeviceExtension,
|
|
IN PVOID Context,
|
|
IN PVOID BusInformation,
|
|
IN PCHAR ArgumentString,
|
|
IN OUT PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
OUT PBOOLEAN Again
|
|
)
|
|
{
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
PHW_CHANNEL chan = NULL;
|
|
#ifndef UNIATA_CORE
|
|
// this buffer must be global for UNIATA_CORE build
|
|
PCI_COMMON_CONFIG pciData;
|
|
#endif //UNIATA_CORE
|
|
ULONG slotNumber;
|
|
ULONG busDataRead;
|
|
ULONG SystemIoBusNumber;
|
|
/*
|
|
UCHAR vendorString[5];
|
|
UCHAR deviceString[5];
|
|
|
|
PUCHAR vendorStrPtr;
|
|
PUCHAR deviceStrPtr;
|
|
*/
|
|
UCHAR BaseClass;
|
|
UCHAR SubClass;
|
|
ULONG VendorID;
|
|
ULONG DeviceID;
|
|
ULONG RevID;
|
|
ULONG dev_id;
|
|
PCI_SLOT_NUMBER slotData;
|
|
|
|
ULONG i;
|
|
ULONG channel;
|
|
ULONG c = 0;
|
|
PUCHAR ioSpace;
|
|
UCHAR statusByte;
|
|
ULONG bm_offset;
|
|
|
|
// UCHAR tmp8;
|
|
// ULONG irq;
|
|
|
|
BOOLEAN found = FALSE;
|
|
BOOLEAN MasterDev;
|
|
BOOLEAN simplexOnly = FALSE;
|
|
#ifndef UNIATA_CORE
|
|
#ifdef UNIATA_INIT_ON_PROBE
|
|
BOOLEAN skip_find_dev = FALSE;
|
|
#endif
|
|
#endif
|
|
BOOLEAN AltInit = FALSE;
|
|
|
|
SCSI_PHYSICAL_ADDRESS IoBasePort1;
|
|
SCSI_PHYSICAL_ADDRESS IoBasePort2;
|
|
|
|
PIDE_BUSMASTER_REGISTERS BaseIoAddressBM_0 = NULL;
|
|
PIDE_REGISTERS_1 BaseIoAddress1[IDE_MAX_CHAN];
|
|
PIDE_REGISTERS_2 BaseIoAddress2[IDE_MAX_CHAN];
|
|
|
|
RtlZeroMemory(&BaseIoAddress1, sizeof(BaseIoAddress1));
|
|
RtlZeroMemory(&BaseIoAddress2, sizeof(BaseIoAddress2));
|
|
|
|
NTSTATUS status;
|
|
PPORT_CONFIGURATION_INFORMATION_COMMON _ConfigInfo =
|
|
(PPORT_CONFIGURATION_INFORMATION_COMMON)ConfigInfo;
|
|
|
|
if(!WinVer_WDM_Model) {
|
|
*Again = FALSE;
|
|
} else {
|
|
*Again = TRUE;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "UniataFindBusMasterController: Context=%x, BMListLen=%d\n", Context, BMListLen));
|
|
|
|
KdPrint2((PRINT_PREFIX "ConfigInfo->Length %x\n", ConfigInfo->Length));
|
|
|
|
if(ForceSimplex) {
|
|
KdPrint2((PRINT_PREFIX "ForceSimplex (1)\n"));
|
|
simplexOnly = TRUE;
|
|
}
|
|
|
|
if(ConfigInfo->AdapterInterfaceType == Isa) {
|
|
KdPrint2((PRINT_PREFIX "AdapterInterfaceType: Isa\n"));
|
|
}
|
|
if(InDriverEntry) {
|
|
i = (ULONG)Context;
|
|
if(i & 0x80000000) {
|
|
AltInit = TRUE;
|
|
}
|
|
i &= ~0x80000000;
|
|
channel = BMList[i].channel;
|
|
} else {
|
|
channel = 0;
|
|
for(i=0; i<BMListLen; i++) {
|
|
if(BMList[i].slotNumber == ConfigInfo->SlotNumber &&
|
|
BMList[i].busNumber == ConfigInfo->SystemIoBusNumber) {
|
|
break;
|
|
}
|
|
}
|
|
if(i >= BMListLen) {
|
|
KdPrint2((PRINT_PREFIX "unexpected device arrival\n"));
|
|
i = (ULONG)Context;
|
|
if(FirstMasterOk) {
|
|
channel = 1;
|
|
}
|
|
i &= ~0x80000000;
|
|
if(i >= BMListLen) {
|
|
KdPrint2((PRINT_PREFIX " => SP_RETURN_NOT_FOUND\n"));
|
|
goto exit_notfound;
|
|
}
|
|
}
|
|
BMList[i].channel = (UCHAR)channel;
|
|
}
|
|
|
|
bm_offset = channel ? ATA_BM_OFFSET1 : 0;
|
|
|
|
KdPrint2((PRINT_PREFIX "bm_offset %x, channel %x \n", bm_offset, channel));
|
|
|
|
if (!deviceExtension) {
|
|
KdPrint2((PRINT_PREFIX "!deviceExtension => SP_RETURN_ERROR\n"));
|
|
return SP_RETURN_ERROR;
|
|
}
|
|
RtlZeroMemory(deviceExtension, sizeof(HW_DEVICE_EXTENSION));
|
|
/*
|
|
vendorStrPtr = vendorString;
|
|
deviceStrPtr = deviceString;
|
|
*/
|
|
slotNumber = BMList[i].slotNumber;
|
|
SystemIoBusNumber = BMList[i].busNumber;
|
|
|
|
|
|
KdPrint2((PRINT_PREFIX "AdapterInterfaceType=%#x\n",ConfigInfo->AdapterInterfaceType));
|
|
KdPrint2((PRINT_PREFIX "IoBusNumber=%#x\n",ConfigInfo->SystemIoBusNumber));
|
|
KdPrint2((PRINT_PREFIX "slotNumber=%#x\n",slotNumber));
|
|
|
|
// this buffer must be global and already filled for UNIATA_CORE build
|
|
busDataRead = HalGetBusData(
|
|
//busDataRead = ScsiPortGetBusData(HwDeviceExtension,
|
|
PCIConfiguration,
|
|
SystemIoBusNumber,
|
|
slotNumber,
|
|
&pciData,
|
|
PCI_COMMON_HDR_LENGTH);
|
|
|
|
#ifndef UNIATA_CORE
|
|
if (busDataRead < (ULONG)PCI_COMMON_HDR_LENGTH) {
|
|
KdPrint2((PRINT_PREFIX "busDataRead < PCI_COMMON_HDR_LENGTH => SP_RETURN_ERROR\n"));
|
|
goto exit_error;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "busDataRead\n"));
|
|
if (pciData.VendorID == PCI_INVALID_VENDORID) {
|
|
KdPrint2((PRINT_PREFIX "PCI_INVALID_VENDORID\n"));
|
|
goto exit_error;
|
|
}
|
|
#endif //UNIATA_CORE
|
|
|
|
VendorID = pciData.VendorID;
|
|
DeviceID = pciData.DeviceID;
|
|
BaseClass = pciData.BaseClass;
|
|
SubClass = pciData.SubClass;
|
|
RevID = pciData.RevisionID;
|
|
dev_id = VendorID | (DeviceID << 16);
|
|
slotData.u.AsULONG = slotNumber;
|
|
KdPrint2((PRINT_PREFIX "DevId = %8.8X Class = %4.4X/%4.4X\n", dev_id, BaseClass, SubClass ));
|
|
|
|
deviceExtension->slotNumber = slotNumber;
|
|
deviceExtension->SystemIoBusNumber = SystemIoBusNumber;
|
|
deviceExtension->DevID = dev_id;
|
|
deviceExtension->RevID = RevID;
|
|
deviceExtension->NumberChannels = IDE_DEFAULT_MAX_CHAN; // default
|
|
deviceExtension->NumberLuns = IDE_MAX_LUN_PER_CHAN; // default
|
|
deviceExtension->DevIndex = i;
|
|
|
|
_snprintf(deviceExtension->Signature, sizeof(deviceExtension->Signature),
|
|
"UATA%8.8x/%1.1x@%8.8x", dev_id, channel, slotNumber);
|
|
|
|
if(BaseClass != PCI_DEV_CLASS_STORAGE) {
|
|
KdPrint2((PRINT_PREFIX "BaseClass != PCI_DEV_CLASS_STORAGE => SP_RETURN_NOT_FOUND\n"));
|
|
goto exit_notfound;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "Storage Class\n"));
|
|
|
|
// look for known chipsets
|
|
if(VendorID != BMList[i].nVendorId ||
|
|
DeviceID != BMList[i].nDeviceId) {
|
|
KdPrint2((PRINT_PREFIX "device not suitable\n"));
|
|
goto exit_notfound;
|
|
}
|
|
|
|
found = UniataCheckPCISubclass(BMList[i].Known, BMList[i].RaidFlags, SubClass);
|
|
if(!found) {
|
|
KdPrint2((PRINT_PREFIX "Subclass not supported\n"));
|
|
goto exit_notfound;
|
|
}
|
|
|
|
ConfigInfo->AlignmentMask = 0x00000003;
|
|
|
|
MasterDev = IsMasterDev(&pciData);
|
|
|
|
if(MasterDev) {
|
|
KdPrint2((PRINT_PREFIX "MasterDev (1)\n"));
|
|
deviceExtension->MasterDev = TRUE;
|
|
KdPrint2((PRINT_PREFIX "Check exclude\n"));
|
|
if(AtapiRegCheckDevValue(deviceExtension, channel, DEVNUM_NOT_SPECIFIED, L"Exclude", 0)) {
|
|
KdPrint2((PRINT_PREFIX "Device excluded\n"));
|
|
goto exit_notfound;
|
|
}
|
|
}
|
|
|
|
status = UniataChipDetect(HwDeviceExtension, &pciData, i, ConfigInfo, &simplexOnly);
|
|
switch(status) {
|
|
case STATUS_SUCCESS:
|
|
found = TRUE;
|
|
break;
|
|
case STATUS_NOT_FOUND:
|
|
found = FALSE;
|
|
break;
|
|
default:
|
|
KdPrint2((PRINT_PREFIX "FAILED => SP_RETURN_ERROR\n"));
|
|
goto exit_error;
|
|
}
|
|
KdPrint2((PRINT_PREFIX "ForceSimplex = %d\n", simplexOnly));
|
|
KdPrint2((PRINT_PREFIX "HwFlags = %x\n (0)", deviceExtension->HwFlags));
|
|
switch(dev_id) {
|
|
/* additional checks for some supported chipsets */
|
|
case 0xc6931080:
|
|
if (SubClass != PCI_DEV_SUBCLASS_IDE) {
|
|
KdPrint2((PRINT_PREFIX "0xc6931080, SubClass != PCI_DEV_SUBCLASS_IDE => found = FALSE\n"));
|
|
found = FALSE;
|
|
} else {
|
|
found = FALSE;
|
|
}
|
|
break;
|
|
|
|
/* unknown chipsets, try generic DMA if it seems possible */
|
|
default:
|
|
if (found)
|
|
break;
|
|
KdPrint2((PRINT_PREFIX "Default device\n"));
|
|
if(Ata_is_supported_dev(&pciData)) {
|
|
KdPrint2((PRINT_PREFIX "Ata_is_supported_dev\n"));
|
|
found = TRUE;
|
|
} else
|
|
if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
KdPrint2((PRINT_PREFIX "AHCI candidate\n"));
|
|
found = TRUE;
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "!Ata_is_supported_dev => found = FALSE\n"));
|
|
found = FALSE;
|
|
}
|
|
deviceExtension->UnknownDev = TRUE;
|
|
break;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "HwFlags = %x\n (1)", deviceExtension->HwFlags));
|
|
if(!found) {
|
|
KdPrint2((PRINT_PREFIX "!found => SP_RETURN_NOT_FOUND\n"));
|
|
goto exit_notfound;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "HwFlags = %x\n (2)", deviceExtension->HwFlags));
|
|
KdPrint2((PRINT_PREFIX "found suitable device\n"));
|
|
|
|
/***********************************************************/
|
|
/***********************************************************/
|
|
/***********************************************************/
|
|
|
|
deviceExtension->UseDpc = TRUE;
|
|
#ifndef UNIATA_CORE
|
|
if (g_Dump) {
|
|
deviceExtension->DriverMustPoll = TRUE;
|
|
deviceExtension->UseDpc = FALSE;
|
|
deviceExtension->simplexOnly = TRUE;
|
|
deviceExtension->HwFlags |= UNIATA_NO_DPC;
|
|
}
|
|
#endif //UNIATA_CORE
|
|
KdPrint2((PRINT_PREFIX "HwFlags = %x\n (3)", deviceExtension->HwFlags));
|
|
if(deviceExtension->HwFlags & UNIATA_NO_DPC) {
|
|
/* CMD 649, ROSB SWK33, ICH4 */
|
|
KdPrint2((PRINT_PREFIX "UniataFindBusMasterController: UNIATA_NO_DPC (0)\n"));
|
|
deviceExtension->UseDpc = FALSE;
|
|
}
|
|
|
|
if(MasterDev) {
|
|
if((WinVer_Id() <= WinVer_NT) && AltInit && FirstMasterOk) {
|
|
// this is the 2nd attempt to init this controller by OUR driver
|
|
KdPrint2((PRINT_PREFIX "Skip primary/secondary claiming checks\n"));
|
|
} else {
|
|
if((channel==0) && ConfigInfo->AtdiskPrimaryClaimed) {
|
|
KdPrint2((PRINT_PREFIX "Error: Primary channel already claimed by another driver\n"));
|
|
goto exit_notfound;
|
|
}
|
|
if((channel==1) && ConfigInfo->AtdiskSecondaryClaimed) {
|
|
KdPrint2((PRINT_PREFIX "Error: Secondary channel already claimed by another driver\n"));
|
|
goto exit_notfound;
|
|
}
|
|
}
|
|
}
|
|
if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
KdPrint2((PRINT_PREFIX " AHCI registers layout\n"));
|
|
} else
|
|
if(deviceExtension->AltRegMap) {
|
|
KdPrint2((PRINT_PREFIX " Non-standard registers layout\n"));
|
|
if(deviceExtension->HwFlags & UNIATA_SATA) {
|
|
KdPrint2((PRINT_PREFIX "UNIATA_SATA -> IsBusMaster == TRUE\n"));
|
|
if(!deviceExtension->BusMaster) {
|
|
deviceExtension->BusMaster = DMA_MODE_BM;
|
|
}
|
|
}
|
|
} else {
|
|
deviceExtension->BusMaster = DMA_MODE_NONE;
|
|
|
|
if(WinVer_WDM_Model && !deviceExtension->UnknownDev) {
|
|
UniataEnableIoPCI(ConfigInfo->SystemIoBusNumber, slotData.u.AsULONG, &pciData);
|
|
}
|
|
// validate Mem/Io ranges
|
|
//no_ranges = TRUE;
|
|
{
|
|
ULONG j;
|
|
for(j=0; j<PCI_TYPE0_ADDRESSES; j++) {
|
|
if(pciData.u.type0.BaseAddresses[j] & ~0x7) {
|
|
//no_ranges = FALSE;
|
|
//break;
|
|
KdPrint2((PRINT_PREFIX "Range %d = %#x\n", j, pciData.u.type0.BaseAddresses[j]));
|
|
}
|
|
}
|
|
}
|
|
|
|
if(IsBusMaster(&pciData)) {
|
|
|
|
KdPrint2((PRINT_PREFIX "IsBusMaster == TRUE\n"));
|
|
BaseIoAddressBM_0 = (PIDE_BUSMASTER_REGISTERS)
|
|
(AtapiGetIoRange(HwDeviceExtension, ConfigInfo, &pciData, SystemIoBusNumber,
|
|
4, bm_offset, MasterDev ? 0x08 : 0x10/*ATA_BMIOSIZE*/)/* - bm_offset*/); //range id
|
|
if(BaseIoAddressBM_0) {
|
|
UniataInitMapBM(deviceExtension,
|
|
BaseIoAddressBM_0,
|
|
(*ConfigInfo->AccessRanges)[4].RangeInMemory ? TRUE : FALSE);
|
|
deviceExtension->BusMaster = DMA_MODE_BM;
|
|
deviceExtension->BaseIoAddressBM_0.Addr = (ULONGIO_PTR)BaseIoAddressBM_0;
|
|
if((*ConfigInfo->AccessRanges)[4].RangeInMemory) {
|
|
deviceExtension->BaseIoAddressBM_0.MemIo = TRUE;
|
|
}
|
|
}
|
|
KdPrint2((PRINT_PREFIX " BusMasterAddress (base): %#x\n", BaseIoAddressBM_0));
|
|
}
|
|
|
|
if(!deviceExtension->BusMaster) {
|
|
KdPrint2((PRINT_PREFIX " !BusMasterAddress -> PIO4\n"));
|
|
deviceExtension->MaxTransferMode = ATA_PIO4;
|
|
}
|
|
|
|
if(deviceExtension->BusMaster && !MasterDev) {
|
|
KdPrint2((PRINT_PREFIX "IsBusMaster == TRUE && !MasterDev\n"));
|
|
statusByte = AtapiReadPort1(&(deviceExtension->chan[0]), IDX_BM_Status);
|
|
KdPrint2((PRINT_PREFIX " BM statusByte = %x\n", statusByte));
|
|
if(statusByte == IDE_STATUS_WRONG) {
|
|
KdPrint2((PRINT_PREFIX " invalid port ?\n"));
|
|
deviceExtension->BusMaster = DMA_MODE_NONE;
|
|
/*
|
|
if(BaseIoAddressBM_0) {
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
BaseIoAddressBM_0);
|
|
BaseIoAddressBM_0 = NULL;
|
|
}
|
|
*/
|
|
} else
|
|
if(statusByte & BM_STATUS_SIMPLEX_ONLY) {
|
|
KdPrint2((PRINT_PREFIX " BM_STATUS => simplexOnly\n"));
|
|
simplexOnly = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* the Cypress chip is a mess, it contains two ATA functions, but
|
|
* both channels are visible on the first one.
|
|
* simply ignore the second function for now, as the right
|
|
* solution (ignoring the second channel on the first function)
|
|
* doesn't work with the crappy ATA interrupt setup on the alpha.
|
|
*/
|
|
if (dev_id == 0xc6931080 && slotData.u.bits.FunctionNumber > 1) {
|
|
KdPrint2((PRINT_PREFIX "dev_id == 0xc6931080 && FunctionNumber > 1 => exit_findbm\n"));
|
|
goto exit_findbm;
|
|
}
|
|
|
|
/* do extra chipset specific setups */
|
|
AtapiReadChipConfig(HwDeviceExtension, i, CHAN_NOT_SPECIFIED);
|
|
AtapiChipInit(HwDeviceExtension, i, CHAN_NOT_SPECIFIED_CHECK_CABLE);
|
|
|
|
simplexOnly |= deviceExtension->simplexOnly;
|
|
deviceExtension->simplexOnly |= simplexOnly;
|
|
|
|
KdPrint2((PRINT_PREFIX "simplexOnly = %d (2)", simplexOnly));
|
|
|
|
//TODO: fix hang with UseDpc=TRUE in Simplex mode
|
|
//deviceExtension->UseDpc = TRUE;
|
|
if(simplexOnly) {
|
|
KdPrint2((PRINT_PREFIX "simplexOnly => UseDpc = FALSE\n"));
|
|
deviceExtension->UseDpc = FALSE;
|
|
}
|
|
|
|
if(simplexOnly && MasterDev) {
|
|
if(deviceExtension->NumberChannels < IDE_DEFAULT_MAX_CHAN) {
|
|
KdPrint2((PRINT_PREFIX "set NumberChannels = %d\n", IDE_DEFAULT_MAX_CHAN));
|
|
deviceExtension->NumberChannels = IDE_DEFAULT_MAX_CHAN;
|
|
if(BaseIoAddressBM_0) {
|
|
UniataInitMapBM(deviceExtension,
|
|
BaseIoAddressBM_0,
|
|
(*ConfigInfo->AccessRanges)[4].RangeInMemory ? TRUE : FALSE);
|
|
}
|
|
}
|
|
}
|
|
if((channel > 0) &&
|
|
(deviceExtension->NumberChannels > 1)) {
|
|
KdPrint2((PRINT_PREFIX "Error: channel > 0 && NumberChannels > 1\n"));
|
|
goto exit_findbm;
|
|
}
|
|
|
|
// Indicate number of buses.
|
|
ConfigInfo->NumberOfBuses = (UCHAR)(deviceExtension->NumberChannels);
|
|
if(!ConfigInfo->InitiatorBusId[0]) {
|
|
ConfigInfo->InitiatorBusId[0] = (CHAR)(IoGetConfigurationInformation()->ScsiPortCount);
|
|
KdPrint2((PRINT_PREFIX "set ConfigInfo->InitiatorBusId[0] = %#x\n", ConfigInfo->InitiatorBusId[0]));
|
|
}
|
|
// Indicate four devices can be attached to the adapter
|
|
ConfigInfo->MaximumNumberOfTargets = (UCHAR)(deviceExtension->NumberLuns);
|
|
|
|
if (MasterDev) {
|
|
KdPrint2((PRINT_PREFIX "MasterDev (2)\n"));
|
|
/*
|
|
if((WinVer_Id() > WinVer_NT) ||
|
|
(deviceExtension->NumberChannels > 1)) {
|
|
|
|
KdPrint2((PRINT_PREFIX "2 channels & 2 irq for 1 controller Win 2000+\n"));
|
|
|
|
if (ConfigInfo->AdapterInterfaceType == MicroChannel) {
|
|
ConfigInfo->InterruptMode2 =
|
|
ConfigInfo->InterruptMode = LevelSensitive;
|
|
} else {
|
|
ConfigInfo->InterruptMode2 =
|
|
ConfigInfo->InterruptMode = Latched;
|
|
}
|
|
ConfigInfo->BusInterruptLevel = 14;
|
|
ConfigInfo->BusInterruptLevel2 = 15;
|
|
} else*/
|
|
if(simplexOnly) {
|
|
|
|
KdPrint2((PRINT_PREFIX "2 channels & 2 irq for 1 controller\n"));
|
|
|
|
if (ConfigInfo->AdapterInterfaceType == MicroChannel) {
|
|
ConfigInfo->InterruptMode2 =
|
|
ConfigInfo->InterruptMode = LevelSensitive;
|
|
} else {
|
|
ConfigInfo->InterruptMode2 =
|
|
ConfigInfo->InterruptMode = Latched;
|
|
}
|
|
ConfigInfo->BusInterruptLevel = 14;
|
|
ConfigInfo->BusInterruptLevel2 = 15;
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "1 channels & 1 irq for 1 controller\n"));
|
|
if (ConfigInfo->AdapterInterfaceType == MicroChannel) {
|
|
ConfigInfo->InterruptMode = LevelSensitive;
|
|
} else {
|
|
ConfigInfo->InterruptMode = Latched;
|
|
}
|
|
ConfigInfo->BusInterruptLevel = (channel == 0 ? 14 : 15);
|
|
}
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "!MasterDev\n"));
|
|
ConfigInfo->SlotNumber = slotNumber;
|
|
ConfigInfo->SystemIoBusNumber = SystemIoBusNumber;
|
|
ConfigInfo->InterruptMode = LevelSensitive;
|
|
|
|
/* primary and secondary channels share the same interrupt */
|
|
if(!ConfigInfo->BusInterruptVector ||
|
|
(ConfigInfo->BusInterruptVector != pciData.u.type0.InterruptLine)) {
|
|
KdPrint2((PRINT_PREFIX "patch irq line = %#x\n", pciData.u.type0.InterruptLine));
|
|
ConfigInfo->BusInterruptVector = pciData.u.type0.InterruptLine; // set default value
|
|
if(!ConfigInfo->BusInterruptVector) {
|
|
KdPrint2((PRINT_PREFIX "patch irq line (2) = 10\n"));
|
|
ConfigInfo->BusInterruptVector = 10;
|
|
}
|
|
}
|
|
}
|
|
ConfigInfo->MultipleRequestPerLu = TRUE;
|
|
ConfigInfo->AutoRequestSense = TRUE;
|
|
ConfigInfo->TaggedQueuing = TRUE;
|
|
|
|
if((WinVer_Id() >= WinVer_NT) ||
|
|
(ConfigInfo->Length >= sizeof(_ConfigInfo->comm) + sizeof(_ConfigInfo->nt4))) {
|
|
KdPrint2((PRINT_PREFIX "update ConfigInfo->nt4\n"));
|
|
_ConfigInfo->nt4.DeviceExtensionSize = sizeof(HW_DEVICE_EXTENSION);
|
|
_ConfigInfo->nt4.SpecificLuExtensionSize = sizeof(HW_LU_EXTENSION);
|
|
//if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
_ConfigInfo->nt4.SrbExtensionSize = sizeof(ATA_REQ);
|
|
//} else {
|
|
// _ConfigInfo->nt4.SrbExtensionSize = FIELD_OFFSET(ATA_REQ, dma_tab) + sizeof(BM_DMA_ENTRY)*ATA_DMA_ENTRIES;
|
|
//}
|
|
KdPrint2((PRINT_PREFIX "using AtaReq sz %x\n", _ConfigInfo->nt4.SrbExtensionSize));
|
|
}
|
|
if((WinVer_Id() > WinVer_2k) ||
|
|
(ConfigInfo->Length >= sizeof(_ConfigInfo->comm) + sizeof(_ConfigInfo->nt4) + sizeof(_ConfigInfo->w2k))) {
|
|
KdPrint2((PRINT_PREFIX "update ConfigInfo->w2k: 64bit %d\n",
|
|
deviceExtension->Host64));
|
|
#ifdef USE_OWN_DMA
|
|
// We need not set Dma64BitAddresses since we perform address translation manually.
|
|
#else
|
|
_ConfigInfo->w2k.Dma64BitAddresses = deviceExtension->Host64;
|
|
#endif //USE_OWN_DMA
|
|
_ConfigInfo->w2k.ResetTargetSupported = TRUE;
|
|
_ConfigInfo->w2k.MaximumNumberOfLogicalUnits = (UCHAR)deviceExtension->NumberLuns;
|
|
}
|
|
|
|
// Save the Interrupe Mode for later use
|
|
deviceExtension->InterruptMode = ConfigInfo->InterruptMode;
|
|
deviceExtension->BusInterruptLevel = ConfigInfo->BusInterruptLevel;
|
|
deviceExtension->BusInterruptVector = ConfigInfo->BusInterruptVector;
|
|
deviceExtension->Channel = channel;
|
|
deviceExtension->DevIndex = i;
|
|
deviceExtension->OrigAdapterInterfaceType
|
|
= ConfigInfo->AdapterInterfaceType;
|
|
deviceExtension->AlignmentMask = ConfigInfo->AlignmentMask;
|
|
deviceExtension->AdapterInterfaceType = PCIBus;
|
|
|
|
KdPrint2((PRINT_PREFIX "chan[%d] InterruptMode: %d, Level %d, Level2 %d, Vector %d, Vector2 %d\n",
|
|
channel,
|
|
ConfigInfo->InterruptMode,
|
|
ConfigInfo->BusInterruptLevel,
|
|
ConfigInfo->BusInterruptLevel2,
|
|
ConfigInfo->BusInterruptVector,
|
|
ConfigInfo->BusInterruptVector2
|
|
));
|
|
|
|
found = FALSE;
|
|
|
|
if(deviceExtension->BusMaster) {
|
|
|
|
KdPrint2((PRINT_PREFIX "Reconstruct ConfigInfo\n"));
|
|
#ifdef USE_OWN_DMA
|
|
ConfigInfo->NeedPhysicalAddresses = FALSE;
|
|
#else
|
|
ConfigInfo->NeedPhysicalAddresses = TRUE;
|
|
#endif //USE_OWN_DMA
|
|
if(!MasterDev) {
|
|
//#ifdef USE_OWN_DMA
|
|
// KdPrint2((PRINT_PREFIX "!MasterDev, own DMA\n"));
|
|
//#else
|
|
KdPrint2((PRINT_PREFIX "set Dma32BitAddresses\n"));
|
|
ConfigInfo->Dma32BitAddresses = TRUE;
|
|
//#endif //USE_OWN_DMA
|
|
}
|
|
|
|
// thanks to Vitaliy Vorobyov aka deathsoft@yandex.ru for
|
|
// better solution:
|
|
|
|
if(AltInit) {
|
|
// I'm sorry, I have to do this
|
|
// when Win doesn't
|
|
|
|
if(ConfigInfo->AdapterInterfaceType == Isa /*&&
|
|
// InDriverEntry*/) {
|
|
KdPrint2((PRINT_PREFIX "AdapterInterfaceType Isa => PCIBus\n"));
|
|
ConfigInfo->AdapterInterfaceType = PCIBus;
|
|
}
|
|
if(ConfigInfo->AdapterInterfaceType == PCIBus /*&&
|
|
// InDriverEntry*/) {
|
|
KdPrint2((PRINT_PREFIX "AdapterInterfaceType PCIBus, update address\n"));
|
|
ConfigInfo->SlotNumber = slotNumber;
|
|
ConfigInfo->SystemIoBusNumber = SystemIoBusNumber;
|
|
}
|
|
}
|
|
|
|
#ifndef USE_OWN_DMA
|
|
ConfigInfo->Master = TRUE;
|
|
ConfigInfo->DmaWidth = Width16Bits;
|
|
#endif //USE_OWN_DMA
|
|
ConfigInfo->ScatterGather = TRUE;
|
|
}
|
|
ConfigInfo->MapBuffers = TRUE; // Need for PIO and OWN_DMA
|
|
ConfigInfo->CachesData = TRUE;
|
|
|
|
KdPrint2((PRINT_PREFIX "BMList[i].channel %#x, NumberChannels %#x, channel %#x\n",BMList[i].channel, deviceExtension->NumberChannels, channel));
|
|
|
|
for (; channel < (BMList[i].channel + deviceExtension->NumberChannels); channel++, c++) {
|
|
|
|
KdPrint2((PRINT_PREFIX "de %#x, Channel %#x\n",deviceExtension, channel));
|
|
//PrintNtConsole("de %#x, Channel %#x, nchan %#x\n",deviceExtension, channel, deviceExtension->NumberChannels);
|
|
chan = &deviceExtension->chan[c];
|
|
|
|
KdPrint2((PRINT_PREFIX "chan = %#x\n", chan));
|
|
//PrintNtConsole("chan = %#x, c=%#x\n", chan, c);
|
|
AtapiSetupLunPtrs(chan, deviceExtension, c);
|
|
|
|
/* do extra channel-specific setups */
|
|
AtapiReadChipConfig(HwDeviceExtension, i, channel);
|
|
//AtapiChipInit(HwDeviceExtension, i, channel);
|
|
if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
KdPrint2((PRINT_PREFIX " No more setup for AHCI channel\n"));
|
|
} else
|
|
if(deviceExtension->AltRegMap) {
|
|
KdPrint2((PRINT_PREFIX " Non-standard registers layout\n"));
|
|
} else {
|
|
// Check if the range specified is not used by another driver
|
|
if(MasterDev) {
|
|
KdPrint2((PRINT_PREFIX "set AccessRanges\n"));
|
|
(*ConfigInfo->AccessRanges)[channel * 2 + 0].RangeStart =
|
|
ScsiPortConvertUlongToPhysicalAddress(channel ? IO_WD2 : IO_WD1);
|
|
(*ConfigInfo->AccessRanges)[channel * 2 + 0].RangeLength = ATA_IOSIZE;
|
|
|
|
(*ConfigInfo->AccessRanges)[channel * 2 + 1].RangeStart =
|
|
ScsiPortConvertUlongToPhysicalAddress((channel ? IO_WD2 : IO_WD1) + ATA_ALTOFFSET);
|
|
(*ConfigInfo->AccessRanges)[channel * 2 + 1].RangeLength = ATA_ALTIOSIZE;
|
|
} else
|
|
if(AltInit &&
|
|
!(*ConfigInfo->AccessRanges)[channel * 2 + 0].RangeStart.QuadPart &&
|
|
!(*ConfigInfo->AccessRanges)[channel * 2 + 1].RangeStart.QuadPart) {
|
|
KdPrint2((PRINT_PREFIX "cheat ScsiPort, sync real PCI and ConfigInfo IO ranges\n"));
|
|
AtapiGetIoRange(HwDeviceExtension, ConfigInfo, &pciData, SystemIoBusNumber,
|
|
channel * 2 + 0, 0, ATA_IOSIZE);
|
|
AtapiGetIoRange(HwDeviceExtension, ConfigInfo, &pciData, SystemIoBusNumber,
|
|
channel * 2 + 1, 0, ATA_ALTIOSIZE);
|
|
}
|
|
|
|
IoBasePort1 = (*ConfigInfo->AccessRanges)[channel * 2 + 0].RangeStart;
|
|
IoBasePort2 = (*ConfigInfo->AccessRanges)[channel * 2 + 1].RangeStart;
|
|
|
|
if(!MasterDev) {
|
|
if(!IoBasePort1.QuadPart || !IoBasePort2.QuadPart) {
|
|
KdPrint2((PRINT_PREFIX "ScsiPortValidateRange failed (1)\n"));
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if(!ScsiPortValidateRange(HwDeviceExtension,
|
|
PCIBus /*ConfigInfo->AdapterInterfaceType*/,
|
|
SystemIoBusNumber /*ConfigInfo->SystemIoBusNumber*/,
|
|
IoBasePort1,
|
|
ATA_IOSIZE,
|
|
TRUE) ) {
|
|
KdPrint2((PRINT_PREFIX "ScsiPortValidateRange failed (1)\n"));
|
|
continue;
|
|
}
|
|
|
|
if(!ScsiPortValidateRange(HwDeviceExtension,
|
|
PCIBus /*ConfigInfo->AdapterInterfaceType*/,
|
|
SystemIoBusNumber /*ConfigInfo->SystemIoBusNumber*/,
|
|
IoBasePort2,
|
|
ATA_ALTIOSIZE,
|
|
TRUE) ) {
|
|
KdPrint2((PRINT_PREFIX "ScsiPortValidateRange failed (2)\n"));
|
|
continue;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "Getting IO ranges\n"));
|
|
|
|
// Ok, translate adresses to io-space
|
|
if(ScsiPortConvertPhysicalAddressToUlong(IoBasePort2)) {
|
|
if(!(MasterDev /* || USE_16_BIT */)) {
|
|
KdPrint2((PRINT_PREFIX "!MasterDev mode\n"));
|
|
IoBasePort2 = ScsiPortConvertUlongToPhysicalAddress(
|
|
ScsiPortConvertPhysicalAddressToUlong(IoBasePort2) + 2);
|
|
}
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "use relative IoBasePort2\n"));
|
|
IoBasePort2 = ScsiPortConvertUlongToPhysicalAddress(
|
|
ScsiPortConvertPhysicalAddressToUlong(IoBasePort1) + ATA_PCCARD_ALTOFFSET);
|
|
}
|
|
|
|
// Get the system physical address for this IO range.
|
|
ioSpace = (PUCHAR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
MasterDev ? ConfigInfo->AdapterInterfaceType : PCIBus /*ConfigInfo->AdapterInterfaceType*/,
|
|
MasterDev ? ConfigInfo->SystemIoBusNumber : SystemIoBusNumber /*ConfigInfo->SystemIoBusNumber*/,
|
|
IoBasePort1,
|
|
ATA_IOSIZE,
|
|
TRUE);
|
|
KdPrint2((PRINT_PREFIX "IO range 1 %#x\n",ioSpace));
|
|
|
|
// Check if ioSpace accessible.
|
|
if (!ioSpace) {
|
|
KdPrint2((PRINT_PREFIX "!ioSpace\n"));
|
|
continue;
|
|
}
|
|
/*
|
|
if(deviceExtension->BusMaster) {
|
|
KdPrint2((PRINT_PREFIX "set BusMaster io-range in DO\n"));
|
|
// bm_offset already includes (channel ? ATA_BM_OFFSET1 : 0)
|
|
deviceExtension->BaseIoAddressBM[c] = (PIDE_BUSMASTER_REGISTERS)
|
|
((ULONG)(deviceExtension->BaseIoAddressBM_0) + bm_offset + (c ? ATA_BM_OFFSET1 : 0));
|
|
}
|
|
*/
|
|
//deviceExtension->BaseIoAddress1[c] = (PIDE_REGISTERS_1)(ioSpace);
|
|
BaseIoAddress1[c] = (PIDE_REGISTERS_1)(ioSpace);
|
|
|
|
// Get the system physical address for the second IO range.
|
|
ioSpace = (PUCHAR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
MasterDev ? ConfigInfo->AdapterInterfaceType : PCIBus /*ConfigInfo->AdapterInterfaceType*/,
|
|
MasterDev ? ConfigInfo->SystemIoBusNumber : SystemIoBusNumber /*ConfigInfo->SystemIoBusNumber*/,
|
|
IoBasePort2,
|
|
ATA_ALTIOSIZE,
|
|
TRUE);
|
|
KdPrint2((PRINT_PREFIX "IO range 2 %#x\n",ioSpace));
|
|
|
|
BaseIoAddress2[c] = (PIDE_REGISTERS_2)(ioSpace);
|
|
if(!ioSpace) {
|
|
// Release all allocated resources
|
|
KdPrint2((PRINT_PREFIX "!deviceExtension->BaseIoAddress2\n"));
|
|
//ioSpace = (PUCHAR)BaseIoAddress1[c];
|
|
// goto free_iospace_1;
|
|
found = FALSE;
|
|
goto exit_findbm;
|
|
}
|
|
UniataInitMapBase(chan, BaseIoAddress1[c], BaseIoAddress2[c]);
|
|
}
|
|
//ioSpace = (PUCHAR)(deviceExtension->BaseIoAddress1[c]);
|
|
|
|
DbgDumpRegTranslation(chan, IDX_IO1);
|
|
DbgDumpRegTranslation(chan, IDX_IO2);
|
|
DbgDumpRegTranslation(chan, IDX_BM_IO);
|
|
DbgDumpRegTranslation(chan, IDX_SATA_IO);
|
|
|
|
if(!(deviceExtension->HwFlags & UNIATA_AHCI)) {
|
|
#ifdef _DEBUG
|
|
UniataDumpATARegs(chan);
|
|
#endif
|
|
|
|
#ifndef UNIATA_CORE
|
|
#ifdef UNIATA_INIT_ON_PROBE
|
|
// if(deviceExtension->HwFlags & UNIATA_SATA) {
|
|
//#endif //UNIATA_INIT_ON_PROBE
|
|
KdPrint2((PRINT_PREFIX "Check drive 0\n"));
|
|
// Check master.
|
|
SelectDrive(chan, 0);
|
|
AtapiStallExecution(10);
|
|
GetBaseStatus(chan, statusByte);
|
|
skip_find_dev = FALSE;
|
|
if(!(deviceExtension->HwFlags & UNIATA_NO_SLAVE) && (deviceExtension->NumberLuns > 1)) {
|
|
if ((statusByte & 0xf8) == 0xf8 ||
|
|
(statusByte == 0xa5)) {
|
|
// Check slave.
|
|
KdPrint2((PRINT_PREFIX "Check drive 1\n"));
|
|
SelectDrive(chan, 1);
|
|
AtapiStallExecution(1);
|
|
GetBaseStatus(chan, statusByte);
|
|
if ((statusByte & 0xf8) == 0xf8 ||
|
|
(statusByte == 0xa5)) {
|
|
// No controller at this base address.
|
|
KdPrint2((PRINT_PREFIX "Empty channel\n"));
|
|
skip_find_dev = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Search for devices on this controller.
|
|
if (!skip_find_dev &&
|
|
FindDevices(HwDeviceExtension,
|
|
0,
|
|
c)) {
|
|
KdPrint2((PRINT_PREFIX "Found some devices\n"));
|
|
found = TRUE;
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "no devices\n"));
|
|
/* KeBugCheckEx(0xc000000e,
|
|
ScsiPortConvertPhysicalAddressToUlong(IoBasePort1),
|
|
ScsiPortConvertPhysicalAddressToUlong(IoBasePort2),
|
|
(ULONG)(deviceExtension->BaseIoAddressBM[c]), skip_find_dev);*/
|
|
}
|
|
//#ifdef UNIATA_INIT_ON_PROBE
|
|
// }
|
|
#else //UNIATA_INIT_ON_PROBE
|
|
KdPrint2((PRINT_PREFIX "clean IDE intr 0\n"));
|
|
|
|
SelectDrive(chan, 0);
|
|
AtapiStallExecution(10);
|
|
GetBaseStatus(chan, statusByte);
|
|
|
|
if(!(deviceExtension->HwFlags & UNIATA_NO_SLAVE) && (deviceExtension->NumberLuns > 1)) {
|
|
KdPrint2((PRINT_PREFIX "clean IDE intr 1\n"));
|
|
|
|
SelectDrive(chan, 1);
|
|
AtapiStallExecution(1);
|
|
GetBaseStatus(chan, statusByte);
|
|
|
|
SelectDrive(chan, 0);
|
|
}
|
|
|
|
statusByte = GetDmaStatus(deviceExtension, c);
|
|
KdPrint2((PRINT_PREFIX " DMA status %#x\n", statusByte));
|
|
if(statusByte & BM_STATUS_INTR) {
|
|
// bullshit, we have DMA interrupt, but had never initiate DMA operation
|
|
KdPrint2((PRINT_PREFIX " clear unexpected DMA intr\n"));
|
|
AtapiDmaDone(deviceExtension, 0, c, NULL);
|
|
GetBaseStatus(chan, statusByte);
|
|
}
|
|
|
|
#endif //UNIATA_INIT_ON_PROBE
|
|
}
|
|
found = TRUE;
|
|
|
|
chan->PrimaryAddress = FALSE;
|
|
// Claim primary or secondary ATA IO range.
|
|
if (MasterDev) {
|
|
KdPrint2((PRINT_PREFIX "claim Compatible controller\n"));
|
|
if (channel == 0) {
|
|
KdPrint2((PRINT_PREFIX "claim Primary\n"));
|
|
AtdiskPrimaryClaimed =
|
|
ConfigInfo->AtdiskPrimaryClaimed = TRUE;
|
|
chan->PrimaryAddress = TRUE;
|
|
|
|
FirstMasterOk = TRUE;
|
|
|
|
} else
|
|
if (channel == 1) {
|
|
KdPrint2((PRINT_PREFIX "claim Secondary\n"));
|
|
AtdiskSecondaryClaimed =
|
|
ConfigInfo->AtdiskSecondaryClaimed = TRUE;
|
|
|
|
FirstMasterOk = TRUE;
|
|
}
|
|
} else {
|
|
if(chan->RegTranslation[IDX_IO1].Addr == IO_WD1 &&
|
|
!chan->RegTranslation[IDX_IO1].MemIo) {
|
|
KdPrint2((PRINT_PREFIX "claim Primary (PCI over ISA range)\n"));
|
|
AtdiskPrimaryClaimed =
|
|
ConfigInfo->AtdiskPrimaryClaimed = TRUE;
|
|
}
|
|
if(chan->RegTranslation[IDX_IO1].Addr == IO_WD2 &&
|
|
!chan->RegTranslation[IDX_IO1].MemIo) {
|
|
KdPrint2((PRINT_PREFIX "claim Secondary (PCI over ISA range)\n"));
|
|
AtdiskSecondaryClaimed =
|
|
ConfigInfo->AtdiskSecondaryClaimed = TRUE;
|
|
}
|
|
}
|
|
|
|
AtapiDmaAlloc(HwDeviceExtension, ConfigInfo, c);
|
|
#else //UNIATA_CORE
|
|
}
|
|
found = TRUE;
|
|
#endif //UNIATA_CORE
|
|
} // end for(channel)
|
|
|
|
exit_findbm:
|
|
|
|
#ifndef UNIATA_CORE
|
|
if(!found) {
|
|
KdPrint2((PRINT_PREFIX "exit: !found\n"));
|
|
if(BaseIoAddress1[0])
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
BaseIoAddress1[0]);
|
|
if(BaseIoAddress2[0])
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
BaseIoAddress2[0]);
|
|
|
|
if(BaseIoAddress1[1])
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
BaseIoAddress1[1]);
|
|
if(BaseIoAddress2[1])
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
BaseIoAddress2[1]);
|
|
|
|
if(BaseIoAddressBM_0)
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
BaseIoAddressBM_0);
|
|
|
|
if(deviceExtension->BaseIoAHCI_0.Addr) {
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
deviceExtension->BaseIoAHCI_0.pAddr);
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "return SP_RETURN_NOT_FOUND\n"));
|
|
goto exit_notfound;
|
|
} else {
|
|
|
|
KdPrint2((PRINT_PREFIX "exit: init spinlock\n"));
|
|
//KeInitializeSpinLock(&(deviceExtension->DpcSpinLock));
|
|
deviceExtension->ActiveDpcChan =
|
|
deviceExtension->FirstDpcChan = CHAN_NOT_SPECIFIED;
|
|
|
|
BMList[i].Isr2Enable = FALSE;
|
|
|
|
KdPrint2((PRINT_PREFIX "MasterDev=%#x, NumberChannels=%#x, Isr2DevObj=%#x\n",
|
|
MasterDev, deviceExtension->NumberChannels, BMList[i].Isr2DevObj));
|
|
|
|
// ConnectIntr2 should be moved to HwInitialize
|
|
status = UniataConnectIntr2(HwDeviceExtension);
|
|
|
|
KdPrint2((PRINT_PREFIX "MasterDev=%#x, NumberChannels=%#x, Isr2DevObj=%#x\n",
|
|
MasterDev, deviceExtension->NumberChannels, BMList[i].Isr2DevObj));
|
|
|
|
if(/*WinVer_WDM_Model &&*/ MasterDev) {
|
|
KdPrint2((PRINT_PREFIX "do not tell system, that we know about PCI IO ranges\n"));
|
|
/* if(BaseIoAddressBM_0) {
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
BaseIoAddressBM_0);
|
|
}*/
|
|
(*ConfigInfo->AccessRanges)[4].RangeStart = ScsiPortConvertUlongToPhysicalAddress(0);
|
|
(*ConfigInfo->AccessRanges)[4].RangeLength = 0;
|
|
(*ConfigInfo->AccessRanges)[5].RangeStart = ScsiPortConvertUlongToPhysicalAddress(0);
|
|
(*ConfigInfo->AccessRanges)[5].RangeLength = 0;
|
|
}
|
|
|
|
if(!NT_SUCCESS(status)) {
|
|
KdPrint2((PRINT_PREFIX "failed\n"));
|
|
found = FALSE;
|
|
goto exit_findbm;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "final chan[%d] InterruptMode: %d, Level %d, Level2 %d, Vector %d, Vector2 %d\n",
|
|
channel,
|
|
ConfigInfo->InterruptMode,
|
|
ConfigInfo->BusInterruptLevel,
|
|
ConfigInfo->BusInterruptLevel2,
|
|
ConfigInfo->BusInterruptVector,
|
|
ConfigInfo->BusInterruptVector2
|
|
));
|
|
|
|
|
|
}
|
|
#endif //UNIATA_CORE
|
|
|
|
KdPrint2((PRINT_PREFIX "return SP_RETURN_FOUND\n"));
|
|
//PrintNtConsole("return SP_RETURN_FOUND, de %#x, c0.lun0 %#x\n", deviceExtension, deviceExtension->chan[0].lun[0]);
|
|
|
|
if(MasterDev) {
|
|
KdPrint2((PRINT_PREFIX "Attempt %d of MasterDev ok\n", AltInit));
|
|
FirstMasterOk = TRUE;
|
|
}
|
|
|
|
ConfigInfo->NumberOfBuses++; // add virtual channel for communication port
|
|
return SP_RETURN_FOUND;
|
|
|
|
exit_error:
|
|
UniataFreeLunExt(deviceExtension);
|
|
return SP_RETURN_ERROR;
|
|
|
|
exit_notfound:
|
|
UniataFreeLunExt(deviceExtension);
|
|
return SP_RETURN_NOT_FOUND;
|
|
|
|
} // end UniataFindBusMasterController()
|
|
|
|
#ifndef UNIATA_CORE
|
|
|
|
/*
|
|
This is for claiming PCI Busmaster in compatible mode under WDM OSes
|
|
*/
|
|
NTSTATUS
|
|
NTAPI
|
|
UniataClaimLegacyPCIIDE(
|
|
ULONG i
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
PCM_RESOURCE_LIST resourceList = NULL;
|
|
UNICODE_STRING devname;
|
|
|
|
KdPrint2((PRINT_PREFIX "UniataClaimLegacyPCIIDE:\n"));
|
|
|
|
if(BMList[i].PciIdeDevObj) {
|
|
KdPrint2((PRINT_PREFIX "Already initialized\n"));
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
RtlInitUnicodeString(&devname, L"\\Device\\uniata_PCIIDE");
|
|
status = IoCreateDevice(SavedDriverObject, sizeof(PCIIDE_DEVICE_EXTENSION),
|
|
/*NULL*/ &devname, FILE_DEVICE_UNKNOWN,
|
|
0, FALSE, &(BMList[i].PciIdeDevObj));
|
|
|
|
if(!NT_SUCCESS(status)) {
|
|
KdPrint2((PRINT_PREFIX "IoCreateDevice failed %#x\n", status));
|
|
return status;
|
|
}
|
|
|
|
resourceList = (PCM_RESOURCE_LIST) ExAllocatePool(PagedPool,
|
|
sizeof(CM_RESOURCE_LIST));
|
|
|
|
if (!resourceList) {
|
|
KdPrint2((PRINT_PREFIX "!resourceList\n"));
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
del_do:
|
|
IoDeleteDevice(BMList[i].PciIdeDevObj);
|
|
BMList[i].PciIdeDevObj = NULL;
|
|
return status;
|
|
}
|
|
|
|
RtlZeroMemory(
|
|
resourceList,
|
|
sizeof(CM_RESOURCE_LIST));
|
|
|
|
// IoReportDetectedDevice() should be used for WDM OSes
|
|
|
|
// TODO: check if resourceList is actually used inside HalAssignSlotResources()
|
|
// Note: with empty resourceList call to HalAssignSlotResources() fails on some HW
|
|
// e.g. Intel ICH4, but works with non-empty.
|
|
|
|
resourceList->Count = 1;
|
|
resourceList->List[0].InterfaceType = PCIBus;
|
|
resourceList->List[0].BusNumber = BMList[i].busNumber;
|
|
// we do not report IO ranges since they are used/claimed by ISA part(s)
|
|
resourceList->List[0].PartialResourceList.Count = 0;
|
|
|
|
RtlInitUnicodeString(&devname, L"PCIIDE");
|
|
status = HalAssignSlotResources(&SavedRegPath,
|
|
&devname,
|
|
SavedDriverObject,
|
|
BMList[i].PciIdeDevObj,
|
|
PCIBus,
|
|
BMList[i].busNumber,
|
|
BMList[i].slotNumber,
|
|
&resourceList);
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
KdPrint2((PRINT_PREFIX "HalAssignSlotResources failed %#x\n", status));
|
|
// this is always deallocated inside HalAssignSlotResources() implementation
|
|
//ExFreePool(resourceList);
|
|
goto del_do;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "ok %#x\n", status));
|
|
BMList[i].ChanInitOk |= 0x80;
|
|
|
|
return status;
|
|
} // end UniataClaimLegacyPCIIDE()
|
|
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function is called to initialize 2nd device object for
|
|
multichannel controllers.
|
|
|
|
Arguments:
|
|
|
|
HwDeviceExtension - HBA miniport driver's adapter data storage
|
|
|
|
Return Value:
|
|
|
|
ULONG
|
|
|
|
--*/
|
|
NTSTATUS
|
|
NTAPI
|
|
UniataConnectIntr2(
|
|
IN PVOID HwDeviceExtension
|
|
)
|
|
{
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
ULONG i = deviceExtension->DevIndex;
|
|
NTSTATUS status;
|
|
PISR2_DEVICE_EXTENSION Isr2DevExt;
|
|
WCHAR devname_str[33];
|
|
UNICODE_STRING devname;
|
|
|
|
KdPrint2((PRINT_PREFIX "Init ISR:\n"));
|
|
|
|
/*
|
|
We MUST register 2nd ISR for multichannel controllers even for UP systems.
|
|
This is needed for cases when
|
|
multichannel controller generate interrupt while we are still in its ISR for
|
|
other channle's interrupt. New interrupt must be detected and queued for
|
|
further processing. If we do not do this, system will not route this
|
|
interrupt to main ISR (since it is busy) and we shall get to infinite loop
|
|
looking for interrupt handler.
|
|
*/
|
|
|
|
if(!deviceExtension->MasterDev && (deviceExtension->NumberChannels > 1) && // do not touch MasterDev
|
|
!deviceExtension->simplexOnly && /* // this is unnecessary on simplex controllers
|
|
!BMList[i].Isr2DevObj*/ // handle re-init under w2k+
|
|
/*!ForceSimplex*/
|
|
/*(CPU_num > 1) && // unnecessary for UP systems*/
|
|
TRUE) {
|
|
// Ok, continue...
|
|
KdPrint2((PRINT_PREFIX "Multichannel native mode, go...\n"));
|
|
#ifndef UNIATA_USE_XXableInterrupts
|
|
// If we raise IRQL to TIMER value, other interrupt cannot occure on the same CPU
|
|
/* if(KeNumberProcessors < 2) {
|
|
KdPrint2((PRINT_PREFIX "Unnecessary (?), UP machine\n"));
|
|
//return STATUS_SUCCESS;
|
|
}*/
|
|
#endif //UNIATA_USE_XXableInterrupts
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "Unnecessary\n"));
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
if(BMList[i].Isr2DevObj) {
|
|
KdPrint2((PRINT_PREFIX "Already initialized [%d] %#x\n", i, BMList[i].Isr2DevObj));
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "Create DO\n"));
|
|
|
|
devname.Length =
|
|
_snwprintf(devname_str, sizeof(devname_str)/sizeof(WCHAR)-1,
|
|
L"\\Device\\uniata%d_2ch", i);
|
|
devname_str[devname.Length] = 0;
|
|
devname.Length *= sizeof(WCHAR);
|
|
devname.MaximumLength = devname.Length;
|
|
devname.Buffer = devname_str;
|
|
|
|
KdPrint2((PRINT_PREFIX "DO name: len(%d, %d), %S\n", devname.Length, devname.MaximumLength, devname.Buffer));
|
|
|
|
status = IoCreateDevice(SavedDriverObject, sizeof(ISR2_DEVICE_EXTENSION),
|
|
/*NULL*/ &devname, FILE_DEVICE_UNKNOWN,
|
|
0, FALSE, &(BMList[i].Isr2DevObj));
|
|
|
|
if(!NT_SUCCESS(status)) {
|
|
KdPrint2((PRINT_PREFIX "IoCreateDevice failed %#x\n", status));
|
|
return status;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "HalGetInterruptVector\n"));
|
|
KdPrint2((PRINT_PREFIX " OrigAdapterInterfaceType=%d\n", deviceExtension->OrigAdapterInterfaceType));
|
|
KdPrint2((PRINT_PREFIX " SystemIoBusNumber=%d\n", deviceExtension->SystemIoBusNumber));
|
|
KdPrint2((PRINT_PREFIX " BusInterruptLevel=%d\n", deviceExtension->BusInterruptLevel));
|
|
KdPrint2((PRINT_PREFIX " BusInterruptVector=%d\n", deviceExtension->BusInterruptVector));
|
|
BMList[i].Isr2Vector = HalGetInterruptVector(
|
|
deviceExtension->OrigAdapterInterfaceType,
|
|
deviceExtension->SystemIoBusNumber,
|
|
deviceExtension->BusInterruptLevel,
|
|
deviceExtension->BusInterruptVector,
|
|
&(BMList[i].Isr2Irql),
|
|
&(BMList[i].Isr2Affinity));
|
|
|
|
Isr2DevExt = (PISR2_DEVICE_EXTENSION)(BMList[i].Isr2DevObj->DeviceExtension);
|
|
Isr2DevExt->HwDeviceExtension = deviceExtension;
|
|
Isr2DevExt->DevIndex = i;
|
|
|
|
KdPrint2((PRINT_PREFIX "isr2_de %#x\n", Isr2DevExt));
|
|
KdPrint2((PRINT_PREFIX "isr2_vector %#x\n", BMList[i].Isr2Vector));
|
|
KdPrint2((PRINT_PREFIX "isr2_irql %#x\n", BMList[i].Isr2Irql));
|
|
KdPrint2((PRINT_PREFIX "isr2_affinity %#x\n", BMList[i].Isr2Affinity));
|
|
|
|
// deviceExtension->QueueNewIrql = BMList[i].Isr2Irql;
|
|
|
|
KdPrint2((PRINT_PREFIX "IoConnectInterrupt\n"));
|
|
status = IoConnectInterrupt(
|
|
&(BMList[i].Isr2InterruptObject),
|
|
AtapiInterrupt2,
|
|
Isr2DevExt,
|
|
NULL,
|
|
BMList[i].Isr2Vector,
|
|
BMList[i].Isr2Irql,
|
|
BMList[i].Isr2Irql,
|
|
(KINTERRUPT_MODE)(deviceExtension->InterruptMode),
|
|
TRUE,
|
|
BMList[i].Isr2Affinity,
|
|
FALSE);
|
|
|
|
if(!NT_SUCCESS(status)) {
|
|
KdPrint2((PRINT_PREFIX "IoConnectInterrupt failed\n"));
|
|
IoDeleteDevice(BMList[i].Isr2DevObj);
|
|
BMList[i].Isr2DevObj = NULL;
|
|
BMList[i].Isr2InterruptObject = NULL;
|
|
return status;
|
|
}
|
|
|
|
//deviceExtension->Isr2DevObj = BMList[i].Isr2DevObj;
|
|
|
|
return status;
|
|
} // end UniataConnectIntr2()
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
UniataDisconnectIntr2(
|
|
IN PVOID HwDeviceExtension
|
|
)
|
|
{
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
ULONG i = deviceExtension->DevIndex;
|
|
// NTSTATUS status;
|
|
|
|
KdPrint2((PRINT_PREFIX "Deinit ISR:\n"));
|
|
|
|
if(!BMList[i].Isr2DevObj) {
|
|
KdPrint2((PRINT_PREFIX "Already uninitialized %#x\n"));
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
IoDisconnectInterrupt(BMList[i].Isr2InterruptObject);
|
|
|
|
BMList[i].Isr2InterruptObject = NULL;
|
|
|
|
IoDeleteDevice(BMList[i].Isr2DevObj);
|
|
|
|
BMList[i].Isr2DevObj = NULL;
|
|
//deviceExtension->Isr2DevObj = NULL;
|
|
|
|
return STATUS_SUCCESS;
|
|
} // end UniataDisconnectIntr2()
|
|
|
|
#endif //UNIATA_CORE
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
AtapiCheckIOInterference(
|
|
IN PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
ULONG portBase) {
|
|
// check if Primary/Secondary Master IDE claimed
|
|
if((portBase == IO_WD1) &&
|
|
(ConfigInfo->AtdiskPrimaryClaimed || AtdiskPrimaryClaimed)) {
|
|
KdPrint2((PRINT_PREFIX "AtapiCheckIOInterference: AtdiskPrimaryClaimed\n"));
|
|
return TRUE;
|
|
} else
|
|
if((portBase == IO_WD2) &&
|
|
(ConfigInfo->AtdiskSecondaryClaimed || AtdiskSecondaryClaimed)) {
|
|
KdPrint2((PRINT_PREFIX "AtapiCheckIOInterference: AtdiskSecondaryClaimed\n"));
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
} // end AtapiCheckIOInterference()
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function is called by the OS-specific port driver after
|
|
the necessary storage has been allocated, to gather information
|
|
about the adapter's configuration.
|
|
|
|
Arguments:
|
|
|
|
HwDeviceExtension - HBA miniport driver's adapter data storage
|
|
Context - Address of adapter count
|
|
ArgumentString - Used to determine whether driver is client of ntldr or crash dump utility.
|
|
ConfigInfo - Configuration information structure describing HBA
|
|
Again - Indicates search for adapters to continue
|
|
|
|
Return Value:
|
|
|
|
ULONG
|
|
|
|
--*/
|
|
ULONG
|
|
NTAPI
|
|
AtapiFindIsaController(
|
|
IN PVOID HwDeviceExtension,
|
|
IN PVOID Context,
|
|
IN PVOID BusInformation,
|
|
IN PCHAR ArgumentString,
|
|
IN OUT PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
OUT PBOOLEAN Again
|
|
)
|
|
{
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
PHW_CHANNEL chan;
|
|
PULONG adapterCount = (PULONG)Context;
|
|
PUCHAR ioSpace = NULL;
|
|
ULONG i;
|
|
ULONG irq=0;
|
|
ULONG portBase=0;
|
|
ULONG retryCount;
|
|
// BOOLEAN atapiOnly;
|
|
UCHAR statusByte, statusByte2;
|
|
BOOLEAN preConfig = FALSE;
|
|
//
|
|
PIDE_REGISTERS_1 BaseIoAddress1;
|
|
PIDE_REGISTERS_2 BaseIoAddress2 = NULL;
|
|
|
|
// The following table specifies the ports to be checked when searching for
|
|
// an IDE controller. A zero entry terminates the search.
|
|
static CONST ULONG AdapterAddresses[5] = {IO_WD1, IO_WD2, IO_WD1-8, IO_WD2-8, 0};
|
|
// CONST UCHAR Channels[5] = {0, 1, 0, 1, 0};
|
|
|
|
// The following table specifies interrupt levels corresponding to the
|
|
// port addresses in the previous table.
|
|
static CONST ULONG InterruptLevels[5] = {14, 15, 11, 10, 0};
|
|
|
|
KdPrint2((PRINT_PREFIX "AtapiFindIsaController (ISA):\n"));
|
|
|
|
if (!deviceExtension) {
|
|
return SP_RETURN_ERROR;
|
|
}
|
|
RtlZeroMemory(deviceExtension, sizeof(HW_DEVICE_EXTENSION));
|
|
|
|
KdPrint2((PRINT_PREFIX " assume max PIO4\n"));
|
|
deviceExtension->MaxTransferMode = ATA_PIO4;
|
|
deviceExtension->NumberChannels = 1;
|
|
deviceExtension->NumberLuns = IDE_MAX_LUN_PER_CHAN; // default
|
|
|
|
if(!UniataAllocateLunExt(deviceExtension, UNIATA_ALLOCATE_NEW_LUNS)) {
|
|
goto exit_error;
|
|
}
|
|
|
|
chan = &(deviceExtension->chan[0]);
|
|
AtapiSetupLunPtrs(chan, deviceExtension, 0);
|
|
|
|
deviceExtension->AdapterInterfaceType =
|
|
deviceExtension->OrigAdapterInterfaceType
|
|
= ConfigInfo->AdapterInterfaceType;
|
|
|
|
#ifndef UNIATA_CORE
|
|
|
|
/* do extra chipset specific setups */
|
|
AtapiReadChipConfig(HwDeviceExtension, DEVNUM_NOT_SPECIFIED, CHAN_NOT_SPECIFIED);
|
|
AtapiChipInit(HwDeviceExtension, DEVNUM_NOT_SPECIFIED, CHAN_NOT_SPECIFIED);
|
|
|
|
// Check to see if this is a special configuration environment.
|
|
portBase = irq = 0;
|
|
if (ArgumentString) {
|
|
|
|
irq = AtapiParseArgumentString(ArgumentString, "Interrupt");
|
|
if (irq ) {
|
|
|
|
// Both parameters must be present to proceed
|
|
portBase = AtapiParseArgumentString(ArgumentString, "BaseAddress");
|
|
if (!portBase) {
|
|
|
|
// Try a default search for the part.
|
|
irq = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif //UNIATA_CORE
|
|
/*
|
|
for(i=0; i<2; i++) {
|
|
if((*ConfigInfo->AccessRanges)[i].RangeStart) {
|
|
KdPrint2((PRINT_PREFIX " IoRange[%d], start %#x, len %#x, mem %#x\n",
|
|
i,
|
|
ScsiPortConvertPhysicalAddressToUlong((*ConfigInfo->AccessRanges)[i].RangeStart),
|
|
(*ConfigInfo->AccessRanges)[i].RangeLength,
|
|
(*ConfigInfo->AccessRanges)[i].RangeInMemory
|
|
));
|
|
}
|
|
}
|
|
*/
|
|
// if((*ConfigInfo->AccessRanges)[0].RangeStart) {
|
|
portBase = ScsiPortConvertPhysicalAddressToUlong((*ConfigInfo->AccessRanges)[0].RangeStart);
|
|
// }
|
|
if(portBase) {
|
|
if(!AtapiCheckIOInterference(ConfigInfo, portBase)) {
|
|
ioSpace = (PUCHAR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
ConfigInfo->AdapterInterfaceType,
|
|
ConfigInfo->SystemIoBusNumber,
|
|
(*ConfigInfo->AccessRanges)[0].RangeStart,
|
|
(*ConfigInfo->AccessRanges)[0].RangeLength,
|
|
(BOOLEAN) !((*ConfigInfo->AccessRanges)[0].RangeInMemory));
|
|
} else {
|
|
// do not touch resources, just fail later inside loop on next call to
|
|
// AtapiCheckIOInterference()
|
|
}
|
|
*Again = FALSE;
|
|
// Since we have pre-configured information we only need to go through this loop once
|
|
preConfig = TRUE;
|
|
KdPrint2((PRINT_PREFIX " preconfig, portBase=%x, len=%x\n", portBase, (*ConfigInfo->AccessRanges)[0].RangeLength));
|
|
}
|
|
|
|
// Scan through the adapter address looking for adapters.
|
|
#ifndef UNIATA_CORE
|
|
while (AdapterAddresses[*adapterCount] != 0) {
|
|
#else
|
|
do {
|
|
#endif //UNIATA_CORE
|
|
|
|
retryCount = 4;
|
|
deviceExtension->DevIndex = (*adapterCount); // this is used inside AtapiRegCheckDevValue()
|
|
KdPrint2((PRINT_PREFIX "AtapiFindIsaController: adapterCount=%d\n", *adapterCount));
|
|
|
|
for (i = 0; i < deviceExtension->NumberLuns; i++) {
|
|
// Zero device fields to ensure that if earlier devices were found,
|
|
// but not claimed, the fields are cleared.
|
|
deviceExtension->lun[i].DeviceFlags &= ~(DFLAGS_ATAPI_DEVICE | DFLAGS_DEVICE_PRESENT | DFLAGS_TAPE_DEVICE);
|
|
}
|
|
// Get the system physical address for this IO range.
|
|
|
|
// Check if configInfo has the default information
|
|
// if not, we go and find ourselves
|
|
if (preConfig == FALSE) {
|
|
|
|
ULONG portBase_reg = 0;
|
|
ULONG irq_reg = 0;
|
|
|
|
if (!portBase) {
|
|
portBase = AdapterAddresses[*adapterCount];
|
|
KdPrint2((PRINT_PREFIX "portBase[%d]=%x\n", *adapterCount, portBase));
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "portBase=%x\n", portBase));
|
|
}
|
|
|
|
portBase_reg = AtapiRegCheckDevValue(deviceExtension, CHAN_NOT_SPECIFIED, DEVNUM_NOT_SPECIFIED, L"PortBase", 0);
|
|
irq_reg = AtapiRegCheckDevValue(deviceExtension, CHAN_NOT_SPECIFIED, DEVNUM_NOT_SPECIFIED, L"Irq", 0);
|
|
if(portBase_reg && irq_reg) {
|
|
KdPrint2((PRINT_PREFIX "use registry settings portBase=%x, irq=%d\n", portBase_reg, irq_reg));
|
|
portBase = portBase_reg;
|
|
irq = irq_reg;
|
|
}
|
|
// check if Primary/Secondary Master IDE claimed
|
|
if(AtapiCheckIOInterference(ConfigInfo, portBase)) {
|
|
goto next_adapter;
|
|
}
|
|
ioSpace = (PUCHAR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
ConfigInfo->AdapterInterfaceType,
|
|
ConfigInfo->SystemIoBusNumber,
|
|
ScsiPortConvertUlongToPhysicalAddress(portBase),
|
|
ATA_IOSIZE,
|
|
TRUE);
|
|
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "preconfig portBase=%x\n", portBase));
|
|
// Check if Primary/Secondary Master IDE claimed
|
|
// We can also get here from preConfig branc with conflicting portBase
|
|
// (and thus, w/o ioSpace allocated)
|
|
if(AtapiCheckIOInterference(ConfigInfo, portBase)) {
|
|
goto not_found;
|
|
}
|
|
}
|
|
BaseIoAddress1 = (PIDE_REGISTERS_1)ioSpace;
|
|
next_adapter:
|
|
// Update the adapter count.
|
|
(*adapterCount)++;
|
|
|
|
// Check if ioSpace accessible.
|
|
if (!ioSpace) {
|
|
KdPrint2((PRINT_PREFIX "AtapiFindIsaController: !ioSpace\n"));
|
|
portBase = 0;
|
|
continue;
|
|
}
|
|
|
|
// Get the system physical address for the second IO range.
|
|
if (BaseIoAddress1) {
|
|
if(preConfig &&
|
|
!ScsiPortConvertPhysicalAddressToUlong((*ConfigInfo->AccessRanges)[1].RangeStart)) {
|
|
KdPrint2((PRINT_PREFIX "AtapiFindIsaController: PCMCIA ?\n"));
|
|
ioSpace = (PUCHAR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
ConfigInfo->AdapterInterfaceType,
|
|
ConfigInfo->SystemIoBusNumber,
|
|
ScsiPortConvertUlongToPhysicalAddress((ULONGIO_PTR)BaseIoAddress1 + 0x0E),
|
|
ATA_ALTIOSIZE,
|
|
TRUE);
|
|
} else {
|
|
ioSpace = (PUCHAR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
ConfigInfo->AdapterInterfaceType,
|
|
ConfigInfo->SystemIoBusNumber,
|
|
ScsiPortConvertUlongToPhysicalAddress((ULONGIO_PTR)BaseIoAddress1 + ATA_ALTOFFSET),
|
|
ATA_ALTIOSIZE,
|
|
TRUE);
|
|
}
|
|
}
|
|
BaseIoAddress2 = (PIDE_REGISTERS_2)ioSpace;
|
|
KdPrint2((PRINT_PREFIX " BaseIoAddress1=%x\n", BaseIoAddress1));
|
|
KdPrint2((PRINT_PREFIX " BaseIoAddress2=%x\n", BaseIoAddress2));
|
|
if(!irq) {
|
|
KdPrint2((PRINT_PREFIX " expected InterruptLevel=%x\n", InterruptLevels[*adapterCount - 1]));
|
|
}
|
|
|
|
UniataInitMapBase(chan, BaseIoAddress1, BaseIoAddress2);
|
|
UniataInitMapBM(deviceExtension, 0, FALSE);
|
|
|
|
#ifdef _DEBUG
|
|
UniataDumpATARegs(chan);
|
|
#endif
|
|
|
|
// Select master.
|
|
SelectDrive(chan, 0);
|
|
|
|
statusByte = AtapiReadPort1(chan, IDX_IO1_i_Status);
|
|
statusByte2 = AtapiReadPort1(chan, IDX_IO2_AltStatus);
|
|
if((statusByte ^ statusByte2) & ~IDE_STATUS_INDEX) {
|
|
KdPrint2((PRINT_PREFIX "AtapiFindIsaController: Status %x vs AltStatus %x missmatch, abort init ?\n", statusByte, statusByte2));
|
|
|
|
if(BaseIoAddress2) {
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
(PCHAR)BaseIoAddress2);
|
|
BaseIoAddress2 = NULL;
|
|
}
|
|
BaseIoAddress2 = (PIDE_REGISTERS_2)((ULONGIO_PTR)BaseIoAddress1 + 0x0E);
|
|
KdPrint2((PRINT_PREFIX " try BaseIoAddress2=%x\n", BaseIoAddress2));
|
|
ioSpace = (PUCHAR)ScsiPortGetDeviceBase(HwDeviceExtension,
|
|
ConfigInfo->AdapterInterfaceType,
|
|
ConfigInfo->SystemIoBusNumber,
|
|
ScsiPortConvertUlongToPhysicalAddress((ULONGIO_PTR)BaseIoAddress2),
|
|
ATA_ALTIOSIZE,
|
|
TRUE);
|
|
if(!ioSpace) {
|
|
BaseIoAddress2 = NULL;
|
|
KdPrint2((PRINT_PREFIX " abort (0)\n"));
|
|
goto not_found;
|
|
}
|
|
UniataInitMapBase(chan, BaseIoAddress1, BaseIoAddress2);
|
|
statusByte = AtapiReadPort1(chan, IDX_IO1_i_Status);
|
|
statusByte2 = AtapiReadPort1(chan, IDX_IO2_AltStatus);
|
|
if((statusByte ^ statusByte2) & ~IDE_STATUS_INDEX) {
|
|
KdPrint2((PRINT_PREFIX " abort: Status %x vs AltStatus %x missmatch\n", statusByte, statusByte2));
|
|
goto not_found;
|
|
}
|
|
}
|
|
|
|
retryIdentifier:
|
|
|
|
// Select master.
|
|
SelectDrive(chan, 0);
|
|
|
|
// Check if card at this address.
|
|
AtapiWritePort1(chan, IDX_IO1_o_CylinderLow, 0xAA);
|
|
statusByte = AtapiReadPort1(chan, IDX_IO1_i_CylinderLow);
|
|
|
|
// Check if indentifier can be read back.
|
|
if (AtapiReadPort1(chan, IDX_IO1_i_CylinderLow) != 0xAA ||
|
|
statusByte == IDE_STATUS_WRONG) {
|
|
|
|
KdPrint2((PRINT_PREFIX "AtapiFindIsaController: Identifier read back from Master (%#x)\n",
|
|
statusByte));
|
|
|
|
statusByte = AtapiReadPort1(chan, IDX_IO2_AltStatus);
|
|
|
|
if (statusByte != IDE_STATUS_WRONG && (statusByte & IDE_STATUS_BUSY)) {
|
|
|
|
i = 0;
|
|
|
|
// Could be the TEAC in a thinkpad. Their dos driver puts it in a sleep-mode that
|
|
// warm boots don't clear.
|
|
do {
|
|
AtapiStallExecution(1000);
|
|
statusByte = AtapiReadPort1(chan, IDX_ATAPI_IO1_i_Status);
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: First access to status %#x\n",
|
|
statusByte));
|
|
} while ((statusByte & IDE_STATUS_BUSY) && ++i < 10);
|
|
|
|
if (retryCount-- && (!(statusByte & IDE_STATUS_BUSY))) {
|
|
goto retryIdentifier;
|
|
}
|
|
}
|
|
|
|
// Select slave.
|
|
SelectDrive(chan, 1);
|
|
statusByte = AtapiReadPort1(chan, IDX_IO2_AltStatus);
|
|
|
|
// See if slave is present.
|
|
AtapiWritePort1(chan, IDX_IO1_o_CylinderLow, 0xAA);
|
|
|
|
if (AtapiReadPort1(chan, IDX_IO1_i_CylinderLow) != 0xAA ||
|
|
statusByte == IDE_STATUS_WRONG) {
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: Identifier read back from Slave (%#x)\n",
|
|
statusByte));
|
|
goto not_found;
|
|
}
|
|
}
|
|
|
|
// Fill in the access array information only if default params are not in there.
|
|
if (preConfig == FALSE) {
|
|
|
|
// An adapter has been found request another call, only if we didn't get preconfigured info.
|
|
*Again = TRUE;
|
|
|
|
if (portBase) {
|
|
(*ConfigInfo->AccessRanges)[0].RangeStart = ScsiPortConvertUlongToPhysicalAddress(portBase);
|
|
} else {
|
|
(*ConfigInfo->AccessRanges)[0].RangeStart =
|
|
ScsiPortConvertUlongToPhysicalAddress(AdapterAddresses[*adapterCount - 1]);
|
|
}
|
|
|
|
(*ConfigInfo->AccessRanges)[0].RangeLength = ATA_IOSIZE;
|
|
(*ConfigInfo->AccessRanges)[0].RangeInMemory = FALSE;
|
|
|
|
if(BaseIoAddress2) {
|
|
if(hasPCI) {
|
|
(*ConfigInfo->AccessRanges)[1].RangeStart = ScsiPortConvertUlongToPhysicalAddress((ULONG)BaseIoAddress2);
|
|
(*ConfigInfo->AccessRanges)[1].RangeLength = ATA_ALTIOSIZE;
|
|
(*ConfigInfo->AccessRanges)[1].RangeInMemory = FALSE;
|
|
} else {
|
|
// NT4 and NT3.51 on ISA-only hardware definitly fail floppy.sys load
|
|
// when this range is claimed by other driver.
|
|
// However, floppy should use only 0x3f0-3f5,3f7
|
|
if((ULONG)BaseIoAddress2 >= 0x3f0 && (ULONG)BaseIoAddress2 <= 0x3f7) {
|
|
KdPrint2((PRINT_PREFIX "!!! Possible AltStatus vs Floppy IO range interference !!!\n"));
|
|
}
|
|
KdPrint2((PRINT_PREFIX "Do not expose to OS on old ISA\n"));
|
|
(*ConfigInfo->AccessRanges)[1].RangeStart = ScsiPortConvertUlongToPhysicalAddress(0);
|
|
(*ConfigInfo->AccessRanges)[1].RangeLength = 0;
|
|
}
|
|
}
|
|
|
|
// Indicate the interrupt level corresponding to this IO range.
|
|
if (irq) {
|
|
ConfigInfo->BusInterruptLevel = irq;
|
|
} else {
|
|
ConfigInfo->BusInterruptLevel = InterruptLevels[*adapterCount - 1];
|
|
}
|
|
|
|
if (ConfigInfo->AdapterInterfaceType == MicroChannel) {
|
|
ConfigInfo->InterruptMode = LevelSensitive;
|
|
} else {
|
|
ConfigInfo->InterruptMode = Latched;
|
|
}
|
|
}
|
|
|
|
ConfigInfo->NumberOfBuses = 1;
|
|
ConfigInfo->MaximumNumberOfTargets = IDE_MAX_LUN_PER_CHAN;
|
|
|
|
// Indicate maximum transfer length is 64k.
|
|
ConfigInfo->MaximumTransferLength = 0x10000;
|
|
deviceExtension->MaximumDmaTransferLength = ConfigInfo->MaximumTransferLength;
|
|
|
|
KdPrint2((PRINT_PREFIX "de %#x, Channel ???\n", deviceExtension));
|
|
//PrintNtConsole("de %#x, Channel %#x, nchan %#x\n",deviceExtension, channel, deviceExtension->NumberChannels);
|
|
|
|
KdPrint2((PRINT_PREFIX "chan = %#x\n", chan));
|
|
//PrintNtConsole("chan = %#x, c=%#x\n", chan, c);
|
|
/*
|
|
// should be already set up in AtapiSetupLunPtrs(chan, deviceExtension, 0);
|
|
|
|
chan->DeviceExtension = deviceExtension;
|
|
chan->lChannel = 0;
|
|
chan->lun[0] = &(deviceExtension->lun[0]);
|
|
chan->lun[1] = &(deviceExtension->lun[1]);*/
|
|
|
|
/* do extra channel-specific setups */
|
|
AtapiReadChipConfig(HwDeviceExtension, DEVNUM_NOT_SPECIFIED, 0);
|
|
AtapiChipInit(HwDeviceExtension, DEVNUM_NOT_SPECIFIED, 0);
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: Found IDE at %#x\n",
|
|
BaseIoAddress1));
|
|
|
|
// For Daytona, the atdisk driver gets the first shot at the
|
|
// primary and secondary controllers.
|
|
if (preConfig == FALSE) {
|
|
|
|
if (*adapterCount - 1 < 2) {
|
|
|
|
// Determine whether this driver is being initialized by the
|
|
// system or as a crash dump driver.
|
|
if (g_Dump) {
|
|
#ifndef UNIATA_CORE
|
|
deviceExtension->DriverMustPoll = TRUE;
|
|
#endif //UNIATA_CORE
|
|
} else {
|
|
deviceExtension->DriverMustPoll = FALSE;
|
|
}
|
|
|
|
} else {
|
|
//atapiOnly = FALSE;
|
|
}
|
|
|
|
} else {
|
|
|
|
//atapiOnly = FALSE;
|
|
deviceExtension->DriverMustPoll = FALSE;
|
|
|
|
}// preConfig check
|
|
|
|
// Save the Interrupe Mode for later use
|
|
deviceExtension->InterruptMode = ConfigInfo->InterruptMode;
|
|
deviceExtension->BusInterruptLevel = ConfigInfo->BusInterruptLevel;
|
|
deviceExtension->BusInterruptVector = ConfigInfo->BusInterruptVector;
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: look for devices\n"));
|
|
// Search for devices on this controller.
|
|
if (FindDevices(HwDeviceExtension,
|
|
0,
|
|
0 /* Channel */)) {
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: detected\n"));
|
|
// Claim primary or secondary ATA IO range.
|
|
if (portBase) {
|
|
switch (portBase) {
|
|
case IO_WD2:
|
|
ConfigInfo->AtdiskSecondaryClaimed = TRUE;
|
|
chan->PrimaryAddress = FALSE;
|
|
break;
|
|
case IO_WD1:
|
|
ConfigInfo->AtdiskPrimaryClaimed = TRUE;
|
|
chan->PrimaryAddress = TRUE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
} else {
|
|
if (*adapterCount == 1) {
|
|
ConfigInfo->AtdiskPrimaryClaimed = TRUE;
|
|
chan->PrimaryAddress = TRUE;
|
|
} else if (*adapterCount == 2) {
|
|
ConfigInfo->AtdiskSecondaryClaimed = TRUE;
|
|
chan->PrimaryAddress = FALSE;
|
|
}
|
|
}
|
|
|
|
if(deviceExtension->AdapterInterfaceType == Isa) {
|
|
IsaCount++;
|
|
} else
|
|
if(deviceExtension->AdapterInterfaceType == MicroChannel) {
|
|
MCACount++;
|
|
}
|
|
|
|
ConfigInfo->NumberOfBuses++; // add virtual channel for communication port
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: return SP_RETURN_FOUND\n"));
|
|
return(SP_RETURN_FOUND);
|
|
} else {
|
|
not_found:
|
|
// No controller at this base address.
|
|
if(BaseIoAddress1) {
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
(PCHAR)BaseIoAddress1);
|
|
BaseIoAddress1 = NULL;
|
|
}
|
|
if(BaseIoAddress2) {
|
|
ScsiPortFreeDeviceBase(HwDeviceExtension,
|
|
(PCHAR)BaseIoAddress2);
|
|
BaseIoAddress2 = NULL;
|
|
}
|
|
for(i=0; i<2; i++) {
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: cleanup AccessRanges %d\n", i));
|
|
(*ConfigInfo->AccessRanges)[i].RangeStart = ScsiPortConvertUlongToPhysicalAddress(0);
|
|
(*ConfigInfo->AccessRanges)[i].RangeLength = 0;
|
|
(*ConfigInfo->AccessRanges)[i].RangeInMemory = FALSE;
|
|
}
|
|
irq = 0;
|
|
portBase = 0;
|
|
}
|
|
#ifndef UNIATA_CORE
|
|
}
|
|
#else
|
|
} while(FALSE);
|
|
#endif //UNIATA_CORE
|
|
|
|
// The entire table has been searched and no adapters have been found.
|
|
// There is no need to call again and the device base can now be freed.
|
|
// Clear the adapter count for the next bus.
|
|
*Again = FALSE;
|
|
*(adapterCount) = 0;
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiFindIsaController: return SP_RETURN_NOT_FOUND\n"));
|
|
UniataFreeLunExt(deviceExtension);
|
|
return(SP_RETURN_NOT_FOUND);
|
|
|
|
exit_error:
|
|
UniataFreeLunExt(deviceExtension);
|
|
return SP_RETURN_ERROR;
|
|
|
|
} // end AtapiFindIsaController()
|
|
|
|
/*
|
|
Do nothing, but parse ScsiPort ArgumentString and setup global variables.
|
|
*/
|
|
|
|
ULONG
|
|
NTAPI
|
|
AtapiReadArgumentString(
|
|
IN PVOID HwDeviceExtension,
|
|
IN PVOID Context,
|
|
IN PVOID BusInformation,
|
|
IN PCHAR ArgumentString,
|
|
IN OUT PPORT_CONFIGURATION_INFORMATION ConfigInfo,
|
|
OUT PBOOLEAN Again
|
|
)
|
|
{
|
|
#ifndef __REACTOS__
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
#endif
|
|
|
|
if (AtapiParseArgumentString(ArgumentString, "dump") == 1) {
|
|
KdPrint2((PRINT_PREFIX
|
|
"AtapiReadArgumentString: Crash dump\n"));
|
|
//atapiOnly = FALSE;
|
|
g_Dump = TRUE;
|
|
}
|
|
return(SP_RETURN_NOT_FOUND);
|
|
} // end AtapiReadArgumentString()
|
|
|
|
ULONG
|
|
NTAPI
|
|
UniataAnybodyHome(
|
|
IN PVOID HwDeviceExtension,
|
|
IN ULONG lChannel,
|
|
IN ULONG deviceNumber
|
|
)
|
|
{
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
PHW_CHANNEL chan = &(deviceExtension->chan[lChannel]);
|
|
//ULONG ldev = GET_LDEV2(lChannel, deviceNumber, 0);
|
|
PHW_LU_EXTENSION LunExt = chan->lun[deviceNumber];
|
|
|
|
SATA_SSTATUS_REG SStatus;
|
|
UCHAR signatureLow;
|
|
UCHAR signatureHigh;
|
|
|
|
if(LunExt->DeviceFlags & DFLAGS_HIDDEN) {
|
|
KdPrint2((PRINT_PREFIX " hidden\n"));
|
|
UniataForgetDevice(LunExt);
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
if(UniataIsSATARangeAvailable(deviceExtension, lChannel)) {
|
|
|
|
SStatus.Reg = UniataSataReadPort4(chan, IDX_SATA_SStatus, deviceNumber);
|
|
KdPrint2((PRINT_PREFIX "SStatus %x\n", SStatus.Reg));
|
|
if(SStatus.DET <= SStatus_DET_Dev_NoPhy) {
|
|
KdPrint2((PRINT_PREFIX " SATA DET <= SStatus_DET_Dev_NoPhy\n"));
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
if(SStatus.SPD < SStatus_SPD_Gen1) {
|
|
KdPrint2((PRINT_PREFIX " SATA SPD < SStatus_SPD_Gen1\n"));
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
if(SStatus.IPM == SStatus_IPM_NoDev) {
|
|
KdPrint2((PRINT_PREFIX " SATA IPN == SStatus_IPM_NoDev\n"));
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
if(!(deviceExtension->HwFlags & UNIATA_AHCI)) {
|
|
// Select the device for legacy.
|
|
goto legacy_select;
|
|
}
|
|
|
|
} else {
|
|
legacy_select:
|
|
// Select the device.
|
|
SelectDrive(chan, deviceNumber);
|
|
AtapiStallExecution(5);
|
|
}
|
|
|
|
if((deviceExtension->HwFlags & UNIATA_AHCI) &&
|
|
UniataIsSATARangeAvailable(deviceExtension, lChannel)) {
|
|
KdPrint2((PRINT_PREFIX " AHCI check\n"));
|
|
ULONG SIG = UniataAhciReadChannelPort4(chan, IDX_AHCI_P_SIG);
|
|
signatureLow = (UCHAR)(SIG >> 16);
|
|
signatureHigh = (UCHAR)(SIG >> 24);
|
|
} else {
|
|
signatureLow = AtapiReadPort1(chan, IDX_IO1_i_CylinderLow);
|
|
signatureHigh = AtapiReadPort1(chan, IDX_IO1_i_CylinderHigh);
|
|
}
|
|
|
|
if (signatureLow == ATAPI_MAGIC_LSB && signatureHigh == ATAPI_MAGIC_MSB) {
|
|
KdPrint2((PRINT_PREFIX " ATAPI at home\n"));
|
|
return ATA_AT_HOME_ATAPI;
|
|
}
|
|
if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
KdPrint2((PRINT_PREFIX " AHCI HDD at home\n"));
|
|
return ATA_AT_HOME_HDD;
|
|
}
|
|
if(g_opt_VirtualMachine > VM_NONE /*== VM_BOCHS ||
|
|
g_opt_VirtualMachine == VM_VBOX*/) {
|
|
GetStatus(chan, signatureLow);
|
|
if(!signatureLow) {
|
|
KdPrint2((PRINT_PREFIX " 0-status VM - not present\n"));
|
|
UniataForgetDevice(LunExt);
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
}
|
|
|
|
AtapiStallExecution(10);
|
|
|
|
AtapiWritePort1(chan, IDX_IO1_o_BlockNumber, 0x55);
|
|
AtapiWritePort1(chan, IDX_IO1_o_BlockNumber, 0x55);
|
|
AtapiStallExecution(5);
|
|
signatureLow = AtapiReadPort1(chan, IDX_IO1_i_BlockNumber);
|
|
if(signatureLow != 0x55) {
|
|
if(signatureLow == 0xff || signatureLow == 0) {
|
|
KdPrint2((PRINT_PREFIX " nobody home! %#x != 0x55\n", signatureLow));
|
|
UniataForgetDevice(LunExt);
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
// another chance
|
|
signatureLow = AtapiReadPort1(chan, IDX_ATAPI_IO1_o_ByteCountHigh);
|
|
signatureLow += 2;
|
|
AtapiWritePort1(chan, IDX_ATAPI_IO1_o_ByteCountHigh, signatureLow);
|
|
AtapiStallExecution(5);
|
|
signatureHigh = AtapiReadPort1(chan, IDX_ATAPI_IO1_o_ByteCountHigh);
|
|
if(signatureLow != signatureHigh) {
|
|
KdPrint2((PRINT_PREFIX " nobody home! last chance failed %#x != %#x\n", signatureLow, signatureHigh));
|
|
UniataForgetDevice(LunExt);
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
return ATA_AT_HOME_XXX;
|
|
}
|
|
|
|
AtapiWritePort1(chan, IDX_IO1_o_BlockNumber, 0xAA);
|
|
AtapiWritePort1(chan, IDX_IO1_o_BlockNumber, 0xAA);
|
|
AtapiStallExecution(5);
|
|
signatureLow = AtapiReadPort1(chan, IDX_IO1_i_BlockNumber);
|
|
if(signatureLow != 0xAA) {
|
|
KdPrint2((PRINT_PREFIX " nobody home! %#x != 0xAA\n", signatureLow));
|
|
UniataForgetDevice(LunExt);
|
|
return ATA_AT_HOME_NOBODY;
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX " HDD at home\n"));
|
|
return ATA_AT_HOME_HDD;
|
|
} // end UniataAnybodyHome()
|
|
|
|
ULONG
|
|
NTAPI
|
|
CheckDevice(
|
|
IN PVOID HwDeviceExtension,
|
|
IN ULONG lChannel,
|
|
IN ULONG deviceNumber,
|
|
IN BOOLEAN ResetDev
|
|
)
|
|
{
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
PHW_CHANNEL chan = &(deviceExtension->chan[lChannel]);
|
|
//ULONG ldev = GET_LDEV2(lChannel, deviceNumber, 0);
|
|
PHW_LU_EXTENSION LunExt;
|
|
|
|
UCHAR signatureLow,
|
|
signatureHigh;
|
|
UCHAR statusByte;
|
|
ULONG RetVal=0;
|
|
ULONG waitCount = g_opt_WaitBusyResetCount;
|
|
ULONG at_home = 0;
|
|
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: Device %#x\n",
|
|
deviceNumber));
|
|
|
|
if(deviceNumber >= chan->NumberLuns) {
|
|
return 0;
|
|
}
|
|
if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
if(!(at_home = UniataAnybodyHome(HwDeviceExtension, lChannel, deviceNumber))) {
|
|
return 0;
|
|
}
|
|
}
|
|
LunExt = chan->lun[deviceNumber];
|
|
|
|
if(ResetDev) {
|
|
LunExt->PowerState = 0;
|
|
}
|
|
|
|
if((deviceExtension->HwFlags & UNIATA_SATA) &&
|
|
!UniataIsSATARangeAvailable(deviceExtension, lChannel) &&
|
|
deviceNumber) {
|
|
KdPrint2((PRINT_PREFIX " SATA w/o i/o registers, check slave presence\n"));
|
|
SelectDrive(chan, deviceNumber & 0x01);
|
|
statusByte = AtapiReadPort1(chan, IDX_ATAPI_IO1_i_DriveSelect);
|
|
KdPrint2((PRINT_PREFIX " DriveSelect: %#x\n", statusByte));
|
|
if((statusByte & IDE_DRIVE_MASK) != IDE_DRIVE_SELECT_2) {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: (no dev)\n"));
|
|
UniataForgetDevice(LunExt);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if(ResetDev && (deviceExtension->HwFlags & UNIATA_AHCI)) {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: reset AHCI dev\n"));
|
|
if(UniataAhciSoftReset(HwDeviceExtension, chan->lChannel, deviceNumber) == (ULONG)(-1)) {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: (no dev)\n"));
|
|
UniataForgetDevice(LunExt);
|
|
return 0;
|
|
}
|
|
} else
|
|
if(ResetDev) {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: reset dev\n"));
|
|
|
|
// Reset device
|
|
AtapiSoftReset(chan, deviceNumber);
|
|
|
|
if(!(at_home = UniataAnybodyHome(HwDeviceExtension, lChannel, deviceNumber))) {
|
|
//KdPrint2((PRINT_PREFIX "CheckDevice: nobody at home 1\n"));
|
|
return 0;
|
|
}
|
|
statusByte = WaitOnBusy(chan);
|
|
|
|
if((statusByte | IDE_STATUS_BUSY) == IDE_STATUS_WRONG) {
|
|
KdPrint2((PRINT_PREFIX
|
|
"CheckDevice: bad status %x\n", statusByte));
|
|
} else
|
|
if(statusByte != IDE_STATUS_WRONG && (statusByte & IDE_STATUS_BUSY)) {
|
|
// Perform hard-reset.
|
|
KdPrint2((PRINT_PREFIX
|
|
"CheckDevice: BUSY\n"));
|
|
|
|
AtapiHardReset(chan, FALSE, 500 * 1000);
|
|
/*
|
|
AtapiWritePort1(chan, IDX_IO2_o_Control, IDE_DC_RESET_CONTROLLER );
|
|
chan->last_devsel = -1;
|
|
AtapiStallExecution(500 * 1000);
|
|
AtapiWritePort1(chan, IDX_IO2_o_Control, IDE_DC_REENABLE_CONTROLLER);
|
|
*/
|
|
SelectDrive(chan, deviceNumber & 0x01);
|
|
|
|
do {
|
|
// Wait for Busy to drop.
|
|
AtapiStallExecution(100);
|
|
GetStatus(chan, statusByte);
|
|
|
|
} while ((statusByte & IDE_STATUS_BUSY) && waitCount--);
|
|
|
|
GetBaseStatus(chan, statusByte);
|
|
KdPrint2((PRINT_PREFIX
|
|
"CheckDevice: status after hard reset %x\n", statusByte));
|
|
}
|
|
|
|
if((statusByte | IDE_STATUS_BUSY) == IDE_STATUS_WRONG) {
|
|
KdPrint2((PRINT_PREFIX
|
|
"CheckDevice: no dev ?\n"));
|
|
UniataForgetDevice(LunExt);
|
|
return 0;
|
|
} else
|
|
if(UniataIsSATARangeAvailable(deviceExtension, lChannel)) {
|
|
//if(deviceExtension->HwFlags & UNIATA_SATA) {
|
|
KdPrint2((PRINT_PREFIX
|
|
"CheckDevice: try enable SATA Phy\n"));
|
|
statusByte = UniataSataPhyEnable(HwDeviceExtension, lChannel, deviceNumber);
|
|
if(statusByte == IDE_STATUS_WRONG) {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: status %#x (no dev)\n", statusByte));
|
|
UniataForgetDevice(LunExt);
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
RetVal = LunExt->DeviceFlags;
|
|
signatureLow = signatureHigh = 0; // make GCC happy
|
|
} else {
|
|
// Select the device.
|
|
SelectDrive(chan, deviceNumber);
|
|
|
|
if(!(at_home = UniataAnybodyHome(HwDeviceExtension, lChannel, deviceNumber))) {
|
|
//KdPrint2((PRINT_PREFIX "CheckDevice: nobody at home 2\n"));
|
|
return 0;
|
|
}
|
|
|
|
statusByte = WaitOnBaseBusyLong(chan);
|
|
|
|
GetBaseStatus(chan, statusByte);
|
|
if(deviceExtension->HwFlags & UNIATA_SATA) {
|
|
UniataSataClearErr(HwDeviceExtension, lChannel, UNIATA_SATA_IGNORE_CONNECT, deviceNumber);
|
|
}
|
|
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: status %#x\n", statusByte));
|
|
if(((statusByte | IDE_STATUS_BUSY) == IDE_STATUS_WRONG) ||
|
|
(statusByte & IDE_STATUS_BUSY)) {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: busy => return\n"));
|
|
UniataForgetDevice(LunExt);
|
|
return 0;
|
|
}
|
|
|
|
signatureLow = AtapiReadPort1(chan, IDX_IO1_i_CylinderLow);
|
|
signatureHigh = AtapiReadPort1(chan, IDX_IO1_i_CylinderHigh);
|
|
}
|
|
|
|
// set default costs
|
|
LunExt->RwSwitchCost = REORDER_COST_SWITCH_RW_HDD;
|
|
LunExt->RwSwitchMCost = REORDER_MCOST_SWITCH_RW_HDD;
|
|
LunExt->SeekBackMCost = REORDER_MCOST_SEEK_BACK_HDD;
|
|
LunExt->AtapiReadyWaitDelay = 0;
|
|
|
|
if(deviceExtension->HwFlags & UNIATA_AHCI) {
|
|
if(RetVal & DFLAGS_DEVICE_PRESENT) {
|
|
if(IssueIdentify(HwDeviceExtension,
|
|
deviceNumber,
|
|
lChannel,
|
|
(RetVal & DFLAGS_ATAPI_DEVICE) ? IDE_COMMAND_ATAPI_IDENTIFY : IDE_COMMAND_IDENTIFY,
|
|
FALSE)) {
|
|
// OK
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: detected AHCI Device %#x\n",
|
|
deviceNumber));
|
|
} else {
|
|
//RetVal &= ~DFLAGS_ATAPI_DEVICE;
|
|
//LunExt->DeviceFlags &= ~DFLAGS_ATAPI_DEVICE;
|
|
|
|
UniataForgetDevice(LunExt);
|
|
RetVal = 0;
|
|
}
|
|
}
|
|
} else
|
|
if (signatureLow == ATAPI_MAGIC_LSB && signatureHigh == ATAPI_MAGIC_MSB) {
|
|
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: ATAPI signature found\n"));
|
|
// ATAPI signature found.
|
|
// Issue the ATAPI identify command if this
|
|
// is not for the crash dump utility.
|
|
try_atapi:
|
|
if (!g_Dump) {
|
|
|
|
// Issue ATAPI packet identify command.
|
|
if (IssueIdentify(HwDeviceExtension,
|
|
deviceNumber,
|
|
lChannel,
|
|
IDE_COMMAND_ATAPI_IDENTIFY, FALSE)) {
|
|
|
|
// Indicate ATAPI device.
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: Device %#x is ATAPI\n",
|
|
deviceNumber));
|
|
|
|
RetVal = DFLAGS_DEVICE_PRESENT | DFLAGS_ATAPI_DEVICE;
|
|
LunExt->DeviceFlags |= (DFLAGS_DEVICE_PRESENT | DFLAGS_ATAPI_DEVICE);
|
|
|
|
// some ATAPI devices doesn't work with DPC on CMD-649
|
|
// and probably some other controllers
|
|
if(deviceExtension->HwFlags & UNIATA_NO_DPC_ATAPI) {
|
|
/* CMD 649, ROSB SWK33, ICH4 */
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: UNIATA_NO_DPC_ATAPI\n"));
|
|
deviceExtension->UseDpc = FALSE;
|
|
}
|
|
|
|
GetStatus(chan, statusByte);
|
|
if (statusByte & IDE_STATUS_ERROR) {
|
|
AtapiSoftReset(chan, deviceNumber);
|
|
}
|
|
|
|
} else {
|
|
forget_device:
|
|
// Indicate no working device.
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: Device %#x not responding\n",
|
|
deviceNumber));
|
|
|
|
UniataForgetDevice(LunExt);
|
|
RetVal = 0;
|
|
}
|
|
GetBaseStatus(chan, statusByte);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: IDE device check\n"));
|
|
// Issue IDE Identify. If an Atapi device is actually present, the signature
|
|
// will be asserted, and the drive will be recognized as such.
|
|
if(deviceExtension->DWordIO) {
|
|
KdPrint2((PRINT_PREFIX " try 32bit IO\n"));
|
|
LunExt->DeviceFlags |= DFLAGS_DWORDIO_ENABLED;
|
|
}
|
|
if (IssueIdentify(HwDeviceExtension,
|
|
deviceNumber,
|
|
lChannel,
|
|
IDE_COMMAND_IDENTIFY, FALSE)) {
|
|
|
|
// IDE drive found.
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: Device %#x is IDE\n",
|
|
deviceNumber));
|
|
|
|
// Indicate IDE - not ATAPI device.
|
|
RetVal = DFLAGS_DEVICE_PRESENT;
|
|
LunExt->DeviceFlags |= DFLAGS_DEVICE_PRESENT;
|
|
LunExt->DeviceFlags &= ~DFLAGS_ATAPI_DEVICE;
|
|
} else
|
|
if(g_opt_VirtualMachine <= VM_NONE) {
|
|
// This can be ATAPI on broken hardware
|
|
GetBaseStatus(chan, statusByte);
|
|
if(!at_home && UniataAnybodyHome(HwDeviceExtension, lChannel, deviceNumber)) {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: nobody at home post IDE\n"));
|
|
goto forget_device;
|
|
}
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: try ATAPI %#x, status %#x\n",
|
|
deviceNumber, statusByte));
|
|
goto try_atapi;
|
|
} else {
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: VM Device %#x not present\n",
|
|
deviceNumber));
|
|
}
|
|
GetBaseStatus(chan, statusByte);
|
|
}
|
|
KdPrint2((PRINT_PREFIX "CheckDevice: check status: %sfound\n", RetVal ? "" : "not "));
|
|
return RetVal;
|
|
} // end CheckDevice()
|
|
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called from AtapiFindXxxController to identify
|
|
devices attached to an IDE controller.
|
|
|
|
Arguments:
|
|
|
|
HwDeviceExtension - HBA miniport driver's adapter data storage
|
|
AtapiOnly - Indicates that routine should return TRUE only if
|
|
an ATAPI device is attached to the controller.
|
|
|
|
Return Value:
|
|
|
|
TRUE - True if devices found.
|
|
|
|
--*/
|
|
BOOLEAN
|
|
NTAPI
|
|
FindDevices(
|
|
IN PVOID HwDeviceExtension,
|
|
IN ULONG Flags,
|
|
IN ULONG Channel
|
|
)
|
|
{
|
|
PHW_DEVICE_EXTENSION deviceExtension = (PHW_DEVICE_EXTENSION)HwDeviceExtension;
|
|
PHW_CHANNEL chan = &(deviceExtension->chan[Channel]);
|
|
PHW_LU_EXTENSION LunExt;
|
|
BOOLEAN deviceResponded = FALSE,
|
|
skipSetParameters = FALSE;
|
|
ULONG waitCount = 10000;
|
|
//ULONG deviceNumber;
|
|
ULONG i;
|
|
UCHAR statusByte;
|
|
ULONG max_ldev;
|
|
BOOLEAN AtapiOnly = FALSE;
|
|
|
|
KdPrint2((PRINT_PREFIX "FindDevices:\n"));
|
|
|
|
// Disable interrupts
|
|
AtapiDisableInterrupts(deviceExtension, Channel);
|
|
// AtapiWritePort1(chan, IDX_IO2_o_Control,IDE_DC_DISABLE_INTERRUPTS | IDE_DC_A_4BIT );
|
|
|
|
// Clear expecting interrupt flag and current SRB field.
|
|
UniataExpectChannelInterrupt(chan, FALSE);
|
|
// chan->CurrentSrb = NULL;
|
|
// max_ldev = (chan->ChannelCtrlFlags & CTRFLAGS_NO_SLAVE) ? 1 : IDE_MAX_LUN_PER_CHAN;
|
|
max_ldev = (chan->ChannelCtrlFlags & CTRFLAGS_NO_SLAVE) ? 1 : deviceExtension->NumberLuns;
|
|
KdPrint2((PRINT_PREFIX " max_ldev %d\n", max_ldev));
|
|
|
|
// Search for devices.
|
|
for (i = 0; i < max_ldev; i++) {
|
|
//AtapiDisableInterrupts(deviceExtension, Channel);
|
|
if(Flags & UNIATA_FIND_DEV_UNHIDE) {
|
|
chan->lun[i]->DeviceFlags &= ~DFLAGS_HIDDEN;
|
|
}
|
|
deviceResponded |=
|
|
(CheckDevice(HwDeviceExtension, Channel, i, TRUE) != 0);
|
|
//AtapiEnableInterrupts(deviceExtension, Channel);
|
|
}
|
|
|
|
if(chan->DeviceExtension->HwFlags & UNIATA_AHCI) {
|
|
AtapiEnableInterrupts(deviceExtension, Channel);
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: returning %d (AHCI)\n",
|
|
deviceResponded));
|
|
return deviceResponded;
|
|
}
|
|
|
|
for (i = 0; i < max_ldev; i++) {
|
|
LunExt = chan->lun[i];
|
|
|
|
if (( LunExt->DeviceFlags & DFLAGS_DEVICE_PRESENT) &&
|
|
!(LunExt->DeviceFlags & DFLAGS_LBA_ENABLED) &&
|
|
!(LunExt->DeviceFlags & DFLAGS_ATAPI_DEVICE) && deviceResponded) {
|
|
|
|
// This hideous hack is to deal with ESDI devices that return
|
|
// garbage geometry in the IDENTIFY data.
|
|
// This is ONLY for the crashdump environment as
|
|
// these are ESDI devices.
|
|
if (LunExt->IdentifyData.SectorsPerTrack == 0x35 &&
|
|
LunExt->IdentifyData.NumberOfHeads == 0x07) {
|
|
|
|
KdPrint2((PRINT_PREFIX "FindDevices: Found nasty Compaq ESDI!\n"));
|
|
|
|
// Change these values to something reasonable.
|
|
LunExt->IdentifyData.SectorsPerTrack = 0x34;
|
|
LunExt->IdentifyData.NumberOfHeads = 0x0E;
|
|
}
|
|
|
|
if (LunExt->IdentifyData.SectorsPerTrack == 0x35 &&
|
|
LunExt->IdentifyData.NumberOfHeads == 0x0F) {
|
|
|
|
KdPrint2((PRINT_PREFIX "FindDevices: Found nasty Compaq ESDI!\n"));
|
|
|
|
// Change these values to something reasonable.
|
|
LunExt->IdentifyData.SectorsPerTrack = 0x34;
|
|
LunExt->IdentifyData.NumberOfHeads = 0x0F;
|
|
}
|
|
|
|
|
|
if (LunExt->IdentifyData.SectorsPerTrack == 0x36 &&
|
|
LunExt->IdentifyData.NumberOfHeads == 0x07) {
|
|
|
|
KdPrint2((PRINT_PREFIX "FindDevices: Found nasty UltraStor ESDI!\n"));
|
|
|
|
// Change these values to something reasonable.
|
|
LunExt->IdentifyData.SectorsPerTrack = 0x3F;
|
|
LunExt->IdentifyData.NumberOfHeads = 0x10;
|
|
skipSetParameters = TRUE;
|
|
}
|
|
|
|
|
|
if (skipSetParameters)
|
|
continue;
|
|
|
|
statusByte = WaitOnBusy(chan);
|
|
|
|
// Select the device.
|
|
SelectDrive(chan, i & 0x01);
|
|
GetStatus(chan, statusByte);
|
|
|
|
if (statusByte & IDE_STATUS_ERROR) {
|
|
|
|
// Reset the device.
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: Resetting controller before SetDriveParameters.\n"));
|
|
|
|
AtapiHardReset(chan, FALSE, 500 * 1000);
|
|
/*
|
|
AtapiWritePort1(chan, IDX_IO2_o_Control, IDE_DC_RESET_CONTROLLER );
|
|
chan->last_devsel = -1;
|
|
AtapiStallExecution(500 * 1000);
|
|
AtapiWritePort1(chan, IDX_IO2_o_Control, IDE_DC_REENABLE_CONTROLLER);
|
|
*/
|
|
SelectDrive(chan, i & 0x01);
|
|
|
|
do {
|
|
// Wait for Busy to drop.
|
|
AtapiStallExecution(100);
|
|
GetStatus(chan, statusByte);
|
|
|
|
} while ((statusByte & IDE_STATUS_BUSY) && waitCount--);
|
|
}
|
|
|
|
statusByte = WaitOnBusy(chan);
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: Status before SetDriveParameters: (%#x) (%#x)\n",
|
|
statusByte,
|
|
AtapiReadPort1(chan, IDX_IO1_i_DriveSelect)));
|
|
|
|
GetBaseStatus(chan, statusByte);
|
|
|
|
// Use the IDENTIFY data to set drive parameters.
|
|
if (!SetDriveParameters(HwDeviceExtension,i,Channel)) {
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: Set drive parameters for device %d failed\n",
|
|
i));
|
|
// Don't use this device as writes could cause corruption.
|
|
LunExt->DeviceFlags &= ~DFLAGS_DEVICE_PRESENT;
|
|
UniataForgetDevice(LunExt);
|
|
continue;
|
|
|
|
}
|
|
if (LunExt->DeviceFlags & DFLAGS_REMOVABLE_DRIVE) {
|
|
|
|
// Pick up ALL IDE removable drives that conform to Yosemite V0.2...
|
|
AtapiOnly = FALSE;
|
|
}
|
|
|
|
// Indicate that a device was found.
|
|
if (!AtapiOnly) {
|
|
deviceResponded = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* // Reset the controller. This is a feeble attempt to leave the ESDI
|
|
// controllers in a state that ATDISK driver will recognize them.
|
|
// The problem in ATDISK has to do with timings as it is not reproducible
|
|
// in debug. The reset should restore the controller to its poweron state
|
|
// and give the system enough time to settle.
|
|
if (!deviceResponded) {
|
|
|
|
AtapiWritePort1(chan, IDX_IO2_o_Control,IDE_DC_RESET_CONTROLLER );
|
|
AtapiStallExecution(50 * 1000);
|
|
AtapiWritePort1(chan, IDX_IO2_o_Control,IDE_DC_REENABLE_CONTROLLER);
|
|
}
|
|
*/
|
|
if(deviceResponded) {
|
|
for (i = 0; i < max_ldev; i++) {
|
|
LunExt = chan->lun[i];
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: select %d dev to clear INTR\n", i));
|
|
SelectDrive(chan, i);
|
|
GetBaseStatus(chan, statusByte);
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: statusByte=%#x\n", statusByte));
|
|
}
|
|
for (i = 0; i < max_ldev; i++) {
|
|
LunExt = chan->lun[i];
|
|
|
|
if(LunExt->DeviceFlags & DFLAGS_DEVICE_PRESENT) {
|
|
// Make sure some device (master is preferred) is selected on exit.
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: select %d dev on exit\n", i));
|
|
SelectDrive(chan, i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
GetBaseStatus(chan, statusByte);
|
|
// Enable interrupts
|
|
AtapiEnableInterrupts(deviceExtension, Channel);
|
|
// AtapiWritePort1(chan, IDX_IO2_o_Control, IDE_DC_A_4BIT );
|
|
GetBaseStatus(chan, statusByte);
|
|
|
|
KdPrint2((PRINT_PREFIX
|
|
"FindDevices: returning %d\n",
|
|
deviceResponded));
|
|
|
|
return deviceResponded;
|
|
|
|
} // end FindDevices()
|