Windows2003-3790/base/boot/lib/i386/wake.c
2020-09-30 16:53:55 +02:00

730 lines
17 KiB
C

/*++
Copyright (c) 1990, 1991 Microsoft Corporation
Module Name:
wake.c
Abstract:
Author:
Ken Reneris
Environment:
Kernel Mode
Revision History:
Steve Deng (sdeng) 20-Aug-2002
Add support for PAE and Amd64. It is assumed that all the
physical pages are below 4GB. Otherwise hibernation feature
should be disabled.
--*/
#include "arccodes.h"
#include "bootx86.h"
extern PHARDWARE_PTE PDE;
extern PHARDWARE_PTE HalPT;
extern ULONG HiberNoMappings;
extern BOOLEAN HiberIoError;
extern ULONG HiberLastRemap;
extern BOOLEAN HiberOutOfRemap;
extern UCHAR BlpEnablePAEStart;
extern UCHAR BlpEnablePAEEnd;
extern UCHAR BlAmd64SwitchToLongModeStart;
extern UCHAR BlAmd64SwitchToLongModeEnd;
PVOID HiberTransVa;
ULONG64 HiberTransVaAmd64;
ULONG HiberCurrentMapIndex;
#define PDE_SHIFT 22
#define PTE_SHIFT 12
#define PTE_INDEX_MASK 0x3ff
#define PDPT_SHIFT_X86PAE 30
#define PDE_SHIFT_X86PAE 21
#define PTE_SHIFT_X86PAE 12
#define PDE_INDEX_MASK_X86PAE 0x1ff
#define PTE_INDEX_MASK_X86PAE 0x1ff
VOID
HiberSetupForWakeDispatchPAE (
VOID
);
VOID
HiberSetupForWakeDispatchX86 (
VOID
);
VOID
HiberSetupForWakeDispatchAmd64 (
VOID
);
PVOID
HbMapPte (
IN ULONG PteToMap,
IN ULONG Page
)
{
PVOID Va;
Va = (PVOID) (HiberVa + (PteToMap << PAGE_SHIFT));
HbSetPte (Va, HiberPtes, PteToMap, Page);
return Va;
}
PVOID
HbNextSharedPage (
IN ULONG PteToMap,
IN ULONG RealPage
)
/*++
Routine Description:
Allocates the next available page in the free and
maps the Hiber pte to the page. The allocated page
is put onto the remap list
Arguments:
PteToMap - Which Hiber PTE to map
RealPage - The page to enter into the remap table for
this allocation
Return Value:
Virtual address of the mapping
--*/
{
PULONG MapPage;
PULONG RemapPage;
ULONG DestPage;
ULONG i;
MapPage = (PULONG) (HiberVa + (PTE_MAP_PAGE << PAGE_SHIFT));
RemapPage = (PULONG) (HiberVa + (PTE_REMAP_PAGE << PAGE_SHIFT));
//
// Loop until we find a free page which is not in
// use by the loader image, then map it
//
while (HiberCurrentMapIndex < HiberNoMappings) {
DestPage = MapPage[HiberCurrentMapIndex];
HiberCurrentMapIndex += 1;
i = HbPageDisposition (DestPage);
if (i == HbPageInvalid) {
HiberIoError = TRUE;
return HiberBuffer;
}
if (i == HbPageNotInUse) {
MapPage[HiberLastRemap] = DestPage;
RemapPage[HiberLastRemap] = RealPage;
HiberLastRemap += 1;
HiberPageFrames[PteToMap] = DestPage;
return HbMapPte(PteToMap, DestPage);
}
}
HiberOutOfRemap = TRUE;
return HiberBuffer;
}
VOID
HbAllocatePtes (
IN ULONG NumberPages,
OUT PVOID *PteAddress,
OUT PVOID *MappedAddress
)
/*++
Routine Description:
Allocated a consecutive chuck of Ptes.
Arguments:
NumberPage - Number of ptes to allocate
PteAddress - Pointer to the first PTE
MappedAddress - Base VA of the address mapped
--*/
{
ULONG i;
ULONG j;
//
// We use the HAL's PDE for mapping. Find enough free PTEs
//
for (i=0; i<=1024-NumberPages; i++) {
for (j=0; j < NumberPages; j++) {
if ((((PULONG)HalPT))[i+j]) {
break;
}
}
if (j == NumberPages) {
*PteAddress = (PVOID) &HalPT[i];
*MappedAddress = (PVOID) (0xffc00000 | (i<<12));
return ;
}
i += j;
}
BlPrint("NoMem");
while (1);
}
VOID
HbSetPte (
IN PVOID Va,
IN PHARDWARE_PTE Pte,
IN ULONG Index,
IN ULONG PageNumber
)
/*++
Routine Description:
Sets the Pte to the corresponding page address
Arguments:
Va - the virtual address of the physical page described by PageNumber
Pte - the base address of Page Table
Index - the index into Page Table
PageNumber - the page frame number of the pysical page to be mapped
Return:
None
--*/
{
Pte[Index].PageFrameNumber = PageNumber;
Pte[Index].Valid = 1;
Pte[Index].Write = 1;
Pte[Index].WriteThrough = 0;
Pte[Index].CacheDisable = 0;
_asm {
mov eax, Va
invlpg [eax]
}
}
VOID
HbMapPagePAE(
PHARDWARE_PTE_X86PAE HbPdpt,
PVOID VirtualAddress,
ULONG PageFrameNumber
)
/*++
Routine Description:
This functions maps a virtural address into PAE page mapping structure.
Arguments:
HbPdpt - Supplies the base address of Page Directory Pointer Table.
VirtualAddress - Supplies the virtual address to map
PageFrameNumber - Supplies the physical page number to map the address to
Return:
None
--*/
{
ULONG index;
PHARDWARE_PTE_X86PAE HbPde, HbPte;
//
// If the PDPT entry is empty, allocate a new page for it.
// Otherwise we simply map the page at this entry.
//
index = ((ULONG) VirtualAddress) >> PDPT_SHIFT_X86PAE;
if(HbPdpt[index].Valid) {
HbPde = HbMapPte (PTE_TRANSFER_PDE, (ULONG)(HbPdpt[index].PageFrameNumber));
} else {
HbPde = HbNextSharedPage(PTE_TRANSFER_PDE, 0);
RtlZeroMemory (HbPde, PAGE_SIZE);
HbPdpt[index].PageFrameNumber = HiberPageFrames[PTE_TRANSFER_PDE];
HbPdpt[index].Valid = 1;
}
//
// If the PDE entry is empty, allocate a new page for it.
// Otherwise we simply map the page at this entry.
//
index = (((ULONG) VirtualAddress) >> PDE_SHIFT_X86PAE) & PDE_INDEX_MASK_X86PAE;
if(HbPde[index].Valid) {
HbPte = HbMapPte (PTE_WAKE_PTE, (ULONG)(HbPde[index].PageFrameNumber));
} else {
HbPte = HbNextSharedPage(PTE_WAKE_PTE, 0);
RtlZeroMemory (HbPte, PAGE_SIZE);
HbPde[index].PageFrameNumber = HiberPageFrames[PTE_WAKE_PTE];
HbPde[index].Write = 1;
HbPde[index].Valid = 1;
}
//
// Locate the PTE of VirtualAddress and set it to PageFrameNumber
//
index = (((ULONG) VirtualAddress) >> PTE_SHIFT_X86PAE) & PDE_INDEX_MASK_X86PAE;
HbPte[index].PageFrameNumber = PageFrameNumber;
HbPte[index].Write = 1;
HbPte[index].Valid = 1;
}
VOID
HiberSetupForWakeDispatch (
VOID
)
{
if(BlAmd64UseLongMode) {
//
// if system was hibernated in long mode
//
HiberSetupForWakeDispatchAmd64();
} else {
if(BlUsePae) {
//
// if system was hibernated in pae mode
//
HiberSetupForWakeDispatchPAE();
} else {
//
// if system was hibernated in 32-bit x86 mode
//
HiberSetupForWakeDispatchX86();
}
}
}
VOID
HiberSetupForWakeDispatchX86 (
VOID
)
{
PHARDWARE_PTE HbPde;
PHARDWARE_PTE HbPte;
PHARDWARE_PTE WakePte;
PHARDWARE_PTE TransVa;
ULONG TransPde;
ULONG WakePde;
ULONG PteEntry;
//
// Allocate a transistion CR3. A page directory and table which
// contains the hibernation PTEs
//
HbPde = HbNextSharedPage(PTE_TRANSFER_PDE, 0);
HbPte = HbNextSharedPage(PTE_WAKE_PTE, 0); // TRANSFER_PTE, 0);
RtlZeroMemory (HbPde, PAGE_SIZE);
RtlZeroMemory (HbPte, PAGE_SIZE);
//
// Set PDE to point to PTE
//
TransPde = ((ULONG) HiberVa) >> PDE_SHIFT;
HbPde[TransPde].PageFrameNumber = HiberPageFrames[PTE_WAKE_PTE];
HbPde[TransPde].Write = 1;
HbPde[TransPde].Valid = 1;
//
// Fill in the hiber PTEs
//
PteEntry = (((ULONG) HiberVa) >> PTE_SHIFT) & PTE_INDEX_MASK;
TransVa = &HbPte[PteEntry];
RtlCopyMemory (TransVa, HiberPtes, HIBER_PTES * sizeof(HARDWARE_PTE));
//
// Make another copy at the Va of the wake image hiber ptes
//
WakePte = HbPte;
WakePde = ((ULONG) HiberIdentityVa) >> PDE_SHIFT;
if (WakePde != TransPde) {
WakePte = HbNextSharedPage(PTE_WAKE_PTE, 0);
HbPde[WakePde].PageFrameNumber = HiberPageFrames[PTE_WAKE_PTE];
HbPde[WakePde].Write = 1;
HbPde[WakePde].Valid = 1;
}
PteEntry = (((ULONG) HiberIdentityVa) >> PTE_SHIFT) & PTE_INDEX_MASK;
TransVa = &WakePte[PteEntry];
RtlCopyMemory (TransVa, HiberPtes, HIBER_PTES * sizeof(HARDWARE_PTE));
//
// Set TransVa to be relative to the va of the transfer Cr3
//
HiberTransVa = (PVOID) (((PUCHAR) TransVa) - HiberVa + (PUCHAR) HiberIdentityVa);
}
VOID
HiberSetupForWakeDispatchPAE (
VOID
)
/*++
Routine Description:
Set up memory mappings for wake dispatch routine. This mapping will
be used in a "transfer mode" in which the mapping of loader has
already been discarded and the mapping of OS is not restored yet.
For PAE systems, PAE is enabled before entering this "transfer mode".
Arguments:
None
Return:
None
--*/
{
PHARDWARE_PTE_X86PAE HbPdpt;
ULONG i, TransferCR3;
//
// Borrow the pte at PTE_SOURCE temporarily for Pdpt
//
HbPdpt = HbNextSharedPage(PTE_SOURCE, 0);
RtlZeroMemory (HbPdpt, PAGE_SIZE);
TransferCR3 = HiberPageFrames[PTE_SOURCE];
//
// Map in the pages where the code of BlpEnablePAE() resides.
// This has to be a 1-1 mapping, i.e. the value of virtual
// address is the same as the value of physical address
//
HbMapPagePAE( HbPdpt,
(PVOID)(&BlpEnablePAEStart),
(ULONG)(&BlpEnablePAEStart) >> PAGE_SHIFT);
HbMapPagePAE( HbPdpt,
(PVOID)(&BlpEnablePAEEnd),
(ULONG)(&BlpEnablePAEEnd) >> PAGE_SHIFT);
//
// Map in the pages that is reserved for hibernation use
//
for (i = 0; i < HIBER_PTES; i++) {
HbMapPagePAE( HbPdpt,
(PUCHAR)HiberVa + PAGE_SIZE * i,
(*((PULONG)(HiberPtes) + i) >> PAGE_SHIFT));
}
for (i = 0; i < HIBER_PTES; i++) {
HbMapPagePAE( HbPdpt,
(PUCHAR)HiberIdentityVa + PAGE_SIZE * i,
(*((PULONG)(HiberPtes) + i) >> PAGE_SHIFT));
}
//
// Update the value of HiberPageFrames[PTE_TRANSFER_PDE] to the
// TransferCR3. The wake dispatch function will read this entry
// for the CR3 value in transfer mode
//
HiberPageFrames[PTE_TRANSFER_PDE] = TransferCR3;
//
// HiberTransVa points to the PTEs reserved for hibernation code.
// HiberTransVa is similar to HiberPtes but it is relative to
// transfer Cr3
//
HiberTransVa = (PVOID)((PUCHAR) HiberIdentityVa +
(PTE_WAKE_PTE << PAGE_SHIFT) +
sizeof(HARDWARE_PTE_X86PAE) *
(((ULONG_PTR)HiberIdentityVa >> PTE_SHIFT_X86PAE) & PTE_INDEX_MASK_X86PAE));
}
#define AMD64_MAPPING_LEVELS 4
typedef struct _HB_AMD64_MAPPING_INFO {
ULONG PteToUse;
ULONG AddressMask;
ULONG AddressShift;
} CONST HB_AMD64_MAPPING_INFO, *PHB_AMD64_MAPPING_INFO;
HB_AMD64_MAPPING_INFO HbAmd64MappingInfo[AMD64_MAPPING_LEVELS] =
{
{ PTE_WAKE_PTE, 0x1ff, 12 },
{ PTE_TRANSFER_PDE, 0x1ff, 21 },
{ PTE_DEST, 0x1ff, 30 },
{ PTE_SOURCE, 0x1ff, 39 }
};
#define _HARDWARE_PTE_WORKING_SET_BITS 11
typedef struct _HARDWARE_PTE_AMD64 {
ULONG64 Valid : 1;
ULONG64 Write : 1; // UP version
ULONG64 Owner : 1;
ULONG64 WriteThrough : 1;
ULONG64 CacheDisable : 1;
ULONG64 Accessed : 1;
ULONG64 Dirty : 1;
ULONG64 LargePage : 1;
ULONG64 Global : 1;
ULONG64 CopyOnWrite : 1; // software field
ULONG64 Prototype : 1; // software field
ULONG64 reserved0 : 1; // software field
ULONG64 PageFrameNumber : 28;
ULONG64 reserved1 : 24 - (_HARDWARE_PTE_WORKING_SET_BITS+1);
ULONG64 SoftwareWsIndex : _HARDWARE_PTE_WORKING_SET_BITS;
ULONG64 NoExecute : 1;
} HARDWARE_PTE_AMD64, *PHARDWARE_PTE_AMD64;
VOID
HbMapPageAmd64(
PHARDWARE_PTE_AMD64 RootLevelPageTable,
ULONGLONG VirtualAddress,
ULONG PageFrameNumber
)
/*++
Routine Description:
This functions maps a virtural address into Amd64 page mapping structure.
Arguments:
RootLevelPageTable - Supplies the base address of Page-Map Level-4 Table.
VirtualAddress - Supplies the virtual address to map
PageFrameNumber - Supplies the physical page number to be mapped
Return:
None
--*/
{
LONG i;
ULONG index, PteToUse;
PHARDWARE_PTE_AMD64 PageTable, PageTableTmp;
PageTable = RootLevelPageTable;
//
// Build a 4-level mapping top-down
//
for(i = AMD64_MAPPING_LEVELS - 1; i >= 0; i--) {
index = (ULONG)((VirtualAddress >> HbAmd64MappingInfo[i].AddressShift) &
HbAmd64MappingInfo[i].AddressMask);
if (i > 0) {
PteToUse = HbAmd64MappingInfo[i-1].PteToUse;
if (PageTable[index].Valid) {
//
// If a page table entry is valid we simply map the page
// at it and go to the next mapping level.
//
PageTable = HbMapPte(PteToUse,
(ULONG)PageTable[index].PageFrameNumber);
} else {
//
// If a page entry is invalid, we allocate a new page
// from the free page list and reference the new page
// from this page entry.
//
PageTableTmp = HbNextSharedPage(PteToUse, 0);
PageTable[index].PageFrameNumber = HiberPageFrames[PteToUse];
PageTable[index].Valid = 1;
PageTable[index].Write = 1;
RtlZeroMemory (PageTableTmp, PAGE_SIZE);
PageTable = PageTableTmp;
}
} else {
//
// Now we come to the PTE level. Set this PTE entry to the
// target page frame number.
//
PageTable[index].PageFrameNumber = PageFrameNumber;
PageTable[index].Valid = 1;
PageTable[index].Write = 1;
}
}
}
VOID
HiberSetupForWakeDispatchAmd64(
VOID
)
/*++
Routine Description:
This function prepares memory mapping for the transition period
where neither loader's nor kernel's mapping is available. The
processor runs in long mode in transition period.
We'll create mapping for the following virtual addresses
- the code handles long mode switch
- loader's HiberVa
- wake image's HiberVa
Arguments:
None
Return:
None
--*/
{
PHARDWARE_PTE_AMD64 HbTopLevelPTE;
ULONG i;
ULONG TransferCR3;
//
// Borrow the pte at PTE_SOURCE for temporary use here
//
HbTopLevelPTE = HbNextSharedPage(PTE_SOURCE, 0);
RtlZeroMemory (HbTopLevelPTE, PAGE_SIZE);
TransferCR3 = HiberPageFrames[PTE_SOURCE];
//
// Map in the pages that is holding thee code of _BlAmd64SwitchToLongMode.
// This has to be a 1-1 mapping, i.e. the virtual address equals physical
// address
//
HbMapPageAmd64(
HbTopLevelPTE,
(ULONGLONG)(&BlAmd64SwitchToLongModeStart),
(ULONG)(&BlAmd64SwitchToLongModeStart) >> PAGE_SHIFT);
HbMapPageAmd64(
HbTopLevelPTE,
(ULONGLONG)(&BlAmd64SwitchToLongModeEnd),
(ULONG)(&BlAmd64SwitchToLongModeEnd) >> PAGE_SHIFT);
//
// Map in the loader's HiberVa
//
for (i = 0; i < HIBER_PTES; i++) {
HbMapPageAmd64(HbTopLevelPTE,
(ULONGLONG)((ULONG_PTR)HiberVa + PAGE_SIZE * i),
(*((PULONG)(HiberPtes) + i) >> PAGE_SHIFT));
}
//
// Map in the wake image's HiberVa. Note that the hiber pte at
// PTE_WAKE_PTE will be set to the the right value as a result
// of this mapping.
//
for (i = 0; i < HIBER_PTES; i++) {
HbMapPageAmd64(HbTopLevelPTE,
HiberIdentityVaAmd64 + PAGE_SIZE * i,
(*((PULONG)(HiberPtes) + i) >> PAGE_SHIFT));
}
//
// Update the value of HiberPageFrames[PTE_TRANSFER_PDE] to the
// TransferCR3. The wake dispatch function will read this entry
// for the CR3 value in transition mode
//
HiberPageFrames[PTE_TRANSFER_PDE] = TransferCR3;
//
// HiberTransVaAmd64 points to the wake image's hiber ptes. It is
// is relative to transfer Cr3
//
HiberTransVaAmd64 = HiberIdentityVaAmd64 +
(PTE_WAKE_PTE << PAGE_SHIFT) +
sizeof(HARDWARE_PTE_AMD64) * ((HiberIdentityVaAmd64 >> HbAmd64MappingInfo[0].AddressShift) & HbAmd64MappingInfo[0].AddressMask);
}