2020-09-30 16:53:55 +02:00

4170 lines
99 KiB
C

/*++
Copyright (c) 1991-2000 Microsoft Corporation
Module Name:
amd64x86.c
Abstract:
This module contains routines necessary to support loading and
transitioning into an AMD64 kernel. The code in this module has
access to x86-specific defines found in i386.h but not to amd64-
specific declarations found in amd64.h.
Author:
Forrest Foltz (forrestf) 20-Apr-2000
Environment:
Revision History:
--*/
#include "amd64prv.h"
#include <pcmp.inc>
#include <ntapic.inc>
#if defined(ROUND_UP)
#undef ROUND_UP
#endif
#include "cmp.h"
#include "arc.h"
#define WANT_BLDRTHNK_FUNCTIONS
#define COPYBUF_MALLOC BlAllocateHeap
#include <amd64thk.h>
#define IMAGE_DEFINITIONS 0
#include <ximagdef.h>
//
// Warning 4152 is "nonstandard extension, function/data pointer conversion
//
#pragma warning(disable:4152)
//
// Private, tempory memory descriptor type
//
#define LoaderAmd64MemoryData (LoaderMaximum + 10)
//
// Array of 64-bit memory descriptors
//
PMEMORY_ALLOCATION_DESCRIPTOR_64 BlAmd64DescriptorArray;
LONG BlAmd64DescriptorArraySize;
//
// Forward declarations for functions local to this module
//
ARC_STATUS
BlAmd64AllocateMemoryAllocationDescriptors(
VOID
);
ARC_STATUS
BlAmd64BuildLdrDataTableEntry64(
IN PLDR_DATA_TABLE_ENTRY DataTableEntry32,
OUT PLDR_DATA_TABLE_ENTRY_64 *DataTableEntry64
);
ARC_STATUS
BlAmd64BuildLoaderBlock64(
VOID
);
ARC_STATUS
BlAmd64BuildLoaderBlockExtension64(
VOID
);
ARC_STATUS
BlAmd64BuildMappingPhase1(
VOID
);
ARC_STATUS
BlAmd64BuildMappingPhase2(
VOID
);
ARC_STATUS
BlAmd64BuildMappingWorker(
VOID
);
BOOLEAN
BlAmd64ContainsResourceList(
IN PCONFIGURATION_COMPONENT_DATA ComponentData32,
OUT PULONG ResourceListSize64
);
ARC_STATUS
BlAmd64FixSharedUserPage(
VOID
);
BOOLEAN
BlAmd64IsPageMapped(
IN ULONG Va,
OUT PFN_NUMBER *Pfn,
OUT PBOOLEAN PageTableMapped
);
ARC_STATUS
BlAmd64PrepareSystemStructures(
VOID
);
VOID
BlAmd64ReplaceMemoryDescriptorType(
IN TYPE_OF_MEMORY Target,
IN TYPE_OF_MEMORY Replacement,
IN BOOLEAN Coallesce
);
VOID
BlAmd64ResetPageTableHeap(
VOID
);
VOID
BlAmd64SwitchToLongMode(
VOID
);
ARC_STATUS
BlAmd64TransferArcDiskInformation(
VOID
);
ARC_STATUS
BlAmd64TransferBootDriverNodes(
VOID
);
ARC_STATUS
BlAmd64TransferConfigurationComponentData(
VOID
);
PCONFIGURATION_COMPONENT_DATA_64
BlAmd64TransferConfigWorker(
IN PCONFIGURATION_COMPONENT_DATA ComponentData32,
IN PCONFIGURATION_COMPONENT_DATA_64 ComponentDataParent64
);
ARC_STATUS
BlAmd64TransferHardwareIdList(
IN PPNP_HARDWARE_ID HardwareId,
OUT POINTER64 *HardwareIdDatabaseList64
);
ARC_STATUS
BlAmd64TransferLoadedModuleState(
VOID
);
ARC_STATUS
BlAmd64TransferMemoryAllocationDescriptors(
VOID
);
ARC_STATUS
BlAmd64TransferNlsData(
VOID
);
VOID
BlAmd64TransferResourceList(
IN PCONFIGURATION_COMPONENT_DATA ComponentData32,
OUT PCONFIGURATION_COMPONENT_DATA_64 ComponentData64
);
ARC_STATUS
BlAmd64TransferSetupLoaderBlock(
VOID
);
#if DBG
PCHAR BlAmd64MemoryDescriptorText[] = {
"LoaderExceptionBlock",
"LoaderSystemBlock",
"LoaderFree",
"LoaderBad",
"LoaderLoadedProgram",
"LoaderFirmwareTemporary",
"LoaderFirmwarePermanent",
"LoaderOsloaderHeap",
"LoaderOsloaderStack",
"LoaderSystemCode",
"LoaderHalCode",
"LoaderBootDriver",
"LoaderConsoleInDriver",
"LoaderConsoleOutDriver",
"LoaderStartupDpcStack",
"LoaderStartupKernelStack",
"LoaderStartupPanicStack",
"LoaderStartupPcrPage",
"LoaderStartupPdrPage",
"LoaderRegistryData",
"LoaderMemoryData",
"LoaderNlsData",
"LoaderSpecialMemory",
"LoaderBBTMemory",
"LoaderReserve"
};
#endif
VOID
NSUnmapFreeDescriptors(
IN PLIST_ENTRY ListHead
);
//
// Data declarations
//
PLOADER_PARAMETER_BLOCK BlAmd64LoaderBlock32;
PLOADER_PARAMETER_BLOCK_64 BlAmd64LoaderBlock64;
//
// Pointer to the top of the 64-bit stack frame to use upon transition
// to long mode.
//
POINTER64 BlAmd64IdleStack64;
//
// GDT and IDT pseudo-descriptor for use with LGDT/LIDT
//
DESCRIPTOR_TABLE_DESCRIPTOR BlAmd64GdtDescriptor;
DESCRIPTOR_TABLE_DESCRIPTOR BlAmd64IdtDescriptor;
DESCRIPTOR_TABLE_DESCRIPTOR BlAmd32GdtDescriptor;
//
// 64-bit pointers to the loader parameter block and kernel
// entry routine
//
POINTER64 BlAmd64LoaderParameterBlock;
POINTER64 BlAmd64KernelEntry;
//
// A private list of page tables used to build the long mode paging
// structures is kept. This is in order to avoid memory allocations while
// the structures are being assembled.
//
// The PT_NODE type as well as the BlAmd64FreePfnList and BlAmd64BusyPfnList
// globals are used to that end.
//
typedef struct _PT_NODE *PPT_NODE;
typedef struct _PT_NODE {
PPT_NODE Next;
PAMD64_PAGE_TABLE PageTable;
} PT_NODE;
PPT_NODE BlAmd64FreePfnList = NULL;
PPT_NODE BlAmd64BusyPfnList = NULL;
//
// Indicate if the system is waking up from hibernate
//
ULONG HiberInProgress = 0;
//
// External data
//
extern ULONG64 BlAmd64_LOCALAPIC;
ARC_STATUS
BlAmd64MapMemoryRegion(
IN ULONG RegionVa,
IN ULONG RegionSize
)
/*++
Routine Description:
This function creates long mode mappings for all valid x86 mappings
within the region described by RegionVa and RegionSize.
Arguments:
RegionVa - Supplies the starting address of the VA region.
RegionSize - Supplies the size of the VA region.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
ULONG va32;
ULONG va32End;
POINTER64 va64;
ARC_STATUS status;
PFN_NUMBER pfn;
BOOLEAN pageMapped;
BOOLEAN pageTableMapped;
ULONG increment;
va32 = RegionVa;
va32End = va32 + RegionSize;
while (va32 < va32End) {
pageMapped = BlAmd64IsPageMapped( va32, &pfn, &pageTableMapped );
if (pageTableMapped != FALSE) {
//
// The page table corresponding to this address is present.
//
if (pageMapped != FALSE) {
//
// The page corresponding to this address is present.
//
if ((va32 & KSEG0_BASE_X86) != 0) {
//
// The address lies within the X86 KSEG0 region. Map
// it to the corresponding address within the AMD64
// KSEG0 region.
//
va64 = PTR_64( (PVOID)va32 );
} else {
//
// Map the VA directly.
//
va64 = (POINTER64)va32;
}
//
// Now create the mapping in the AMD64 page table structure.
//
status = BlAmd64CreateMapping( va64, pfn );
if (status != ESUCCESS) {
return status;
}
}
//
// Check the next page.
//
increment = PAGE_SIZE;
} else {
//
// Not only is the page not mapped but neither is the page table.
// Skip to the next page table address boundary.
//
increment = 1 << PDI_SHIFT;
}
//
// Advance to the next VA to check, checking for overflow.
//
va32 = (va32 + increment) & ~(increment - 1);
if (va32 == 0) {
break;
}
}
return ESUCCESS;
}
BOOLEAN
BlAmd64IsPageMapped(
IN ULONG Va,
OUT PFN_NUMBER *Pfn,
OUT PBOOLEAN PageTableMapped
)
/*++
Routine Description:
This function accepts a 32-bit virtual address, determines whether it
is a valid address, and if so returns the Pfn associated with it.
Addresses that are within the recursive mapping are treated as NOT
mapped.
Arguments:
None.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
ULONG pdeIndex;
ULONG pteIndex;
PHARDWARE_PTE pde;
PHARDWARE_PTE pte;
BOOLEAN dummy;
PBOOLEAN pageTableMapped;
//
// Point the output parameter pointer as appropriate.
//
if (ARGUMENT_PRESENT(PageTableMapped)) {
pageTableMapped = PageTableMapped;
} else {
pageTableMapped = &dummy;
}
//
// Pages that are a part of the X86 32-bit mapping structure ARE
// IGNORED.
//
if (Va >= PTE_BASE && Va <= PTE_TOP) {
*pageTableMapped = TRUE;
return FALSE;
}
//
// Determine whether the mapping PDE is present
//
pdeIndex = Va >> PDI_SHIFT;
pde = &((PHARDWARE_PTE)PDE_BASE)[ pdeIndex ];
if (pde->Valid == 0) {
*pageTableMapped = FALSE;
return FALSE;
}
//
// Indicate that the page table for this address is mapped.
//
*pageTableMapped = TRUE;
//
// It is, now get the page present status
//
pteIndex = Va >> PTI_SHIFT;
pte = &((PHARDWARE_PTE)PTE_BASE)[ pteIndex ];
if (pte->Valid == 0) {
return FALSE;
}
*Pfn = pte->PageFrameNumber;
return TRUE;
}
PAMD64_PAGE_TABLE
BlAmd64AllocatePageTable(
VOID
)
/*++
Routine Description:
This function allocates and initializes a PAGE_TABLE structure.
Arguments:
None.
Return Value:
Returns a pointer to the allocated page table structure, or NULL
if the allocation failed.
--*/
{
ARC_STATUS status;
ULONG descriptor;
PPT_NODE ptNode;
PAMD64_PAGE_TABLE pageTable;
//
// Pull a page table off of the free list, if one exists
//
ptNode = BlAmd64FreePfnList;
if (ptNode != NULL) {
BlAmd64FreePfnList = ptNode->Next;
} else {
//
// The free page table list is empty, allocate a new
// page table and node to track it with.
//
status = BlAllocateDescriptor( LoaderAmd64MemoryData,
0,
1,
&descriptor );
if (status != ESUCCESS) {
return NULL;
}
ptNode = BlAllocateHeap( sizeof(PT_NODE) );
if (ptNode == NULL) {
return NULL;
}
ptNode->PageTable = (PAMD64_PAGE_TABLE)(descriptor << PAGE_SHIFT);
}
ptNode->Next = BlAmd64BusyPfnList;
BlAmd64BusyPfnList = ptNode;
pageTable = ptNode->PageTable;
RtlZeroMemory( pageTable, PAGE_SIZE );
return pageTable;
}
ARC_STATUS
BlAmd64TransferToKernel(
IN PTRANSFER_ROUTINE SystemEntry,
IN PLOADER_PARAMETER_BLOCK BlLoaderBlock
)
/*++
Routine Description:
This routine prepares the AMD64 data structures required for kernel
execution, including page table structures and 64-bit loader block,
and transfers control to the kernel.
This routine returns only upon an error.
Arguments:
SystemEntry - Pointer to the kernel entry point.
BlLoaderBlock - Pointer to the 32-bit loader block structure.
Return Value:
No return on success. On failure, returns the status of the operation.
--*/
{
UNREFERENCED_PARAMETER( BlLoaderBlock );
BlAmd64LoaderParameterBlock = PTR_64(BlAmd64LoaderBlock64);
BlAmd64KernelEntry = PTR_64(SystemEntry);
BlAmd64SwitchToLongMode();
return EINVAL;
}
ARC_STATUS
BlAmd64PrepForTransferToKernelPhase1(
IN PLOADER_PARAMETER_BLOCK BlLoaderBlock
)
/*++
Routine Description:
This routine prepares the AMD64 data structures required for kernel
execution, including page table structures and 64-bit loader block.
This is the first of two phases of preperation. This phase is executed
while heap and descriptor allocations are still permitted.
Arguments:
BlLoaderBlock - Pointer to the 32-bit loader block structure.
Return Value:
No return on success. On failure, returns the status of the operation.
--*/
{
ARC_STATUS status;
//
// This is the main routine called to do preperatory work before
// transitioning into the AMD64 kernel.
//
BlAmd64LoaderBlock32 = BlLoaderBlock;
//
// Build a 64-bit copy of the loader parameter block.
//
status = BlAmd64BuildLoaderBlock64();
if (status != ESUCCESS) {
return status;
}
//
// Process the loaded modules.
//
status = BlAmd64TransferLoadedModuleState();
if (status != ESUCCESS) {
return status;
}
//
// Next the boot driver nodes
//
status = BlAmd64TransferBootDriverNodes();
if (status != ESUCCESS) {
return status;
}
//
// NLS data
//
status = BlAmd64TransferNlsData();
if (status != ESUCCESS) {
return status;
}
//
// Configuration component data tree
//
status = BlAmd64TransferConfigurationComponentData();
if (status != ESUCCESS) {
return status;
}
//
// ARC disk information
//
status = BlAmd64TransferArcDiskInformation();
if (status != ESUCCESS) {
return status;
}
//
// Setup loader block
//
status = BlAmd64TransferSetupLoaderBlock();
if (status != ESUCCESS) {
return status;
}
//
// Allocate structures needed by the kernel: TSS, stacks etc.
//
status = BlAmd64PrepareSystemStructures();
if (status != ESUCCESS) {
return status;
}
//
// Mark the descriptor for the shared user page so that it will
// not be freed by the kernel.
//
status = BlAmd64FixSharedUserPage();
if (status != ESUCCESS) {
return status;
}
//
// Pre-allocate any pages needed for the long mode paging structures.
//
status = BlAmd64BuildMappingPhase1();
if (status != ESUCCESS) {
return status;
}
//
// Pre-allocate the 64-bit memory allocation descriptors that will be
// used by BlAmd64TransferMemoryAllocationDescriptors().
//
status = BlAmd64AllocateMemoryAllocationDescriptors();
if (status != ESUCCESS) {
return status;
}
return status;
}
VOID
BlAmd64PrepForTransferToKernelPhase2(
IN PLOADER_PARAMETER_BLOCK BlLoaderBlock
)
/*++
Routine Description:
This routine prepares the AMD64 data structures required for kernel
execution, including page table structures and 64-bit loader block.
This is the second of two phases of preperation. This phase is executed
after the 32-bit page tables have been purged of any unused mappings.
Note that descriptor and heap allocations are not permitted at this
point. Any necessary storage must have been preallocated during phase 1.
Arguments:
BlLoaderBlock - Pointer to the 32-bit loader block structure.
Return Value:
No return on success. On failure, returns the status of the operation.
--*/
{
PLOADER_PARAMETER_EXTENSION_64 extension;
ARC_STATUS status;
UNREFERENCED_PARAMETER( BlLoaderBlock );
//
// At this point everything has been preallocated, nothing can fail.
//
status = BlAmd64BuildMappingPhase2();
ASSERT(status == ESUCCESS);
//
// Transfer the memory descriptor state.
//
status = BlAmd64TransferMemoryAllocationDescriptors();
ASSERT(status == ESUCCESS);
//
// Set LoaderPagesSpanned in the 64-bit loader block.
//
extension = PTR_32(BlAmd64LoaderBlock64->Extension);
extension->LoaderPagesSpanned = BlHighestPage+1;
}
ARC_STATUS
BlAmd64BuildMappingPhase1(
VOID
)
/*++
Routine Description:
This routine performs the first of the two-phase long mode mapping
structure creation process now, while memory allocations are still
possible. It simply calls BlAmd64BuilMappingWorker() which in fact
creates the mapping structures, and (more importantly) allocates all
of the page tables required to do so.
Arguments:
None.
Return Value:
None.
--*/
{
ARC_STATUS status;
//
// While it is possible to perform memory allocations, reserve enough
// page tables to build the AMD64 paging structures.
//
// The easiest way to calculate the maximum number of pages needed is
// to actually build the structures. We do that now with the first of
// two calls to BlAmd64BuildMappingWorker().
//
status = BlAmd64BuildMappingWorker();
if (status != ESUCCESS) {
return status;
}
return ESUCCESS;
}
ARC_STATUS
BlAmd64BuildMappingPhase2(
VOID
)
/*++
Routine Description:
This routine performs the second of the two-phase long mode mapping
structure creation process. All page tables will have been preallocated
as a result of the work performed by BlAmd64BuildMappingPhase1().
Arguments:
None.
Return Value:
None.
--*/
{
ARC_STATUS status;
//
// Reset the Amd64 paging structures
//
BlAmd64ResetPageTableHeap();
//
// All necessary page tables can now be found on BlAmd64FreePfnList.
// On this, the second call to BlAmd64BuildMappingWorker(), those are the
// pages that will be used to perform the mapping.
//
status = BlAmd64BuildMappingWorker();
if (status != ESUCCESS) {
return status;
}
return ESUCCESS;
}
ARC_STATUS
BlAmd64BuildMappingWorker(
VOID
)
/*++
Routine Description:
This routine creates any necessary memory mappings in the long-mode
page table structure. It is called twice, once from
BlAmd64BuildMappingPhase1() and again from BlAmd64BuildMappingPhase2().
Any additional memory mapping that must be carried out should go in
this routine.
Arguments:
None.
Return Value:
None.
--*/
{
ARC_STATUS status;
PFN_NUMBER pfn;
//
// Any long mode mapping code goes here. This routine is called twice:
// once from BlAmd64BuildMappingPhase1(), and again from
// BlAmd64BuildMappingPhase2().
//
//
// Transfer any mappings in the first 32MB of identity mapping.
//
status = BlAmd64MapMemoryRegion( 0,
32 * 1024 * 1024 );
if (status != ESUCCESS) {
return status;
}
//
// Transfer any mappings in the 1GB region starting at KSEG0_BASE_X86.
//
status = BlAmd64MapMemoryRegion( KSEG0_BASE_X86,
0x40000000 );
if (status != ESUCCESS) {
return status;
}
//
// "Map" the HAL va
//
status = BlAmd64MapHalVaSpace();
if (status != ESUCCESS) {
return status;
}
//
// Map the shared user data page
//
BlAmd64IsPageMapped( KI_USER_SHARED_DATA, &pfn, NULL );
status = BlAmd64CreateMapping( KI_USER_SHARED_DATA_64, pfn );
if (status != ESUCCESS) {
return status;
}
return ESUCCESS;
}
VOID
BlAmd64ResetPageTableHeap(
VOID
)
/*++
Routine Description:
This function is called as part of the two-phase page table creation
process. Its purpose is to move all of the PFNs required to build
the long mode page tables back to the free list, and to otherwise
initialize the long mode paging structure.
Arguments:
None.
Return Value:
None.
--*/
{
PPT_NODE ptNodeLast;
//
// Move the page table nodes from the busy list to the free list.
//
if (BlAmd64BusyPfnList != NULL) {
//
// A tail pointer is not kept, so find the tail node here.
//
ptNodeLast = BlAmd64BusyPfnList;
while (ptNodeLast->Next != NULL) {
ptNodeLast = ptNodeLast->Next;
}
ptNodeLast->Next = BlAmd64FreePfnList;
BlAmd64FreePfnList = BlAmd64BusyPfnList;
BlAmd64BusyPfnList = NULL;
}
//
// Zero the top-level pte declared in amd64.c
//
BlAmd64ClearTopLevelPte();
}
ARC_STATUS
BlAmd64TransferHardwareIdList(
IN PPNP_HARDWARE_ID HardwareId,
OUT POINTER64 *HardwareIdDatabaseList64
)
/*++
Routine Description:
This routine walks the singly-linked list of PNP_HARDWARE_ID structures
and for each one found, creates a 64-bit PNP_HARDWARE_ID_64 structure and
inserts it on a list of same.
The resultant 64-bit list is in the same order as the supplied 32-bit
list.
Arguments:
HardwareId - Supplies a pointer to the head of the singly-linked list of
PNP_HARDWARE_ID structures.
HardwareIdDatabaseList64 -
Supplies a pointer to a POINTER64 which upon successful
completion of this routine will contain a 64-bit KSEG0
pointer to the created 64-bit PNP_HARDWARE_ID_64 list.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
PPNP_HARDWARE_ID_64 hardwareId64;
ARC_STATUS status;
//
// Walk the id list backwards. To do this we call ourselves
// recursively until we find the end of the list, then process the nodes
// on the way back up.
//
if (HardwareId == NULL) {
return ESUCCESS;
}
status = BlAmd64TransferHardwareIdList( HardwareId->Next,
HardwareIdDatabaseList64 );
if (status != ESUCCESS) {
return status;
}
hardwareId64 = BlAllocateHeap(sizeof(PNP_HARDWARE_ID_64));
if (hardwareId64 == NULL) {
return ENOMEM;
}
status = Copy_PNP_HARDWARE_ID( HardwareId, hardwareId64 );
if (status != ESUCCESS) {
return status;
}
//
// Link it into the front of the 64-bit list.
//
hardwareId64->Next = *HardwareIdDatabaseList64;
*HardwareIdDatabaseList64 = PTR_64(hardwareId64);
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferDeviceRegistryList(
IN PDETECTED_DEVICE_REGISTRY DetectedDeviceRegistry32,
OUT POINTER64 *DetectedDeviceRegistry64
)
/*++
Routine Description:
This routine walks the singly-linked list of DETECTED_DEVICE_REGISTRY
structures and for each one found, creates a 64-bit
DETECTED_DEVICE_REGISTRY_64 structure and inserts it on a list of same.
The resultant 64-bit list is in the same order as the supplied 32-bit
list.
Arguments:
DetectedDeviceRegistry32 - Supplies a pointer to the head of the singly-linked list of
DETECTED_DEVICE_REGISTRY structures.
DetectedDeviceRegistry64 -
Supplies a pointer to a POINTER64 which upon successful
completion of this routine will contain a 64-bit KSEG0
pointer to the created 64-bit DETECTED_DEVICE_REGISTRY_64
list.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
PDETECTED_DEVICE_REGISTRY_64 registry64;
ARC_STATUS status;
//
// Walk the registry list backwards. To do this we call ourselves
// recursively until we find the end of the list, then process the nodes
// on the way back up.
//
if (DetectedDeviceRegistry32 == NULL) {
return ESUCCESS;
}
status = BlAmd64TransferDeviceRegistryList( DetectedDeviceRegistry32->Next,
DetectedDeviceRegistry64 );
if (status != ESUCCESS) {
return status;
}
//
// Allocate a 64-bit registry structure and copy the contents
// of the 32-bit one in.
//
registry64 = BlAllocateHeap(sizeof(DETECTED_DEVICE_REGISTRY_64));
if (registry64 == NULL) {
return ENOMEM;
}
status = Copy_DETECTED_DEVICE_REGISTRY( DetectedDeviceRegistry32, registry64 );
if (status != ESUCCESS) {
return status;
}
//
// Link it into the front of the 64-bit list.
//
registry64->Next = *DetectedDeviceRegistry64;
*DetectedDeviceRegistry64 = PTR_64(registry64);
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferDeviceFileList(
IN PDETECTED_DEVICE_FILE DetectedDeviceFile32,
OUT POINTER64 *DetectedDeviceFile64
)
/*++
Routine Description:
This routine walks the singly-linked list of DETECTED_DEVICE_FILE
structures and for each one found, creates a 64-bit
DETECTED_DEVICE_FILE_64 structure and inserts it on a list of same.
The resultant 64-bit list is in the same order as the supplied 32-bit
list.
Arguments:
DetectedDeviceFile32 - Supplies a pointer to the head of the singly-linked
list of DETECTED_DEVICE_FILE structures.
DetectedDeviceFile64 -
Supplies a pointer to a POINTER64 which upon successful
completion of this routine will contain a 64-bit KSEG0
pointer to the created 64-bit DETECTED_DEVICE_FILE_64
list.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
PDETECTED_DEVICE_FILE_64 file64;
ARC_STATUS status;
//
// Walk the file list backwards. To do this we call ourselves
// recursively until we find the end of the list, then process the nodes
// on the way back up.
//
if (DetectedDeviceFile32 == NULL) {
return ESUCCESS;
}
status = BlAmd64TransferDeviceFileList( DetectedDeviceFile32->Next,
DetectedDeviceFile64 );
if (status != ESUCCESS) {
return status;
}
//
// Allocate a 64-bit file structure and copy the contents
// of the 32-bit one in.
//
file64 = BlAllocateHeap(sizeof(DETECTED_DEVICE_FILE_64));
if (file64 == NULL) {
return ENOMEM;
}
status = Copy_DETECTED_DEVICE_FILE( DetectedDeviceFile32, file64 );
if (status != ESUCCESS) {
return status;
}
//
// Transfer the singly-linked list of DETECTED_DEVICE_REGISTRY structures
// linked to this DETECTED_DEVICE_FILE structure.
//
status = BlAmd64TransferDeviceRegistryList(
DetectedDeviceFile32->RegistryValueList,
&file64->RegistryValueList );
if (status != ESUCCESS) {
return status;
}
//
// Link it into the front of the 64-bit list.
//
file64->Next = *DetectedDeviceFile64;
*DetectedDeviceFile64 = PTR_64(file64);
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferDeviceList(
IN PDETECTED_DEVICE DetectedDevice32,
OUT POINTER64 *DetectedDeviceList64
)
/*++
Routine Description:
This routine walks the singly-linked list of DETECTED_DEVICE
structures and for each one found, creates a 64-bit
DETECTED_DEVICE_64 structure and inserts it on a list of same.
The resultant 64-bit list is in the same order as the supplied 32-bit
list.
Arguments:
DetectedDevice32 - Supplies a pointer to the head of the singly-linked
list of DETECTED_DEVICE structures.
DetectedDeviceList64 -
Supplies a pointer to a POINTER64 which upon successful
completion of this routine will contain a 64-bit KSEG0
pointer to the created 64-bit DETECTED_DEVICE_64
list.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
PDETECTED_DEVICE_64 device64;
ARC_STATUS status;
//
// Walk the device list backwards. To do this we call ourselves
// recursively until we find the end of the list, then process the nodes
// on the way back up.
//
if (DetectedDevice32 == NULL) {
return ESUCCESS;
}
status = BlAmd64TransferDeviceList( DetectedDevice32->Next,
DetectedDeviceList64 );
if (status != ESUCCESS) {
return status;
}
//
// Allocate a 64-bit device structure and copy the contents
// of the 32-bit one in.
//
device64 = BlAllocateHeap(sizeof(DETECTED_DEVICE_64));
if (device64 == NULL) {
return ENOMEM;
}
status = Copy_DETECTED_DEVICE( DetectedDevice32, device64 );
if (status != ESUCCESS) {
return status;
}
//
// Transfer any PROTECTED_DEVICE_FILE structures
//
status = BlAmd64TransferDeviceFileList( DetectedDevice32->Files,
&device64->Files );
if (status != ESUCCESS) {
return status;
}
//
// Link it into the front of the 64-bit list.
//
device64->Next = *DetectedDeviceList64;
*DetectedDeviceList64 = PTR_64(device64);
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferSetupLoaderBlock(
VOID
)
/*++
Routine Description:
This routine creates a SETUP_LOADER_BLOCK_64 structure that is the
equivalent of the 32-bit SETUP_LOADER_BLOCK structure referenced within
the 32-bit setup loader block.
Arguments:
None.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
PSETUP_LOADER_BLOCK setupBlock32;
PSETUP_LOADER_BLOCK_64 setupBlock64;
ARC_STATUS status;
setupBlock32 = BlAmd64LoaderBlock32->SetupLoaderBlock;
if (setupBlock32 == NULL) {
return ESUCCESS;
}
setupBlock64 = BlAllocateHeap(sizeof(SETUP_LOADER_BLOCK_64));
if (setupBlock64 == NULL) {
return ENOMEM;
}
status = Copy_SETUP_LOADER_BLOCK( setupBlock32, setupBlock64 );
if (status != ESUCCESS) {
return status;
}
{
#define TRANSFER_DEVICE_LIST(x) \
setupBlock64->x = PTR_64(NULL); \
status = BlAmd64TransferDeviceList( setupBlock32->x, \
&setupBlock64->x ); \
if (status != ESUCCESS) return status;
TRANSFER_DEVICE_LIST(KeyboardDevices);
TRANSFER_DEVICE_LIST(ScsiDevices);
TRANSFER_DEVICE_LIST(BootBusExtenders);
TRANSFER_DEVICE_LIST(BusExtenders);
TRANSFER_DEVICE_LIST(InputDevicesSupport);
#undef TRANSFER_DEVICE_LIST
}
setupBlock64->HardwareIdDatabase = PTR_64(NULL);
status = BlAmd64TransferHardwareIdList( setupBlock32->HardwareIdDatabase,
&setupBlock64->HardwareIdDatabase );
if (status != ESUCCESS) {
return status;
}
BlAmd64LoaderBlock64->SetupLoaderBlock = PTR_64(setupBlock64);
return status;
}
ARC_STATUS
BlAmd64TransferArcDiskInformation(
VOID
)
/*++
Routine Description:
This routine creates an ARC_DISK_INFORMATION_64 structure that is the
equivalent of the 32-bit ARC_DISK_INFORMATION structure referenced within
the 32-bit loader block.
Arguments:
None.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
ARC_STATUS status;
PLIST_ENTRY listHead;
PLIST_ENTRY listEntry;
PARC_DISK_INFORMATION diskInfo32;
PARC_DISK_INFORMATION_64 diskInfo64;
PARC_DISK_SIGNATURE diskSignature32;
PARC_DISK_SIGNATURE_64 diskSignature64;
//
// Create a 64-bit ARC_DISK_INFORMATION structure
//
diskInfo32 = BlAmd64LoaderBlock32->ArcDiskInformation;
if (diskInfo32 == NULL) {
return ESUCCESS;
}
diskInfo64 = BlAllocateHeap(sizeof(ARC_DISK_INFORMATION_64));
if (diskInfo64 == NULL) {
return ENOMEM;
}
status = Copy_ARC_DISK_INFORMATION( diskInfo32, diskInfo64 );
if (status != ESUCCESS) {
return status;
}
InitializeListHead64( &diskInfo64->DiskSignatures );
//
// Walk the 32-bit list of ARC_DISK_SIGNATURE nodes and create
// a 64-bit version of each
//
listHead = &diskInfo32->DiskSignatures;
listEntry = listHead->Flink;
while (listEntry != listHead) {
diskSignature32 = CONTAINING_RECORD( listEntry,
ARC_DISK_SIGNATURE,
ListEntry );
diskSignature64 = BlAllocateHeap(sizeof(ARC_DISK_SIGNATURE_64));
if (diskSignature64 == NULL) {
return ENOMEM;
}
status = Copy_ARC_DISK_SIGNATURE( diskSignature32, diskSignature64 );
if (status != ESUCCESS) {
return status;
}
InsertTailList64( &diskInfo64->DiskSignatures,
&diskSignature64->ListEntry );
listEntry = listEntry->Flink;
}
BlAmd64LoaderBlock64->ArcDiskInformation = PTR_64(diskInfo64);
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferConfigurationComponentData(
VOID
)
/*++
Routine Description:
This routine creates a CONFIGURATION_COMPONENT_DATA_64 structure tree
that is the equivalent of the 32-bit CONFIGURATION_COMPONENT_DATA
structure tree referenced within the 32-bit loader block.
Arguments:
None.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
PCONFIGURATION_COMPONENT_DATA_64 rootComponent64;
if (BlAmd64LoaderBlock32->ConfigurationRoot == NULL) {
return ESUCCESS;
}
rootComponent64 =
BlAmd64TransferConfigWorker( BlAmd64LoaderBlock32->ConfigurationRoot,
NULL );
if (rootComponent64 == NULL) {
return ENOMEM;
}
BlAmd64LoaderBlock64->ConfigurationRoot = PTR_64(rootComponent64);
return ESUCCESS;
}
PCONFIGURATION_COMPONENT_DATA_64
BlAmd64TransferConfigWorker(
IN PCONFIGURATION_COMPONENT_DATA ComponentData32,
IN PCONFIGURATION_COMPONENT_DATA_64 ComponentDataParent64
)
/*++
Routine Description:
Given a 32-bit CONFIGURATION_COMPONENT_DATA structure, this routine
creates an equivalent 64-bit CONFIGURATION_COMPONENT_DATA structure
for the supplied structure, as well as for all of its children and
siblings.
This routine calls itself recursively for each sibling and child.
Arguments:
ComponentData32 - Supplies a pointer to the 32-bit structure to transfer.
ComponentDataParent64 - Supplies a pointer to the current 64-bit parent
structure.
Return Value:
Returns a pointer to the created 64-bit structure, or NULL if a failure
was encountered.
--*/
{
ARC_STATUS status;
ULONG componentDataSize64;
ULONG partialResourceListSize64;
BOOLEAN thunkResourceList;
PCONFIGURATION_COMPONENT_DATA_64 componentData64;
PCONFIGURATION_COMPONENT_DATA_64 newCompData64;
//
// Create and copy configuration component data node
//
componentDataSize64 = sizeof(CONFIGURATION_COMPONENT_DATA_64);
thunkResourceList = BlAmd64ContainsResourceList(ComponentData32,
&partialResourceListSize64);
if (thunkResourceList != FALSE) {
//
// This node contains a CM_PARTIAL_RESOURCE_LIST structure.
// partialResourceListSize64 contains the number of bytes beyond the
// CONFIGURATION_COMPONENT_DATA header that must be allocated in order to
// thunk the CM_PARTIAL_RESOURCE_LIST into a 64-bit version.
//
componentDataSize64 += partialResourceListSize64;
}
componentData64 = BlAllocateHeap(componentDataSize64);
if (componentData64 == NULL) {
return NULL;
}
status = Copy_CONFIGURATION_COMPONENT_DATA( ComponentData32,
componentData64 );
if (status != ESUCCESS) {
return NULL;
}
if (thunkResourceList != FALSE) {
//
// Update the configuration component data size
//
componentData64->ComponentEntry.ConfigurationDataLength =
partialResourceListSize64;
}
componentData64->Parent = PTR_64(ComponentDataParent64);
if (thunkResourceList != FALSE) {
//
// Now transfer the resource list.
//
BlAmd64TransferResourceList(ComponentData32,componentData64);
}
//
// Process the child (and recursively, all children)
//
if (ComponentData32->Child != NULL) {
newCompData64 = BlAmd64TransferConfigWorker( ComponentData32->Child,
componentData64 );
if (newCompData64 == NULL) {
return newCompData64;
}
componentData64->Child = PTR_64(newCompData64);
}
//
// Process the sibling (and recursively, all siblings)
//
if (ComponentData32->Sibling != NULL) {
newCompData64 = BlAmd64TransferConfigWorker( ComponentData32->Sibling,
ComponentDataParent64 );
if (newCompData64 == NULL) {
return newCompData64;
}
componentData64->Sibling = PTR_64(newCompData64);
}
return componentData64;
}
VOID
BlAmd64TransferResourceList(
IN PCONFIGURATION_COMPONENT_DATA ComponentData32,
OUT PCONFIGURATION_COMPONENT_DATA_64 ComponentData64
)
/*++
Routine Description:
This routine transfers the 32-bit CM_PARTIAL_RESOURCE_LIST structure that
immediately follows ComponentData32 to the memory immediately after
ComponentData64.
Arguments:
ComponentData32 - Supplies a pointer to the 32-bit structure to transfer from.
ComponentData64 - Supplies a pointer to the 64-bit structure to transfer to.
Return Value:
None.
--*/
{
PCM_PARTIAL_RESOURCE_LIST resourceList32;
PCM_PARTIAL_RESOURCE_LIST_64 resourceList64;
PCM_PARTIAL_RESOURCE_DESCRIPTOR resourceDesc32;
PCM_PARTIAL_RESOURCE_DESCRIPTOR_64 resourceDesc64;
PVOID descBody32;
PVOID descBody64;
PUCHAR tail32;
PUCHAR tail64;
ULONG tailSize;
ULONG descriptorCount;
//
// Calculate pointers to the source and target descriptor lists.
//
resourceList32 = (PCM_PARTIAL_RESOURCE_LIST)ComponentData32->ConfigurationData;
resourceList64 = (PCM_PARTIAL_RESOURCE_LIST_64)(ComponentData64 + 1);
//
// Update ComponentData64 to refer to it's new data area, which will be immediately
// following the component data structure.
//
ComponentData64->ConfigurationData = PTR_64(resourceList64);
//
// Copy the resource list header information
//
Copy_CM_PARTIAL_RESOURCE_LIST(resourceList32,resourceList64);
//
// Now thunk each of the resource descriptors
//
descriptorCount = resourceList32->Count;
resourceDesc32 = resourceList32->PartialDescriptors;
resourceDesc64 = &resourceList64->PartialDescriptors;
while (descriptorCount > 0) {
//
// Transfer the common header information
//
Copy_CM_PARTIAL_RESOURCE_DESCRIPTOR(resourceDesc32,resourceDesc64);
descBody32 = &resourceDesc32->u;
descBody64 = &resourceDesc64->u;
RtlZeroMemory(descBody64,
sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR_64) -
FIELD_OFFSET(CM_PARTIAL_RESOURCE_DESCRIPTOR_64,u));
//
// Transfer the body according to the type
//
switch(resourceDesc32->Type) {
case CmResourceTypeNull:
break;
case CmResourceTypePort:
Copy_CM_PRD_PORT(descBody32,descBody64);
break;
case CmResourceTypeInterrupt:
Copy_CM_PRD_INTERRUPT(descBody32,descBody64);
break;
case CmResourceTypeMemory:
Copy_CM_PRD_MEMORY(descBody32,descBody64);
break;
case CmResourceTypeDma:
Copy_CM_PRD_DMA(descBody32,descBody64);
break;
case CmResourceTypeDeviceSpecific:
Copy_CM_PRD_DEVICESPECIFICDATA(descBody32,descBody64);
break;
case CmResourceTypeBusNumber:
Copy_CM_PRD_BUSNUMBER(descBody32,descBody64);
break;
default:
Copy_CM_PRD_GENERIC(descBody32,descBody64);
break;
}
resourceDesc32 += 1;
resourceDesc64 += 1;
descriptorCount -= 1;
}
//
// Calculate how much data, if any, is appended to the resource list.
//
tailSize = ComponentData32->ComponentEntry.ConfigurationDataLength +
(PUCHAR)resourceList32 -
(PUCHAR)resourceDesc32;
if (tailSize > 0) {
//
// Some data is there, append it as-is to the 64-bit structure.
//
tail32 = (PUCHAR)resourceDesc32;
tail64 = (PUCHAR)resourceDesc64;
RtlCopyMemory(tail64,tail32,tailSize);
}
}
BOOLEAN
BlAmd64ContainsResourceList(
IN PCONFIGURATION_COMPONENT_DATA ComponentData32,
OUT PULONG ResourceListSize64
)
/*++
Routine Description:
Given a 32-bit CONFIGURATION_COMPONENT_DATA structure, this routine
determines whether the data associated with the structure contains a
CM_PARTIAL_RESOURCE_LIST structure.
If it does, the size of the 64-bit representation of this structure is calculated,
added to any data that might be appended to the resource list structure, and
returned in ResourceListSize64.
Arguments:
ComponentData32 - Supplies a pointer to the 32-bit structure to transfer.
ResourceListSize64 - Supplies a pointer to a ULONG in which the necessary
additional data size is returned.
Return Value:
Returns TRUE if the CONFIGURATION_COMPONENT_DATA stucture refers to a
CM_PARTIAL_RESOURCE_LIST structure, FALSE otherwise.
--*/
{
ULONG configDataLen;
PCM_PARTIAL_RESOURCE_LIST resourceList;
ULONG resourceCount;
PCM_PARTIAL_RESOURCE_DESCRIPTOR resourceDescriptor;
PCM_PARTIAL_RESOURCE_DESCRIPTOR lastResourceDescriptor;
configDataLen = ComponentData32->ComponentEntry.ConfigurationDataLength;
if (configDataLen < sizeof(CM_PARTIAL_RESOURCE_LIST)) {
//
// Data not large enough to contain the smallest possible resource list
//
return FALSE;
}
resourceList = (PCM_PARTIAL_RESOURCE_LIST)ComponentData32->ConfigurationData;
if (resourceList->Version != 0 || resourceList->Revision != 0) {
//
// Unrecognized version.
//
return FALSE;
}
configDataLen -= FIELD_OFFSET(CM_PARTIAL_RESOURCE_LIST,PartialDescriptors);
resourceCount = resourceList->Count;
if (configDataLen < sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR) * resourceCount) {
//
// Config data len is not large enough to contain a CM_PARTIAL_RESOURCE_LIST
// as large as this one claims to be.
//
return FALSE;
}
//
// Validate each of the CM_PARTIAL_RESOURCE_DESCRIPTOR structures in the list
//
resourceDescriptor = resourceList->PartialDescriptors;
lastResourceDescriptor = resourceDescriptor + resourceCount;
while (resourceDescriptor < lastResourceDescriptor) {
if (resourceDescriptor->Type > CmResourceTypeMaximum) {
return FALSE;
}
resourceDescriptor += 1;
}
//
// Looks like this is an actual resource list. Calculate the size of any remaining
// data after the CM_PARTIAL_RESOURCE_LIST structure.
//
configDataLen -= sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR) * resourceCount;
*ResourceListSize64 = sizeof(CM_PARTIAL_RESOURCE_LIST_64) +
sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR_64) * (resourceCount - 1) +
configDataLen;
return TRUE;
}
ARC_STATUS
BlAmd64TransferNlsData(
VOID
)
/*++
Routine Description:
This routine creates an NLS_DATA_BLOCK64 structure that is the
equivalent of the 32-bit NLS_DATA_BLOCK structure referenced within
the 32-bit loader block.
Arguments:
None.
Return Value:
ARC_STATUS - Status of operation.
--*/
{
ARC_STATUS status;
PNLS_DATA_BLOCK nlsDataBlock32;
PNLS_DATA_BLOCK_64 nlsDataBlock64;
nlsDataBlock32 = BlAmd64LoaderBlock32->NlsData;
if (nlsDataBlock32 == NULL) {
return ESUCCESS;
}
nlsDataBlock64 = BlAllocateHeap(sizeof(NLS_DATA_BLOCK_64));
if (nlsDataBlock64 == NULL) {
return ENOMEM;
}
status = Copy_NLS_DATA_BLOCK( nlsDataBlock32, nlsDataBlock64 );
if (status != ESUCCESS) {
return status;
}
BlAmd64LoaderBlock64->NlsData = PTR_64( nlsDataBlock64 );
return ESUCCESS;
}
ARC_STATUS
BlAmd64BuildLoaderBlock64(
VOID
)
/*++
Routine Description:
This routine allocates a 64-bit loader parameter block and copies the
contents of the 32-bit loader parameter block into it.
Arguments:
None.
Return Value:
The status of the operation.
--*/
{
ARC_STATUS status;
//
// Allocate the loader block and extension
//
BlAmd64LoaderBlock64 = BlAllocateHeap(sizeof(LOADER_PARAMETER_BLOCK_64));
if (BlAmd64LoaderBlock64 == NULL) {
return ENOMEM;
}
//
// Copy the contents of the 32-bit loader parameter block to the
// 64-bit version
//
status = Copy_LOADER_PARAMETER_BLOCK( BlAmd64LoaderBlock32, BlAmd64LoaderBlock64 );
if (status != ESUCCESS) {
return status;
}
//
// Build the loader block extension
//
status = BlAmd64BuildLoaderBlockExtension64();
if (status != ESUCCESS) {
return status;
}
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferMemoryAllocationDescriptors(
VOID
)
/*++
Routine Description:
This routine transfers all of the 32-bit memory allocation descriptors
to a 64-bit list.
The storage for the 64-bit memory allocation descriptors has been
preallocated by a previous call to
BlAmd64AllocateMemoryAllocationDescriptors(). This memory is described
by BlAmd64DescriptorArray and BlAmd64DescriptorArraySize.
Arguments:
None.
Return Value:
The status of the operation.
--*/
{
ARC_STATUS status;
PMEMORY_ALLOCATION_DESCRIPTOR memDesc32;
PMEMORY_ALLOCATION_DESCRIPTOR_64 memDesc64;
PLIST_ENTRY listHead;
PLIST_ENTRY listEntry;
LONG descriptorCount;
//
// Modify some descriptor types. All of the descriptors of type
// LoaderMemoryData really contain things that won't be used in 64-bit
// mode, such as 32-bit page tables and the like.
//
// The descriptors that we really want to stick around are allocated with
// LoaderAmd64MemoryData.
//
// Perform two memory descriptor list search-and-replacements:
//
// LoaderMemoryData -> LoaderOSLoaderHeap
//
// These desriptors will be freed during kernel init phase 1
//
// LoaderAmd64MemoryData -> LoaderMemoryData
//
// This stuff will be kept around
//
//
// All existing LoaderMemoryData refers to structures that are not useful
// once running in long mode. However, we're using some of the structures
// now (32-bit page tables for example), so convert them to
// type LoaderOsloaderHeap, which will be eventually freed by the kernel.
//
BlAmd64ReplaceMemoryDescriptorType(LoaderMemoryData,
LoaderOsloaderHeap,
TRUE);
//
// Same for LoaderStartupPcrPage
//
BlAmd64ReplaceMemoryDescriptorType(LoaderStartupPcrPage,
LoaderOsloaderHeap,
TRUE);
//
// All of the permanent structures that need to be around for longmode
// were temporarily allocated with LoaderAmd64MemoryData. Convert all
// of those to LoaderMemoryData now.
//
BlAmd64ReplaceMemoryDescriptorType(LoaderAmd64MemoryData,
LoaderMemoryData,
TRUE);
//
// Now walk the 32-bit memory descriptors, filling in and inserting a
// 64-bit version into BlAmd64LoaderBlock64.
//
InitializeListHead64( &BlAmd64LoaderBlock64->MemoryDescriptorListHead );
memDesc64 = BlAmd64DescriptorArray;
descriptorCount = BlAmd64DescriptorArraySize;
listHead = &BlAmd64LoaderBlock32->MemoryDescriptorListHead;
listEntry = listHead->Flink;
while (listEntry != listHead && descriptorCount > 0) {
memDesc32 = CONTAINING_RECORD( listEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry );
status = Copy_MEMORY_ALLOCATION_DESCRIPTOR( memDesc32, memDesc64 );
if (status != ESUCCESS) {
return status;
}
#if DBG
DbgPrint("Base 0x%08x size 0x%02x %s\n",
memDesc32->BasePage,
memDesc32->PageCount,
BlAmd64MemoryDescriptorText[memDesc32->MemoryType]);
#endif
InsertTailList64( &BlAmd64LoaderBlock64->MemoryDescriptorListHead,
&memDesc64->ListEntry );
listEntry = listEntry->Flink;
memDesc64 = memDesc64 + 1;
descriptorCount -= 1;
}
ASSERT( descriptorCount >= 0 && listEntry == listHead );
return ESUCCESS;
}
ARC_STATUS
BlAmd64AllocateMemoryAllocationDescriptors(
VOID
)
/*++
Routine Description:
This routine preallocates a quantity of memory sufficient to contain
a 64-bit version of each memory allocation descriptor.
The resultant memory is described in two globals: BlAmd64DescriptorArray
and BlAmd64DescriptorArrayCount.
Arguments:
None.
Return Value:
The status of the operation.
--*/
{
PLIST_ENTRY listHead;
PLIST_ENTRY listEntry;
ULONG descriptorCount;
ULONG arraySize;
PMEMORY_ALLOCATION_DESCRIPTOR_64 descriptorArray;
//
// Count the number of descriptors needed.
//
descriptorCount = 0;
listHead = &BlAmd64LoaderBlock32->MemoryDescriptorListHead;
listEntry = listHead->Flink;
while (listEntry != listHead) {
descriptorCount += 1;
listEntry = listEntry->Flink;
}
//
// Allocate memory sufficient to contain them all in 64-bit form.
//
arraySize = descriptorCount *
sizeof(MEMORY_ALLOCATION_DESCRIPTOR_64);
descriptorArray = BlAllocateHeap(arraySize);
if (descriptorArray == NULL) {
return ENOMEM;
}
BlAmd64DescriptorArray = descriptorArray;
BlAmd64DescriptorArraySize = descriptorCount;
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferLoadedModuleState(
VOID
)
/*++
Routine Description:
This routine transfers the 32-bit list of LDR_DATA_TABLE_ENTRY structures
to an equivalent 64-bit list.
Arguments:
None.
Return Value:
The status of the operation.
--*/
{
PLDR_DATA_TABLE_ENTRY dataTableEntry32;
PLDR_DATA_TABLE_ENTRY_64 dataTableEntry64;
PLIST_ENTRY listEntry;
PLIST_ENTRY listHead;
ARC_STATUS status;
InitializeListHead64( &BlAmd64LoaderBlock64->LoadOrderListHead );
//
// For each of the LDR_DATA_TABLE_ENTRY structures in the 32-bit
// loader parameter block, create a 64-bit LDR_DATA_TABLE_ENTRY
// and queue it on the 64-bit loader parameter block.
//
listHead = &BlAmd64LoaderBlock32->LoadOrderListHead;
listEntry = listHead->Flink;
while (listEntry != listHead) {
dataTableEntry32 = CONTAINING_RECORD( listEntry,
LDR_DATA_TABLE_ENTRY,
InLoadOrderLinks );
status = BlAmd64BuildLdrDataTableEntry64( dataTableEntry32,
&dataTableEntry64 );
if (status != ESUCCESS) {
return status;
}
//
// Insert it into the 64-bit loader block's data table queue.
//
InsertTailList64( &BlAmd64LoaderBlock64->LoadOrderListHead,
&dataTableEntry64->InLoadOrderLinks );
listEntry = listEntry->Flink;
}
return ESUCCESS;
}
ARC_STATUS
BlAmd64BuildLdrDataTableEntry64(
IN PLDR_DATA_TABLE_ENTRY DataTableEntry32,
OUT PLDR_DATA_TABLE_ENTRY_64 *DataTableEntry64
)
/*++
Routine Description:
This routine transfers the contents of a single 32-bit
LDR_DATA_TABLE_ENTRY structure to the 64-bit equivalent.
Arguments:
DataTableEntry32 - Supplies a pointer to the source structure.
DataTableEntry64 - Supplies a pointer to the destination pointer to
the created structure.
Return Value:
The status of the operation.
--*/
{
ARC_STATUS status;
PLDR_DATA_TABLE_ENTRY_64 dataTableEntry64;
//
// Allocate a 64-bit data table entry and transfer the 32-bit
// contents
//
dataTableEntry64 = BlAllocateHeap( sizeof(LDR_DATA_TABLE_ENTRY_64) );
if (dataTableEntry64 == NULL) {
return ENOMEM;
}
status = Copy_LDR_DATA_TABLE_ENTRY( DataTableEntry32, dataTableEntry64 );
if (status != ESUCCESS) {
return status;
}
*DataTableEntry64 = dataTableEntry64;
//
// Later on, we'll need to determine the 64-bit copy of this data
// table entry. Store the 64-bit pointer to the copy here.
//
*((POINTER64 *)&DataTableEntry32->DllBase) = PTR_64(dataTableEntry64);
return ESUCCESS;
}
ARC_STATUS
BlAmd64BuildLoaderBlockExtension64(
VOID
)
/*++
Routine Description:
This routine transfers the contents of the 32-bit loader block
extension to a 64-bit equivalent.
Arguments:
None.
Return Value:
The status of the operation.
--*/
{
PLOADER_PARAMETER_EXTENSION_64 loaderExtension;
ARC_STATUS status;
//
// Allocate the 64-bit extension and transfer the contents of the
// 32-bit block.
//
loaderExtension = BlAllocateHeap( sizeof(LOADER_PARAMETER_EXTENSION_64) );
if (loaderExtension == NULL) {
return ENOMEM;
}
//
// Perform automatic copy of most fields
//
status = Copy_LOADER_PARAMETER_EXTENSION( BlLoaderBlock->Extension,
loaderExtension );
if (status != ESUCCESS) {
return status;
}
//
// Initialize Symbol list head properly
//
InitializeListHead64( &loaderExtension->FirmwareDescriptorListHead );
//
// Manually fix up remaining fields
//
loaderExtension->Size = sizeof(LOADER_PARAMETER_EXTENSION_64);
BlAmd64LoaderBlock64->Extension = PTR_64(loaderExtension);
return ESUCCESS;
}
ARC_STATUS
BlAmd64TransferBootDriverNodes(
VOID
)
/*++
Routine Description:
This routine transfers the 32-bit list of BOOT_DRIVER_NODE structures
to an equivalent 64-bit list.
Arguments:
None.
Return Value:
The status of the operation.
--*/
{
PBOOT_DRIVER_LIST_ENTRY driverListEntry32;
PBOOT_DRIVER_NODE driverNode32;
PBOOT_DRIVER_NODE_64 driverNode64;
POINTER64 dataTableEntry64;
PKLDR_DATA_TABLE_ENTRY dataTableEntry;
PLIST_ENTRY listEntry;
PLIST_ENTRY listHead;
ARC_STATUS status;
InitializeListHead64( &BlAmd64LoaderBlock64->BootDriverListHead );
//
// For each of the BOOT_DRIVER_NODE structures in the 32-bit
// loader parameter block, create a 64-bit BOOT_DRIVER_NODE
// and (possibly) associated LDR_DATA_TABLE_ENTRY structure.
//
listHead = &BlAmd64LoaderBlock32->BootDriverListHead;
listEntry = listHead->Flink;
while (listEntry != listHead) {
driverListEntry32 = CONTAINING_RECORD( listEntry,
BOOT_DRIVER_LIST_ENTRY,
Link );
driverNode32 = CONTAINING_RECORD( driverListEntry32,
BOOT_DRIVER_NODE,
ListEntry );
driverNode64 = BlAllocateHeap( sizeof(BOOT_DRIVER_NODE_64) );
if (driverNode64 == NULL) {
return ENOMEM;
}
status = Copy_BOOT_DRIVER_NODE( driverNode32, driverNode64 );
if (status != ESUCCESS) {
return status;
}
dataTableEntry = driverNode32->ListEntry.LdrEntry;
if (dataTableEntry != NULL) {
//
// There is already a 64-bit copy of this table entry, and we
// stored a pointer to it at DllBase.
//
dataTableEntry64 = *((POINTER64 *)&dataTableEntry->DllBase);
driverNode64->ListEntry.LdrEntry = dataTableEntry64;
}
//
// Now insert the driver list entry into the 64-bit loader block.
//
InsertTailList64( &BlAmd64LoaderBlock64->BootDriverListHead,
&driverNode64->ListEntry.Link );
listEntry = listEntry->Flink;
}
return ESUCCESS;
}
ARC_STATUS
BlAmd64CheckForLongMode(
IN ULONG LoadDeviceId,
IN OUT PCHAR KernelPath,
IN PCHAR KernelFileName
)
/*++
Routine Description:
This routine examines a kernel image and determines whether it was
compiled for AMD64. The global BlAmd64UseLongMode is set to non-zero
if a long-mode kernel is discovered.
Arguments:
LoadDeviceId - Supplies the load device identifier.
KernelPath - Supplies a pointer to the path to the kernel directory.
Upon successful return, KernelFileName will be appended
to this path.
KernelFileName - Supplies a pointer to the name of the kernel file.
Note: If KernelPath already contains the full path and filename of
the kernel image to check, pass a pointer to "\0" for
KernelFileName.
Return Value:
The status of the operation. Upon successful completion ESUCCESS
is returned, whether long mode capability was detected or not.
--*/
{
CHAR localBufferSpace[ SECTOR_SIZE * 2 + SECTOR_SIZE - 1 ];
PCHAR localBuffer;
ULONG fileId;
PIMAGE_NT_HEADERS32 ntHeaders;
ARC_STATUS status;
ULONG bytesRead;
PCHAR kernelNameTarget;
//
// File I/O here must be sector-aligned.
//
localBuffer = (PCHAR)
(((ULONG)localBufferSpace + SECTOR_SIZE - 1) & ~(SECTOR_SIZE - 1));
//
// Build the path to the kernel and open it.
//
kernelNameTarget = KernelPath + strlen(KernelPath);
strcpy(kernelNameTarget, KernelFileName);
status = BlOpen( LoadDeviceId, KernelPath, ArcOpenReadOnly, &fileId );
*kernelNameTarget = '\0'; // Restore the kernel path, assuming
// failure.
if (status != ESUCCESS) {
return status;
}
//
// Read the PE image header
//
status = BlRead( fileId, localBuffer, SECTOR_SIZE * 2, &bytesRead );
BlClose( fileId );
//
// Determine whether the image header is valid, and if so whether
// the image is AMD64, I386 or something else.
//
ntHeaders = RtlImageNtHeader( localBuffer );
if (ntHeaders == NULL) {
return EBADF;
}
if (IMAGE_64BIT(ntHeaders)) {
//
// Return with the kernel name appended to the path
//
if (BlIsAmd64Supported() != FALSE) {
strcpy(kernelNameTarget, KernelFileName);
BlAmd64UseLongMode = TRUE;
status = ESUCCESS;
} else {
//
// We have an AMD64 image, but the processor does not support
// AMD64. There is nothing we can do.
//
status = EBADF;
}
} else if (IMAGE_32BIT(ntHeaders)) {
ASSERT( BlAmd64UseLongMode == FALSE );
status = ESUCCESS;
} else {
status = EBADF;
}
return status;
}
ARC_STATUS
BlAmd64PrepareSystemStructures(
VOID
)
/*++
Routine Description:
This routine allocates and initializes several structures necessary
for transfer to an AMD64 kernel. These structures include:
GDT
IDT
KTSS64
Idle thread stack
DPC stack
Double fault stack
MCA exception stack
Arguments:
None.
Return Value:
The status of the operation.
--*/
{
PCHAR processorData;
ULONG dataSize;
ULONG descriptor;
ULONG stackOffset;
PKTSS64_64 sysTss64;
PCHAR idleStack;
PCHAR dpcStack;
PCHAR doubleFaultStack;
PCHAR mcaStack;
PVOID gdt64;
PVOID idt64;
ARC_STATUS status;
//
// Calculate the cumulative, rounded size of the various structures that
// we need, and allocate a sufficient number of pages.
//
dataSize = ROUNDUP16(GDT_64_SIZE) +
ROUNDUP16(IDT_64_SIZE) +
ROUNDUP16(sizeof(KTSS64_64));
dataSize = ROUNDUP_PAGE(dataSize);
stackOffset = dataSize;
dataSize += KERNEL_STACK_SIZE_64 + // Idle thread stack
KERNEL_STACK_SIZE_64 + // DPC stack
DOUBLE_FAULT_STACK_SIZE_64 + // Double fault stack
MCA_EXCEPTION_STACK_SIZE_64; // MCA exception stack
//
// dataSize is still page aligned.
//
status = BlAllocateDescriptor( LoaderAmd64MemoryData,
0,
dataSize / PAGE_SIZE,
&descriptor );
if (status != ESUCCESS) {
return status;
}
processorData = (PCHAR)(descriptor * PAGE_SIZE | KSEG0_BASE_X86);
//
// Zero the block that was just allocated, then get local pointers to the
// various structures within.
//
RtlZeroMemory( processorData, dataSize );
//
// Assign the stack pointers. Stack pointers start at the TOP of their
// respective stack areas.
//
idleStack = processorData + stackOffset + KERNEL_STACK_SIZE_64;
dpcStack = idleStack + KERNEL_STACK_SIZE_64;
doubleFaultStack = dpcStack + DOUBLE_FAULT_STACK_SIZE_64;
mcaStack = doubleFaultStack + MCA_EXCEPTION_STACK_SIZE_64;
//
// Record the idle stack base so that we can switch to it in amd64s.asm
//
BlAmd64IdleStack64 = PTR_64(idleStack);
//
// Assign pointers to GDT, IDT and KTSS64.
//
gdt64 = (PVOID)processorData;
processorData += ROUNDUP16(GDT_64_SIZE);
idt64 = (PVOID)processorData;
processorData += ROUNDUP16(IDT_64_SIZE);
sysTss64 = (PKTSS64_64)processorData;
processorData += ROUNDUP16(sizeof(KTSS64_64));
//
// Build the GDT. This is done in amd64.c as it involves AMD64
// structure definitions. The IDT remains zeroed.
//
BlAmd64BuildAmd64GDT( sysTss64, gdt64 );
//
// Build the pseudo-descriptors for the GDT and IDT. These will
// be referenced during the long-mode transition in amd64s.asm.
//
BlAmd64GdtDescriptor.Limit = (USHORT)(GDT_64_SIZE - 1);
BlAmd64GdtDescriptor.Base = PTR_64(gdt64);
BlAmd64IdtDescriptor.Limit = (USHORT)(IDT_64_SIZE - 1);
BlAmd64IdtDescriptor.Base = PTR_64(idt64);
//
// Build another GDT pseudo-descriptor, this one with a 32-bit
// base. This base address must be a 32-bit address that is addressible
// from long mode during init, so use the mapping in the identity mapped
// region.
//
BlAmd32GdtDescriptor.Limit = (USHORT)(GDT_64_SIZE - 1);
BlAmd32GdtDescriptor.Base = (ULONG)gdt64 ^ KSEG0_BASE_X86;
//
// Initialize the system TSS
//
sysTss64->Rsp0 = PTR_64(idleStack);
sysTss64->Ist[TSS64_IST_PANIC] = PTR_64(doubleFaultStack);
sysTss64->Ist[TSS64_IST_MCA] = PTR_64(mcaStack);
//
// Fill required fields within the loader block
//
BlAmd64LoaderBlock64->KernelStack = PTR_64(dpcStack);
return ESUCCESS;
}
VOID
BlAmd64ReplaceMemoryDescriptorType(
IN TYPE_OF_MEMORY Target,
IN TYPE_OF_MEMORY Replacement,
IN BOOLEAN Coallesce
)
/*++
Routine Description:
This routine walks the 32-bit memory allocation descriptor list and
performs a "search and replace" of the types therein.
Optionally, it will coallesce each successful replacement with
adjacent descriptors of like type.
Arguments:
Target - The descriptor type to search for
Replacement - The type with which to replace each located Target type.
Coallesce - If !FALSE, indicates that each successful replacement should
be coallesced with any like-typed neighbors.
Return Value:
None.
--*/
{
PMEMORY_ALLOCATION_DESCRIPTOR descriptor;
PMEMORY_ALLOCATION_DESCRIPTOR adjacentDescriptor;
PLIST_ENTRY listHead;
PLIST_ENTRY listEntry;
PLIST_ENTRY adjacentListEntry;
listHead = &BlAmd64LoaderBlock32->MemoryDescriptorListHead;
listEntry = listHead;
while (TRUE) {
listEntry = listEntry->Flink;
if (listEntry == listHead) {
break;
}
descriptor = CONTAINING_RECORD(listEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
if (descriptor->MemoryType != Target) {
continue;
}
descriptor->MemoryType = Replacement;
if (Coallesce == FALSE) {
//
// Do not attempt to coallesce
//
continue;
}
//
// Now attempt to coallesce the descriptor. First try the
// next descriptor.
//
adjacentListEntry = listEntry->Flink;
if (adjacentListEntry != listHead) {
adjacentDescriptor = CONTAINING_RECORD(adjacentListEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
if (adjacentDescriptor->MemoryType == descriptor->MemoryType &&
descriptor->BasePage + descriptor->PageCount ==
adjacentDescriptor->BasePage) {
descriptor->PageCount += adjacentDescriptor->PageCount;
BlRemoveDescriptor(adjacentDescriptor);
}
}
//
// Now try the previous descriptor.
//
adjacentListEntry = listEntry->Blink;
if (adjacentListEntry != listHead) {
adjacentDescriptor = CONTAINING_RECORD(adjacentListEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
if (adjacentDescriptor->MemoryType == descriptor->MemoryType &&
adjacentDescriptor->BasePage + adjacentDescriptor->PageCount ==
descriptor->BasePage) {
descriptor->PageCount += adjacentDescriptor->PageCount;
descriptor->BasePage -= adjacentDescriptor->PageCount;
BlRemoveDescriptor(adjacentDescriptor);
}
}
}
}
ARC_STATUS
BlAmd64FixSharedUserPage(
VOID
)
{
PFN_NUMBER pfn;
PMEMORY_ALLOCATION_DESCRIPTOR descriptor;
ARC_STATUS status;
//
// The shared user page is allocated as LoaderMemoryData. All
// LoaderMemoryData descriptors will be converted to LoaderOsloaderHeap
// during the transition to 64-bit mode, as the assumption is made that
// all of the old structures will no longer be needed.
//
// The shared user page is the exception to this rule, so it must be
// found and placed into an appropriately marked descriptor.
//
//
// Get the pfn of the shared user page, find its descriptor,
// carve out a new descriptor that contains just that page and give
// it a type of LoaderAmd64MemoryData.
//
BlAmd64IsPageMapped( KI_USER_SHARED_DATA, &pfn, NULL );
descriptor = BlFindMemoryDescriptor( pfn );
status = BlGenerateDescriptor(descriptor,
LoaderAmd64MemoryData,
pfn,
1);
return status;
}
BOOLEAN
BlAmd64Setup (
IN PCHAR SetupDevice
)
/*++
Routine Description:
This routine determines whether we are installing an I386 or AMD64 build.
If the directory "\\AMD64" exists at the root of DriveId then it is
assumed that an AMD64 installation is being performed.
Arguments:
SetupDevice - Supplies the ARC path to the setup device. This parameter
need only be supplied on the first invocation of this routine. The
result of the first call is cached for subsequent invocations.
Return Value:
TRUE - An AMD64 installation is being performed.
FALSE - An I386 installation is being performed.
--*/
{
ULONG deviceId;
ULONG dirId;
ARC_STATUS status;
static BOOLEAN alreadyDetected = FALSE;
static BOOLEAN detectedAmd64 = FALSE;
if (alreadyDetected == FALSE) {
ASSERT(SetupDevice != NULL);
status = ArcOpen(SetupDevice, ArcOpenReadOnly, &deviceId);
if (status == ESUCCESS) {
status = BlOpen(deviceId, "\\AMD64", ArcOpenDirectory, &dirId);
if (status == ESUCCESS) {
detectedAmd64 = TRUE;
BlClose(dirId);
}
ArcClose(deviceId);
}
alreadyDetected = TRUE;
}
return detectedAmd64;
}
VOID
BlCheckForAmd64Image(
PPO_MEMORY_IMAGE MemImage
)
/*++
Routine Description:
This routine determines whether a hibernate file was created for
Amd64 platform. BlAmd64UseLongMode will be set accordingly.
Arguments:
MemImage - Header of hibernate image file.
Return Value:
None.
--*/
{
//
// It is assumed that "version" and "LengthSelf" field can be reference
// in same way between a x86 and an Amd64 image header.
//
if((MemImage->Version == 0) &&
(MemImage->LengthSelf == sizeof(PO_MEMORY_IMAGE_64))) {
BlAmd64UseLongMode = TRUE;
}
}
ULONG
BlAmd64FieldOffset_PO_MEMORY_IMAGE(
ULONG offset32
)
/*++
Routine Description:
This routine helps to access 64-bit version of PO_MEMORY_IMAGE from
its 32-bit definition. It calculates the offset of a field in 64-bit
definition from the offset of same field in 32-bit definiation.
Arguments:
offset32 - Field offset of 32-bit definiation.
Return Value:
Field offset of 64-bit definiation.
--*/
{
PCOPY_REC copyRec;
copyRec = cr3264_PO_MEMORY_IMAGE;
while (copyRec->Size32 != 0) {
if (copyRec->Offset32 == offset32) {
return copyRec->Offset64;
}
copyRec++;
}
return 0;
}
ULONG
BlAmd64FieldOffset_PO_MEMORY_RANGE_ARRAY_LINK(
ULONG offset32
)
/*++
Routine Description:
This routine helps to access 64-bit version of PO_MEMORY_RANGE_ARRAY_LINK
from its 32-bit definition. It calculates the offset of a field in 64-bit
definition from the offset of same field in 32-bit definiation.
Arguments:
offset32 - Field offset of 32-bit definiation.
Return Value:
Field offset of 64-bit definiation.
--*/
{
PCOPY_REC copyRec;
copyRec = cr3264_PO_MEMORY_RANGE_ARRAY_LINK;
while (copyRec->Size32 != 0) {
if (copyRec->Offset32 == offset32) {
return copyRec->Offset64;
}
copyRec++;
}
return 0;
}
ULONG
BlAmd64FieldOffset_PO_MEMORY_RANGE_ARRAY_RANGE(
ULONG offset32
)
/*++
Routine Description:
This routine helps to access 64-bit version of PO_MEMORY_RANGE_ARRAY_RANGE
from its 32-bit definition. It calculates the offset of a field in 64-bit
definition from the offset of same field in 32-bit definiation.
Arguments:
offset32 - Field offset of 32-bit definiation.
Return Value:
Field offset of 64-bit definiation.
--*/
{
PCOPY_REC copyRec;
copyRec = cr3264_PO_MEMORY_RANGE_ARRAY_RANGE;
while (copyRec->Size32 != 0) {
if (copyRec->Offset32 == offset32) {
return copyRec->Offset64;
}
copyRec++;
}
return 0;
}
ULONG
BlAmd64ElementOffset_PO_MEMORY_RANGE_ARRAY_LINK(
ULONG index
)
/*++
Routine Description:
This routine calculates the offset of a element in a structure array.
Each element in this array is defined as PO_MORY_RANGE_ARRAY_LINK in
it 64-bit format.
Arguments:
index - Supplies the index of a element.
Return Value:
Offset of the element from base address of the array.
--*/
{
return (ULONG)(&(((PO_MEMORY_RANGE_ARRAY_LINK_64 *)0)[index]));
}
ULONG
BlAmd64ElementOffset_PO_MEMORY_RANGE_ARRAY_RANGE(
ULONG index
)
/*++
Routine Description:
This routine calculates the offset of a element in a structure array.
Each element in this array is defined as PO_MEMORY_RANGE_ARRAY_RANGE
in its 64-bit format.
Arguments:
index - Supplies the index of a element.
Return Value:
Offset of the element from base address of the array.
--*/
{
return (ULONG)(&(((PO_MEMORY_RANGE_ARRAY_RANGE_64 *)0)[index]));
}
#define BL_ENABLE_REMAP
#if defined(BL_ENABLE_REMAP)
#include "hammernb.h"
#include "acpitabl.h"
#include "pci.h"
#include "ntacpi.h"
#if !defined(_4G)
#define _4G (1UI64 << 32)
#endif
extern PRSDP BlRsdp;
extern PRSDT BlRsdt;
extern PXSDT BlXsdt;
BOOLEAN
BlAmd64GetNode1Info (
OUT ULONG *Base,
OUT ULONG *Size
);
BOOLEAN
BlAmd64RelocateAcpi (
ULONG Node0Base,
ULONG Node0Limit,
ULONG Node1Base,
ULONG Node1Limit
);
BOOLEAN
BlAmd64RemapMTRRs (
IN ULONG OldBase,
IN ULONG NewBase,
IN ULONG Size
);
BOOLEAN
BlAmd64RemapNode1Dram (
IN ULONG NewBase
);
ULONG
StringToUlong (
IN PCHAR String
);
BOOLEAN
BlAmd64RemapDram (
IN PCHAR LoaderOptions
)
/*++
Routine Description:
This routine looks for the /RELOCATEPHYSICAL= switch supplied as a loader
option. The option directs the loader to relocate node 1's physical memory
to the address supplied.
The address represents the new physical memory base in 1GB units.
For example, to relocate node 1's physical memory to 128GB, use:
/RELOCATEPHYSICAL=128
Arguments:
LoaderOptions - Supplies a pointer to the loader option string
Return Value:
TRUE if the relocation was performed, FALSE otherwise
--*/
{
BOOLEAN result;
ULONG oldBase;
ULONG oldLimit;
ULONG newBase;
ULONG newBasePage;
ULONG size;
PMEMORY_ALLOCATION_DESCRIPTOR descriptor;
PLIST_ENTRY listHead;
PLIST_ENTRY listEntry;
ULONG descriptorBase;
ULONG descriptorLimit;
ARC_STATUS status;
PCHAR pch;
ULONG type;
newBase = 0;
if (LoaderOptions != NULL) {
pch = strstr(LoaderOptions,"RELOCATEPHYSICAL=");
if (pch != NULL) {
newBase = StringToUlong( pch + strlen( "RELOCATEPHYSICAL=" ));
}
}
if (newBase == 0) {
return FALSE;
}
//
// The parameter is supplied in GB, convert to 16MB chunks for internal
// use
//
newBase *= 64;
//
// Determine the chunk of physical memory associated with node 1.
// Note that this routine will relocate the ACPI tables if a suitable
// node 1 bridge device was found.
//
result = BlAmd64GetNode1Info( &oldBase, &size );
if (result == FALSE) {
return FALSE;
}
newBasePage = newBase << (24 - 12);
oldLimit = oldBase + size - 1;
//
// Make sure that the descriptors describing that physical memory
// haven't already been allocated. Acceptable descriptor
// types are Free, LoaderReserve and SpecialMemory.
//
listHead = &BlLoaderBlock->MemoryDescriptorListHead;
listEntry = listHead->Flink;
while (listEntry != listHead) {
descriptor = CONTAINING_RECORD(listEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
descriptorBase = descriptor->BasePage;
descriptorLimit = descriptorBase + descriptor->PageCount - 1;
if ((descriptorBase <= oldLimit) && (descriptorLimit >= oldBase)) {
//
// Some or all of this memory descriptor lies within the
// relocated region.
//
if (descriptor->MemoryType != LoaderFree &&
descriptor->MemoryType != LoaderSpecialMemory &&
descriptor->MemoryType != LoaderReserve) {
return FALSE;
}
}
listEntry = listEntry->Flink;
}
//
// From the loader perspective everything looks good, perform the remap.
//
result = BlAmd64RemapNode1Dram( newBase );
if (result == FALSE) {
return FALSE;
}
//
// The bridge(s) have been reprogrammed. Now walk the memory descriptor
// lists, performing necessary relocations.
//
listHead = &BlLoaderBlock->MemoryDescriptorListHead;
listEntry = listHead->Flink;
while (listEntry != listHead) {
descriptor = CONTAINING_RECORD(listEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
descriptorBase = descriptor->BasePage;
descriptorLimit = descriptorBase + descriptor->PageCount - 1;
if ((descriptorBase <= oldLimit) && (descriptorLimit >= oldBase)) {
//
// Some or all of this memory descriptor lies within the
// relocated region.
//
if (descriptorBase >= oldBase && descriptorLimit <= oldLimit) {
//
// The descriptor lies entirely within the relocation range
// so relocate the whole thing.
//
} else {
//
// Only part of the descriptor lies within the relocation
// range, so a new descriptor must be allocated.
//
if (descriptorBase < oldBase) {
descriptorBase = oldBase;
}
if (descriptorLimit > oldLimit) {
descriptorLimit = oldLimit;
}
type = descriptor->MemoryType;
status = BlGenerateDescriptor( descriptor,
LoaderSpecialMemory,
descriptorBase,
descriptorLimit - descriptorBase + 1 );
ASSERT(status == ESUCCESS);
listEntry = listEntry->Flink;
descriptor = CONTAINING_RECORD(listEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
descriptor->MemoryType = type;
}
listEntry = listEntry->Flink;
BlRemoveDescriptor( descriptor );
descriptor->BasePage = descriptor->BasePage - oldBase + newBasePage;
BlInsertDescriptor( descriptor );
} else {
listEntry = listEntry->Flink;
}
}
//
// Recalculate BlHighestPage
//
BlHighestPage = 0;
listHead = &BlLoaderBlock->MemoryDescriptorListHead;
listEntry = listHead->Flink;
while (listEntry != listHead) {
descriptor = CONTAINING_RECORD(listEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
descriptorBase = descriptor->BasePage;
descriptorLimit = descriptorBase + descriptor->PageCount - 1;
if (descriptor->MemoryType != LoaderSpecialMemory &&
descriptor->MemoryType != LoaderReserve &&
descriptorLimit > BlHighestPage) {
BlHighestPage = descriptorLimit;
}
listEntry = listEntry->Flink;
}
//
// Remap the MTRRs
//
result = BlAmd64RemapMTRRs( oldBase, newBase, oldLimit - oldBase + 1 );
if (result == FALSE) {
return FALSE;
}
return TRUE;
}
ULONG
StringToUlong (
IN PCHAR String
)
/*++
Routine Description:
This routine converts a hexadecimal or decimal string to
a 32-bit unsigned integer.
Arguments:
String - Supplies a null-terminated ASCII string in decimal or
hexadecimal format.
01234567 - decimal format
0x01234567 - hexadecimal format
The input string is processed until an invalid character or
the end of the string is encountered.
Return Value:
Returns the value of the parsed string.
--*/
{
CHAR ch;
PCHAR pch;
ULONG result;
int radix;
UCHAR digit;
pch = String;
result = 0;
radix = 10;
while (TRUE) {
ch = (char)toupper(*pch);
if ((ch >= '0' && ch <= '9') ||
(ch >= 'A' && ch <= 'F')) {
if (ch >= '0' && ch <= '9') {
digit = ch - '0';
} else {
digit = ch - 'A' + 10;
}
result = result * radix + digit;
} else if (ch == 'X') {
if (result == 0) {
radix = 16;
} else {
break;
}
} else {
break;
}
pch += 1;
}
return result;
}
BOOLEAN
BlAmd64RemapNode1Dram (
IN ULONG NewBase
)
/*++
Routine Description:
Relocates node 1 memory to a new physical address and reprograms
MSRs related to the physical memory map.
Arguments:
NewBase - Supplies bits [39:24] of the desired new physical base address
of the memory associated with node 1.
Return Value:
TRUE if the operation was successful, FALSE otherwise.
--*/
{
AMD_NB_FUNC1_CONFIG nodeConfigArray[8];
PAMD_NB_FUNC1_CONFIG nodeConfig;
PAMD_NB_DRAM_MAP dramMap;
ULONG length;
PCI_SLOT_NUMBER slotNumber;
ULONG nodeCount;
ULONG nodeIndex;
ULONG span;
ULONG oldBase;
ULONG oldLimit;
ULONG newLimit;
ULONG64 topMem;
ULONG64 topMem4G;
ULONG64 msrValue;
ULONG64 base64;
ULONG64 limit64;
//
// NewBase supplies the new DRAM base[39:24]
//
nodeCount = 0;
nodeConfig = nodeConfigArray;
do {
slotNumber.u.AsULONG = 0;
slotNumber.u.bits.DeviceNumber = NB_DEVICE_BASE + nodeCount;
slotNumber.u.bits.FunctionNumber = 1;
length = HalGetBusDataByOffset( PCIConfiguration,
0,
slotNumber.u.AsULONG,
nodeConfig,
0,
sizeof(*nodeConfig) );
if (length != sizeof(*nodeConfig)) {
break;
}
if (BlAmd64ValidateBridgeDevice( nodeConfig ) == FALSE) {
break;
}
#if 0
for (mapIndex = 0; mapIndex < 8; mapIndex += 1) {
if (nodeConfig->DRAMMap[mapIndex].ReadEnable != 0) {
limit = nodeConfig->DRAMMap[mapIndex].Limit;
if (limit > NewBase) {
//
// The new base was found to conflict with existing
// ram.
//
return FALSE;
}
}
}
#endif
nodeCount += 1;
nodeConfig += 1;
} while (nodeCount <= 8);
if (nodeCount < 2) {
//
// This remap can only be performed on systems with more than
// two nodes.
//
return FALSE;
}
//
// We always remap the second node's memory (node 1).
//
nodeConfig = nodeConfigArray;
dramMap = &nodeConfig->DRAMMap[1];
oldBase = dramMap->Base;
oldLimit = dramMap->Limit;
span = oldLimit - oldBase;
newLimit = NewBase + span;
for (nodeIndex = 0; nodeIndex < nodeCount; nodeIndex += 1) {
ASSERT(dramMap->Base == oldBase);
ASSERT(dramMap->Limit == oldLimit);
dramMap->Base = NewBase;
dramMap->Limit = newLimit;
slotNumber.u.AsULONG = 0;
slotNumber.u.bits.DeviceNumber = NB_DEVICE_BASE + nodeIndex;
slotNumber.u.bits.FunctionNumber = 1;
length = HalSetBusDataByOffset( PCIConfiguration,
0,
slotNumber.u.AsULONG,
dramMap,
FIELD_OFFSET(AMD_NB_FUNC1_CONFIG,DRAMMap[1]),
sizeof(*dramMap) );
if (length != sizeof(*dramMap)) {
//
// We may be severely hosed here, if we have already
// reprogrammed some of the bridges.
//
return FALSE;
}
nodeConfig += 1;
dramMap = &nodeConfig->DRAMMap[1];
}
//
// Determine the address of the last byte of ram under 4G and the last
// byte of ram overall.
//
topMem = 0;
topMem4G = 0;
for (nodeIndex = 0; nodeIndex < nodeCount; nodeIndex += 1) {
base64 = nodeConfigArray[0].DRAMMap[nodeIndex].Base;
base64 <<= 24;
limit64 = nodeConfigArray[0].DRAMMap[nodeIndex].Limit;
limit64 = (limit64 + 1) << 24;
if (base64 < _4G) {
if (topMem4G < limit64) {
topMem4G = limit64;
}
}
if (topMem < limit64) {
topMem = limit64;
}
}
//
// Indicate whether a memory hole exists below 4G.
//
if (topMem4G < _4G) {
msrValue = RDMSR(MSR_TOP_MEM);
WRMSR(MSR_TOP_MEM,topMem4G & MSR_TOP_MEM_MASK);
}
//
// If memory above _4G was located then enable and program TOP_MEM_2
//
if (topMem > _4G) {
msrValue = RDMSR(MSR_SYSCFG);
msrValue |= SYSCFG_MTRRTOM2EN;
WRMSR(MSR_TOP_MEM_2, topMem & MSR_TOP_MEM_MASK);
WRMSR(MSR_SYSCFG,msrValue);
}
return TRUE;
}
BOOLEAN
BlAmd64GetNode1Info (
OUT ULONG *Base,
OUT ULONG *Size
)
/*++
Routine Description:
This routine determines the configuration of the block of physical
memory associated with node 1 (the second northbridge).
It also relocates the ACPI tables in node 1 to memory in node 0.
Arguments:
Base - supplies a pointer to the location in which to store the base
PFN of the node 1 memory block.
Size - supplies a pointer to the location in which to store the size,
in pages, of the node 1 memory block.
Return Value:
TRUE - A suitable second northbridge was found and the ACPI tables therein
were relocated if necessary.
FALSE - A suitable second northbridge was not found.
--*/
{
AMD_NB_FUNC1_CONFIG nodeConfig;
PCI_SLOT_NUMBER slotNumber;
ULONG length;
ULONG base;
ULONG size;
ULONG node0Base;
ULONG node0Size;
//
// Get the configuration of northbridge 1.
//
slotNumber.u.AsULONG = 0;
slotNumber.u.bits.DeviceNumber = NB_DEVICE_BASE + 1;
slotNumber.u.bits.FunctionNumber = 1;
length = HalGetBusDataByOffset( PCIConfiguration,
0,
slotNumber.u.AsULONG,
&nodeConfig,
0,
sizeof(nodeConfig) );
if (length != sizeof(nodeConfig)) {
return FALSE;
}
if (BlAmd64ValidateBridgeDevice( &nodeConfig ) == FALSE) {
return FALSE;
}
//
// A second northbridge exists, the relocation can be performed.
//
base = nodeConfig.DRAMMap[1].Base;
size = nodeConfig.DRAMMap[1].Limit - base + 1;
*Base = base << (24 - 12);
*Size = size << (24 - 12);
node0Base = nodeConfig.DRAMMap[0].Base;
node0Size = nodeConfig.DRAMMap[0].Limit - node0Base + 1;
node0Base <<= (24 - 12);
node0Size <<= (24 - 12);
BlAmd64RelocateAcpi( node0Base,
node0Base + node0Size - 1,
*Base,
*Base + *Size - 1 );
return TRUE;
}
BOOLEAN
BlAmd64RemapMTRRs (
IN ULONG OldBase,
IN ULONG NewBase,
IN ULONG Size
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
//
// All parameters expressed in pages
//
ULONG mtrrCount;
ULONG index;
MTRR_CAPABILITIES mtrrCapabilities;
PMTRR_VARIABLE_BASE baseArray;
PMTRR_VARIABLE_MASK maskArray;
ULONG allocationSize;
UNREFERENCED_PARAMETER(Size);
UNREFERENCED_PARAMETER(OldBase);
UNREFERENCED_PARAMETER(NewBase);
//
// Determine how many variable MTRRs are supported and
// allocate enough storage for all of them
//
mtrrCapabilities.QuadPart = RDMSR(MTRR_MSR_CAPABILITIES);
mtrrCount = (ULONG)mtrrCapabilities.Vcnt;
allocationSize = sizeof(*baseArray) * mtrrCount * 2;
baseArray = _alloca(allocationSize);
maskArray = (PMTRR_VARIABLE_MASK)(baseArray + mtrrCount);
RtlZeroMemory(baseArray,allocationSize);
//
// Read the variable MTRRSs. At the same time, look for the
// MTRR register that contains the old region, and a free
// one as well.
//
for (index = 0; index < mtrrCount; index += 1) {
baseArray[index].QuadPart = RDMSR(MTRR_MSR_VARIABLE_BASE + index * 2);
maskArray[index].QuadPart = RDMSR(MTRR_MSR_VARIABLE_MASK + index * 2);
}
//
// For now just clear the mask bits in MTRR register 0. This expands the
// first MTRR region so that it covers all memory.
//
maskArray[0].Mask = 0;
//
// Now reprogram the modified MTRR table
//
for (index = 0; index < mtrrCount; index += 1) {
WRMSR(MTRR_MSR_VARIABLE_BASE + index * 2,baseArray[index].QuadPart);
WRMSR(MTRR_MSR_VARIABLE_MASK + index * 2,maskArray[index].QuadPart);
}
return TRUE;
}
BOOLEAN
BlAmd64UpdateAcpiConfigurationEntry (
ULONG NewPhysical
)
/*++
Routine Description:
NTDETECT located the physical pointer to the ACPI RSDT table and passed it
up as a configuration node.
This routine finds that configuration node and replaces the physical address
therein with a new address.
This routine would be called after relocating the ACPI tables.
Arguments:
NewPhysical - Supplies the new physical address of the relocated ACPI tables.
Return Value:
TRUE if the relocation was performed, FALSE otherwise.
--*/
{
PCONFIGURATION_COMPONENT_DATA component;
PCONFIGURATION_COMPONENT_DATA resume;
PCM_PARTIAL_RESOURCE_LIST prl;
PCM_PARTIAL_RESOURCE_DESCRIPTOR prd;
PACPI_BIOS_MULTI_NODE rsdp;
resume = NULL;
while (TRUE) {
component = KeFindConfigurationNextEntry( BlLoaderBlock->ConfigurationRoot,
AdapterClass,
MultiFunctionAdapter,
NULL,
&resume );
if (component == NULL) {
return FALSE;
}
if (strcmp(component->ComponentEntry.Identifier,"ACPI BIOS") == 0) {
break;
}
resume = component;
}
prl = (PCM_PARTIAL_RESOURCE_LIST)component->ConfigurationData;
prd = prl->PartialDescriptors;
rsdp = (PACPI_BIOS_MULTI_NODE)(prd + 1);
rsdp->RsdtAddress.QuadPart = NewPhysical;
return TRUE;
}
BOOLEAN
BlAmd64RelocateAcpi (
ULONG Node0Base,
ULONG Node0Limit,
ULONG Node1Base,
ULONG Node1Limit
)
/*++
Routine Description:
This routine looks for ACPI tables within node 1's physical memory and,
if found, relocates them to node 0 memory.
Arguments:
Node0Base - Supplies the lowest PFN of node 0 memory
Node0Limit - Supplies the highest PFN of node 0 memory
Node1Base - Supplies the lowest PFN of node 1 memory (before relocation)
Node1Limit - Supplies the highest PFN of node 1 memory (before relocation)
Return Value:
Returns TRUE if successful, FALSE if a problem was encountered.
--*/
{
ULONG oldRsdtPhysical;
ULONG oldRsdtPhysicalPage;
ULONG newBasePage;
ULONG descriptorSize;
PLIST_ENTRY listHead;
PLIST_ENTRY listEntry;
PMEMORY_ALLOCATION_DESCRIPTOR oldAcpiDescriptor;
PMEMORY_ALLOCATION_DESCRIPTOR newAcpiDescriptor;
PMEMORY_ALLOCATION_DESCRIPTOR descriptor;
PCHAR oldAcpiVa;
PCHAR newAcpiVa;
PHYSICAL_ADDRESS physicalAddress;
ULONG vaBias;
PDESCRIPTION_HEADER descriptionHeader;
ULONG physAddr;
PFADT fadt;
ARC_STATUS status;
ULONG index;
ULONG rsdtPhysical;
ULONG rsdtLength;
//
// Add physicalBias to an ACPI physical pointer to relocate it
//
ULONG physicalBias;
oldRsdtPhysical = BlRsdp->RsdtAddress;
oldRsdtPhysicalPage = oldRsdtPhysical >> PAGE_SHIFT;
//
// Determine whether the descriptor resides in node 1's physical memory.
// If it does not then it does not need to be relocated.
//
if (oldRsdtPhysicalPage < Node1Base ||
oldRsdtPhysicalPage > Node1Limit) {
return TRUE;
}
//
// Find the descriptor that contains the ACPI tables
//
oldAcpiDescriptor = BlFindMemoryDescriptor( oldRsdtPhysicalPage );
//
// Find a descriptor in node 0 memory that is suitable for
// allocating the new ACPI tables from
//
listHead = &BlLoaderBlock->MemoryDescriptorListHead;
listEntry = listHead->Blink;
while (TRUE) {
descriptor = CONTAINING_RECORD(listEntry,
MEMORY_ALLOCATION_DESCRIPTOR,
ListEntry);
if ((descriptor->MemoryType == LoaderReserve) &&
(descriptor->BasePage > Node0Base) &&
((descriptor->BasePage + oldAcpiDescriptor->PageCount) < Node0Limit)) {
break;
}
listEntry = listEntry->Blink;
if (listEntry == listHead) {
return FALSE;
}
}
//
// Carve out the new ACPI descriptor
//
newBasePage = Node0Limit - oldAcpiDescriptor->PageCount + 1;
if ((newBasePage + oldAcpiDescriptor->PageCount) >
(descriptor->BasePage + descriptor->PageCount)) {
newBasePage = descriptor->BasePage +
descriptor->PageCount -
oldAcpiDescriptor->PageCount;
}
status = BlGenerateDescriptor( descriptor,
LoaderSpecialMemory,
newBasePage,
oldAcpiDescriptor->PageCount );
ASSERT( status == ESUCCESS );
newAcpiDescriptor = BlFindMemoryDescriptor( newBasePage );
ASSERT( newAcpiDescriptor != NULL );
//
// Unmap the old RSDT
//
MmUnmapIoSpace( BlRsdt, BlRsdt->Header.Length );
//
// Map both descriptors, copy data from new to old, then unmap
// and free the old descriptor.
//
descriptorSize = oldAcpiDescriptor->PageCount << PAGE_SHIFT;
physicalAddress.QuadPart = oldAcpiDescriptor->BasePage << PAGE_SHIFT;
oldAcpiVa = MmMapIoSpace (physicalAddress, descriptorSize, MmCached);
physicalAddress.QuadPart = newAcpiDescriptor->BasePage << PAGE_SHIFT;
newAcpiVa = MmMapIoSpace (physicalAddress, descriptorSize, MmCached);
RtlCopyMemory( newAcpiVa, oldAcpiVa, descriptorSize );
MmUnmapIoSpace( oldAcpiVa, descriptorSize );
oldAcpiDescriptor->MemoryType = LoaderReserve;
//
// Now thunk the new ACPI tables.
//
physicalBias = (newAcpiDescriptor->BasePage - oldAcpiDescriptor->BasePage) << PAGE_SHIFT;
vaBias = (ULONG)newAcpiVa - (newAcpiDescriptor->BasePage << PAGE_SHIFT);
#define PHYS_TO_VA(p) (PVOID)(p + vaBias)
rsdtPhysical = BlRsdp->RsdtAddress + physicalBias;
BlRsdp->RsdtAddress = rsdtPhysical;
ASSERT(BlXsdt == NULL);
BlRsdt = (PRSDT)PHYS_TO_VA(rsdtPhysical);
//
// Thunk the phys mem pointer array at the end of the RSDT
//
for (index = 0; index < NumTableEntriesFromRSDTPointer(BlRsdt); index += 1) {
physAddr = BlRsdt->Tables[index];
physAddr += physicalBias;
BlRsdt->Tables[index] = physAddr;
//
// Look for tables that themselves have physical pointers that require thunking
//
descriptionHeader = (PDESCRIPTION_HEADER)(PHYS_TO_VA(physAddr));
if (descriptionHeader->Signature == FADT_SIGNATURE) {
fadt = (PFADT)descriptionHeader;
fadt->facs += physicalBias;
fadt->dsdt += physicalBias;
}
}
//
// Now unmap the ACPI tables and remap just the RSDT table
//
rsdtLength = BlRsdt->Header.Length;
MmUnmapIoSpace( newAcpiVa, descriptorSize );
physicalAddress.QuadPart = rsdtPhysical;
BlRsdt = MmMapIoSpace( physicalAddress, rsdtLength, MmCached );
//
// Find the ACPI BIOS configuration entry and update it with the new
// RSDT physical address
//
BlAmd64UpdateAcpiConfigurationEntry( rsdtPhysical );
//
// That's it.
//
return TRUE;
}
#else // BL_ENABLE_REMAP
BOOLEAN
BlAmd64RemapDram (
IN PCHAR LoaderOptions
)
{
return FALSE;
}
#endif // BL_ENABLE_REMAP