CloverBootloader/MemoryFix/AptioMemoryFix/VMem.c

549 lines
20 KiB
C
Raw Normal View History

/**
Virtual memory functions.
by dmazar
**/
#include <Library/UefiLib.h>
#include <Library/BaseMemoryLib.h>
2019-10-04 22:32:02 +02:00
#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)) {
2019-10-04 22:32:02 +02:00
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());
}