mirror of
https://github.com/Fennix-Project/Kernel.git
synced 2025-05-25 22:14:37 +00:00
531 lines
16 KiB
C++
531 lines
16 KiB
C++
/*
|
|
This file is part of Fennix Kernel.
|
|
|
|
Fennix Kernel is free software: you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License as
|
|
published by the Free Software Foundation, either version 3 of
|
|
the License, or (at your option) any later version.
|
|
|
|
Fennix Kernel is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Fennix Kernel. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <memory.hpp>
|
|
|
|
#include <convert.h>
|
|
#include <debug.h>
|
|
|
|
namespace Memory
|
|
{
|
|
bool Virtual::Check(void *VirtualAddress, PTFlag Flag)
|
|
{
|
|
uintptr_t Address = (uintptr_t)VirtualAddress;
|
|
Address &= 0xFFFFFFFFFFFFF000;
|
|
|
|
PageMapIndexer Index = PageMapIndexer(Address);
|
|
|
|
PageDirectoryPointerTableEntryPtr *PDPTE = nullptr;
|
|
PageDirectoryEntryPtr *PDE = nullptr;
|
|
PageTableEntryPtr *PTE = nullptr;
|
|
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
if (!PML4->Present)
|
|
{
|
|
debug("PML4 not present for %#lx", VirtualAddress);
|
|
return false;
|
|
}
|
|
|
|
PDPTE = (PageDirectoryPointerTableEntryPtr *)((uintptr_t)PML4->GetAddress() << 12);
|
|
if (!PDPTE)
|
|
{
|
|
debug("Failed to get PDPTE for %#lx", VirtualAddress);
|
|
return false;
|
|
}
|
|
|
|
if (PDPTE->Entries[Index.PDPTEIndex].PageSize)
|
|
{
|
|
bool result = PDPTE->Entries[Index.PDPTEIndex].raw & Flag;
|
|
if (!result)
|
|
{
|
|
debug("Failed to check %#lx for %#lx (raw: %#lx)", VirtualAddress, Flag,
|
|
PDPTE->Entries[Index.PDPTEIndex].raw);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
PDE = (PageDirectoryEntryPtr *)((uintptr_t)PDPTE->Entries[Index.PDPTEIndex].GetAddress() << 12);
|
|
if (!PDE)
|
|
{
|
|
debug("Failed to get PDE for %#lx", VirtualAddress);
|
|
return false;
|
|
}
|
|
|
|
if (PDE->Entries[Index.PDEIndex].PageSize)
|
|
{
|
|
bool result = PDE->Entries[Index.PDEIndex].raw & Flag;
|
|
if (!result)
|
|
{
|
|
debug("Failed to check %#lx for %#lx (raw: %#lx)", VirtualAddress, Flag,
|
|
PDE->Entries[Index.PDEIndex].raw);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
PTE = (PageTableEntryPtr *)((uintptr_t)PDE->Entries[Index.PDEIndex].GetAddress() << 12);
|
|
if (!PTE)
|
|
{
|
|
debug("Failed to get PTE for %#lx", VirtualAddress);
|
|
return false;
|
|
}
|
|
|
|
bool result = PTE->Entries[Index.PTEIndex].raw & Flag;
|
|
if (!result)
|
|
{
|
|
debug("Failed to check %#lx for %#lx (raw: %#lx)", VirtualAddress, Flag,
|
|
PTE->Entries[Index.PTEIndex].raw);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void *Virtual::GetPhysical(void *VirtualAddress)
|
|
{
|
|
uintptr_t Address = (uintptr_t)VirtualAddress;
|
|
Address &= 0xFFFFFFFFFFFFF000;
|
|
|
|
PageMapIndexer Index = PageMapIndexer(Address);
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
|
|
PageDirectoryPointerTableEntryPtr *PDPTE = nullptr;
|
|
PageDirectoryEntryPtr *PDE = nullptr;
|
|
PageTableEntryPtr *PTE = nullptr;
|
|
|
|
if (PML4->Present)
|
|
{
|
|
PDPTE = (PageDirectoryPointerTableEntryPtr *)((uintptr_t)PML4->GetAddress() << 12);
|
|
if (PDPTE)
|
|
{
|
|
if (PDPTE->Entries[Index.PDPTEIndex].Present)
|
|
{
|
|
if (PDPTE->Entries[Index.PDPTEIndex].PageSize)
|
|
return (void *)((uintptr_t)PDPTE->Entries[Index.PDPTEIndex].GetAddress() << 12);
|
|
|
|
PDE = (PageDirectoryEntryPtr *)((uintptr_t)PDPTE->Entries[Index.PDPTEIndex].GetAddress() << 12);
|
|
if (PDE)
|
|
{
|
|
if (PDE->Entries[Index.PDEIndex].Present)
|
|
{
|
|
if (PDE->Entries[Index.PDEIndex].PageSize)
|
|
return (void *)((uintptr_t)PDE->Entries[Index.PDEIndex].GetAddress() << 12);
|
|
|
|
PTE = (PageTableEntryPtr *)((uintptr_t)PDE->Entries[Index.PDEIndex].GetAddress() << 12);
|
|
if (PTE)
|
|
{
|
|
if (PTE->Entries[Index.PTEIndex].Present)
|
|
return (void *)((uintptr_t)PTE->Entries[Index.PTEIndex].GetAddress() << 12);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Virtual::MapType Virtual::GetMapType(void *VirtualAddress)
|
|
{
|
|
uintptr_t Address = (uintptr_t)VirtualAddress;
|
|
Address &= 0xFFFFFFFFFFFFF000;
|
|
|
|
PageMapIndexer Index = PageMapIndexer(Address);
|
|
|
|
PageDirectoryPointerTableEntryPtr *PDPTE = nullptr;
|
|
PageDirectoryEntryPtr *PDE = nullptr;
|
|
PageTableEntryPtr *PTE = nullptr;
|
|
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
if (!PML4->Present)
|
|
goto ReturnLogError;
|
|
|
|
PDPTE = (PageDirectoryPointerTableEntryPtr *)((uintptr_t)PML4->GetAddress() << 12);
|
|
if (!PDPTE || !PDPTE->Entries[Index.PDPTEIndex].Present)
|
|
goto ReturnLogError;
|
|
|
|
if (PDPTE->Entries[Index.PDPTEIndex].PageSize)
|
|
return MapType::OneGiB;
|
|
|
|
PDE = (PageDirectoryEntryPtr *)((uintptr_t)PDPTE->Entries[Index.PDPTEIndex].GetAddress() << 12);
|
|
if (!PDE || !PDE->Entries[Index.PDEIndex].Present)
|
|
goto ReturnLogError;
|
|
|
|
if (PDE->Entries[Index.PDEIndex].PageSize)
|
|
return MapType::TwoMiB;
|
|
|
|
PTE = (PageTableEntryPtr *)((uintptr_t)PDE->Entries[Index.PDEIndex].GetAddress() << 12);
|
|
if (!PTE)
|
|
goto ReturnLogError;
|
|
|
|
if (PTE->Entries[Index.PTEIndex].Present)
|
|
return MapType::FourKiB;
|
|
|
|
ReturnLogError:
|
|
return MapType::NoMapType;
|
|
}
|
|
|
|
PageMapLevel5 *Virtual::GetPML5(void *VirtualAddress, MapType Type)
|
|
{
|
|
UNUSED(VirtualAddress);
|
|
UNUSED(Type);
|
|
stub; /* TODO */
|
|
return nullptr;
|
|
}
|
|
|
|
PageMapLevel4 *Virtual::GetPML4(void *VirtualAddress, MapType Type)
|
|
{
|
|
UNUSED(Type);
|
|
uintptr_t Address = (uintptr_t)VirtualAddress;
|
|
Address &= 0xFFFFFFFFFFFFF000;
|
|
|
|
PageMapIndexer Index = PageMapIndexer(Address);
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
if (PML4->Present)
|
|
return PML4;
|
|
|
|
debug("PML4 not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageDirectoryPointerTableEntry *Virtual::GetPDPTE(void *VirtualAddress, MapType Type)
|
|
{
|
|
UNUSED(Type);
|
|
uintptr_t Address = (uintptr_t)VirtualAddress;
|
|
Address &= 0xFFFFFFFFFFFFF000;
|
|
|
|
PageMapIndexer Index = PageMapIndexer(Address);
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
if (!PML4->Present)
|
|
{
|
|
debug("PML4 not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageDirectoryPointerTableEntryPtr *PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)((uintptr_t)PML4->Address << 12);
|
|
PageDirectoryPointerTableEntry *PDPTE = &PDPTEPtr->Entries[Index.PDPTEIndex];
|
|
if (PDPTE->Present)
|
|
return PDPTE;
|
|
|
|
debug("PDPTE not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageDirectoryEntry *Virtual::GetPDE(void *VirtualAddress, MapType Type)
|
|
{
|
|
UNUSED(Type);
|
|
uintptr_t Address = (uintptr_t)VirtualAddress;
|
|
Address &= 0xFFFFFFFFFFFFF000;
|
|
|
|
PageMapIndexer Index = PageMapIndexer(Address);
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
if (!PML4->Present)
|
|
{
|
|
debug("PML4 not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageDirectoryPointerTableEntryPtr *PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)((uintptr_t)PML4->Address << 12);
|
|
PageDirectoryPointerTableEntry *PDPTE = &PDPTEPtr->Entries[Index.PDPTEIndex];
|
|
if (!PDPTE->Present)
|
|
{
|
|
debug("PDPTE not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageDirectoryEntryPtr *PDEPtr = (PageDirectoryEntryPtr *)(PDPTE->GetAddress() << 12);
|
|
PageDirectoryEntry *PDE = &PDEPtr->Entries[Index.PDEIndex];
|
|
if (PDE->Present)
|
|
return PDE;
|
|
|
|
debug("PDE not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageTableEntry *Virtual::GetPTE(void *VirtualAddress, MapType Type)
|
|
{
|
|
UNUSED(Type);
|
|
uintptr_t Address = (uintptr_t)VirtualAddress;
|
|
Address &= 0xFFFFFFFFFFFFF000;
|
|
|
|
PageMapIndexer Index = PageMapIndexer(Address);
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
if (!PML4->Present)
|
|
{
|
|
debug("PML4 not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageDirectoryPointerTableEntryPtr *PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)((uintptr_t)PML4->Address << 12);
|
|
PageDirectoryPointerTableEntry *PDPTE = &PDPTEPtr->Entries[Index.PDPTEIndex];
|
|
if (!PDPTE->Present)
|
|
{
|
|
debug("PDPTE not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageDirectoryEntryPtr *PDEPtr = (PageDirectoryEntryPtr *)(PDPTE->GetAddress() << 12);
|
|
PageDirectoryEntry *PDE = &PDEPtr->Entries[Index.PDEIndex];
|
|
if (!PDE->Present)
|
|
{
|
|
debug("PDE not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
PageTableEntryPtr *PTEPtr = (PageTableEntryPtr *)(PDE->GetAddress() << 12);
|
|
PageTableEntry *PTE = &PTEPtr->Entries[Index.PTEIndex];
|
|
if (PTE->Present)
|
|
return PTE;
|
|
|
|
debug("PTE not present for %#lx", VirtualAddress);
|
|
return nullptr;
|
|
}
|
|
|
|
void Virtual::Map(void *VirtualAddress, void *PhysicalAddress, uint64_t Flags, MapType Type)
|
|
{
|
|
SmartLock(this->MemoryLock);
|
|
if (unlikely(!this->pTable))
|
|
{
|
|
error("No page table");
|
|
return;
|
|
}
|
|
|
|
Flags |= PTFlag::P;
|
|
|
|
PageMapIndexer Index = PageMapIndexer((uintptr_t)VirtualAddress);
|
|
// Clear any flags that are not 1 << 0 (Present) - 1 << 5 (Accessed) because rest are for page table entries only
|
|
uint64_t DirectoryFlags = Flags & 0x3F;
|
|
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
PageDirectoryPointerTableEntryPtr *PDPTEPtr = nullptr;
|
|
if (!PML4->Present)
|
|
{
|
|
PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)KernelAllocator.RequestPages(TO_PAGES(sizeof(PageDirectoryPointerTableEntryPtr) + 1));
|
|
memset(PDPTEPtr, 0, sizeof(PageDirectoryPointerTableEntryPtr));
|
|
PML4->Present = true;
|
|
PML4->SetAddress((uintptr_t)PDPTEPtr >> 12);
|
|
}
|
|
else
|
|
PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)(PML4->GetAddress() << 12);
|
|
PML4->raw |= DirectoryFlags;
|
|
|
|
PageDirectoryPointerTableEntry *PDPTE = &PDPTEPtr->Entries[Index.PDPTEIndex];
|
|
if (Type == MapType::OneGiB)
|
|
{
|
|
PDPTE->raw |= Flags;
|
|
PDPTE->PageSize = true;
|
|
PDPTE->SetAddress((uintptr_t)PhysicalAddress >> 12);
|
|
debug("Mapped 1GB page at %p to %p", VirtualAddress, PhysicalAddress);
|
|
return;
|
|
}
|
|
|
|
PageDirectoryEntryPtr *PDEPtr = nullptr;
|
|
if (!PDPTE->Present)
|
|
{
|
|
PDEPtr = (PageDirectoryEntryPtr *)KernelAllocator.RequestPages(TO_PAGES(sizeof(PageDirectoryEntryPtr) + 1));
|
|
memset(PDEPtr, 0, sizeof(PageDirectoryEntryPtr));
|
|
PDPTE->Present = true;
|
|
PDPTE->SetAddress((uintptr_t)PDEPtr >> 12);
|
|
}
|
|
else
|
|
PDEPtr = (PageDirectoryEntryPtr *)(PDPTE->GetAddress() << 12);
|
|
PDPTE->raw |= DirectoryFlags;
|
|
|
|
PageDirectoryEntry *PDE = &PDEPtr->Entries[Index.PDEIndex];
|
|
if (Type == MapType::TwoMiB)
|
|
{
|
|
PDE->raw |= Flags;
|
|
PDE->PageSize = true;
|
|
PDE->SetAddress((uintptr_t)PhysicalAddress >> 12);
|
|
debug("Mapped 2MB page at %p to %p", VirtualAddress, PhysicalAddress);
|
|
return;
|
|
}
|
|
|
|
PageTableEntryPtr *PTEPtr = nullptr;
|
|
if (!PDE->Present)
|
|
{
|
|
PTEPtr = (PageTableEntryPtr *)KernelAllocator.RequestPages(TO_PAGES(sizeof(PageTableEntryPtr) + 1));
|
|
memset(PTEPtr, 0, sizeof(PageTableEntryPtr));
|
|
PDE->Present = true;
|
|
PDE->SetAddress((uintptr_t)PTEPtr >> 12);
|
|
}
|
|
else
|
|
PTEPtr = (PageTableEntryPtr *)(PDE->GetAddress() << 12);
|
|
PDE->raw |= DirectoryFlags;
|
|
|
|
PageTableEntry *PTE = &PTEPtr->Entries[Index.PTEIndex];
|
|
PTE->Present = true;
|
|
PTE->raw |= Flags;
|
|
PTE->SetAddress((uintptr_t)PhysicalAddress >> 12);
|
|
CPU::x64::invlpg(VirtualAddress);
|
|
|
|
#ifdef DEBUG
|
|
/* https://stackoverflow.com/a/3208376/9352057 */
|
|
#define BYTE_TO_BINARY_PATTERN "%c%c%c%c%c%c%c%c"
|
|
#define BYTE_TO_BINARY(byte) \
|
|
(byte & 0x80 ? '1' : '0'), \
|
|
(byte & 0x40 ? '1' : '0'), \
|
|
(byte & 0x20 ? '1' : '0'), \
|
|
(byte & 0x10 ? '1' : '0'), \
|
|
(byte & 0x08 ? '1' : '0'), \
|
|
(byte & 0x04 ? '1' : '0'), \
|
|
(byte & 0x02 ? '1' : '0'), \
|
|
(byte & 0x01 ? '1' : '0')
|
|
|
|
if (!this->Check(VirtualAddress, (PTFlag)Flags)) // quick workaround just to see where it fails
|
|
warn("Failed to map v:%#lx p:%#lx with flags: " BYTE_TO_BINARY_PATTERN, VirtualAddress, PhysicalAddress, BYTE_TO_BINARY(Flags));
|
|
#endif
|
|
}
|
|
|
|
void Virtual::Unmap(void *VirtualAddress, MapType Type)
|
|
{
|
|
SmartLock(this->MemoryLock);
|
|
if (!this->pTable)
|
|
{
|
|
error("No page table");
|
|
return;
|
|
}
|
|
|
|
PageMapIndexer Index = PageMapIndexer((uintptr_t)VirtualAddress);
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
if (!PML4->Present)
|
|
{
|
|
warn("Page %#lx not present", PML4->GetAddress());
|
|
return;
|
|
}
|
|
|
|
PageDirectoryPointerTableEntryPtr *PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)((uintptr_t)PML4->Address << 12);
|
|
PageDirectoryPointerTableEntry *PDPTE = &PDPTEPtr->Entries[Index.PDPTEIndex];
|
|
if (!PDPTE->Present)
|
|
{
|
|
warn("Page %#lx not present", PDPTE->GetAddress());
|
|
return;
|
|
}
|
|
|
|
if (Type == MapType::OneGiB && PDPTE->PageSize)
|
|
{
|
|
PDPTE->Present = false;
|
|
return;
|
|
}
|
|
|
|
PageDirectoryEntryPtr *PDEPtr = (PageDirectoryEntryPtr *)((uintptr_t)PDPTE->Address << 12);
|
|
PageDirectoryEntry *PDE = &PDEPtr->Entries[Index.PDEIndex];
|
|
if (!PDE->Present)
|
|
{
|
|
warn("Page %#lx not present", PDE->GetAddress());
|
|
return;
|
|
}
|
|
|
|
if (Type == MapType::TwoMiB && PDE->PageSize)
|
|
{
|
|
PDE->Present = false;
|
|
return;
|
|
}
|
|
|
|
PageTableEntryPtr *PTEPtr = (PageTableEntryPtr *)((uintptr_t)PDE->Address << 12);
|
|
PageTableEntry PTE = PTEPtr->Entries[Index.PTEIndex];
|
|
if (!PTE.Present)
|
|
{
|
|
warn("Page %#lx not present", PTE.GetAddress());
|
|
return;
|
|
}
|
|
|
|
PTE.Present = false;
|
|
PTEPtr->Entries[Index.PTEIndex] = PTE;
|
|
CPU::x64::invlpg(VirtualAddress);
|
|
}
|
|
|
|
void Virtual::Remap(void *VirtualAddress, void *PhysicalAddress, uint64_t Flags, MapType Type)
|
|
{
|
|
SmartLock(this->MemoryLock);
|
|
if (unlikely(!this->pTable))
|
|
{
|
|
error("No page table");
|
|
return;
|
|
}
|
|
|
|
Flags |= PTFlag::P;
|
|
|
|
PageMapIndexer Index = PageMapIndexer((uintptr_t)VirtualAddress);
|
|
// Clear any flags that are not 1 << 0 (Present) - 1 << 5 (Accessed) because rest are for page table entries only
|
|
uint64_t DirectoryFlags = Flags & 0x3F;
|
|
|
|
PageMapLevel4 *PML4 = &this->pTable->Entries[Index.PMLIndex];
|
|
PageDirectoryPointerTableEntryPtr *PDPTEPtr = nullptr;
|
|
if (!PML4->Present)
|
|
{
|
|
PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)KernelAllocator.RequestPages(TO_PAGES(sizeof(PageDirectoryPointerTableEntryPtr) + 1));
|
|
memset(PDPTEPtr, 0, sizeof(PageDirectoryPointerTableEntryPtr));
|
|
PML4->Present = true;
|
|
PML4->SetAddress((uintptr_t)PDPTEPtr >> 12);
|
|
}
|
|
else
|
|
PDPTEPtr = (PageDirectoryPointerTableEntryPtr *)(PML4->GetAddress() << 12);
|
|
PML4->raw |= DirectoryFlags;
|
|
|
|
PageDirectoryPointerTableEntry *PDPTE = &PDPTEPtr->Entries[Index.PDPTEIndex];
|
|
if (Type == MapType::OneGiB)
|
|
{
|
|
PDPTE->raw &= 0xFFF;
|
|
PDPTE->raw |= Flags;
|
|
PDPTE->PageSize = true;
|
|
PDPTE->SetAddress((uintptr_t)PhysicalAddress >> 12);
|
|
debug("Mapped 1GB page at %p to %p", VirtualAddress, PhysicalAddress);
|
|
return;
|
|
}
|
|
|
|
PageDirectoryEntryPtr *PDEPtr = nullptr;
|
|
if (!PDPTE->Present)
|
|
{
|
|
PDEPtr = (PageDirectoryEntryPtr *)KernelAllocator.RequestPages(TO_PAGES(sizeof(PageDirectoryEntryPtr) + 1));
|
|
memset(PDEPtr, 0, sizeof(PageDirectoryEntryPtr));
|
|
PDPTE->Present = true;
|
|
PDPTE->SetAddress((uintptr_t)PDEPtr >> 12);
|
|
}
|
|
else
|
|
PDEPtr = (PageDirectoryEntryPtr *)(PDPTE->GetAddress() << 12);
|
|
PDPTE->raw |= DirectoryFlags;
|
|
|
|
PageDirectoryEntry *PDE = &PDEPtr->Entries[Index.PDEIndex];
|
|
if (Type == MapType::TwoMiB)
|
|
{
|
|
PDE->raw &= 0xFFF;
|
|
PDE->raw |= Flags;
|
|
PDE->PageSize = true;
|
|
PDE->SetAddress((uintptr_t)PhysicalAddress >> 12);
|
|
debug("Mapped 2MB page at %p to %p", VirtualAddress, PhysicalAddress);
|
|
return;
|
|
}
|
|
|
|
PageTableEntryPtr *PTEPtr = nullptr;
|
|
if (!PDE->Present)
|
|
{
|
|
PTEPtr = (PageTableEntryPtr *)KernelAllocator.RequestPages(TO_PAGES(sizeof(PageTableEntryPtr) + 1));
|
|
memset(PTEPtr, 0, sizeof(PageTableEntryPtr));
|
|
PDE->Present = true;
|
|
PDE->SetAddress((uintptr_t)PTEPtr >> 12);
|
|
}
|
|
else
|
|
PTEPtr = (PageTableEntryPtr *)(PDE->GetAddress() << 12);
|
|
PDE->raw |= DirectoryFlags;
|
|
|
|
PageTableEntry *PTE = &PTEPtr->Entries[Index.PTEIndex];
|
|
PTE->raw &= 0xFFF;
|
|
PTE->raw |= Flags;
|
|
PTE->Present = true;
|
|
PTE->SetAddress((uintptr_t)PhysicalAddress >> 12);
|
|
CPU::x64::invlpg(VirtualAddress);
|
|
}
|
|
}
|