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.
431 lines
12 KiB
431 lines
12 KiB
/*++
|
|
|
|
Copyright (c) 1997 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
mphibrnt.c
|
|
|
|
Abstract:
|
|
|
|
This file provides the code that changes the system from
|
|
the ACPI S0 (running) state to S4 (hibernated).
|
|
|
|
Author:
|
|
|
|
Jake Oshins (jakeo) May 6, 1997
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "halp.h"
|
|
#include "apic.inc"
|
|
#include "pcmp_nt.inc"
|
|
#include "ixsleep.h"
|
|
|
|
NTSTATUS
|
|
HaliLegacyHibernate(
|
|
IN PVOID Context,
|
|
IN PENTER_STATE_SYSTEM_HANDLER SystemHandler OPTIONAL,
|
|
IN PVOID SystemContext,
|
|
IN LONG NumberProcessors,
|
|
IN volatile PLONG Number
|
|
);
|
|
|
|
ULONG
|
|
DetectMPS (
|
|
OUT PBOOLEAN IsConfiguredMp
|
|
);
|
|
|
|
volatile extern BOOLEAN HalpHiberInProgress;
|
|
extern BOOLEAN HalpDisableHibernate;
|
|
extern UCHAR HalpLastEnumeratedActualProcessor;
|
|
|
|
struct PcMpTable *PcMpTablePtr;
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE, HalpRegisterHibernate)
|
|
#pragma alloc_text(PAGELK, HaliLegacyHibernate)
|
|
#endif
|
|
|
|
|
|
VOID
|
|
HalpRegisterHibernate(
|
|
VOID
|
|
)
|
|
/*++
|
|
Routine Description:
|
|
|
|
This function registers a hibernation handler (for
|
|
state S4) with the Policy Manager.
|
|
|
|
Arguments:
|
|
|
|
--*/
|
|
{
|
|
POWER_STATE_HANDLER powerState;
|
|
OBJECT_ATTRIBUTES objAttributes;
|
|
PCALLBACK_OBJECT callback;
|
|
UNICODE_STRING callbackName;
|
|
PSYSTEM_POWER_STATE_DISABLE_REASON pReasonNoOSPM,pReasonBios;
|
|
SYSTEM_POWER_LOGGING_ENTRY PowerLoggingEntry;
|
|
NTSTATUS ReasonStatus;
|
|
|
|
PAGED_CODE();
|
|
|
|
//
|
|
// 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 (HalpDisableHibernate == FALSE) {
|
|
|
|
//
|
|
// Register the hibernation handler.
|
|
//
|
|
|
|
powerState.Type = PowerStateSleeping4;
|
|
powerState.RtcWake = FALSE;
|
|
powerState.Handler = &HaliLegacyHibernate;
|
|
powerState.Context = 0;
|
|
|
|
ZwPowerInformation(SystemPowerStateHandler,
|
|
&powerState,
|
|
sizeof(POWER_STATE_HANDLER),
|
|
NULL,
|
|
0);
|
|
} else {
|
|
|
|
//
|
|
// we're not enabling hibernate because there is a hackflag
|
|
// that disallows hibernate. let the power manager know why.
|
|
//
|
|
pReasonBios = ExAllocatePoolWithTag(
|
|
PagedPool,
|
|
sizeof(SYSTEM_POWER_STATE_DISABLE_REASON),
|
|
HAL_POOL_TAG
|
|
);
|
|
if (pReasonBios) {
|
|
RtlZeroMemory(pReasonBios, sizeof(SYSTEM_POWER_STATE_DISABLE_REASON));
|
|
pReasonBios->AffectedState[PowerStateSleeping4] = TRUE;
|
|
pReasonBios->PowerReasonCode = SPSD_REASON_BIOSINCOMPATIBLE;
|
|
|
|
PowerLoggingEntry.LoggingType = LOGGING_TYPE_SPSD;
|
|
PowerLoggingEntry.LoggingEntry = pReasonBios;
|
|
|
|
ReasonStatus = ZwPowerInformation(
|
|
SystemPowerLoggingEntry,
|
|
&PowerLoggingEntry,
|
|
sizeof(PowerLoggingEntry),
|
|
NULL,
|
|
0 );
|
|
|
|
if (ReasonStatus != STATUS_SUCCESS) {
|
|
ExFreePool(pReasonBios);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
//
|
|
//
|
|
// we're on a non-ACPI system (This function is called in an #ifndef
|
|
// ACPI_HAL block.) This means there is no S1-S3 handler registered.
|
|
// If the user wants "standby" support, they must use APM. Let the power
|
|
// manager interface know that we're on a legacy platform so we can inform
|
|
// the user (for instance the user might think they are running in ACPI
|
|
// mode but they are not.)
|
|
//
|
|
pReasonNoOSPM = ExAllocatePoolWithTag(
|
|
PagedPool,
|
|
sizeof(SYSTEM_POWER_STATE_DISABLE_REASON),
|
|
HAL_POOL_TAG
|
|
);
|
|
|
|
if (pReasonNoOSPM) {
|
|
RtlZeroMemory(pReasonNoOSPM, sizeof(SYSTEM_POWER_STATE_DISABLE_REASON));
|
|
pReasonNoOSPM->AffectedState[PowerStateSleeping1] = TRUE;
|
|
pReasonNoOSPM->AffectedState[PowerStateSleeping2] = TRUE;
|
|
pReasonNoOSPM->AffectedState[PowerStateSleeping3] = TRUE;
|
|
pReasonNoOSPM->PowerReasonCode = SPSD_REASON_NOOSPM;
|
|
|
|
PowerLoggingEntry.LoggingType = LOGGING_TYPE_SPSD;
|
|
PowerLoggingEntry.LoggingEntry = pReasonNoOSPM;
|
|
|
|
ReasonStatus = ZwPowerInformation(
|
|
SystemPowerLoggingEntry,
|
|
&PowerLoggingEntry,
|
|
sizeof(PowerLoggingEntry),
|
|
NULL,
|
|
0 );
|
|
|
|
if (ReasonStatus != STATUS_SUCCESS) {
|
|
ExFreePool(pReasonNoOSPM);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
NTSTATUS
|
|
HaliLegacyHibernate (
|
|
IN PVOID Context,
|
|
IN PENTER_STATE_SYSTEM_HANDLER SystemHandler OPTIONAL,
|
|
IN PVOID SystemContext,
|
|
IN LONG NumberProcessors,
|
|
IN volatile PLONG Number
|
|
)
|
|
/*++
|
|
Routine Description:
|
|
|
|
This function is called to hibernate legacy PCs. It saves
|
|
hardware state and waits here for the user to power off the system.
|
|
|
|
Arguments:
|
|
|
|
|
|
--*/
|
|
{
|
|
volatile ULONG ThisProcessor;
|
|
static volatile ULONG Barrier = 0;
|
|
LONG ii;
|
|
KIRQL oldIrql, dummyIrql;
|
|
LOADER_PARAMETER_BLOCK LoaderBlock;
|
|
KPRCB Prcb;
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
BOOLEAN IsMpSystem;
|
|
KAFFINITY SavedActiveProcessors;
|
|
extern ULONG HalpProfileRunning;
|
|
|
|
ASSERT(SystemHandler);
|
|
|
|
ThisProcessor = KeGetPcr()->Prcb->Number;
|
|
|
|
if (ThisProcessor == 0) {
|
|
|
|
HalpHiberInProgress = TRUE;
|
|
|
|
if ((NumberProcessors > 1) &&
|
|
(HalpHiberProcState == NULL)) {
|
|
|
|
//
|
|
// We could not allocate memory to save processor state.
|
|
//
|
|
|
|
HalpHiberInProgress = FALSE;
|
|
}
|
|
}
|
|
|
|
oldIrql = KeGetCurrentIrql();
|
|
|
|
//
|
|
// Wait for all processors to arrive here.
|
|
//
|
|
|
|
InterlockedDecrement(Number);
|
|
while (*Number != 0);
|
|
|
|
if (!HalpHiberInProgress) {
|
|
|
|
//
|
|
// We could not allocate memory to save processor state.
|
|
//
|
|
|
|
return(STATUS_INSUFFICIENT_RESOURCES);
|
|
}
|
|
|
|
|
|
//
|
|
// Save non-boot processor state
|
|
//
|
|
|
|
if (ThisProcessor != 0) {
|
|
|
|
//
|
|
// Save processor state and wait here.
|
|
// N.B. We wait here rather than returning to the kernel because
|
|
// the stack pointer in the saved processor context points to the
|
|
// current stack and we want to resume execution in this routine
|
|
// with our current stack.
|
|
//
|
|
|
|
HalpSaveProcessorStateAndWait(&HalpHiberProcState[ThisProcessor],
|
|
(PULONG)&Barrier);
|
|
|
|
//
|
|
// Barrier will be 0 when we return from this function before
|
|
// hibernating. It will non-zero the second time this
|
|
// function returns.
|
|
//
|
|
// N.B. The non-boot processors will spin in HalpSaveProcessorState
|
|
// until Barrier is zeroed.
|
|
//
|
|
|
|
if (Barrier == 0) {
|
|
return STATUS_DEVICE_DOES_NOT_EXIST;
|
|
} else {
|
|
goto HalpPnHiberResume;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Save motherboard state.
|
|
//
|
|
|
|
HalpSaveDmaControllerState();
|
|
|
|
//
|
|
// Wait for all the non-boot procs to finish saving state.
|
|
//
|
|
|
|
while (Barrier != (ULONG)NumberProcessors - 1);
|
|
|
|
//
|
|
// Change HAL's picture of the world to single processor while
|
|
// the hibernate file is written.
|
|
//
|
|
|
|
SavedActiveProcessors = HalpActiveProcessors;
|
|
HalpActiveProcessors = KeGetCurrentPrcb()->SetMember;
|
|
|
|
//
|
|
// If there's a system handler, invoke it. The system handler will
|
|
// write the hibernation file to disk
|
|
//
|
|
|
|
if (SystemHandler) {
|
|
status = SystemHandler(SystemContext);
|
|
}
|
|
|
|
//
|
|
// Hibernation is over. Boot processor gets control here. The
|
|
// non boot processors are in the state that BIOS left them.
|
|
//
|
|
|
|
HalpActiveProcessors = SavedActiveProcessors;
|
|
Barrier = 0;
|
|
|
|
//
|
|
// If this returns success, then the system is now effectively
|
|
// hibernated. On the other hand, if this function returns something other
|
|
// than success, then it means that we have just un-hibernated,
|
|
// so restore state.
|
|
//
|
|
|
|
|
|
if ((status == STATUS_SUCCESS) ||
|
|
(status == STATUS_DEVICE_DOES_NOT_EXIST)) {
|
|
|
|
return STATUS_DEVICE_DOES_NOT_EXIST;
|
|
}
|
|
|
|
//
|
|
// If you are remapping local apic, io apic and MPS table
|
|
// resources, you first have to unmap the current resources!!!
|
|
// The BIOS may have created the MPS table at a different place or may
|
|
// have changed values like processor local APIC IDs. Reparse it.
|
|
//
|
|
|
|
HalpUnMapIOApics();
|
|
HalpUnMapPhysicalRange(PcMpTablePtr,
|
|
(PcMpTablePtr->TableLength + PcMpTablePtr->ExtTableLength));
|
|
DetectMPS(&IsMpSystem);
|
|
HalpMpInfoTable.NtProcessors = NumberProcessors;
|
|
HalpIpiClock = 0;
|
|
RtlZeroMemory(&LoaderBlock, sizeof(LoaderBlock));
|
|
RtlZeroMemory(&Prcb, sizeof(Prcb));
|
|
LoaderBlock.Prcb = (ULONG) &Prcb;
|
|
|
|
//
|
|
// Reset Processor enumeration (so it starts at the beginning).
|
|
//
|
|
|
|
HalpLastEnumeratedActualProcessor = 0;
|
|
|
|
//
|
|
// Initialize minimum global hardware state needed.
|
|
//
|
|
|
|
HalpInitializeIOUnits();
|
|
HalpInitializePICs(FALSE);
|
|
|
|
//
|
|
// Restore DMA controller state
|
|
//
|
|
|
|
HalpRestoreDmaControllerState();
|
|
|
|
//
|
|
// Initialize boot processor's local APIC so it can wake other processors
|
|
//
|
|
|
|
HalpInitializeLocalUnit ();
|
|
KeRaiseIrql(HIGH_LEVEL, &dummyIrql);
|
|
|
|
//
|
|
// Wake up the other processors
|
|
//
|
|
|
|
for(ii = 1; ii < NumberProcessors; ++ii) {
|
|
|
|
// Set processor number in dummy loader parameter block
|
|
|
|
Prcb.Number = (UCHAR) ii;
|
|
CurTiledCr3LowPart = HalpTiledCr3Addresses[ii].LowPart;
|
|
if (!HalStartNextProcessor(&LoaderBlock, &HalpHiberProcState[ii])) {
|
|
|
|
//
|
|
// We could not start a processor. This is a fatal error but
|
|
// don't bail out yet until you try the remaining processors.
|
|
//
|
|
|
|
DBGMSG("HAL: Cannot start processor after hibernate resume\n");
|
|
}
|
|
}
|
|
|
|
HalpPnHiberResume:
|
|
|
|
//
|
|
// Finish up all the MP stuff that happens across multiple
|
|
// HALs.
|
|
//
|
|
|
|
HalpPostSleepMP(NumberProcessors, Number);
|
|
|
|
if (KeGetPcr()->Prcb->Number == 0) {
|
|
|
|
//
|
|
// Restore the IO APIC state
|
|
//
|
|
|
|
HalpRestoreIoApicRedirTable();
|
|
|
|
if (HalpProfileRunning == 1) {
|
|
HalStartProfileInterrupt(0);
|
|
}
|
|
|
|
}
|
|
|
|
KfLowerIrql(oldIrql);
|
|
|
|
return(status);
|
|
}
|