2001-08-21 20:18:27 +00:00
|
|
|
/*
|
2006-11-27 19:26:31 +00:00
|
|
|
* PROJECT: ReactOS HAL
|
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
2015-11-04 13:30:52 +00:00
|
|
|
* FILE: hal/halx86/legacy/bus/pcibus.c
|
2006-11-27 19:26:31 +00:00
|
|
|
* PURPOSE: PCI Bus Support (Configuration Space, Resource Allocation)
|
|
|
|
* PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
|
2001-08-21 20:18:27 +00:00
|
|
|
*/
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* INCLUDES ******************************************************************/
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2005-06-18 14:29:31 +00:00
|
|
|
#include <hal.h>
|
2005-06-19 22:53:49 +00:00
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* GLOBALS *******************************************************************/
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
extern BOOLEAN HalpPciLockSettings;
|
[HAL]: Bus support in the HAL actually creates a further wedge between the different x86 HALs: There are actually two dinstinct implementations. On the ACPI HAL, the system is assumed not to have things like special ISA, MCA, EISA buses, and a PCI driver is used in combination with the ACPI Interface for PCI Bus support. On non-ACPI systems, the legacy "Bus Handler" library is used, and the HAL provides a core set of CMOS, EISA, ISA, MCA and PCI bus handlers, each with their own routines and specific code. Additionally, PCI IRQ Routing and other PCI bus internals are handled directly by the HAL -- on the ACPI HAL, the PCI Bus support is implemented through a "Fake"/static bus handler, just to keep the functions shared. On ReactOS, both the ACPI and non-ACPI HAL were currently using a mix of both HAL bus handling types, mostly implemented the "ACPI way" (with a fake PCI bus handler and such).
As a result, none of the Hal*Bus HALDISPATCH routines were implemented, which bus drivers expect to find when they're not on ACPI systems (ReactOS today). eVb's new PCI driver was crashing, for example.
Furthermore, legacy systems suffer, because the ACPI HAL Bus routines (that we currently have) expect perfect ACPI-style-compliant systems, not the legacy crap from the early 90ies. This works fine in VMs and new hardware, but old hardware is left behind.
This patch basically corrects the first part of the problem, by making the bus handling support separate between ACPI and non-ACPI HALs. For now, the code remains 100% the same in functionality between both.
However, I have started adding the first few elements:
[HAL]: Implement HalRegisterBusHandler HALDISPATCH routine.
[HAL]: On legacy HALs, register the CMOS, ISA, SYSTEM handlers.
[HAL]: Add cmosbus.c. Stub all bus-specific bus handler routines in the xxxbus.c files.
No real functionality change occurs with this patch, yet.
svn path=/trunk/; revision=47649
2010-06-07 01:09:41 +00:00
|
|
|
ULONG HalpBusType;
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
BOOLEAN HalpPCIConfigInitialized;
|
|
|
|
ULONG HalpMinPciBus, HalpMaxPciBus;
|
|
|
|
KSPIN_LOCK HalpPCIConfigLock;
|
|
|
|
PCI_CONFIG_HANDLER PCIConfigHandler;
|
2003-02-17 21:24:13 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* PCI Operation Matrix */
|
|
|
|
UCHAR PCIDeref[4][4] =
|
|
|
|
{
|
|
|
|
{0, 1, 2, 2}, // ULONG-aligned offset
|
|
|
|
{1, 1, 1, 1}, // UCHAR-aligned offset
|
|
|
|
{2, 1, 2, 2}, // USHORT-aligned offset
|
|
|
|
{1, 1, 1, 1} // UCHAR-aligned offset
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Type 1 PCI Bus */
|
|
|
|
PCI_CONFIG_HANDLER PCIConfigHandlerType1 =
|
|
|
|
{
|
|
|
|
/* Synchronization */
|
|
|
|
(FncSync)HalpPCISynchronizeType1,
|
|
|
|
(FncReleaseSync)HalpPCIReleaseSynchronzationType1,
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Read */
|
|
|
|
{
|
|
|
|
(FncConfigIO)HalpPCIReadUlongType1,
|
|
|
|
(FncConfigIO)HalpPCIReadUcharType1,
|
|
|
|
(FncConfigIO)HalpPCIReadUshortType1
|
|
|
|
},
|
2003-02-17 21:24:13 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Write */
|
|
|
|
{
|
|
|
|
(FncConfigIO)HalpPCIWriteUlongType1,
|
|
|
|
(FncConfigIO)HalpPCIWriteUcharType1,
|
|
|
|
(FncConfigIO)HalpPCIWriteUshortType1
|
|
|
|
}
|
|
|
|
};
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Type 2 PCI Bus */
|
|
|
|
PCI_CONFIG_HANDLER PCIConfigHandlerType2 =
|
|
|
|
{
|
|
|
|
/* Synchronization */
|
|
|
|
(FncSync)HalpPCISynchronizeType2,
|
2007-12-13 15:34:02 +00:00
|
|
|
(FncReleaseSync)HalpPCIReleaseSynchronizationType2,
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Read */
|
|
|
|
{
|
|
|
|
(FncConfigIO)HalpPCIReadUlongType2,
|
|
|
|
(FncConfigIO)HalpPCIReadUcharType2,
|
|
|
|
(FncConfigIO)HalpPCIReadUshortType2
|
|
|
|
},
|
2003-02-17 21:24:13 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Write */
|
|
|
|
{
|
|
|
|
(FncConfigIO)HalpPCIWriteUlongType2,
|
|
|
|
(FncConfigIO)HalpPCIWriteUcharType2,
|
|
|
|
(FncConfigIO)HalpPCIWriteUshortType2
|
|
|
|
}
|
|
|
|
};
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
PCIPBUSDATA HalpFakePciBusData =
|
|
|
|
{
|
|
|
|
{
|
|
|
|
PCI_DATA_TAG,
|
|
|
|
PCI_DATA_VERSION,
|
|
|
|
HalpReadPCIConfig,
|
|
|
|
HalpWritePCIConfig,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
2008-12-03 17:31:38 +00:00
|
|
|
{{{0, 0, 0}}},
|
2006-11-27 19:26:31 +00:00
|
|
|
{0, 0, 0, 0}
|
|
|
|
},
|
2008-12-03 17:31:38 +00:00
|
|
|
{{0, 0}},
|
2006-11-27 19:26:31 +00:00
|
|
|
32,
|
|
|
|
};
|
|
|
|
|
|
|
|
BUS_HANDLER HalpFakePciBusHandler =
|
|
|
|
{
|
|
|
|
1,
|
|
|
|
PCIBus,
|
|
|
|
PCIConfiguration,
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&HalpFakePciBusData,
|
|
|
|
0,
|
2009-10-30 22:07:26 +00:00
|
|
|
NULL,
|
2006-11-27 19:26:31 +00:00
|
|
|
{0, 0, 0, 0},
|
2009-10-30 22:07:26 +00:00
|
|
|
(PGETSETBUSDATA)HalpGetPCIData,
|
|
|
|
(PGETSETBUSDATA)HalpSetPCIData,
|
2006-11-27 19:26:31 +00:00
|
|
|
NULL,
|
|
|
|
HalpAssignPCISlotResources,
|
|
|
|
NULL,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
/* TYPE 1 FUNCTIONS **********************************************************/
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpPCISynchronizeType1(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PCI_SLOT_NUMBER Slot,
|
2021-06-06 13:37:47 +00:00
|
|
|
OUT PKIRQL OldIrql,
|
|
|
|
OUT PPCI_TYPE1_CFG_BITS PciCfg1)
|
2001-08-21 20:18:27 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Setup the PCI Configuration Register */
|
|
|
|
PciCfg1->u.AsULONG = 0;
|
|
|
|
PciCfg1->u.bits.BusNumber = BusHandler->BusNumber;
|
|
|
|
PciCfg1->u.bits.DeviceNumber = Slot.u.bits.DeviceNumber;
|
|
|
|
PciCfg1->u.bits.FunctionNumber = Slot.u.bits.FunctionNumber;
|
|
|
|
PciCfg1->u.bits.Enable = TRUE;
|
|
|
|
|
|
|
|
/* Acquire the lock */
|
2021-06-06 13:37:47 +00:00
|
|
|
KeRaiseIrql(HIGH_LEVEL, OldIrql);
|
|
|
|
KeAcquireSpinLockAtDpcLevel(&HalpPCIConfigLock);
|
2001-08-21 20:18:27 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpPCIReleaseSynchronzationType1(IN PBUS_HANDLER BusHandler,
|
2021-06-06 13:37:47 +00:00
|
|
|
IN KIRQL OldIrql)
|
2001-08-21 20:18:27 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
PCI_TYPE1_CFG_BITS PciCfg1;
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Clear the PCI Configuration Register */
|
|
|
|
PciCfg1.u.AsULONG = 0;
|
|
|
|
WRITE_PORT_ULONG(((PPCIPBUSDATA)BusHandler->BusData)->Config.Type1.Address,
|
|
|
|
PciCfg1.u.AsULONG);
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Release the lock */
|
2021-06-06 13:37:47 +00:00
|
|
|
KeReleaseSpinLock(&HalpPCIConfigLock, OldIrql);
|
2001-08-21 20:18:27 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
TYPE1_READ(HalpPCIReadUcharType1, UCHAR)
|
|
|
|
TYPE1_READ(HalpPCIReadUshortType1, USHORT)
|
|
|
|
TYPE1_READ(HalpPCIReadUlongType1, ULONG)
|
|
|
|
TYPE1_WRITE(HalpPCIWriteUcharType1, UCHAR)
|
|
|
|
TYPE1_WRITE(HalpPCIWriteUshortType1, USHORT)
|
|
|
|
TYPE1_WRITE(HalpPCIWriteUlongType1, ULONG)
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* TYPE 2 FUNCTIONS **********************************************************/
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpPCISynchronizeType2(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PCI_SLOT_NUMBER Slot,
|
2021-06-06 13:37:47 +00:00
|
|
|
OUT PKIRQL OldIrql,
|
|
|
|
OUT PPCI_TYPE2_ADDRESS_BITS PciCfg)
|
2001-08-21 20:18:27 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
PCI_TYPE2_CSE_BITS PciCfg2Cse;
|
|
|
|
PPCIPBUSDATA BusData = (PPCIPBUSDATA)BusHandler->BusData;
|
|
|
|
|
|
|
|
/* Setup the configuration register */
|
|
|
|
PciCfg->u.AsUSHORT = 0;
|
|
|
|
PciCfg->u.bits.Agent = (USHORT)Slot.u.bits.DeviceNumber;
|
|
|
|
PciCfg->u.bits.AddressBase = (USHORT)BusData->Config.Type2.Base;
|
|
|
|
|
|
|
|
/* Acquire the lock */
|
2021-06-06 13:37:47 +00:00
|
|
|
KeRaiseIrql(HIGH_LEVEL, OldIrql);
|
|
|
|
KeAcquireSpinLockAtDpcLevel(&HalpPCIConfigLock);
|
2006-11-27 19:26:31 +00:00
|
|
|
|
|
|
|
/* Setup the CSE Register */
|
|
|
|
PciCfg2Cse.u.AsUCHAR = 0;
|
|
|
|
PciCfg2Cse.u.bits.Enable = TRUE;
|
|
|
|
PciCfg2Cse.u.bits.FunctionNumber = (UCHAR)Slot.u.bits.FunctionNumber;
|
|
|
|
PciCfg2Cse.u.bits.Key = -1;
|
|
|
|
|
|
|
|
/* Write the bus number and CSE */
|
|
|
|
WRITE_PORT_UCHAR(BusData->Config.Type2.Forward,
|
|
|
|
(UCHAR)BusHandler->BusNumber);
|
|
|
|
WRITE_PORT_UCHAR(BusData->Config.Type2.CSE, PciCfg2Cse.u.AsUCHAR);
|
2001-08-21 20:18:27 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
2007-12-13 15:34:02 +00:00
|
|
|
HalpPCIReleaseSynchronizationType2(IN PBUS_HANDLER BusHandler,
|
2021-06-06 13:37:47 +00:00
|
|
|
IN KIRQL OldIrql)
|
2001-08-21 20:18:27 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
PCI_TYPE2_CSE_BITS PciCfg2Cse;
|
|
|
|
PPCIPBUSDATA BusData = (PPCIPBUSDATA)BusHandler->BusData;
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Clear CSE and bus number */
|
|
|
|
PciCfg2Cse.u.AsUCHAR = 0;
|
|
|
|
WRITE_PORT_UCHAR(BusData->Config.Type2.CSE, PciCfg2Cse.u.AsUCHAR);
|
|
|
|
WRITE_PORT_UCHAR(BusData->Config.Type2.Forward, 0);
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Release the lock */
|
2021-06-06 13:37:47 +00:00
|
|
|
KeReleaseSpinLock(&HalpPCIConfigLock, OldIrql);
|
2001-08-21 20:18:27 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
TYPE2_READ(HalpPCIReadUcharType2, UCHAR)
|
|
|
|
TYPE2_READ(HalpPCIReadUshortType2, USHORT)
|
|
|
|
TYPE2_READ(HalpPCIReadUlongType2, ULONG)
|
|
|
|
TYPE2_WRITE(HalpPCIWriteUcharType2, UCHAR)
|
|
|
|
TYPE2_WRITE(HalpPCIWriteUshortType2, USHORT)
|
|
|
|
TYPE2_WRITE(HalpPCIWriteUlongType2, ULONG)
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* PCI CONFIGURATION SPACE ***************************************************/
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpPCIConfig(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PCI_SLOT_NUMBER Slot,
|
|
|
|
IN PUCHAR Buffer,
|
|
|
|
IN ULONG Offset,
|
|
|
|
IN ULONG Length,
|
|
|
|
IN FncConfigIO *ConfigIO)
|
2001-08-21 20:18:27 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
KIRQL OldIrql;
|
|
|
|
ULONG i;
|
|
|
|
UCHAR State[20];
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Synchronize the operation */
|
|
|
|
PCIConfigHandler.Synchronize(BusHandler, Slot, &OldIrql, State);
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Loop every increment */
|
|
|
|
while (Length)
|
|
|
|
{
|
|
|
|
/* Find out the type of read/write we need to do */
|
|
|
|
i = PCIDeref[Offset % sizeof(ULONG)][Length % sizeof(ULONG)];
|
|
|
|
|
|
|
|
/* Do the read/write and return the number of bytes */
|
|
|
|
i = ConfigIO[i]((PPCIPBUSDATA)BusHandler->BusData,
|
|
|
|
State,
|
|
|
|
Buffer,
|
|
|
|
Offset);
|
|
|
|
|
|
|
|
/* Increment the buffer position and offset, and decrease the length */
|
|
|
|
Offset += i;
|
|
|
|
Buffer += i;
|
|
|
|
Length -= i;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Release the lock and PCI bus */
|
|
|
|
PCIConfigHandler.ReleaseSynchronzation(BusHandler, OldIrql);
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpReadPCIConfig(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PCI_SLOT_NUMBER Slot,
|
|
|
|
IN PVOID Buffer,
|
|
|
|
IN ULONG Offset,
|
|
|
|
IN ULONG Length)
|
2001-08-21 20:18:27 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Validate the PCI Slot */
|
|
|
|
if (!HalpValidPCISlot(BusHandler, Slot))
|
|
|
|
{
|
|
|
|
/* Fill the buffer with invalid data */
|
|
|
|
RtlFillMemory(Buffer, Length, -1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Send the request */
|
|
|
|
HalpPCIConfig(BusHandler,
|
|
|
|
Slot,
|
|
|
|
Buffer,
|
|
|
|
Offset,
|
|
|
|
Length,
|
|
|
|
PCIConfigHandler.ConfigRead);
|
|
|
|
}
|
2001-08-21 20:18:27 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpWritePCIConfig(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PCI_SLOT_NUMBER Slot,
|
|
|
|
IN PVOID Buffer,
|
|
|
|
IN ULONG Offset,
|
|
|
|
IN ULONG Length)
|
|
|
|
{
|
|
|
|
/* Validate the PCI Slot */
|
|
|
|
if (HalpValidPCISlot(BusHandler, Slot))
|
|
|
|
{
|
|
|
|
/* Send the request */
|
|
|
|
HalpPCIConfig(BusHandler,
|
|
|
|
Slot,
|
|
|
|
Buffer,
|
|
|
|
Offset,
|
|
|
|
Length,
|
|
|
|
PCIConfigHandler.ConfigWrite);
|
|
|
|
}
|
|
|
|
}
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2019-08-19 21:48:49 +00:00
|
|
|
#ifdef SARCH_XBOX
|
2021-09-06 18:42:23 +00:00
|
|
|
static
|
2019-08-19 21:48:49 +00:00
|
|
|
BOOLEAN
|
2021-09-06 18:42:23 +00:00
|
|
|
HalpXboxBlacklistedPCISlot(
|
|
|
|
_In_ ULONG BusNumber,
|
|
|
|
_In_ PCI_SLOT_NUMBER Slot)
|
2019-08-19 21:48:49 +00:00
|
|
|
{
|
|
|
|
/* Trying to get PCI config data from devices 0:0:1 and 0:0:2 will completely
|
|
|
|
* hang the Xbox. Also, the device number doesn't seem to be decoded for the
|
|
|
|
* video card, so it appears to be present on 1:0:0 - 1:31:0.
|
|
|
|
* We hack around these problems by indicating "device not present" for devices
|
|
|
|
* 0:0:1, 0:0:2, 1:1:0, 1:2:0, 1:3:0, ...., 1:31:0 */
|
2021-09-06 18:42:23 +00:00
|
|
|
if ((BusNumber == 0 && Slot.u.bits.DeviceNumber == 0 &&
|
2019-08-19 21:48:49 +00:00
|
|
|
(Slot.u.bits.FunctionNumber == 1 || Slot.u.bits.FunctionNumber == 2)) ||
|
2021-09-06 18:42:23 +00:00
|
|
|
(BusNumber == 1 && Slot.u.bits.DeviceNumber != 0))
|
2019-08-19 21:48:49 +00:00
|
|
|
{
|
2021-09-06 18:42:23 +00:00
|
|
|
DPRINT("Blacklisted PCI slot (%d:%d:%d)\n",
|
|
|
|
BusNumber, Slot.u.bits.DeviceNumber, Slot.u.bits.FunctionNumber);
|
2019-08-19 21:48:49 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
BOOLEAN
|
|
|
|
NTAPI
|
|
|
|
HalpValidPCISlot(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PCI_SLOT_NUMBER Slot)
|
2001-08-21 20:18:27 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
PCI_SLOT_NUMBER MultiSlot;
|
|
|
|
PPCIPBUSDATA BusData = (PPCIPBUSDATA)BusHandler->BusData;
|
|
|
|
UCHAR HeaderType;
|
2011-09-11 00:40:20 +00:00
|
|
|
//ULONG Device;
|
2006-11-27 19:26:31 +00:00
|
|
|
|
|
|
|
/* Simple validation */
|
|
|
|
if (Slot.u.bits.Reserved) return FALSE;
|
|
|
|
if (Slot.u.bits.DeviceNumber >= BusData->MaxDevice) return FALSE;
|
|
|
|
|
2019-08-17 17:03:39 +00:00
|
|
|
#ifdef SARCH_XBOX
|
2021-09-06 18:42:23 +00:00
|
|
|
if (HalpXboxBlacklistedPCISlot(BusHandler->BusNumber, Slot))
|
|
|
|
return FALSE;
|
2019-08-17 17:03:39 +00:00
|
|
|
#endif
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Function 0 doesn't need checking */
|
|
|
|
if (!Slot.u.bits.FunctionNumber) return TRUE;
|
|
|
|
|
|
|
|
/* Functions 0+ need Multi-Function support, so check the slot */
|
2011-09-11 00:40:20 +00:00
|
|
|
//Device = Slot.u.bits.DeviceNumber;
|
2006-11-27 19:26:31 +00:00
|
|
|
MultiSlot = Slot;
|
|
|
|
MultiSlot.u.bits.FunctionNumber = 0;
|
|
|
|
|
|
|
|
/* Send function 0 request to get the header back */
|
|
|
|
HalpReadPCIConfig(BusHandler,
|
|
|
|
MultiSlot,
|
|
|
|
&HeaderType,
|
|
|
|
FIELD_OFFSET(PCI_COMMON_CONFIG, HeaderType),
|
|
|
|
sizeof(UCHAR));
|
|
|
|
|
|
|
|
/* Now make sure the header is multi-function */
|
|
|
|
if (!(HeaderType & PCI_MULTIFUNCTION) || (HeaderType == 0xFF)) return FALSE;
|
|
|
|
return TRUE;
|
2001-08-21 20:18:27 +00:00
|
|
|
}
|
|
|
|
|
2021-09-06 18:42:23 +00:00
|
|
|
CODE_SEG("INIT")
|
|
|
|
ULONG
|
|
|
|
HalpPhase0GetPciDataByOffset(
|
|
|
|
_In_ ULONG Bus,
|
|
|
|
_In_ PCI_SLOT_NUMBER PciSlot,
|
|
|
|
_Out_writes_bytes_all_(Length) PVOID Buffer,
|
|
|
|
_In_ ULONG Offset,
|
|
|
|
_In_ ULONG Length)
|
|
|
|
{
|
|
|
|
ULONG BytesLeft = Length;
|
|
|
|
PUCHAR BufferPtr = Buffer;
|
|
|
|
PCI_TYPE1_CFG_BITS PciCfg;
|
|
|
|
|
|
|
|
#ifdef SARCH_XBOX
|
|
|
|
if (HalpXboxBlacklistedPCISlot(Bus, PciSlot))
|
|
|
|
{
|
|
|
|
RtlFillMemory(Buffer, Length, 0xFF);
|
|
|
|
return Length;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
PciCfg.u.AsULONG = 0;
|
|
|
|
PciCfg.u.bits.BusNumber = Bus;
|
|
|
|
PciCfg.u.bits.DeviceNumber = PciSlot.u.bits.DeviceNumber;
|
|
|
|
PciCfg.u.bits.FunctionNumber = PciSlot.u.bits.FunctionNumber;
|
|
|
|
PciCfg.u.bits.Enable = TRUE;
|
|
|
|
|
|
|
|
while (BytesLeft)
|
|
|
|
{
|
|
|
|
ULONG i;
|
|
|
|
|
|
|
|
PciCfg.u.bits.RegisterNumber = Offset / sizeof(ULONG);
|
|
|
|
WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDRESS_PORT, PciCfg.u.AsULONG);
|
|
|
|
|
|
|
|
i = PCIDeref[Offset % sizeof(ULONG)][BytesLeft % sizeof(ULONG)];
|
|
|
|
switch (i)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
{
|
|
|
|
*(PULONG)BufferPtr = READ_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT);
|
|
|
|
|
|
|
|
/* Number of bytes read */
|
|
|
|
i = sizeof(ULONG);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
*BufferPtr = READ_PORT_UCHAR((PUCHAR)(PCI_TYPE1_DATA_PORT +
|
|
|
|
Offset % sizeof(ULONG)));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
*(PUSHORT)BufferPtr = READ_PORT_USHORT((PUSHORT)(PCI_TYPE1_DATA_PORT +
|
|
|
|
Offset % sizeof(ULONG)));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEFAULT_UNREACHABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
Offset += i;
|
|
|
|
BufferPtr += i;
|
|
|
|
BytesLeft -= i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Length;
|
|
|
|
}
|
|
|
|
|
|
|
|
CODE_SEG("INIT")
|
|
|
|
ULONG
|
|
|
|
HalpPhase0SetPciDataByOffset(
|
|
|
|
_In_ ULONG Bus,
|
|
|
|
_In_ PCI_SLOT_NUMBER PciSlot,
|
|
|
|
_In_reads_bytes_(Length) PVOID Buffer,
|
|
|
|
_In_ ULONG Offset,
|
|
|
|
_In_ ULONG Length)
|
|
|
|
{
|
|
|
|
ULONG BytesLeft = Length;
|
|
|
|
PUCHAR BufferPtr = Buffer;
|
|
|
|
PCI_TYPE1_CFG_BITS PciCfg;
|
|
|
|
|
|
|
|
#ifdef SARCH_XBOX
|
|
|
|
if (HalpXboxBlacklistedPCISlot(Bus, PciSlot))
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
PciCfg.u.AsULONG = 0;
|
|
|
|
PciCfg.u.bits.BusNumber = Bus;
|
|
|
|
PciCfg.u.bits.DeviceNumber = PciSlot.u.bits.DeviceNumber;
|
|
|
|
PciCfg.u.bits.FunctionNumber = PciSlot.u.bits.FunctionNumber;
|
|
|
|
PciCfg.u.bits.Enable = TRUE;
|
|
|
|
|
|
|
|
while (BytesLeft)
|
|
|
|
{
|
|
|
|
ULONG i;
|
|
|
|
|
|
|
|
PciCfg.u.bits.RegisterNumber = Offset / sizeof(ULONG);
|
|
|
|
WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDRESS_PORT, PciCfg.u.AsULONG);
|
|
|
|
|
|
|
|
i = PCIDeref[Offset % sizeof(ULONG)][BytesLeft % sizeof(ULONG)];
|
|
|
|
switch (i)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
{
|
|
|
|
WRITE_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT, *(PULONG)BufferPtr);
|
|
|
|
|
|
|
|
/* Number of bytes written */
|
|
|
|
i = sizeof(ULONG);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
WRITE_PORT_UCHAR((PUCHAR)(PCI_TYPE1_DATA_PORT + Offset % sizeof(ULONG)),
|
|
|
|
*BufferPtr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
WRITE_PORT_USHORT((PUSHORT)(PCI_TYPE1_DATA_PORT + Offset % sizeof(ULONG)),
|
|
|
|
*(PUSHORT)BufferPtr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEFAULT_UNREACHABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
Offset += i;
|
|
|
|
BufferPtr += i;
|
|
|
|
BytesLeft -= i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Length;
|
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* HAL PCI CALLBACKS *********************************************************/
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
ULONG
|
|
|
|
NTAPI
|
|
|
|
HalpGetPCIData(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
2014-09-21 16:00:27 +00:00
|
|
|
IN ULONG SlotNumber,
|
2009-10-30 22:07:26 +00:00
|
|
|
IN PVOID Buffer,
|
2006-11-27 19:26:31 +00:00
|
|
|
IN ULONG Offset,
|
|
|
|
IN ULONG Length)
|
2001-11-07 02:27:33 +00:00
|
|
|
{
|
2014-09-21 16:00:27 +00:00
|
|
|
PCI_SLOT_NUMBER Slot;
|
2006-11-27 19:26:31 +00:00
|
|
|
UCHAR PciBuffer[PCI_COMMON_HDR_LENGTH];
|
|
|
|
PPCI_COMMON_CONFIG PciConfig = (PPCI_COMMON_CONFIG)PciBuffer;
|
|
|
|
ULONG Len = 0;
|
|
|
|
|
2014-09-21 16:00:27 +00:00
|
|
|
Slot.u.AsULONG = SlotNumber;
|
2007-12-14 08:53:56 +00:00
|
|
|
#ifdef SARCH_XBOX
|
2021-09-06 18:42:23 +00:00
|
|
|
if (HalpXboxBlacklistedPCISlot(BusHandler->BusNumber, Slot))
|
2007-12-14 08:53:56 +00:00
|
|
|
{
|
2021-09-06 18:42:23 +00:00
|
|
|
RtlFillMemory(Buffer, Length, 0xFF);
|
|
|
|
return Length;
|
2007-12-14 08:53:56 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Normalize the length */
|
|
|
|
if (Length > sizeof(PCI_COMMON_CONFIG)) Length = sizeof(PCI_COMMON_CONFIG);
|
|
|
|
|
|
|
|
/* Check if this is a vendor-specific read */
|
|
|
|
if (Offset >= PCI_COMMON_HDR_LENGTH)
|
|
|
|
{
|
|
|
|
/* Read the header */
|
|
|
|
HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, sizeof(ULONG));
|
|
|
|
|
|
|
|
/* Make sure the vendor is valid */
|
|
|
|
if (PciConfig->VendorID == PCI_INVALID_VENDORID) return 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Read the entire header */
|
|
|
|
Len = PCI_COMMON_HDR_LENGTH;
|
|
|
|
HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, Len);
|
|
|
|
|
|
|
|
/* Validate the vendor ID */
|
|
|
|
if (PciConfig->VendorID == PCI_INVALID_VENDORID)
|
|
|
|
{
|
2010-05-12 09:45:43 +00:00
|
|
|
/* It's invalid, but we want to return this much */
|
|
|
|
Len = sizeof(USHORT);
|
2006-11-27 19:26:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Now check if there's space left */
|
|
|
|
if (Len < Offset) return 0;
|
|
|
|
|
|
|
|
/* There is, so return what's after the offset and normalize */
|
|
|
|
Len -= Offset;
|
|
|
|
if (Len > Length) Len = Length;
|
|
|
|
|
|
|
|
/* Copy the data into the caller's buffer */
|
|
|
|
RtlMoveMemory(Buffer, PciBuffer + Offset, Len);
|
|
|
|
|
|
|
|
/* Update buffer and offset, decrement total length */
|
|
|
|
Offset += Len;
|
2009-10-30 22:07:26 +00:00
|
|
|
Buffer = (PVOID)((ULONG_PTR)Buffer + Len);
|
2006-11-27 19:26:31 +00:00
|
|
|
Length -= Len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now we still have something to copy */
|
|
|
|
if (Length)
|
|
|
|
{
|
|
|
|
/* Check if it's vendor-specific data */
|
|
|
|
if (Offset >= PCI_COMMON_HDR_LENGTH)
|
|
|
|
{
|
|
|
|
/* Read it now */
|
|
|
|
HalpReadPCIConfig(BusHandler, Slot, Buffer, Offset, Length);
|
|
|
|
Len += Length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Update the total length read */
|
|
|
|
return Len;
|
2001-11-07 02:27:33 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
ULONG
|
|
|
|
NTAPI
|
|
|
|
HalpSetPCIData(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
2014-09-21 16:00:27 +00:00
|
|
|
IN ULONG SlotNumber,
|
2009-10-30 22:07:26 +00:00
|
|
|
IN PVOID Buffer,
|
2006-11-27 19:26:31 +00:00
|
|
|
IN ULONG Offset,
|
|
|
|
IN ULONG Length)
|
2002-12-09 19:45:55 +00:00
|
|
|
{
|
2014-09-21 16:00:27 +00:00
|
|
|
PCI_SLOT_NUMBER Slot;
|
2006-11-27 19:26:31 +00:00
|
|
|
UCHAR PciBuffer[PCI_COMMON_HDR_LENGTH];
|
|
|
|
PPCI_COMMON_CONFIG PciConfig = (PPCI_COMMON_CONFIG)PciBuffer;
|
|
|
|
ULONG Len = 0;
|
|
|
|
|
2014-09-21 16:00:27 +00:00
|
|
|
Slot.u.AsULONG = SlotNumber;
|
2007-12-14 08:53:56 +00:00
|
|
|
#ifdef SARCH_XBOX
|
2021-09-06 18:42:23 +00:00
|
|
|
if (HalpXboxBlacklistedPCISlot(BusHandler->BusNumber, Slot))
|
|
|
|
return 0;
|
2007-12-14 08:53:56 +00:00
|
|
|
#endif
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Normalize the length */
|
|
|
|
if (Length > sizeof(PCI_COMMON_CONFIG)) Length = sizeof(PCI_COMMON_CONFIG);
|
|
|
|
|
|
|
|
/* Check if this is a vendor-specific read */
|
|
|
|
if (Offset >= PCI_COMMON_HDR_LENGTH)
|
|
|
|
{
|
|
|
|
/* Read the header */
|
|
|
|
HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, sizeof(ULONG));
|
2002-12-09 19:45:55 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Make sure the vendor is valid */
|
|
|
|
if (PciConfig->VendorID == PCI_INVALID_VENDORID) return 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Read the entire header and validate the vendor ID */
|
|
|
|
Len = PCI_COMMON_HDR_LENGTH;
|
|
|
|
HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, Len);
|
|
|
|
if (PciConfig->VendorID == PCI_INVALID_VENDORID) return 0;
|
|
|
|
|
|
|
|
/* Return what's after the offset and normalize */
|
|
|
|
Len -= Offset;
|
|
|
|
if (Len > Length) Len = Length;
|
2002-12-09 19:45:55 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Copy the specific caller data */
|
|
|
|
RtlMoveMemory(PciBuffer + Offset, Buffer, Len);
|
|
|
|
|
|
|
|
/* Write the actual configuration data */
|
|
|
|
HalpWritePCIConfig(BusHandler, Slot, PciBuffer + Offset, Offset, Len);
|
|
|
|
|
|
|
|
/* Update buffer and offset, decrement total length */
|
|
|
|
Offset += Len;
|
2009-10-30 22:07:26 +00:00
|
|
|
Buffer = (PVOID)((ULONG_PTR)Buffer + Len);
|
2006-11-27 19:26:31 +00:00
|
|
|
Length -= Len;
|
|
|
|
}
|
2002-12-09 19:45:55 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Now we still have something to copy */
|
|
|
|
if (Length)
|
|
|
|
{
|
|
|
|
/* Check if it's vendor-specific data */
|
|
|
|
if (Offset >= PCI_COMMON_HDR_LENGTH)
|
|
|
|
{
|
|
|
|
/* Read it now */
|
|
|
|
HalpWritePCIConfig(BusHandler, Slot, Buffer, Offset, Length);
|
|
|
|
Len += Length;
|
|
|
|
}
|
|
|
|
}
|
2002-12-09 19:45:55 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Update the total length read */
|
|
|
|
return Len;
|
2002-12-09 19:45:55 +00:00
|
|
|
}
|
2021-06-03 09:19:30 +00:00
|
|
|
#ifndef _MINIHAL_
|
2010-06-07 03:19:20 +00:00
|
|
|
ULONG
|
|
|
|
NTAPI
|
|
|
|
HalpGetPCIIntOnISABus(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
|
|
|
IN ULONG BusInterruptLevel,
|
|
|
|
IN ULONG BusInterruptVector,
|
|
|
|
OUT PKIRQL Irql,
|
|
|
|
OUT PKAFFINITY Affinity)
|
|
|
|
{
|
2010-06-07 20:37:45 +00:00
|
|
|
/* Validate the level first */
|
|
|
|
if (BusInterruptLevel < 1) return 0;
|
|
|
|
|
|
|
|
/* PCI has its IRQs on top of ISA IRQs, so pass it on to the ISA handler */
|
|
|
|
return HalGetInterruptVector(Isa,
|
|
|
|
0,
|
|
|
|
BusInterruptLevel,
|
|
|
|
0,
|
|
|
|
Irql,
|
|
|
|
Affinity);
|
2010-06-07 03:19:20 +00:00
|
|
|
}
|
2021-06-03 09:19:30 +00:00
|
|
|
#endif // _MINIHAL_
|
2010-06-07 03:19:20 +00:00
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpPCIPin2ISALine(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
|
|
|
IN PCI_SLOT_NUMBER SlotNumber,
|
|
|
|
IN PPCI_COMMON_CONFIG PciData)
|
|
|
|
{
|
2013-01-10 01:45:22 +00:00
|
|
|
UNIMPLEMENTED_DBGBREAK();
|
2010-06-07 03:19:20 +00:00
|
|
|
}
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-06-07 03:19:20 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpPCIISALine2Pin(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
|
|
|
IN PCI_SLOT_NUMBER SlotNumber,
|
|
|
|
IN PPCI_COMMON_CONFIG PciNewData,
|
|
|
|
IN PPCI_COMMON_CONFIG PciOldData)
|
|
|
|
{
|
2013-01-10 01:45:22 +00:00
|
|
|
UNIMPLEMENTED_DBGBREAK();
|
2010-06-07 03:19:20 +00:00
|
|
|
}
|
|
|
|
|
2021-06-03 09:19:30 +00:00
|
|
|
#ifndef _MINIHAL_
|
2010-06-07 03:19:20 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
HalpGetISAFixedPCIIrq(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
|
|
|
IN PCI_SLOT_NUMBER PciSlot,
|
|
|
|
OUT PSUPPORTED_RANGE *Range)
|
|
|
|
{
|
2010-09-21 17:07:40 +00:00
|
|
|
PCI_COMMON_HEADER PciData;
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Read PCI configuration data */
|
|
|
|
HalGetBusData(PCIConfiguration,
|
|
|
|
BusHandler->BusNumber,
|
|
|
|
PciSlot.u.AsULONG,
|
|
|
|
&PciData,
|
|
|
|
PCI_COMMON_HDR_LENGTH);
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Make sure it's a real device */
|
|
|
|
if (PciData.VendorID == PCI_INVALID_VENDORID) return STATUS_UNSUCCESSFUL;
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Allocate the supported range structure */
|
2011-05-26 22:22:37 +00:00
|
|
|
*Range = ExAllocatePoolWithTag(PagedPool, sizeof(SUPPORTED_RANGE), TAG_HAL);
|
2010-09-21 17:07:40 +00:00
|
|
|
if (!*Range) return STATUS_INSUFFICIENT_RESOURCES;
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Set it up */
|
|
|
|
RtlZeroMemory(*Range, sizeof(SUPPORTED_RANGE));
|
|
|
|
(*Range)->Base = 1;
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* If the PCI device has no IRQ, nothing to do */
|
|
|
|
if (!PciData.u.type0.InterruptPin) return STATUS_SUCCESS;
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* FIXME: The PCI IRQ Routing Miniport should be called */
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Also if the INT# seems bogus, nothing to do either */
|
|
|
|
if ((PciData.u.type0.InterruptLine == 0) ||
|
|
|
|
(PciData.u.type0.InterruptLine == 255))
|
|
|
|
{
|
|
|
|
/* Fake success */
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Otherwise, the INT# should be valid, return it to the caller */
|
|
|
|
(*Range)->Base = PciData.u.type0.InterruptLine;
|
|
|
|
(*Range)->Limit = PciData.u.type0.InterruptLine;
|
2010-06-07 03:19:20 +00:00
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
2021-06-03 09:19:30 +00:00
|
|
|
#endif // _MINIHAL_
|
2010-04-01 19:42:07 +00:00
|
|
|
|
2008-11-29 23:16:39 +00:00
|
|
|
static ULONG NTAPI
|
2007-06-29 17:20:26 +00:00
|
|
|
PciSize(ULONG Base, ULONG Mask)
|
|
|
|
{
|
|
|
|
ULONG Size = Mask & Base; /* Find the significant bits */
|
|
|
|
Size = Size & ~(Size - 1); /* Get the lowest of them to find the decode size */
|
|
|
|
return Size;
|
|
|
|
}
|
|
|
|
|
2010-06-07 03:19:20 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
HalpAdjustPCIResourceList(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
|
|
|
IN OUT PIO_RESOURCE_REQUIREMENTS_LIST *pResourceList)
|
|
|
|
{
|
2010-09-21 17:07:40 +00:00
|
|
|
PPCIPBUSDATA BusData;
|
|
|
|
PCI_SLOT_NUMBER SlotNumber;
|
|
|
|
PSUPPORTED_RANGE Interrupt;
|
|
|
|
NTSTATUS Status;
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Get PCI bus data */
|
|
|
|
BusData = BusHandler->BusData;
|
|
|
|
SlotNumber.u.AsULONG = (*pResourceList)->SlotNumber;
|
2012-03-28 12:15:54 +00:00
|
|
|
|
2010-09-21 17:07:40 +00:00
|
|
|
/* Get the IRQ supported range */
|
|
|
|
Status = BusData->GetIrqRange(BusHandler, RootHandler, SlotNumber, &Interrupt);
|
|
|
|
if (!NT_SUCCESS(Status)) return Status;
|
|
|
|
#ifndef _MINIHAL_
|
|
|
|
/* Handle the /PCILOCK feature */
|
|
|
|
if (HalpPciLockSettings)
|
|
|
|
{
|
|
|
|
/* /PCILOCK is not yet supported */
|
2013-01-10 01:45:22 +00:00
|
|
|
UNIMPLEMENTED_DBGBREAK("/PCILOCK boot switch is not yet supported.");
|
2010-09-21 17:07:40 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* Now create the correct resource list based on the supported bus ranges */
|
|
|
|
#if 0
|
|
|
|
Status = HaliAdjustResourceListRange(BusHandler->BusAddresses,
|
|
|
|
Interrupt,
|
|
|
|
pResourceList);
|
|
|
|
#else
|
|
|
|
DPRINT1("HAL: No PCI Resource Adjustment done! Hardware may malfunction\n");
|
|
|
|
Status = STATUS_SUCCESS;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Return to caller */
|
|
|
|
ExFreePool(Interrupt);
|
|
|
|
return Status;
|
2010-06-07 03:19:20 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
HalpAssignPCISlotResources(IN PBUS_HANDLER BusHandler,
|
|
|
|
IN PBUS_HANDLER RootHandler,
|
|
|
|
IN PUNICODE_STRING RegistryPath,
|
|
|
|
IN PUNICODE_STRING DriverClassName OPTIONAL,
|
|
|
|
IN PDRIVER_OBJECT DriverObject,
|
|
|
|
IN PDEVICE_OBJECT DeviceObject OPTIONAL,
|
|
|
|
IN ULONG Slot,
|
2007-06-29 17:20:26 +00:00
|
|
|
IN OUT PCM_RESOURCE_LIST *AllocatedResources)
|
2003-02-17 21:24:13 +00:00
|
|
|
{
|
2007-06-29 17:20:26 +00:00
|
|
|
PCI_COMMON_CONFIG PciConfig;
|
|
|
|
SIZE_T Address;
|
2012-03-28 12:15:54 +00:00
|
|
|
ULONG ResourceCount;
|
2007-06-29 17:20:26 +00:00
|
|
|
ULONG Size[PCI_TYPE0_ADDRESSES];
|
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
UCHAR Offset;
|
|
|
|
PCM_PARTIAL_RESOURCE_DESCRIPTOR Descriptor;
|
|
|
|
PCI_SLOT_NUMBER SlotNumber;
|
|
|
|
ULONG WriteBuffer;
|
2010-06-07 21:26:09 +00:00
|
|
|
DPRINT1("WARNING: PCI Slot Resource Assignment is FOOBAR\n");
|
2007-06-29 17:20:26 +00:00
|
|
|
|
|
|
|
/* FIXME: Should handle 64-bit addresses */
|
|
|
|
|
|
|
|
/* Read configuration data */
|
|
|
|
SlotNumber.u.AsULONG = Slot;
|
|
|
|
HalpReadPCIConfig(BusHandler, SlotNumber, &PciConfig, 0, PCI_COMMON_HDR_LENGTH);
|
|
|
|
|
|
|
|
/* Check if we read it correctly */
|
|
|
|
if (PciConfig.VendorID == PCI_INVALID_VENDORID)
|
|
|
|
return STATUS_NO_SUCH_DEVICE;
|
|
|
|
|
|
|
|
/* Read the PCI configuration space for the device and store base address and
|
|
|
|
size information in temporary storage. Count the number of valid base addresses */
|
|
|
|
ResourceCount = 0;
|
|
|
|
for (Address = 0; Address < PCI_TYPE0_ADDRESSES; Address++)
|
|
|
|
{
|
|
|
|
if (0xffffffff == PciConfig.u.type0.BaseAddresses[Address])
|
|
|
|
PciConfig.u.type0.BaseAddresses[Address] = 0;
|
|
|
|
|
|
|
|
/* Memory resource */
|
|
|
|
if (0 != PciConfig.u.type0.BaseAddresses[Address])
|
|
|
|
{
|
|
|
|
ResourceCount++;
|
|
|
|
|
2009-09-26 13:41:57 +00:00
|
|
|
Offset = (UCHAR)FIELD_OFFSET(PCI_COMMON_CONFIG, u.type0.BaseAddresses[Address]);
|
2007-06-29 17:20:26 +00:00
|
|
|
|
|
|
|
/* Write 0xFFFFFFFF there */
|
|
|
|
WriteBuffer = 0xffffffff;
|
|
|
|
HalpWritePCIConfig(BusHandler, SlotNumber, &WriteBuffer, Offset, sizeof(ULONG));
|
|
|
|
|
|
|
|
/* Read that figure back from the config space */
|
|
|
|
HalpReadPCIConfig(BusHandler, SlotNumber, &Size[Address], Offset, sizeof(ULONG));
|
|
|
|
|
|
|
|
/* Write back initial value */
|
|
|
|
HalpWritePCIConfig(BusHandler, SlotNumber, &PciConfig.u.type0.BaseAddresses[Address], Offset, sizeof(ULONG));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Interrupt resource */
|
2010-10-12 20:02:15 +00:00
|
|
|
if (0 != PciConfig.u.type0.InterruptPin &&
|
|
|
|
0 != PciConfig.u.type0.InterruptLine &&
|
|
|
|
0xFF != PciConfig.u.type0.InterruptLine)
|
2007-06-29 17:20:26 +00:00
|
|
|
ResourceCount++;
|
|
|
|
|
|
|
|
/* Allocate output buffer and initialize */
|
|
|
|
*AllocatedResources = ExAllocatePoolWithTag(
|
|
|
|
PagedPool,
|
|
|
|
sizeof(CM_RESOURCE_LIST) +
|
|
|
|
(ResourceCount - 1) * sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR),
|
2011-05-26 22:22:37 +00:00
|
|
|
TAG_HAL);
|
2007-06-29 17:20:26 +00:00
|
|
|
|
|
|
|
if (NULL == *AllocatedResources)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
(*AllocatedResources)->Count = 1;
|
|
|
|
(*AllocatedResources)->List[0].InterfaceType = PCIBus;
|
|
|
|
(*AllocatedResources)->List[0].BusNumber = BusHandler->BusNumber;
|
|
|
|
(*AllocatedResources)->List[0].PartialResourceList.Version = 1;
|
|
|
|
(*AllocatedResources)->List[0].PartialResourceList.Revision = 1;
|
|
|
|
(*AllocatedResources)->List[0].PartialResourceList.Count = ResourceCount;
|
|
|
|
Descriptor = (*AllocatedResources)->List[0].PartialResourceList.PartialDescriptors;
|
|
|
|
|
|
|
|
/* Store configuration information */
|
|
|
|
for (Address = 0; Address < PCI_TYPE0_ADDRESSES; Address++)
|
|
|
|
{
|
|
|
|
if (0 != PciConfig.u.type0.BaseAddresses[Address])
|
|
|
|
{
|
2010-02-18 17:00:09 +00:00
|
|
|
if (PCI_ADDRESS_MEMORY_SPACE ==
|
2007-06-29 17:20:26 +00:00
|
|
|
(PciConfig.u.type0.BaseAddresses[Address] & 0x1))
|
|
|
|
{
|
|
|
|
Descriptor->Type = CmResourceTypeMemory;
|
|
|
|
Descriptor->ShareDisposition = CmResourceShareDeviceExclusive; /* FIXME I have no idea... */
|
|
|
|
Descriptor->Flags = CM_RESOURCE_MEMORY_READ_WRITE; /* FIXME Just a guess */
|
|
|
|
Descriptor->u.Memory.Start.QuadPart = (PciConfig.u.type0.BaseAddresses[Address] & PCI_ADDRESS_MEMORY_ADDRESS_MASK);
|
|
|
|
Descriptor->u.Memory.Length = PciSize(Size[Address], PCI_ADDRESS_MEMORY_ADDRESS_MASK);
|
|
|
|
}
|
|
|
|
else if (PCI_ADDRESS_IO_SPACE ==
|
|
|
|
(PciConfig.u.type0.BaseAddresses[Address] & 0x1))
|
|
|
|
{
|
|
|
|
Descriptor->Type = CmResourceTypePort;
|
|
|
|
Descriptor->ShareDisposition = CmResourceShareDeviceExclusive; /* FIXME I have no idea... */
|
|
|
|
Descriptor->Flags = CM_RESOURCE_PORT_IO; /* FIXME Just a guess */
|
|
|
|
Descriptor->u.Port.Start.QuadPart = PciConfig.u.type0.BaseAddresses[Address] &= PCI_ADDRESS_IO_ADDRESS_MASK;
|
|
|
|
Descriptor->u.Port.Length = PciSize(Size[Address], PCI_ADDRESS_IO_ADDRESS_MASK & 0xffff);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ASSERT(FALSE);
|
|
|
|
return STATUS_UNSUCCESSFUL;
|
|
|
|
}
|
|
|
|
Descriptor++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-12 20:02:15 +00:00
|
|
|
if (0 != PciConfig.u.type0.InterruptPin &&
|
|
|
|
0 != PciConfig.u.type0.InterruptLine &&
|
|
|
|
0xFF != PciConfig.u.type0.InterruptLine)
|
2007-06-29 17:20:26 +00:00
|
|
|
{
|
|
|
|
Descriptor->Type = CmResourceTypeInterrupt;
|
|
|
|
Descriptor->ShareDisposition = CmResourceShareShared; /* FIXME Just a guess */
|
|
|
|
Descriptor->Flags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE; /* FIXME Just a guess */
|
|
|
|
Descriptor->u.Interrupt.Level = PciConfig.u.type0.InterruptLine;
|
|
|
|
Descriptor->u.Interrupt.Vector = PciConfig.u.type0.InterruptLine;
|
|
|
|
Descriptor->u.Interrupt.Affinity = 0xFFFFFFFF;
|
|
|
|
|
|
|
|
Descriptor++;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(Descriptor == (*AllocatedResources)->List[0].PartialResourceList.PartialDescriptors + ResourceCount);
|
|
|
|
|
|
|
|
/* FIXME: Should store the resources in the registry resource map */
|
|
|
|
|
|
|
|
return Status;
|
2003-02-17 21:24:13 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
ULONG
|
|
|
|
NTAPI
|
|
|
|
HaliPciInterfaceReadConfig(IN PBUS_HANDLER RootBusHandler,
|
|
|
|
IN ULONG BusNumber,
|
|
|
|
IN PCI_SLOT_NUMBER SlotNumber,
|
|
|
|
IN PVOID Buffer,
|
|
|
|
IN ULONG Offset,
|
|
|
|
IN ULONG Length)
|
2003-02-17 21:24:13 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
BUS_HANDLER BusHandler;
|
2003-11-05 22:39:01 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Setup fake PCI Bus handler */
|
|
|
|
RtlCopyMemory(&BusHandler, &HalpFakePciBusHandler, sizeof(BUS_HANDLER));
|
|
|
|
BusHandler.BusNumber = BusNumber;
|
|
|
|
|
|
|
|
/* Read configuration data */
|
|
|
|
HalpReadPCIConfig(&BusHandler, SlotNumber, Buffer, Offset, Length);
|
|
|
|
|
|
|
|
/* Return length */
|
|
|
|
return Length;
|
|
|
|
}
|
|
|
|
|
2020-10-06 19:44:01 +00:00
|
|
|
CODE_SEG("INIT")
|
2006-11-27 19:26:31 +00:00
|
|
|
PPCI_REGISTRY_INFO_INTERNAL
|
|
|
|
NTAPI
|
|
|
|
HalpQueryPciRegistryInfo(VOID)
|
|
|
|
{
|
2010-03-03 21:56:52 +00:00
|
|
|
#ifndef _MINIHAL_
|
2006-11-27 19:26:31 +00:00
|
|
|
WCHAR NameBuffer[8];
|
2009-11-02 18:05:07 +00:00
|
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
2006-11-27 19:26:31 +00:00
|
|
|
UNICODE_STRING KeyName, ConfigName, IdentName;
|
2009-11-02 18:05:07 +00:00
|
|
|
HANDLE KeyHandle, BusKeyHandle, CardListHandle;
|
2006-11-27 19:26:31 +00:00
|
|
|
NTSTATUS Status;
|
2009-11-02 18:05:07 +00:00
|
|
|
UCHAR KeyBuffer[sizeof(CM_FULL_RESOURCE_DESCRIPTOR) + 100];
|
2006-11-27 19:26:31 +00:00
|
|
|
PKEY_VALUE_FULL_INFORMATION ValueInfo = (PVOID)KeyBuffer;
|
2009-11-02 18:05:07 +00:00
|
|
|
UCHAR PartialKeyBuffer[sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
|
|
|
|
sizeof(PCI_CARD_DESCRIPTOR)];
|
|
|
|
PKEY_VALUE_PARTIAL_INFORMATION PartialValueInfo = (PVOID)PartialKeyBuffer;
|
|
|
|
KEY_FULL_INFORMATION KeyInformation;
|
2006-11-27 19:26:31 +00:00
|
|
|
ULONG ResultLength;
|
|
|
|
PWSTR Tag;
|
2009-11-02 18:05:07 +00:00
|
|
|
ULONG i, ElementCount;
|
2006-11-27 19:26:31 +00:00
|
|
|
PCM_FULL_RESOURCE_DESCRIPTOR FullDescriptor;
|
|
|
|
PCM_PARTIAL_RESOURCE_DESCRIPTOR PartialDescriptor;
|
|
|
|
PPCI_REGISTRY_INFO PciRegInfo;
|
|
|
|
PPCI_REGISTRY_INFO_INTERNAL PciRegistryInfo;
|
2009-11-02 18:05:07 +00:00
|
|
|
PPCI_CARD_DESCRIPTOR CardDescriptor;
|
2006-11-27 19:26:31 +00:00
|
|
|
|
|
|
|
/* Setup the object attributes for the key */
|
|
|
|
RtlInitUnicodeString(&KeyName,
|
|
|
|
L"\\Registry\\Machine\\Hardware\\Description\\"
|
|
|
|
L"System\\MultiFunctionAdapter");
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
|
|
&KeyName,
|
|
|
|
OBJ_CASE_INSENSITIVE,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
/* Open the key */
|
|
|
|
Status = ZwOpenKey(&KeyHandle, KEY_READ, &ObjectAttributes);
|
|
|
|
if (!NT_SUCCESS(Status)) return NULL;
|
|
|
|
|
|
|
|
/* Setup the receiving string */
|
|
|
|
KeyName.Buffer = NameBuffer;
|
|
|
|
KeyName.MaximumLength = sizeof(NameBuffer);
|
|
|
|
|
|
|
|
/* Setup the configuration and identifier key names */
|
2009-11-02 18:05:07 +00:00
|
|
|
RtlInitUnicodeString(&ConfigName, L"Configuration Data");
|
2006-11-27 19:26:31 +00:00
|
|
|
RtlInitUnicodeString(&IdentName, L"Identifier");
|
|
|
|
|
|
|
|
/* Keep looping for each ID */
|
|
|
|
for (i = 0; TRUE; i++)
|
2003-11-05 22:39:01 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Setup the key name */
|
|
|
|
RtlIntegerToUnicodeString(i, 10, &KeyName);
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
|
|
&KeyName,
|
|
|
|
OBJ_CASE_INSENSITIVE,
|
|
|
|
KeyHandle,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
/* Open it */
|
|
|
|
Status = ZwOpenKey(&BusKeyHandle, KEY_READ, &ObjectAttributes);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* None left, fail */
|
|
|
|
ZwClose(KeyHandle);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Read the registry data */
|
|
|
|
Status = ZwQueryValueKey(BusKeyHandle,
|
|
|
|
&IdentName,
|
|
|
|
KeyValueFullInformation,
|
|
|
|
ValueInfo,
|
|
|
|
sizeof(KeyBuffer),
|
|
|
|
&ResultLength);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Failed, try the next one */
|
|
|
|
ZwClose(BusKeyHandle);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the PCI Tag and validate it */
|
|
|
|
Tag = (PWSTR)((ULONG_PTR)ValueInfo + ValueInfo->DataOffset);
|
|
|
|
if ((Tag[0] != L'P') ||
|
|
|
|
(Tag[1] != L'C') ||
|
|
|
|
(Tag[2] != L'I') ||
|
|
|
|
(Tag[3]))
|
|
|
|
{
|
|
|
|
/* Not a valid PCI entry, skip it */
|
|
|
|
ZwClose(BusKeyHandle);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now read our PCI structure */
|
|
|
|
Status = ZwQueryValueKey(BusKeyHandle,
|
|
|
|
&ConfigName,
|
|
|
|
KeyValueFullInformation,
|
|
|
|
ValueInfo,
|
|
|
|
sizeof(KeyBuffer),
|
|
|
|
&ResultLength);
|
|
|
|
ZwClose(BusKeyHandle);
|
|
|
|
if (!NT_SUCCESS(Status)) continue;
|
|
|
|
|
|
|
|
/* We read it OK! Get the actual resource descriptors */
|
|
|
|
FullDescriptor = (PCM_FULL_RESOURCE_DESCRIPTOR)
|
|
|
|
((ULONG_PTR)ValueInfo + ValueInfo->DataOffset);
|
|
|
|
PartialDescriptor = (PCM_PARTIAL_RESOURCE_DESCRIPTOR)
|
|
|
|
((ULONG_PTR)FullDescriptor->
|
|
|
|
PartialResourceList.PartialDescriptors);
|
|
|
|
|
|
|
|
/* Check if this is our PCI Registry Information */
|
|
|
|
if (PartialDescriptor->Type == CmResourceTypeDeviceSpecific)
|
|
|
|
{
|
2009-11-02 18:05:07 +00:00
|
|
|
/* It is, stop searching */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2006-11-27 19:26:31 +00:00
|
|
|
|
2009-11-02 18:05:07 +00:00
|
|
|
/* Close the key */
|
|
|
|
ZwClose(KeyHandle);
|
2006-11-27 19:26:31 +00:00
|
|
|
|
2009-11-02 18:05:07 +00:00
|
|
|
/* Save the PCI information for later */
|
|
|
|
PciRegInfo = (PPCI_REGISTRY_INFO)(PartialDescriptor + 1);
|
2006-11-27 19:26:31 +00:00
|
|
|
|
2009-11-02 18:05:07 +00:00
|
|
|
/* Assume no Card List entries */
|
|
|
|
ElementCount = 0;
|
2006-11-27 19:26:31 +00:00
|
|
|
|
2009-11-02 18:05:07 +00:00
|
|
|
/* Set up for checking the PCI Card List key */
|
|
|
|
RtlInitUnicodeString(&KeyName,
|
|
|
|
L"\\Registry\\Machine\\System\\CurrentControlSet\\"
|
|
|
|
L"Control\\PnP\\PCI\\CardList");
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
|
|
&KeyName,
|
|
|
|
OBJ_CASE_INSENSITIVE,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
/* Attempt to open it */
|
|
|
|
Status = ZwOpenKey(&CardListHandle, KEY_READ, &ObjectAttributes);
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* It exists, so let's query it */
|
|
|
|
Status = ZwQueryKey(CardListHandle,
|
|
|
|
KeyFullInformation,
|
|
|
|
&KeyInformation,
|
|
|
|
sizeof(KEY_FULL_INFORMATION),
|
|
|
|
&ResultLength);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Failed to query, so no info */
|
|
|
|
PciRegistryInfo = NULL;
|
2006-11-27 19:26:31 +00:00
|
|
|
}
|
2009-11-02 18:05:07 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Allocate the full structure */
|
|
|
|
PciRegistryInfo =
|
|
|
|
ExAllocatePoolWithTag(NonPagedPool,
|
|
|
|
sizeof(PCI_REGISTRY_INFO_INTERNAL) +
|
|
|
|
(KeyInformation.Values *
|
|
|
|
sizeof(PCI_CARD_DESCRIPTOR)),
|
2011-05-26 22:22:37 +00:00
|
|
|
TAG_HAL);
|
2009-11-02 18:05:07 +00:00
|
|
|
if (PciRegistryInfo)
|
|
|
|
{
|
|
|
|
/* Get the first card descriptor entry */
|
|
|
|
CardDescriptor = (PPCI_CARD_DESCRIPTOR)(PciRegistryInfo + 1);
|
|
|
|
|
|
|
|
/* Loop all the values */
|
|
|
|
for (i = 0; i < KeyInformation.Values; i++)
|
|
|
|
{
|
|
|
|
/* Attempt to get the value */
|
|
|
|
Status = ZwEnumerateValueKey(CardListHandle,
|
|
|
|
i,
|
|
|
|
KeyValuePartialInformation,
|
|
|
|
PartialValueInfo,
|
|
|
|
sizeof(PartialKeyBuffer),
|
|
|
|
&ResultLength);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Something went wrong, stop the search */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure it is correctly sized */
|
|
|
|
if (PartialValueInfo->DataLength == sizeof(PCI_CARD_DESCRIPTOR))
|
|
|
|
{
|
|
|
|
/* Sure is, copy it over */
|
|
|
|
*CardDescriptor = *(PPCI_CARD_DESCRIPTOR)
|
|
|
|
PartialValueInfo->Data;
|
|
|
|
|
|
|
|
/* One more Card List entry */
|
|
|
|
ElementCount++;
|
|
|
|
|
|
|
|
/* Move to the next descriptor */
|
|
|
|
CardDescriptor = (CardDescriptor + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Close the Card List key */
|
|
|
|
ZwClose(CardListHandle);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* No key, no Card List */
|
|
|
|
PciRegistryInfo = NULL;
|
2003-11-05 22:39:01 +00:00
|
|
|
}
|
2009-11-02 18:05:07 +00:00
|
|
|
|
|
|
|
/* Check if we failed to get the full structure */
|
|
|
|
if (!PciRegistryInfo)
|
|
|
|
{
|
|
|
|
/* Just allocate the basic structure then */
|
|
|
|
PciRegistryInfo = ExAllocatePoolWithTag(NonPagedPool,
|
|
|
|
sizeof(PCI_REGISTRY_INFO_INTERNAL),
|
2011-05-26 22:22:37 +00:00
|
|
|
TAG_HAL);
|
2009-11-02 18:05:07 +00:00
|
|
|
if (!PciRegistryInfo) return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Save the info we got */
|
|
|
|
PciRegistryInfo->MajorRevision = PciRegInfo->MajorRevision;
|
|
|
|
PciRegistryInfo->MinorRevision = PciRegInfo->MinorRevision;
|
|
|
|
PciRegistryInfo->NoBuses = PciRegInfo->NoBuses;
|
|
|
|
PciRegistryInfo->HardwareMechanism = PciRegInfo->HardwareMechanism;
|
|
|
|
PciRegistryInfo->ElementCount = ElementCount;
|
|
|
|
|
|
|
|
/* Return it */
|
|
|
|
return PciRegistryInfo;
|
2010-03-03 21:56:52 +00:00
|
|
|
#else
|
|
|
|
return NULL;
|
|
|
|
#endif
|
2006-11-27 19:26:31 +00:00
|
|
|
}
|
2003-11-05 22:39:01 +00:00
|
|
|
|
2020-10-06 19:44:01 +00:00
|
|
|
CODE_SEG("INIT")
|
2006-11-27 19:26:31 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
HalpInitializePciStubs(VOID)
|
|
|
|
{
|
|
|
|
PPCI_REGISTRY_INFO_INTERNAL PciRegistryInfo;
|
|
|
|
UCHAR PciType;
|
|
|
|
PPCIPBUSDATA BusData = (PPCIPBUSDATA)HalpFakePciBusHandler.BusData;
|
|
|
|
ULONG i;
|
|
|
|
PCI_SLOT_NUMBER j;
|
|
|
|
ULONG VendorId = 0;
|
2010-05-12 02:34:04 +00:00
|
|
|
ULONG MaxPciBusNumber;
|
2006-11-27 19:26:31 +00:00
|
|
|
|
|
|
|
/* Query registry information */
|
|
|
|
PciRegistryInfo = HalpQueryPciRegistryInfo();
|
|
|
|
if (!PciRegistryInfo)
|
2003-02-17 21:24:13 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Assume type 1 */
|
|
|
|
PciType = 1;
|
2010-05-12 02:34:04 +00:00
|
|
|
|
|
|
|
/* Force a manual bus scan later */
|
|
|
|
MaxPciBusNumber = MAXULONG;
|
2003-02-17 21:24:13 +00:00
|
|
|
}
|
2006-11-27 19:26:31 +00:00
|
|
|
else
|
2003-02-17 21:24:13 +00:00
|
|
|
{
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Get the PCI type */
|
2006-11-27 19:26:31 +00:00
|
|
|
PciType = PciRegistryInfo->HardwareMechanism & 0xF;
|
2010-05-12 02:34:04 +00:00
|
|
|
|
|
|
|
/* Get MaxPciBusNumber and make it 0-based */
|
|
|
|
MaxPciBusNumber = PciRegistryInfo->NoBuses - 1;
|
|
|
|
|
|
|
|
/* Free the info structure */
|
2011-05-26 22:22:37 +00:00
|
|
|
ExFreePoolWithTag(PciRegistryInfo, TAG_HAL);
|
2003-02-17 21:24:13 +00:00
|
|
|
}
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Initialize the PCI lock */
|
|
|
|
KeInitializeSpinLock(&HalpPCIConfigLock);
|
|
|
|
|
|
|
|
/* Check the type of PCI bus */
|
|
|
|
switch (PciType)
|
2003-11-05 22:39:01 +00:00
|
|
|
{
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Type 1 PCI Bus */
|
|
|
|
case 1:
|
2003-11-05 22:39:01 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Copy the Type 1 handler data */
|
|
|
|
RtlCopyMemory(&PCIConfigHandler,
|
|
|
|
&PCIConfigHandlerType1,
|
|
|
|
sizeof(PCIConfigHandler));
|
2003-02-17 21:24:13 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Set correct I/O Ports */
|
|
|
|
BusData->Config.Type1.Address = PCI_TYPE1_ADDRESS_PORT;
|
|
|
|
BusData->Config.Type1.Data = PCI_TYPE1_DATA_PORT;
|
|
|
|
break;
|
2003-02-17 21:24:13 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Type 2 PCI Bus */
|
|
|
|
case 2:
|
2003-02-17 21:24:13 +00:00
|
|
|
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Copy the Type 2 handler data */
|
2006-11-27 19:26:31 +00:00
|
|
|
RtlCopyMemory(&PCIConfigHandler,
|
|
|
|
&PCIConfigHandlerType2,
|
|
|
|
sizeof (PCIConfigHandler));
|
2001-11-07 02:27:33 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Set correct I/O Ports */
|
|
|
|
BusData->Config.Type2.CSE = PCI_TYPE2_CSE_PORT;
|
|
|
|
BusData->Config.Type2.Forward = PCI_TYPE2_FORWARD_PORT;
|
|
|
|
BusData->Config.Type2.Base = PCI_TYPE2_ADDRESS_BASE;
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Only 16 devices supported, not 32 */
|
|
|
|
BusData->MaxDevice = 16;
|
|
|
|
break;
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
default:
|
2004-11-15 09:18:19 +00:00
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* Invalid type */
|
2009-11-02 18:05:07 +00:00
|
|
|
DbgPrint("HAL: Unknown PCI type\n");
|
2006-11-27 19:26:31 +00:00
|
|
|
}
|
2004-11-15 09:18:19 +00:00
|
|
|
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Run a forced bus scan if needed */
|
|
|
|
if (MaxPciBusNumber == MAXULONG)
|
2004-12-04 22:52:59 +00:00
|
|
|
{
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Initialize the max bus number to 0xFF */
|
|
|
|
HalpMaxPciBus = 0xFF;
|
|
|
|
|
|
|
|
/* Initialize the counter */
|
|
|
|
MaxPciBusNumber = 0;
|
|
|
|
|
|
|
|
/* Loop all possible buses */
|
|
|
|
for (i = 0; i < HalpMaxPciBus; i++)
|
2006-11-27 19:26:31 +00:00
|
|
|
{
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Loop all devices */
|
|
|
|
for (j.u.AsULONG = 0; j.u.AsULONG < BusData->MaxDevice; j.u.AsULONG++)
|
2006-11-27 19:26:31 +00:00
|
|
|
{
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Query the interface */
|
|
|
|
if (HaliPciInterfaceReadConfig(NULL,
|
|
|
|
i,
|
|
|
|
j,
|
|
|
|
&VendorId,
|
|
|
|
0,
|
|
|
|
sizeof(ULONG)))
|
2006-11-27 19:26:31 +00:00
|
|
|
{
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Validate the vendor ID */
|
2011-11-20 17:54:52 +00:00
|
|
|
if ((VendorId & 0xFFFF) != PCI_INVALID_VENDORID)
|
2010-05-12 02:34:04 +00:00
|
|
|
{
|
|
|
|
/* Set this as the maximum ID */
|
|
|
|
MaxPciBusNumber = i;
|
|
|
|
break;
|
|
|
|
}
|
2006-11-27 19:26:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-12-04 22:52:59 +00:00
|
|
|
}
|
2001-08-21 20:18:27 +00:00
|
|
|
|
2010-05-12 02:34:04 +00:00
|
|
|
/* Set the real max bus number */
|
|
|
|
HalpMaxPciBus = MaxPciBusNumber;
|
|
|
|
|
2006-11-27 19:26:31 +00:00
|
|
|
/* We're done */
|
|
|
|
HalpPCIConfigInitialized = TRUE;
|
|
|
|
}
|
|
|
|
|
2010-06-07 03:19:20 +00:00
|
|
|
/* EOF */
|
|
|
|
|