mirror of
https://github.com/CloverHackyColor/CloverBootloader.git
synced 2024-11-23 11:35:19 +01:00
549 lines
20 KiB
C
549 lines
20 KiB
C
/**
|
|
|
|
Virtual memory functions.
|
|
|
|
by dmazar
|
|
|
|
**/
|
|
|
|
#include <Library/UefiLib.h>
|
|
#include <Library/BaseMemoryLib.h>
|
|
#include <Library/DebugLib.h>
|
|
|
|
#include "Config.h"
|
|
#include "VMem.h"
|
|
#include "MemoryMap.h"
|
|
|
|
/** Memory allocation for VM map pages that we will create with VmMapVirtualPage.
|
|
* We need to have it preallocated during boot services.
|
|
*/
|
|
UINT8 *VmMemoryPool = NULL;
|
|
INTN VmMemoryPoolFreePages = 0;
|
|
|
|
VOID
|
|
GetCurrentPageTable (
|
|
PAGE_MAP_AND_DIRECTORY_POINTER **PageTable,
|
|
UINTN *Flags
|
|
)
|
|
{
|
|
UINTN CR3;
|
|
|
|
CR3 = AsmReadCr3();
|
|
DEBUG ((DEBUG_VERBOSE, "GetCurrentPageTable: CR3 = 0x%lx\n", CR3));
|
|
*PageTable = (PAGE_MAP_AND_DIRECTORY_POINTER*)(UINTN)(CR3 & CR3_ADDR_MASK);
|
|
*Flags = CR3 & (CR3_FLAG_PWT | CR3_FLAG_PCD);
|
|
}
|
|
|
|
VOID
|
|
PrintPageTablePTE (
|
|
PAGE_TABLE_4K_ENTRY *PTE,
|
|
VIRTUAL_ADDR VA
|
|
)
|
|
{
|
|
#if !defined (MDEPKG_NDEBUG)
|
|
UINTN Index;
|
|
UINT64 Start;
|
|
|
|
for (Index = 0; Index < 10; Index++) {
|
|
VA.Pg4K.PTOffset = Index;
|
|
DEBUG ((DEBUG_VERBOSE, " PTE %03x at %p = %lx => ", Index, PTE, PTE->Uint64));
|
|
// 4KB PTE
|
|
Start = (PTE->Uint64 & PT_ADDR_MASK_4K);
|
|
DEBUG ((DEBUG_VERBOSE, "4KB Fl: %lx VA: %lx - %lx ==> PH %lx - %lx\n",
|
|
(PTE->Uint64 & ~PT_ADDR_MASK_4K), VA.Uint64, VA.Uint64 + 0x1000 - 1, Start, Start + 0x1000 - 1));
|
|
PTE++;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
VOID
|
|
PrintPageTablePDE (
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDE,
|
|
VIRTUAL_ADDR VA
|
|
)
|
|
{
|
|
#if !defined (MDEPKG_NDEBUG)
|
|
UINTN Index;
|
|
PAGE_TABLE_2M_ENTRY *PT2M;
|
|
PAGE_TABLE_4K_ENTRY *PTE;
|
|
UINT64 Start;
|
|
|
|
for (Index = 0; Index < 10; Index++) {
|
|
VA.Pg4K.PDOffset = Index;
|
|
DEBUG ((DEBUG_VERBOSE, " PDE %03x at %p = %lx => ", Index, PDE, PDE->Uint64));
|
|
if (PDE->Bits.MustBeZero & 0x1) {
|
|
// 2MB PDE
|
|
PT2M = (PAGE_TABLE_2M_ENTRY *)PDE;
|
|
Start = (PT2M->Uint64 & PT_ADDR_MASK_2M);
|
|
DEBUG ((DEBUG_VERBOSE, "2MB Fl: %lx VA: %lx - %lx ==> PH %lx - %lx\n",
|
|
(PT2M->Uint64 & ~PT_ADDR_MASK_2M), VA.Uint64, VA.Uint64 + 0x200000 - 1, Start, Start + 0x200000 - 1));
|
|
} else {
|
|
DEBUG ((DEBUG_VERBOSE, " Fl: %lx %lx ->\n", (PDE->Uint64 & ~PT_ADDR_MASK_4K), (PDE->Uint64 & PT_ADDR_MASK_4K)));
|
|
PTE = (PAGE_TABLE_4K_ENTRY *)(PDE->Uint64 & PT_ADDR_MASK_4K);
|
|
PrintPageTablePTE(PTE, VA);
|
|
}
|
|
PDE++;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
VOID
|
|
PrintPageTablePDPE (
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDPE,
|
|
VIRTUAL_ADDR VA
|
|
)
|
|
{
|
|
#if !defined (MDEPKG_NDEBUG)
|
|
UINTN Index;
|
|
PAGE_TABLE_1G_ENTRY *PT1G;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDE;
|
|
UINT64 Start;
|
|
|
|
for (Index = 0; Index < 10; Index++) {
|
|
VA.Pg4K.PDPOffset = Index;
|
|
DEBUG ((DEBUG_VERBOSE, " PDPE %03x at %p = %lx => ", Index, PDPE, PDPE->Uint64));
|
|
if (PDPE->Bits.MustBeZero & 0x1) {
|
|
// 1GB PDPE
|
|
PT1G = (PAGE_TABLE_1G_ENTRY *)PDPE;
|
|
Start = (PT1G->Uint64 & PT_ADDR_MASK_1G);
|
|
DEBUG ((DEBUG_VERBOSE, "1GB Fl: %lx VA: %lx - %lx ==> PH %lx - %lx\n",
|
|
(PT1G->Uint64 & ~PT_ADDR_MASK_1G), VA.Uint64, VA.Uint64 + 0x40000000 - 1, Start, Start + 0x40000000 - 1));
|
|
} else {
|
|
DEBUG ((DEBUG_VERBOSE, " Fl: %lx %lx ->\n", (PDPE->Uint64 & ~PT_ADDR_MASK_4K), (PDPE->Uint64 & PT_ADDR_MASK_4K)));
|
|
PDE = (PAGE_MAP_AND_DIRECTORY_POINTER *)(PDPE->Uint64 & PT_ADDR_MASK_4K);
|
|
PrintPageTablePDE(PDE, VA);
|
|
}
|
|
PDPE++;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
VOID
|
|
PrintPageTable (
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PageTable,
|
|
UINTN Flags
|
|
)
|
|
{
|
|
UINTN Index;
|
|
VIRTUAL_ADDR VA;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PML4;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDPE;
|
|
|
|
DEBUG ((DEBUG_VERBOSE, "PrintPageTable: %p, Flags: PWT: %d, PCD: %d\n", PageTable, (Flags & CR3_FLAG_PWT), (Flags & CR3_FLAG_PCD)));
|
|
PML4 = PageTable;
|
|
for (Index = 0; Index < 3; Index++) {
|
|
VA.Uint64 = 0;
|
|
VA.Pg4K.PML4Offset = Index;
|
|
VA_FIX_SIGN_EXTEND(VA);
|
|
DEBUG ((DEBUG_VERBOSE, "PML4 %03x at %p = %lx => Fl: %lx %lx ->\n", Index, PML4, PML4->Uint64, (PML4->Uint64 & ~PT_ADDR_MASK_4K), (PML4->Uint64 & PT_ADDR_MASK_4K)));
|
|
PDPE = (PAGE_MAP_AND_DIRECTORY_POINTER *)(PML4->Uint64 & PT_ADDR_MASK_4K);
|
|
PrintPageTablePDPE(PDPE, VA);
|
|
PML4++;
|
|
}
|
|
}
|
|
|
|
EFI_STATUS
|
|
GetPhysicalAddr (
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PageTable,
|
|
EFI_VIRTUAL_ADDRESS VirtualAddr,
|
|
EFI_PHYSICAL_ADDRESS *PhysicalAddr
|
|
)
|
|
{
|
|
EFI_PHYSICAL_ADDRESS Start;
|
|
VIRTUAL_ADDR VA;
|
|
VIRTUAL_ADDR VAStart;
|
|
VIRTUAL_ADDR VAEnd;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PML4;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDPE;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDE;
|
|
PAGE_TABLE_4K_ENTRY *PTE4K;
|
|
PAGE_TABLE_2M_ENTRY *PTE2M;
|
|
PAGE_TABLE_1G_ENTRY *PTE1G;
|
|
|
|
VA.Uint64 = (UINT64)VirtualAddr;
|
|
//VA_FIX_SIGN_EXTEND(VA);
|
|
DEBUG ((DEBUG_VERBOSE, "PageTable: %p\n", PageTable));
|
|
DEBUG ((DEBUG_VERBOSE, "VA: %lx => Indexes PML4=%x, PDP=%x, PD=%x, PT=%x\n",
|
|
VA.Uint64, VA.Pg4K.PML4Offset, VA.Pg4K.PDPOffset, VA.Pg4K.PDOffset, VA.Pg4K.PTOffset));
|
|
|
|
// PML4
|
|
PML4 = PageTable;
|
|
PML4 += VA.Pg4K.PML4Offset;
|
|
// prepare region start and end
|
|
VAStart.Uint64 = 0;
|
|
VAStart.Pg4K.PML4Offset = VA.Pg4K.PML4Offset;
|
|
VA_FIX_SIGN_EXTEND(VAStart);
|
|
VAEnd.Uint64 = ~(UINT64)0;
|
|
VAEnd.Pg4K.PML4Offset = VA.Pg4K.PML4Offset;
|
|
VA_FIX_SIGN_EXTEND(VAEnd);
|
|
// print it
|
|
DEBUG ((DEBUG_VERBOSE, "PML4[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PML4Offset, PML4, PML4->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (!PML4->Bits.Present) {
|
|
DEBUG ((DEBUG_VERBOSE, "-> Mapping not present!\n"));
|
|
return EFI_NO_MAPPING;
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PML4->Bits.Nx, PML4->Bits.Accessed,
|
|
PML4->Bits.CacheDisabled, PML4->Bits.WriteThrough,
|
|
PML4->Bits.UserSupervisor, PML4->Bits.ReadWrite, PML4->Bits.Present,
|
|
(PML4->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
|
|
// PDPE
|
|
PDPE = (PAGE_MAP_AND_DIRECTORY_POINTER *)(PML4->Uint64 & PT_ADDR_MASK_4K);
|
|
PDPE += VA.Pg4K.PDPOffset;
|
|
VAStart.Pg4K.PDPOffset = VA.Pg4K.PDPOffset;
|
|
VAEnd.Pg4K.PDPOffset = VA.Pg4K.PDPOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "PDPE[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PDPOffset, PDPE, PDPE->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (!PDPE->Bits.Present) {
|
|
DEBUG ((DEBUG_VERBOSE, "-> Mapping not present!\n"));
|
|
return EFI_NO_MAPPING;
|
|
}
|
|
if (PDPE->Bits.MustBeZero & 0x1) {
|
|
// 1GB PDPE
|
|
PTE1G = (PAGE_TABLE_1G_ENTRY *)PDPE;
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|G:%x|PAT:%x|D:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x\n",
|
|
PTE1G->Bits.Nx, PTE1G->Bits.Global, PTE1G->Bits.PAT,
|
|
PTE1G->Bits.Dirty, PTE1G->Bits.Accessed,
|
|
PTE1G->Bits.CacheDisabled, PTE1G->Bits.WriteThrough,
|
|
PTE1G->Bits.UserSupervisor, PTE1G->Bits.ReadWrite, PTE1G->Bits.Present
|
|
));
|
|
Start = (PTE1G->Uint64 & PT_ADDR_MASK_1G);
|
|
*PhysicalAddr = Start + VA.Pg1G.PhysPgOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "-> 1GB page %lx - %lx => %lx\n", Start, Start + 0x40000000 - 1, *PhysicalAddr));
|
|
return EFI_SUCCESS;
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PDPE->Bits.Nx, PDPE->Bits.Accessed,
|
|
PDPE->Bits.CacheDisabled, PDPE->Bits.WriteThrough,
|
|
PDPE->Bits.UserSupervisor, PDPE->Bits.ReadWrite, PDPE->Bits.Present,
|
|
(PDPE->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
|
|
// PDE
|
|
PDE = (PAGE_MAP_AND_DIRECTORY_POINTER *)(PDPE->Uint64 & PT_ADDR_MASK_4K);
|
|
PDE += VA.Pg4K.PDOffset;
|
|
VAStart.Pg4K.PDOffset = VA.Pg4K.PDOffset;
|
|
VAEnd.Pg4K.PDOffset = VA.Pg4K.PDOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "PDE[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PDOffset, PDE, PDE->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (!PDE->Bits.Present) {
|
|
DEBUG ((DEBUG_VERBOSE, "-> Mapping not present!\n"));
|
|
return EFI_NO_MAPPING;
|
|
}
|
|
if (PDE->Bits.MustBeZero & 0x1) {
|
|
// 2MB PDE
|
|
PTE2M = (PAGE_TABLE_2M_ENTRY *)PDE;
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|G:%x|PAT:%x|D:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x\n",
|
|
PTE2M->Bits.Nx, PTE2M->Bits.Global, PTE2M->Bits.PAT,
|
|
PTE2M->Bits.Dirty, PTE2M->Bits.Accessed,
|
|
PTE2M->Bits.CacheDisabled, PTE2M->Bits.WriteThrough,
|
|
PTE2M->Bits.UserSupervisor, PTE2M->Bits.ReadWrite, PTE2M->Bits.Present
|
|
));
|
|
Start = (PTE2M->Uint64 & PT_ADDR_MASK_2M);
|
|
*PhysicalAddr = Start + VA.Pg2M.PhysPgOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "-> 2MB page %lx - %lx => %lx\n", Start, Start + 0x200000 - 1, *PhysicalAddr));
|
|
return EFI_SUCCESS;
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PDE->Bits.Nx, PDE->Bits.Accessed,
|
|
PDE->Bits.CacheDisabled, PDE->Bits.WriteThrough,
|
|
PDE->Bits.UserSupervisor, PDE->Bits.ReadWrite, PDE->Bits.Present,
|
|
(PDE->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
|
|
// PTE
|
|
PTE4K = (PAGE_TABLE_4K_ENTRY *)(PDE->Uint64 & PT_ADDR_MASK_4K);
|
|
PTE4K += VA.Pg4K.PTOffset;
|
|
VAStart.Pg4K.PTOffset = VA.Pg4K.PTOffset;
|
|
VAEnd.Pg4K.PTOffset = VA.Pg4K.PTOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "PTE[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PTOffset, PTE4K, PTE4K->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (!PTE4K->Bits.Present) {
|
|
DEBUG ((DEBUG_VERBOSE, "-> Mapping not present!\n"));
|
|
return EFI_NO_MAPPING;
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|G:%x|PAT:%x|D:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PTE4K->Bits.Nx, PTE4K->Bits.Global, PTE4K->Bits.PAT,
|
|
PTE4K->Bits.Dirty, PTE4K->Bits.Accessed,
|
|
PTE4K->Bits.CacheDisabled, PTE4K->Bits.WriteThrough,
|
|
PTE4K->Bits.UserSupervisor, PTE4K->Bits.ReadWrite, PTE4K->Bits.Present,
|
|
(PTE4K->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
Start = (PTE4K->Uint64 & PT_ADDR_MASK_4K);
|
|
*PhysicalAddr = Start + VA.Pg4K.PhysPgOffset;
|
|
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
/** Inits vm memory pool. Should be called while boot services are still usable. */
|
|
EFI_STATUS
|
|
VmAllocateMemoryPool (
|
|
VOID
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_PHYSICAL_ADDRESS Addr;
|
|
|
|
if (VmMemoryPool != NULL) {
|
|
// already allocated
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
VmMemoryPoolFreePages = 0x200; // 2 MB should be enough
|
|
Addr = BASE_4GB; // max address
|
|
|
|
Status = AllocatePagesFromTop (EfiBootServicesData, VmMemoryPoolFreePages, &Addr, FALSE);
|
|
if (EFI_ERROR (Status)) {
|
|
Print (L"AMF: vm memory pool allocation failure - %r\n", Status);
|
|
} else {
|
|
VmMemoryPool = (UINT8*)Addr;
|
|
DEBUG ((DEBUG_VERBOSE, "VmMemoryPool = %lx - %lx\n", VmMemoryPool, VmMemoryPool + EFI_PAGES_TO_SIZE(VmMemoryPoolFreePages) - 1));
|
|
}
|
|
return Status;
|
|
}
|
|
|
|
/** Central method for allocating pages for VM page maps. */
|
|
VOID *
|
|
VmAllocatePages (
|
|
UINTN NumPages
|
|
)
|
|
{
|
|
VOID *AllocatedPages = NULL;
|
|
|
|
if (VmMemoryPoolFreePages >= (INTN)NumPages) {
|
|
AllocatedPages = VmMemoryPool;
|
|
VmMemoryPool += EFI_PAGES_TO_SIZE(NumPages);
|
|
VmMemoryPoolFreePages -= NumPages;
|
|
} else {
|
|
DEBUG ((DEBUG_INFO, "VmAllocatePages - no more pages!\n"));
|
|
CpuDeadLoop();
|
|
}
|
|
return AllocatedPages;
|
|
}
|
|
|
|
/** Maps (remaps) 4K page given by VirtualAddr to PhysicalAddr page in PageTable. */
|
|
EFI_STATUS
|
|
VmMapVirtualPage (
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PageTable,
|
|
EFI_VIRTUAL_ADDRESS VirtualAddr,
|
|
EFI_PHYSICAL_ADDRESS PhysicalAddr
|
|
)
|
|
{
|
|
EFI_PHYSICAL_ADDRESS Start;
|
|
VIRTUAL_ADDR VA;
|
|
VIRTUAL_ADDR VAStart;
|
|
VIRTUAL_ADDR VAEnd;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PML4;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDPE;
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PDE;
|
|
PAGE_TABLE_4K_ENTRY *PTE4K;
|
|
PAGE_TABLE_4K_ENTRY *PTE4KTmp;
|
|
PAGE_TABLE_2M_ENTRY *PTE2M;
|
|
PAGE_TABLE_1G_ENTRY *PTE1G;
|
|
UINTN Index;
|
|
|
|
VA.Uint64 = (UINT64)VirtualAddr;
|
|
//VA_FIX_SIGN_EXTEND(VA);
|
|
DEBUG ((DEBUG_VERBOSE, "VmMapVirtualPage VA %lx => PA %lx\nPageTable: %p\n", VirtualAddr, PhysicalAddr, PageTable));
|
|
DEBUG ((DEBUG_VERBOSE, "VA: %lx => Indexes PML4=%x, PDP=%x, PD=%x, PT=%x\n",
|
|
VA.Uint64, VA.Pg4K.PML4Offset, VA.Pg4K.PDPOffset, VA.Pg4K.PDOffset, VA.Pg4K.PTOffset));
|
|
|
|
// PML4
|
|
PML4 = PageTable;
|
|
PML4 += VA.Pg4K.PML4Offset;
|
|
// there is a problem if our PML4 points to the same table as first PML4 entry
|
|
// since we may mess the mapping of first virtual region (happens in VBox and probably DUET).
|
|
// check for this on first call and if true, just clear our PML4 - we'll rebuild it in later step
|
|
if (PML4 != PageTable && PML4->Bits.Present && PageTable->Bits.PageTableBaseAddress == PML4->Bits.PageTableBaseAddress) {
|
|
DEBUG ((DEBUG_VERBOSE, "PML4 points to the same table as first PML4 - releasing it and rebuiding in a separate table\n"));
|
|
PML4->Uint64 = 0;
|
|
}
|
|
|
|
// prepare region start and end
|
|
VAStart.Uint64 = 0;
|
|
VAStart.Pg4K.PML4Offset = VA.Pg4K.PML4Offset;
|
|
VA_FIX_SIGN_EXTEND(VAStart);
|
|
VAEnd.Uint64 = ~(UINT64)0;
|
|
VAEnd.Pg4K.PML4Offset = VA.Pg4K.PML4Offset;
|
|
VA_FIX_SIGN_EXTEND(VAEnd);
|
|
// print it
|
|
DEBUG ((DEBUG_VERBOSE, "PML4[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PML4Offset, PML4, PML4->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (!PML4->Bits.Present) {
|
|
DEBUG ((DEBUG_VERBOSE, "-> Mapping not present, creating new PML4 entry and page with PDPE entries!\n"));
|
|
PDPE = (PAGE_MAP_AND_DIRECTORY_POINTER *)VmAllocatePages(1);
|
|
if (PDPE == NULL) {
|
|
DEBUG ((DEBUG_VERBOSE, "No memory - exiting.\n"));
|
|
return EFI_NO_MAPPING;
|
|
}
|
|
|
|
ZeroMem(PDPE, EFI_PAGE_SIZE);
|
|
// init this whole 512GB region with 512 1GB entry pages to map first 512GB phys space
|
|
PTE1G = (PAGE_TABLE_1G_ENTRY *)PDPE;
|
|
Start = 0;
|
|
for (Index = 0; Index < 512; Index++) {
|
|
PTE1G->Uint64 = Start & PT_ADDR_MASK_1G;
|
|
PTE1G->Bits.ReadWrite = 1;
|
|
PTE1G->Bits.Present = 1;
|
|
PTE1G->Bits.MustBe1 = 1;
|
|
PTE1G++;
|
|
Start += 0x40000000;
|
|
}
|
|
|
|
// put it to PML4
|
|
PML4->Uint64 = ((UINT64)PDPE) & PT_ADDR_MASK_4K;
|
|
PML4->Bits.ReadWrite = 1;
|
|
PML4->Bits.Present = 1;
|
|
DEBUG ((DEBUG_VERBOSE, "added to PLM4 as %lx\n", PML4->Uint64));
|
|
// and continue with mapping ...
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PML4->Bits.Nx, PML4->Bits.Accessed,
|
|
PML4->Bits.CacheDisabled, PML4->Bits.WriteThrough,
|
|
PML4->Bits.UserSupervisor, PML4->Bits.ReadWrite, PML4->Bits.Present,
|
|
(PML4->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
|
|
// PDPE
|
|
PDPE = (PAGE_MAP_AND_DIRECTORY_POINTER *)(PML4->Uint64 & PT_ADDR_MASK_4K);
|
|
PDPE += VA.Pg4K.PDPOffset;
|
|
VAStart.Pg4K.PDPOffset = VA.Pg4K.PDPOffset;
|
|
VAEnd.Pg4K.PDPOffset = VA.Pg4K.PDPOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "PDPE[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PDPOffset, PDPE, PDPE->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (!PDPE->Bits.Present || (PDPE->Bits.MustBeZero & 0x1)) {
|
|
DEBUG ((DEBUG_VERBOSE, "-> Mapping not present or mapped as 1GB page, creating new PDPE entry and page with PDE entries!\n"));
|
|
PDE = (PAGE_MAP_AND_DIRECTORY_POINTER *)VmAllocatePages(1);
|
|
if (PDE == NULL) {
|
|
DEBUG ((DEBUG_VERBOSE, "No memory - exiting.\n"));
|
|
return EFI_NO_MAPPING;
|
|
}
|
|
ZeroMem(PDE, EFI_PAGE_SIZE);
|
|
|
|
if (PDPE->Bits.MustBeZero & 0x1) {
|
|
// was 1GB page - init new PDE array to get the same mapping but with 2MB pages
|
|
DEBUG ((DEBUG_VERBOSE, "-> was 1GB page, initing new PDE array to get the same mapping but with 2MB pages!\n"));
|
|
PTE2M = (PAGE_TABLE_2M_ENTRY *)PDE;
|
|
Start = (PDPE->Uint64 & PT_ADDR_MASK_1G);
|
|
for (Index = 0; Index < 512; Index++) {
|
|
PTE2M->Uint64 = Start & PT_ADDR_MASK_2M;
|
|
PTE2M->Bits.ReadWrite = 1;
|
|
PTE2M->Bits.Present = 1;
|
|
PTE2M->Bits.MustBe1 = 1;
|
|
PTE2M++;
|
|
Start += 0x200000;
|
|
}
|
|
}
|
|
|
|
// put it to PDPE
|
|
PDPE->Uint64 = ((UINT64)PDE) & PT_ADDR_MASK_4K;
|
|
PDPE->Bits.ReadWrite = 1;
|
|
PDPE->Bits.Present = 1;
|
|
DEBUG ((DEBUG_VERBOSE, "added to PDPE as %lx\n", PDPE->Uint64));
|
|
// and continue with mapping ...
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PDPE->Bits.Nx, PDPE->Bits.Accessed,
|
|
PDPE->Bits.CacheDisabled, PDPE->Bits.WriteThrough,
|
|
PDPE->Bits.UserSupervisor, PDPE->Bits.ReadWrite, PDPE->Bits.Present,
|
|
(PDPE->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
|
|
// PDE
|
|
PDE = (PAGE_MAP_AND_DIRECTORY_POINTER *)(PDPE->Uint64 & PT_ADDR_MASK_4K);
|
|
PDE += VA.Pg4K.PDOffset;
|
|
VAStart.Pg4K.PDOffset = VA.Pg4K.PDOffset;
|
|
VAEnd.Pg4K.PDOffset = VA.Pg4K.PDOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "PDE[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PDOffset, PDE, PDE->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (!PDE->Bits.Present || (PDE->Bits.MustBeZero & 0x1)) {
|
|
DEBUG ((DEBUG_VERBOSE, "-> Mapping not present or mapped as 2MB page, creating new PDE entry and page with PTE4K entries!\n"));
|
|
PTE4K = (PAGE_TABLE_4K_ENTRY *)VmAllocatePages(1);
|
|
if (PTE4K == NULL) {
|
|
DEBUG ((DEBUG_VERBOSE, "No memory - exiting.\n"));
|
|
return EFI_NO_MAPPING;
|
|
}
|
|
ZeroMem(PTE4K, EFI_PAGE_SIZE);
|
|
|
|
if (PDE->Bits.MustBeZero & 0x1) {
|
|
// was 2MB page - init new PTE array to get the same mapping but with 4KB pages
|
|
DEBUG ((DEBUG_VERBOSE, "-> was 2MB page - initing new PTE array to get the same mapping but with 4KB pages!\n"));
|
|
PTE4KTmp = (PAGE_TABLE_4K_ENTRY *)PTE4K;
|
|
Start = (PDE->Uint64 & PT_ADDR_MASK_2M);
|
|
for (Index = 0; Index < 512; Index++) {
|
|
PTE4KTmp->Uint64 = Start & PT_ADDR_MASK_4K;
|
|
PTE4KTmp->Bits.ReadWrite = 1;
|
|
PTE4KTmp->Bits.Present = 1;
|
|
PTE4KTmp++;
|
|
Start += 0x1000;
|
|
}
|
|
}
|
|
|
|
// put it to PDE
|
|
PDE->Uint64 = ((UINT64)PTE4K) & PT_ADDR_MASK_4K;
|
|
PDE->Bits.ReadWrite = 1;
|
|
PDE->Bits.Present = 1;
|
|
DEBUG ((DEBUG_VERBOSE, "added to PDE as %lx\n", PDE->Uint64));
|
|
// and continue with mapping ...
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PDE->Bits.Nx, PDE->Bits.Accessed,
|
|
PDE->Bits.CacheDisabled, PDE->Bits.WriteThrough,
|
|
PDE->Bits.UserSupervisor, PDE->Bits.ReadWrite, PDE->Bits.Present,
|
|
(PDE->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
|
|
// PTE
|
|
PTE4K = (PAGE_TABLE_4K_ENTRY *)(PDE->Uint64 & PT_ADDR_MASK_4K);
|
|
PTE4K += VA.Pg4K.PTOffset;
|
|
VAStart.Pg4K.PTOffset = VA.Pg4K.PTOffset;
|
|
VAEnd.Pg4K.PTOffset = VA.Pg4K.PTOffset;
|
|
DEBUG ((DEBUG_VERBOSE, "PTE[%03x] at %p = %lx Region: %lx - %lx\n", VA.Pg4K.PTOffset, PTE4K, PTE4K->Uint64, VAStart.Uint64, VAEnd.Uint64));
|
|
if (PTE4K->Bits.Present) {
|
|
DEBUG ((DEBUG_VERBOSE, "mapping already present - remapping!\n"));
|
|
}
|
|
DEBUG ((DEBUG_VERBOSE, "-> Nx:%x|G:%x|PAT:%x|D:%x|A:%x|PCD:%x|PWT:%x|US:%x|RW:%x|P:%x -> %lx\n",
|
|
PTE4K->Bits.Nx, PTE4K->Bits.Global, PTE4K->Bits.PAT,
|
|
PTE4K->Bits.Dirty, PTE4K->Bits.Accessed,
|
|
PTE4K->Bits.CacheDisabled, PTE4K->Bits.WriteThrough,
|
|
PTE4K->Bits.UserSupervisor, PTE4K->Bits.ReadWrite, PTE4K->Bits.Present,
|
|
(PTE4K->Uint64 & PT_ADDR_MASK_4K)
|
|
));
|
|
// put it to PTE
|
|
PTE4K->Uint64 = ((UINT64)PhysicalAddr) & PT_ADDR_MASK_4K;
|
|
PTE4K->Bits.ReadWrite = 1;
|
|
PTE4K->Bits.Present = 1;
|
|
DEBUG ((DEBUG_VERBOSE, "added to PTE4K as %lx\n", PTE4K->Uint64));
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
/** Maps (remaps) NumPages 4K pages given by VirtualAddr to PhysicalAddr pages in PageTable. */
|
|
EFI_STATUS
|
|
VmMapVirtualPages (
|
|
PAGE_MAP_AND_DIRECTORY_POINTER *PageTable,
|
|
EFI_VIRTUAL_ADDRESS VirtualAddr,
|
|
UINTN NumPages,
|
|
EFI_PHYSICAL_ADDRESS PhysicalAddr
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
|
|
Status = EFI_SUCCESS;
|
|
while (NumPages > 0 && (Status == EFI_SUCCESS)) {
|
|
Status = VmMapVirtualPage(PageTable, VirtualAddr, PhysicalAddr);
|
|
VirtualAddr += 0x1000;
|
|
PhysicalAddr += 0x1000;
|
|
NumPages--;
|
|
DEBUG ((DEBUG_VERBOSE, "NumPages: %d, %lx => %lx\n", NumPages, VirtualAddr, PhysicalAddr));
|
|
}
|
|
return Status;
|
|
}
|
|
|
|
/** Flashes TLB caches. */
|
|
VOID
|
|
VmFlushCaches (
|
|
VOID
|
|
)
|
|
{
|
|
// just reload CR3
|
|
AsmWriteCr3(AsmReadCr3());
|
|
}
|