Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1731 lines
45 KiB

/*++
Copyright (c) 1997 Microsoft Corporation
Module Name:
xxacpi.c
Abstract:
Implements various ACPI utility functions.
Author:
Jake Oshins (jakeo) 12-Feb-1997
Environment:
Kernel mode only.
Revision History:
Todd Kjos (HP) (v-tkjos) 1-Jun-1998 : Added IA64 support
--*/
#include "halp.h"
#include "acpitabl.h"
#include "xxacpi.h"
#include "pci.h"
//#define DUMP_FADT
VOID
HalAcpiTimerCarry(
VOID
);
VOID
HalaAcpiTimerInit(
#if defined(ACPI64)
ULONG_PTR TimerPort,
#else
ULONG TimerPort,
#endif
BOOLEAN TimerValExt
);
ULONG
HaliAcpiQueryFlags(
VOID
);
VOID
HaliAcpiTimerInit(
// *** TBD should be ULONG_PTR
ULONG TimerPort OPTIONAL,
IN BOOLEAN TimerValExt
);
VOID
HaliAcpiMachineStateInit(
IN PPROCESSOR_INIT ProcInit,
IN PHAL_SLEEP_VAL SleepValues,
OUT PULONG PicVal
);
BOOLEAN
FASTCALL
HalAcpiC1Idle(
OUT PPROCESSOR_IDLE_TIMES IdleTimes
);
BOOLEAN
FASTCALL
HalAcpiC2Idle(
OUT PPROCESSOR_IDLE_TIMES IdleTimes
);
BOOLEAN
FASTCALL
HalAcpiC3ArbdisIdle(
OUT PPROCESSOR_IDLE_TIMES IdleTimes
);
BOOLEAN
FASTCALL
HalAcpiC3WbinvdIdle(
OUT PPROCESSOR_IDLE_TIMES IdleTimes
);
VOID
FASTCALL
HalProcessorThrottle(
IN UCHAR Throttle
);
NTSTATUS
HaliSetWakeAlarm (
IN ULONGLONG WakeSystemTime,
IN PTIME_FIELDS WakeTimeFields OPTIONAL
);
VOID
HaliSetWakeEnable(
IN BOOLEAN Enable
);
VOID
HalpSetInterruptControllerWakeupState(
ULONG Context
);
PVOID
HalpRemapVirtualAddress(
IN PVOID VirtualAddress,
IN PVOID PhysicalAddress,
IN BOOLEAN WriteThrough
);
ULONG
HaliPciInterfaceReadConfig(
IN PVOID Context,
IN UCHAR BusOffset,
IN ULONG Slot,
IN PVOID Buffer,
IN ULONG Offset,
IN ULONG Length
);
ULONG
HaliPciInterfaceWriteConfig(
IN PVOID Context,
IN UCHAR BusOffset,
IN ULONG Slot,
IN PVOID Buffer,
IN ULONG Offset,
IN ULONG Length
);
VOID
HalpNumaInitializeStaticConfiguration(
IN PLOADER_PARAMETER_BLOCK
);
//
// HAL Hack flags
//
typedef enum {
HalHackAddFakeSleepHandlersS1 = 1,
HalHackAddFakeSleepHandlersS2 = 2,
HalHackAddFakeSleepHandlersS3 = 4
} HALHACKFLAGS;
extern HALHACKFLAGS HalpHackFlags = 0;
//
// Externs
//
extern ULONG HalpAcpiFlags;
extern PHYSICAL_ADDRESS HalpAcpiRsdt;
extern SLEEP_STATE_CONTEXT HalpShutdownContext;
//
// Globals
//
ULONG HalpInvalidAcpiTable;
PRSDT HalpAcpiRsdtVA;
PXSDT HalpAcpiXsdtVA;
PIPPT_TABLE HalpPlatformPropertiesTable;
//
// This is the dispatch table used by the ACPI driver
//
HAL_ACPI_DISPATCH_TABLE HalAcpiDispatchTable;
PPM_DISPATCH_TABLE PmAcpiDispatchTable = NULL;
NTSTATUS
HalpQueryAcpiResourceRequirements(
IN PIO_RESOURCE_REQUIREMENTS_LIST *Requirements
);
NTSTATUS
HalpBuildAcpiResourceList(
OUT PIO_RESOURCE_REQUIREMENTS_LIST List
);
NTSTATUS
HalpAcpiDetectResourceListSize(
OUT PULONG ResourceListSize
);
VOID
HalpRestoreInterruptControllerState(
VOID
);
ULONG
HalpGetPCIData (
IN PBUS_HANDLER BusHandler,
IN PBUS_HANDLER RootHandler,
IN PCI_SLOT_NUMBER SlotNumber,
IN PVOID Buffer,
IN ULONG Offset,
IN ULONG Length
);
VOID
HalpReadRegistryAndApplyHacks (
VOID
);
#define LOW_MEMORY 0x000100000
#define MAX(a, b) \
((a) > (b) ? (a) : (b))
#define MIN(a, b) \
((a) < (b) ? (a) : (b))
// ADRIAO 01/12/98 - We no longer having the HAL declare the IO ports
// specified in the FADT. These will be declared in a future
// defined PNP0Cxx node (for new, in PNP0C02). This is done
// because we cannot know at the hal level what bus the ACPI
// FADT resources refer to. We can only the translated resource info.
// Hence....
#define DECLARE_FADT_RESOURCES_AT_ROOT 0
#ifdef ALLOC_PRAGMA
#pragma alloc_text(INIT, HalpGetAcpiTablePhase0)
#pragma alloc_text(INIT, HalpSetupAcpiPhase0)
#pragma alloc_text(PAGE, HaliInitPowerManagement)
#pragma alloc_text(PAGE, HalpQueryAcpiResourceRequirements)
#pragma alloc_text(PAGE, HalpBuildAcpiResourceList)
#pragma alloc_text(PAGE, HalpGetCrossPartitionIpiInterface)
#pragma alloc_text(PAGE, HalpAcpiDetectResourceListSize)
#pragma alloc_text(PAGE, HalpReadRegistryAndApplyHacks)
#pragma alloc_text(PAGE, HaliAcpiTimerInit)
#pragma alloc_text(PAGE, HaliAcpiMachineStateInit)
#pragma alloc_text(PAGE, HaliAcpiQueryFlags)
#pragma alloc_text(PAGE, HaliSetWakeEnable)
#endif
PVOID
HalpGetAcpiTablePhase0(
IN PLOADER_PARAMETER_BLOCK LoaderBlock,
IN ULONG Signature
)
/*++
Routine Description:
This function returns a pointer to the ACPI table that is
identified by Signature.
Arguments:
Signature - A four byte value that identifies the ACPI table
Return Value:
Pointer to a copy of the table
--*/
{
PRSDT rsdt;
PXSDT xsdt;
ULONG entry, rsdtEntries;
PVOID table;
PHYSICAL_ADDRESS physicalAddr;
PDESCRIPTION_HEADER header;
NTSTATUS status;
ULONG lengthInPages;
ULONG offset;
physicalAddr.QuadPart = 0;
header = NULL;
if ((HalpAcpiRsdtVA == NULL) && (HalpAcpiXsdtVA == NULL)) {
//
// Find and map the RSDT once. This mapping is reused on
// subsequent calls to this routine.
//
status = HalpAcpiFindRsdtPhase0(LoaderBlock);
if (!(NT_SUCCESS(status))) {
HalDebugPrint(( HAL_INFO, "HAL: *** make sure you are using ntdetect.com v5.0 ***\n" ));
KeBugCheckEx(MISMATCHED_HAL,
4, 0xac31, 0, 0);
}
xsdt = HalpMapPhysicalMemory(HalpAcpiRsdt, 2, MmCached);
if (!xsdt) {
return NULL;
}
//
// Do a sanity check on the RSDT.
//
if ((xsdt->Header.Signature != RSDT_SIGNATURE) &&
(xsdt->Header.Signature != XSDT_SIGNATURE)) {
HalDisplayString("HAL: Bad RSDT pointer\n");
KeBugCheckEx(MISMATCHED_HAL,
4, 0xac31, 1, 0);
}
//
// Remap the (X)RSDT now that we know how long it is.
//
offset = HalpAcpiRsdt.LowPart & (PAGE_SIZE - 1);
lengthInPages = (offset + xsdt->Header.Length + (PAGE_SIZE - 1))
>> PAGE_SHIFT;
if (lengthInPages > 2) {
HalpUnmapVirtualAddress(xsdt, 2);
xsdt = HalpMapPhysicalMemory(HalpAcpiRsdt, lengthInPages, MmCached);
if (!xsdt) {
DbgPrint("HAL: Couldn't remap RSDT\n");
return NULL;
}
}
if (xsdt->Header.Signature == XSDT_SIGNATURE) {
HalpAcpiXsdtVA = xsdt;
} else {
HalpAcpiRsdtVA = (PRSDT)xsdt;
HalpAcpiXsdtVA = NULL;
}
}
xsdt = HalpAcpiXsdtVA;
rsdt = HalpAcpiRsdtVA;
rsdtEntries = xsdt ?
NumTableEntriesFromXSDTPointer(xsdt) :
NumTableEntriesFromRSDTPointer(rsdt);
//
// Look down the pointer in each entry to see if it points to
// the table we are looking for.
//
for (entry = 0; entry < rsdtEntries; entry++) {
physicalAddr.QuadPart = xsdt ?
xsdt->Tables[entry].QuadPart :
rsdt->Tables[entry];
if (header != NULL) {
HalpUnmapVirtualAddress(header, 2);
}
header = HalpMapPhysicalMemory(physicalAddr, 2, MmCached);
if (!header) {
return NULL;
}
if (header->Signature == Signature) {
break;
}
}
if (entry == rsdtEntries) {
//
// Signature not found, free the PTR for the last entry
// examined and indicate failure to the caller.
//
HalpUnmapVirtualAddress(header, 2);
return NULL;
}
//
// Make sure we have mapped enough memory to cover the entire
// table.
//
offset = (ULONG)((ULONG_PTR)header & (PAGE_SIZE - 1));
lengthInPages = (header->Length + offset + (PAGE_SIZE - 1)) >> PAGE_SHIFT;
if (lengthInPages > 2) {
HalpUnmapVirtualAddress(header, 2);
header = HalpMapPhysicalMemory( physicalAddr, lengthInPages, MmCached);
}
//
// Validate the table's checksum.
// N.B. We expect the checksum to be wrong on some early versions
// of the FADT.
//
if ((header != NULL) &&
((header->Signature != FADT_SIGNATURE) || (header->Revision > 2))) {
PUCHAR c = (PUCHAR)header + header->Length;
UCHAR s = 0;
if (header->Length) {
do {
s += *--c;
} while (c != (PUCHAR)header);
}
if ((s != 0) || (header->Length == 0)) {
//
// This table is not valid.
//
HalpInvalidAcpiTable = header->Signature;
#if 0
//
// Don't return this table.
//
HalpUnmapVirtualAddress(header, lengthInPages);
return NULL;
#endif
}
}
return header;
}
NTSTATUS
HalpSetupAcpiPhase0(
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
Save some information from the ACPI tables before they get
destroyed.
Arguments:
none
Return Value:
none
--*/
{
NTSTATUS status;
ULONG entry;
PVOID table;
PVOID physicalAddr;
PDESCRIPTION_HEADER header;
ULONG blkSize;
PHYSICAL_ADDRESS rawAddr;
//
// Copy the Fixed Acpi Descriptor Table (FADT) to a permanent
// home.
//
header = HalpGetAcpiTablePhase0(LoaderBlock, FADT_SIGNATURE);
if (header == NULL) {
HalDebugPrint(( HAL_INFO, "HAL: Didn't find the FACP\n" ));
return STATUS_NOT_FOUND;
}
RtlCopyMemory(&HalpFixedAcpiDescTable,
header,
MIN(header->Length, sizeof(HalpFixedAcpiDescTable)));
if (header->Revision < 3) {
KeBugCheckEx(ACPI_BIOS_ERROR, 0x11, 9, header->Revision, 0);
}
// Check for MMIO addresses that need to be mapped.
blkSize = HalpFixedAcpiDescTable.pm1_evt_len;
ASSERT(blkSize);
if ((HalpFixedAcpiDescTable.x_pm1a_evt_blk.AddressSpaceID == AcpiGenericSpaceMemory) &&
(blkSize > 0)) {
rawAddr = HalpFixedAcpiDescTable.x_pm1a_evt_blk.Address;
HalpFixedAcpiDescTable.x_pm1a_evt_blk.Address.QuadPart =
(LONGLONG)HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
blkSize = HalpFixedAcpiDescTable.pm1_ctrl_len;
ASSERT(blkSize);
if ((HalpFixedAcpiDescTable.x_pm1a_ctrl_blk.AddressSpaceID == AcpiGenericSpaceMemory) &&
(blkSize > 0)) {
rawAddr = HalpFixedAcpiDescTable.x_pm1a_ctrl_blk.Address;
HalpFixedAcpiDescTable.x_pm1a_ctrl_blk.Address.QuadPart =
(LONGLONG) HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
blkSize = HalpFixedAcpiDescTable.pm_tmr_len;
ASSERT(blkSize);
if ((HalpFixedAcpiDescTable.x_pm_tmr_blk.AddressSpaceID == AcpiGenericSpaceMemory) &&
(blkSize > 0)) {
rawAddr = HalpFixedAcpiDescTable.x_pm_tmr_blk.Address;
HalpFixedAcpiDescTable.x_pm_tmr_blk.Address.QuadPart =
(LONGLONG) HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
// The rest of these ACPI blocks are optional so test if they exist before mapping them
if (HalpFixedAcpiDescTable.x_pm1b_evt_blk.Address.QuadPart) {
if (HalpFixedAcpiDescTable.x_pm1b_evt_blk.AddressSpaceID == AcpiGenericSpaceMemory) {
blkSize = HalpFixedAcpiDescTable.pm1_evt_len;
rawAddr = HalpFixedAcpiDescTable.x_pm1b_evt_blk.Address;
HalpFixedAcpiDescTable.x_pm1b_evt_blk.Address.QuadPart =
(LONGLONG) HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
}
if (HalpFixedAcpiDescTable.x_pm1b_ctrl_blk.Address.QuadPart) {
if (HalpFixedAcpiDescTable.x_pm1b_ctrl_blk.AddressSpaceID == AcpiGenericSpaceMemory) {
blkSize = HalpFixedAcpiDescTable.pm1_ctrl_len;
rawAddr = HalpFixedAcpiDescTable.x_pm1b_ctrl_blk.Address;
HalpFixedAcpiDescTable.x_pm1b_ctrl_blk.Address.QuadPart =
(LONGLONG) HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
}
if (HalpFixedAcpiDescTable.x_pm2_ctrl_blk.Address.QuadPart) {
if (HalpFixedAcpiDescTable.x_pm2_ctrl_blk.AddressSpaceID == AcpiGenericSpaceMemory) {
blkSize = HalpFixedAcpiDescTable.pm2_ctrl_len;
rawAddr = HalpFixedAcpiDescTable.x_pm2_ctrl_blk.Address;
HalpFixedAcpiDescTable.x_pm2_ctrl_blk.Address.QuadPart =
(LONGLONG) HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
}
if (HalpFixedAcpiDescTable.x_gp0_blk.Address.QuadPart) {
if (HalpFixedAcpiDescTable.x_gp0_blk.AddressSpaceID == AcpiGenericSpaceMemory) {
blkSize = HalpFixedAcpiDescTable.gp0_blk_len;
rawAddr = HalpFixedAcpiDescTable.x_gp0_blk.Address;
HalpFixedAcpiDescTable.x_gp0_blk.Address.QuadPart =
(LONGLONG) HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
}
if (HalpFixedAcpiDescTable.x_gp1_blk.Address.QuadPart) {
if (HalpFixedAcpiDescTable.x_gp1_blk.AddressSpaceID == AcpiGenericSpaceMemory) {
blkSize = HalpFixedAcpiDescTable.gp1_blk_len;
rawAddr = HalpFixedAcpiDescTable.x_gp1_blk.Address;
HalpFixedAcpiDescTable.x_gp1_blk.Address.QuadPart =
(LONGLONG) HalpMapPhysicalMemory(rawAddr,ADDRESS_AND_SIZE_TO_SPAN_PAGES(rawAddr.LowPart, blkSize),MmCached);
}
}
//
// See if Static Resource Affinity Table is present.
//
HalpNumaInitializeStaticConfiguration(LoaderBlock);
//
// See if Windows Platform Properties Table is present.
//
HalpPlatformPropertiesTable =
HalpGetAcpiTablePhase0(LoaderBlock, IPPT_SIGNATURE);
//
// Enable ACPI counter code since we need it in the boot
// process.
//
HaliAcpiTimerInit(0, FALSE);
//
// Claim a page of memory below 1MB to be used for transitioning
// a sleeping processor back from real mode to protected mode
//
#ifdef IA64
HalDebugPrint(( HAL_INFO, "HAL: WARNING - HalpSetupAcpi - Sleep transitions not yet implemented\n" ));
#else
// check first to see if this has already been done by MP startup code
if (!HalpLowStubPhysicalAddress) {
HalpLowStubPhysicalAddress = (PVOID)HalpAllocPhysicalMemory (LoaderBlock,
LOW_MEMORY, 1, FALSE);
if (HalpLowStubPhysicalAddress) {
HalpLowStub = HalpMapPhysicalMemory(HalpLowStubPhysicalAddress, 1, MmCached);
}
}
//
// Claim a PTE that will be used for cache flushing in states S2 and S3.
//
HalpVirtAddrForFlush = HalpMapPhysicalMemory((PVOID)LOW_MEMORY, 1, MmCached);
HalpPteForFlush = MiGetPteAddress(HalpVirtAddrForFlush);
#endif
return STATUS_SUCCESS;
}
VOID
HaliAcpiTimerInit(
// *** TBD should be ULONG_PTR
IN ULONG TimerPort OPTIONAL,
IN BOOLEAN TimerValExt
)
/*++
Routine Description:
This routine initializes the ACPI timer.
Arguments:
TimerPort - The address in I/O space of the ACPI timer. If this is
0, then the values from the cached FADT will be used.
TimerValExt - signifies whether the timer is 24 or 32 bits.
--*/
{
#if defined(ACPI64)
ULONG_PTR port = TimerPort;
#else
ULONG port = TimerPort;
#endif
BOOLEAN ext = TimerValExt;
PAGED_CODE();
if (port == 0) {
port = HalpFixedAcpiDescTable.x_pm_tmr_blk.Address.LowPart;
if (HalpFixedAcpiDescTable.flags & TMR_VAL_EXT) {
ext = TRUE;
} else {
ext = FALSE;
}
}
HalaAcpiTimerInit(port,
ext);
}
VOID
HaliAcpiMachineStateInit(
IN PPROCESSOR_INIT ProcInit,
IN PHAL_SLEEP_VAL SleepValues,
OUT PULONG PicVal
)
/*++
Routine Description:
This function is a callback used by the ACPI driver
to notify the HAL with the processor blocks.
Arguments:
--*/
{
POWER_STATE_HANDLER powerState;
SLEEP_STATE_CONTEXT sleepContext;
NTSTATUS status;
PAGED_CODE();
UNREFERENCED_PARAMETER(ProcInit);
*PicVal = 1; // We only support APIC on IA64
RtlZeroMemory (&sleepContext, sizeof (sleepContext));
powerState.Context = NULL;
//
// Set up fake handlers that do nothing so testing device power
// transitions isn't blocked. Only if hack flag is set, though
//
HalpReadRegistryAndApplyHacks ();
if (HalpHackFlags & HalHackAddFakeSleepHandlersS1) {
powerState.Type = PowerStateSleeping1;
powerState.RtcWake = TRUE;
powerState.Handler = &HaliAcpiFakeSleep;
status = ZwPowerInformation(SystemPowerStateHandler,
&powerState,
sizeof(POWER_STATE_HANDLER),
NULL,
0);
}
if (HalpHackFlags & HalHackAddFakeSleepHandlersS2) {
powerState.Type = PowerStateSleeping2;
powerState.RtcWake = TRUE;
powerState.Handler = &HaliAcpiFakeSleep;
status = ZwPowerInformation(SystemPowerStateHandler,
&powerState,
sizeof(POWER_STATE_HANDLER),
NULL,
0);
}
if (HalpHackFlags & HalHackAddFakeSleepHandlersS3) {
powerState.Type = PowerStateSleeping3;
powerState.RtcWake = TRUE;
powerState.Handler = &HaliAcpiFakeSleep;
powerState.Context = ULongToPtr(sleepContext.AsULONG);
status = ZwPowerInformation(SystemPowerStateHandler,
&powerState,
sizeof(POWER_STATE_HANDLER),
NULL,
0);
}
//
// For now all we are going to do is register a shutdown handler
// that will call shutdown-restart
//
if (SleepValues[4].Supported) {
powerState.Type = PowerStateShutdownOff;
powerState.RtcWake = FALSE;
powerState.Handler = &HaliAcpiSleep;
sleepContext.bits.Pm1aVal = SleepValues[4].Pm1aVal;
sleepContext.bits.Pm1bVal = SleepValues[4].Pm1bVal;
sleepContext.bits.Flags = SLEEP_STATE_OFF;
HalpShutdownContext = sleepContext;
powerState.Context = ULongToPtr(sleepContext.AsULONG);
status = ZwPowerInformation(SystemPowerStateHandler,
&powerState,
sizeof(POWER_STATE_HANDLER),
NULL,
0);
ASSERT(NT_SUCCESS(status));
}
}
ULONG
HaliAcpiQueryFlags(
VOID
)
/*++
Routine Description:
This routine is temporary is used to report the presence of the
boot.ini switch
Arguments:
None
Return Value:
TRUE, if switch present
--*/
{
return HalpAcpiFlags;
}
NTSTATUS
HaliInitPowerManagement(
IN PPM_DISPATCH_TABLE PmDriverDispatchTable,
IN OUT PPM_DISPATCH_TABLE *PmHalDispatchTable
)
/*++
Routine Description:
This is called by the ACPI driver to start the PM
code.
Arguments:
PmDriverDispatchTable - table of functions provided
by the ACPI driver for the HAL
PmHalDispatchTable - table of functions provided by
the HAL for the ACPI driver
Return Value:
status
--*/
{
OBJECT_ATTRIBUTES objAttributes;
PCALLBACK_OBJECT callback;
PHYSICAL_ADDRESS pAddr;
UNICODE_STRING callbackName;
NTSTATUS status;
PFACS facs;
PAGED_CODE();
//
// Keep a pointer to the driver's dispatch table.
//
// ASSERT(PmDriverDispatchTable);
// ASSERT(PmDriverDispatchTable->Signature == ACPI_HAL_DISPATCH_SIGNATURE);
PmAcpiDispatchTable = PmDriverDispatchTable;
//
// Fill in the function table
//
HalAcpiDispatchTable.Signature = HAL_ACPI_DISPATCH_SIGNATURE;
HalAcpiDispatchTable.Version = HAL_ACPI_DISPATCH_VERSION;
HalAcpiDispatchTable.HalpAcpiTimerInit = &HaliAcpiTimerInit;
HalAcpiDispatchTable.HalpAcpiTimerInterrupt =
(pHalAcpiTimerInterrupt)&HalAcpiTimerCarry;
HalAcpiDispatchTable.HalpAcpiMachineStateInit = &HaliAcpiMachineStateInit;
HalAcpiDispatchTable.HalpAcpiQueryFlags = &HaliAcpiQueryFlags;
HalAcpiDispatchTable.HalxPicStateIntact = &HalpAcpiPicStateIntact;
HalAcpiDispatchTable.HalxRestorePicState = &HalpRestoreInterruptControllerState;
HalAcpiDispatchTable.HalpSetVectorState = &HaliSetVectorState;
HalAcpiDispatchTable.HalpPciInterfaceReadConfig = &HaliPciInterfaceReadConfig;
HalAcpiDispatchTable.HalpPciInterfaceWriteConfig = &HaliPciInterfaceWriteConfig;
HalAcpiDispatchTable.HalpSetMaxLegacyPciBusNumber = &HalpSetMaxLegacyPciBusNumber;
HalAcpiDispatchTable.HalpIsVectorValid = &HaliIsVectorValid;
*PmHalDispatchTable = (PPM_DISPATCH_TABLE)&HalAcpiDispatchTable;
//
// Fill in Hal's private dispatch table
//
HalSetWakeEnable = HaliSetWakeEnable;
HalSetWakeAlarm = HaliSetWakeAlarm;
//
// Register callback that tells us to make
// anything we need for sleeping non-pageable.
//
RtlInitUnicodeString(&callbackName, L"\\Callback\\PowerState");
InitializeObjectAttributes(
&objAttributes,
&callbackName,
OBJ_CASE_INSENSITIVE | OBJ_PERMANENT,
NULL,
NULL
);
ExCreateCallback(&callback,
&objAttributes,
FALSE,
TRUE);
ExRegisterCallback(callback,
(PCALLBACK_FUNCTION)&HalpPowerStateCallback,
NULL);
#if 0
//
// Find the location of the firmware waking vector.
// N.B. If any of this fails, then HalpWakeVector will be NULL
// and we won't support S2 or S3.
//
if (HalpFixedAcpiDescTable.x_firmware_ctrl.Address.QuadPart) {
facs = HalpMapPhysicalMemory(HalpFixedAcpiDescTable.x_firmware_ctrl.Address,
ADDRESS_AND_SIZE_TO_SPAN_PAGES(HalpFixedAcpiDescTable.x_firmware_ctrl.Address.LowPart, sizeof(FACS)),
MmCached);
if (facs) {
if (facs->Signature == FACS_SIGNATURE) {
HalpWakeVector = &facs->x_FirmwareWakingVector;
}
}
}
#endif
return STATUS_SUCCESS;
}
NTSTATUS
HalpQueryAcpiResourceRequirements(
IN PIO_RESOURCE_REQUIREMENTS_LIST *Requirements
)
/*++
Routine Description:
This routine is a temporary stub that tries to detect the presence
of an ACPI controller within the system. This code is meant to be
inserted within NT's root system enumerator.
Arguents:
Requirements - pointer to list of resources
Return Value:
STATUS_SUCCESS - If we found a device object
STATUS_NO_SUCH_DEVICE - If we can't find info about the new PDO
--*/
{
NTSTATUS ntStatus;
PIO_RESOURCE_REQUIREMENTS_LIST resourceList;
ULONG resourceListSize;
PAGED_CODE();
//
// Now figure out the number of resource that we need
//
ntStatus = HalpAcpiDetectResourceListSize(
&resourceListSize
);
//
// Convert this resourceListSize into the number of bytes that we
// must allocate
//
resourceListSize = sizeof(IO_RESOURCE_REQUIREMENTS_LIST) +
( (resourceListSize - 1) * sizeof(IO_RESOURCE_DESCRIPTOR) );
//
// Allocate the correct number of bytes of the Resource List
//
resourceList = ExAllocatePoolWithTag(
PagedPool,
resourceListSize,
HAL_POOL_TAG
);
//
// This call must have succeeded or we cannot lay claim to ACPI
//
if (resourceList == NULL) {
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Set up the ListSize in the structure
//
RtlZeroMemory(resourceList, resourceListSize);
resourceList->ListSize = resourceListSize;
//
// Build the ResourceList here
//
ntStatus = HalpBuildAcpiResourceList(resourceList);
//
// Did we build the list okay?
//
if (!NT_SUCCESS(ntStatus)) {
//
// Free memory and exit
//
ExFreePool(resourceList);
return STATUS_NO_SUCH_DEVICE;
}
*Requirements = resourceList;
return ntStatus;
}
NTSTATUS
HalpBuildAcpiResourceList(
OUT PIO_RESOURCE_REQUIREMENTS_LIST List
)
/*++
Routine Description:
This is the routine that builds the ResourceList given the FADT and
an arbitrary number of ResourceDescriptors. We assume that the
ResourceList has been properly allocated and sized
Arguments:
List - The list to fill in
Return Value:
STATUS_SUCCESS if okay
STATUS_UNSUCCESSUL if not
--*/
{
PIO_RESOURCE_DESCRIPTOR partialResource;
ULONG count = 0;
PAGED_CODE();
ASSERT( List != NULL );
//
// Specify some default values (for now) to determine the Bus Type and
// the bus number
//
List->AlternativeLists = 1;
List->InterfaceType = Isa;
List->BusNumber = 0;
List->List[0].Version = 1;
List->List[0].Revision = 1;
//
// Is there an interrupt resource required?
//
if (HalpFixedAcpiDescTable.sci_int_vector != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypeInterrupt;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareShared;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
List->List[0].Descriptors[count].u.Interrupt.MinimumVector =
List->List[0].Descriptors[count].u.Interrupt.MaximumVector =
HalpFixedAcpiDescTable.sci_int_vector;
List->List[0].Count++;
count++;
}
#if DECLARE_FADT_RESOURCES_AT_ROOT
//
// Is there an SMI CMD IO Port?
//
if (HalpFixedAcpiDescTable.smi_cmd_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags =CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
PtrToUlong(HalpFixedAcpiDescTable.smi_cmd_io_port);
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
PtrToUlong(HalpFixedAcpiDescTable.smi_cmd_io_port);
List->List[0].Descriptors[count].u.Port.Length = 1;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there an PM1A Event Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1a_evt_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.pm1a_evt_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.pm1a_evt_blk_io_port + (ULONG) HalpFixedAcpiDescTable.pm1_evt_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.pm1_evt_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there a PM1B Event Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1b_evt_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.pm1b_evt_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.pm1b_evt_blk_io_port + (ULONG) HalpFixedAcpiDescTable.pm1_evt_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.pm1_evt_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there a PM1A Control Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1a_ctrl_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.pm1a_ctrl_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.pm1a_ctrl_blk_io_port + (ULONG) HalpFixedAcpiDescTable.pm1_ctrl_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.pm1_ctrl_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there a PM1B Control Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1b_ctrl_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.pm1b_ctrl_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.pm1b_ctrl_blk_io_port + (ULONG) HalpFixedAcpiDescTable.pm1_ctrl_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.pm1_ctrl_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there a PM2 Control Block IO Port?
//
if (HalpFixedAcpiDescTable.pm2_ctrl_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.pm2_ctrl_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.pm2_ctrl_blk_io_port + (ULONG) HalpFixedAcpiDescTable.pm2_ctrl_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.pm2_ctrl_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there a PM Timer Block IO Port?
//
if (HalpFixedAcpiDescTable.pm_tmr_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.pm_tmr_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.pm_tmr_blk_io_port + (ULONG) HalpFixedAcpiDescTable.pm_tmr_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.pm_tmr_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there a GP0 Block IO Port?
//
if (HalpFixedAcpiDescTable.gp0_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.gp0_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.gp0_blk_io_port + (ULONG) HalpFixedAcpiDescTable.gp0_blk_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.gp0_blk_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
//
// Is there a GP1 Block IO port?
//
if (HalpFixedAcpiDescTable.gp1_blk_io_port != 0) {
List->List[0].Descriptors[count].Type = CmResourceTypePort;
List->List[0].Descriptors[count].ShareDisposition = CmResourceShareDeviceExclusive;
List->List[0].Descriptors[count].Flags = CM_RESOURCE_PORT_IO;
List->List[0].Descriptors[count].u.Port.MinimumAddress.LowPart =
HalpFixedAcpiDescTable.gp1_blk_io_port;
List->List[0].Descriptors[count].u.Port.MaximumAddress.LowPart =
HalpFixedAcpiDescTable.gp1_blk_io_port + (ULONG) HalpFixedAcpiDescTable.gp1_blk_len - 1;
List->List[0].Descriptors[count].u.Port.Length = (ULONG) HalpFixedAcpiDescTable.gp1_blk_len;
List->List[0].Descriptors[count].u.Port.Alignment = 1;
List->List[0].Count++;
count++;
}
#endif // DECLARE_FADT_RESOURCES_AT_ROOT
return STATUS_SUCCESS;
}
NTSTATUS
HalpAcpiDetectResourceListSize(
OUT PULONG ResourceListSize
)
/*++
Routine Description:
Given a pointer to an FADT, determine the number of
CM_PARTIAL_RESOURCE_DESCRIPTORS that are required to
describe all the resource mentioned in the FADT
Arguments:
ResourceListSize - Location to store the answer
Return Value:
STATUS_SUCCESS if everything went okay
--*/
{
PAGED_CODE();
//
// First of all, assume that we need no resources
//
*ResourceListSize = 0;
//
// Is there an interrupt resource required?
//
if (HalpFixedAcpiDescTable.sci_int_vector != 0) {
*ResourceListSize += 1;
}
#if DECLARE_FADT_RESOURCES_AT_ROOT
//
// Is there an SMI CMD IO Port?
//
if (HalpFixedAcpiDescTable.smi_cmd_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there an PM1A Event Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1a_evt_blk_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there a PM1B Event Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1b_evt_blk_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there a PM1A Control Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1a_ctrl_blk_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there a PM1B Control Block IO Port?
//
if (HalpFixedAcpiDescTable.pm1b_ctrl_blk_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there a PM2 Control Block IO Port?
//
if (HalpFixedAcpiDescTable.pm2_ctrl_blk_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there a PM Timer Block IO Port?
//
if (HalpFixedAcpiDescTable.pm_tmr_blk_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there a GP0 Block IO Port?
//
if (HalpFixedAcpiDescTable.gp0_blk_io_port != 0) {
*ResourceListSize += 1;
}
//
// Is there a GP1 Block IO Port?
//
if (HalpFixedAcpiDescTable.gp1_blk_io_port != 0) {
*ResourceListSize += 1;
}
#endif // DECALRE_FADT_RESOURCES_AT_ROOT
return STATUS_SUCCESS;
}
/*++
Routine Description:
Scans the registry for TestFlags and sets the global
hackflag to whatever the registry value is, if it
exists.
Arguments:
None
Return Value:
None
--*/
VOID
HalpReadRegistryAndApplyHacks (
VOID
)
{
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING UnicodeString;
HANDLE BaseHandle = NULL;
NTSTATUS status;
KEY_VALUE_PARTIAL_INFORMATION hackflags;
ULONG resultlength = 0;
HalpHackFlags = 0;
RtlInitUnicodeString (&UnicodeString,
L"\\REGISTRY\\MACHINE\\SYSTEM\\CURRENTCONTROLSET\\Control\\HAL");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
(PSECURITY_DESCRIPTOR) NULL);
status = ZwOpenKey (&BaseHandle,
KEY_READ,
&ObjectAttributes);
if (!NT_SUCCESS (status)) {
return;
}
RtlInitUnicodeString (&UnicodeString,
L"TestFlags");
status = ZwQueryValueKey (BaseHandle,
&UnicodeString,
KeyValuePartialInformation,
&hackflags,
sizeof (hackflags),
&resultlength);
if (!NT_SUCCESS (status)) {
return;
}
if (hackflags.Type != REG_DWORD || hackflags.DataLength != sizeof (ULONG)) {
return;
}
HalpHackFlags = (ULONG) *hackflags.Data;
}
ULONG
HalpReadGenAddr(
IN PGEN_ADDR GenAddr
)
{
ULONG i, result = 0, bitWidth, mask = 0;
//
// Figure out how wide our target register is.
//
bitWidth = GenAddr->BitWidth +
GenAddr->BitOffset;
if (bitWidth > 16) {
bitWidth = 32;
} else if (bitWidth <= 8) {
bitWidth = 8;
} else {
bitWidth = 16;
}
switch (GenAddr->AddressSpaceID) {
case AcpiGenericSpaceIO:
ASSERT(!(GenAddr->Address.LowPart & 0Xffff0000));
ASSERT(GenAddr->Address.HighPart == 0);
switch (bitWidth) {
case 8:
result = READ_PORT_UCHAR((PUCHAR)(UINT_PTR)GenAddr->Address.LowPart);
break;
case 16:
result = READ_PORT_USHORT((PUSHORT)(UINT_PTR)GenAddr->Address.LowPart);
break;
case 32:
result = READ_PORT_ULONG((PULONG)(UINT_PTR)GenAddr->Address.LowPart);
break;
default:
return 0;
}
break;
case AcpiGenericSpaceMemory:
//
// This code path depends on the fact that the addresses
// in these structures have already been converted to
// virtual addresses.
//
switch (bitWidth) {
case 8:
result = READ_REGISTER_UCHAR((PUCHAR)GenAddr->Address.QuadPart);
break;
case 16:
result = READ_REGISTER_USHORT((PUSHORT)GenAddr->Address.QuadPart);
break;
case 32:
result = READ_REGISTER_ULONG((PULONG)GenAddr->Address.QuadPart);
break;
default:
return 0;
}
break;
default:
return 0;
}
//
// If the register is not actually byte-aligned, correct for
// that.
//
if (result && (bitWidth != GenAddr->BitWidth)) {
result >>= GenAddr->BitOffset;
result &= ((0x1ul << GenAddr->BitWidth) - 1);
}
return result;
}
VOID
HalpWriteGenAddr(
IN PGEN_ADDR GenAddr,
IN ULONG Value
)
{
ULONG i, result = 0, bitWidth, data, mask = 0;
data = 0;
//
// Figure out how wide our target register is.
//
bitWidth = GenAddr->BitWidth +
GenAddr->BitOffset;
if (bitWidth > 16) {
bitWidth = 32;
} else if (bitWidth <= 8) {
bitWidth = 8;
} else {
bitWidth = 16;
}
switch (GenAddr->AddressSpaceID) {
case AcpiGenericSpaceIO:
ASSERT(!(GenAddr->Address.LowPart & 0Xffff0000));
ASSERT(GenAddr->Address.HighPart == 0);
switch (bitWidth) {
case 8:
ASSERT(!(Value & 0xffffff00));
if ((GenAddr->BitOffset != 0) ||
(GenAddr->BitWidth != bitWidth)) {
data = READ_PORT_UCHAR((PUCHAR)(UINT_PTR)GenAddr->Address.LowPart);
mask = (UCHAR)~0 >> (8 - GenAddr->BitWidth);
mask = (UCHAR)~(mask << GenAddr->BitOffset);
data &= mask;
data |= (UCHAR)Value << GenAddr->BitOffset;
} else {
data = Value;
}
WRITE_PORT_UCHAR((PUCHAR)(UINT_PTR)GenAddr->Address.LowPart,
(UCHAR)data);
break;
case 16:
ASSERT(!(Value & 0xffff0000));
if ((GenAddr->BitOffset != 0) ||
(GenAddr->BitWidth != bitWidth)) {
data = READ_PORT_USHORT((PUSHORT)(UINT_PTR)GenAddr->Address.LowPart);
mask = (USHORT)~0 >> (16 - GenAddr->BitWidth);
mask = (USHORT)~(mask << GenAddr->BitOffset);
data &= mask;
data |= (USHORT)Value << GenAddr->BitOffset;
} else {
data = Value;
}
WRITE_PORT_USHORT((PUSHORT)(UINT_PTR)GenAddr->Address.LowPart,
(USHORT)data);
break;
case 32:
if ((GenAddr->BitOffset != 0) ||
(GenAddr->BitWidth != bitWidth)) {
data = READ_PORT_ULONG((PULONG)(UINT_PTR)GenAddr->Address.LowPart);
mask = (ULONG)~0 >> (32 - GenAddr->BitWidth);
mask = ~(mask << GenAddr->BitOffset);
data &= mask;
data |= Value << GenAddr->BitOffset;
} else {
data = Value;
}
WRITE_PORT_ULONG((PULONG)(UINT_PTR)GenAddr->Address.LowPart,
data);
break;
default:
return;
}
break;
case AcpiGenericSpaceMemory:
//
// This code path depends on the fact that the addresses
// in these structures have already been converted to
// virtual addresses.
//
switch (bitWidth) {
case 8:
ASSERT(!(Value & 0xffffff00));
if ((GenAddr->BitOffset != 0) ||
(GenAddr->BitWidth != bitWidth)) {
data = READ_REGISTER_UCHAR((PUCHAR)GenAddr->Address.QuadPart);
mask = (UCHAR)~0 >> (8 - GenAddr->BitWidth);
mask = (UCHAR)~(mask << GenAddr->BitOffset);
data &= mask;
data |= (UCHAR)Value << GenAddr->BitOffset;
} else {
data = Value;
}
WRITE_REGISTER_UCHAR((PUCHAR)GenAddr->Address.QuadPart,
(UCHAR)data);
break;
case 16:
ASSERT(!(Value & 0xffff0000));
if ((GenAddr->BitOffset != 0) ||
(GenAddr->BitWidth != bitWidth)) {
data = READ_REGISTER_USHORT((PUSHORT)GenAddr->Address.QuadPart);
mask = (USHORT)~0 >> (16 - GenAddr->BitWidth);
mask = (USHORT)~(mask << GenAddr->BitOffset);
data &= mask;
data |= (USHORT)Value << GenAddr->BitOffset;
} else {
data = Value;
}
WRITE_REGISTER_USHORT((PUSHORT)GenAddr->Address.QuadPart,
(USHORT)data);
break;
case 32:
if ((GenAddr->BitOffset != 0) ||
(GenAddr->BitWidth != bitWidth)) {
data = READ_REGISTER_ULONG((PULONG)GenAddr->Address.QuadPart);
mask = (ULONG)~0 >> (32 - GenAddr->BitWidth);
mask = ~(mask << GenAddr->BitOffset);
data &= mask;
data |= Value << GenAddr->BitOffset;
} else {
data = Value;
}
WRITE_REGISTER_ULONG((PULONG)GenAddr->Address.QuadPart, data);
break;
default:
return;
}
break;
default:
return;
}
}
NTSTATUS
HalpGetPlatformProperties(
OUT PULONG Properties
)
/*++
Routine Description:
This function retrieves the platform properties as specified in
the ACPI-style IPPT table if present on this platform. The table
itself would've been retrieved earlier.
Arguments:
Properties - Pointer to a ULONG that will be updated
to reflect the platform property flags if present.
Return Value:
NTSTATUS - STATUS_SUCCESS indicates the table was present and the
ULONG pointed to by Properties contains valid data.
--*/
{
if (HalpPlatformPropertiesTable) {
*Properties = HalpPlatformPropertiesTable->Flags;
return STATUS_SUCCESS;
} else {
return STATUS_UNSUCCESSFUL;
}
}
NTSTATUS
HalpGetCrossPartitionIpiInterface(
OUT HAL_CROSS_PARTITION_IPI_INTERFACE * IpiInterface
)
/*++
Routine Description:
This function fills in the HAL_CROSS_PARTITION_IPI_INTERFACE
structure pointed to by the argument with the appropriate hal
function pointers.
Arguments:
IpiInterface - Pointer to HAL_CROSS_PARTITION_IPI_INTERFACE
structure.
Return Value:
NTSTATUS
--*/
{
PAGED_CODE();
IpiInterface->HalSendCrossPartitionIpi = HalpSendCrossPartitionIpi;
IpiInterface->HalReserveCrossPartitionInterruptVector =
HalpReserveCrossPartitionInterruptVector;
return STATUS_SUCCESS;
}