2022-11-13 13:29:15 +00:00
|
|
|
#include "arch/MMU.h"
|
2022-11-19 16:59:49 +00:00
|
|
|
#include "memory/MemoryManager.h"
|
2022-11-13 14:33:53 +00:00
|
|
|
#include <String.h>
|
2022-11-13 13:29:15 +00:00
|
|
|
|
|
|
|
#define PAGE_SIZE 4096
|
|
|
|
|
2022-11-16 19:37:32 +00:00
|
|
|
const usize ARCH_PAGE_SIZE = PAGE_SIZE;
|
|
|
|
|
2022-11-13 13:29:15 +00:00
|
|
|
const u64 rindex = 0776; // recursive index
|
|
|
|
const u64 sign = 0177777UL << 48; // sign extension
|
|
|
|
|
|
|
|
struct [[gnu::packed]] PageTableEntry
|
|
|
|
{
|
|
|
|
bool present : 1;
|
|
|
|
bool read_write : 1;
|
|
|
|
bool user : 1;
|
|
|
|
bool write_through : 1;
|
|
|
|
bool cache_disabled : 1;
|
|
|
|
bool accessed : 1;
|
|
|
|
bool ignore0 : 1;
|
|
|
|
bool larger_pages : 1;
|
|
|
|
bool ignore1 : 1;
|
|
|
|
u8 available : 3;
|
|
|
|
u64 address : 48;
|
|
|
|
u8 available2 : 3;
|
|
|
|
bool no_execute : 1;
|
|
|
|
|
2022-11-13 15:31:32 +00:00
|
|
|
void set_address(u64 addr);
|
|
|
|
u64 get_address();
|
2022-11-13 13:29:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#pragma GCC push_options
|
|
|
|
#pragma GCC diagnostic ignored "-Wconversion"
|
|
|
|
|
2022-11-13 15:31:32 +00:00
|
|
|
void PageTableEntry::set_address(u64 addr)
|
2022-11-13 13:29:15 +00:00
|
|
|
{
|
|
|
|
this->address = (addr >> 12);
|
|
|
|
}
|
|
|
|
|
2022-11-13 15:31:32 +00:00
|
|
|
u64 PageTableEntry::get_address()
|
2022-11-13 13:29:15 +00:00
|
|
|
{
|
2022-11-13 15:31:32 +00:00
|
|
|
return (u64)this->address << 12;
|
2022-11-13 13:29:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#pragma GCC pop_options
|
|
|
|
|
|
|
|
struct alignas(PAGE_SIZE) PageDirectory
|
|
|
|
{
|
|
|
|
PageTableEntry entries[512];
|
|
|
|
};
|
|
|
|
|
|
|
|
static_assert(sizeof(PageTableEntry) == 8UL);
|
|
|
|
static_assert(sizeof(PageDirectory) == PAGE_SIZE);
|
|
|
|
|
|
|
|
namespace MMU
|
|
|
|
{
|
|
|
|
|
|
|
|
PageDirectory* l4_table()
|
|
|
|
{
|
|
|
|
u64 l4 = sign | (rindex << 39) | (rindex << 30) | (rindex << 21) | (rindex << 12);
|
|
|
|
return (PageDirectory*)l4;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 l4_index(u64 addr)
|
|
|
|
{
|
|
|
|
return (addr >> 39) & 0777;
|
|
|
|
}
|
|
|
|
|
|
|
|
PageTableEntry& l4_entry(u64 addr)
|
|
|
|
{
|
|
|
|
return l4_table()->entries[l4_index(addr)];
|
|
|
|
}
|
|
|
|
|
|
|
|
PageDirectory* l3_table(u64 addr)
|
|
|
|
{
|
|
|
|
u64 l4 = l4_index(addr);
|
|
|
|
u64 l3 = sign | (rindex << 39) | (rindex << 30) | (rindex << 21) | (l4 << 12);
|
|
|
|
return (PageDirectory*)l3;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 l3_index(u64 addr)
|
|
|
|
{
|
|
|
|
return (addr >> 30) & 0777;
|
|
|
|
}
|
|
|
|
|
|
|
|
PageTableEntry& l3_entry(u64 addr)
|
|
|
|
{
|
|
|
|
return l3_table(addr)->entries[l3_index(addr)];
|
|
|
|
}
|
|
|
|
|
|
|
|
PageDirectory* l2_table(u64 addr)
|
|
|
|
{
|
|
|
|
u64 l4 = l4_index(addr);
|
|
|
|
u64 l3 = l3_index(addr);
|
|
|
|
u64 l2 = sign | (rindex << 39) | (rindex << 30) | (l4 << 21) | (l3 << 12);
|
|
|
|
return (PageDirectory*)l2;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 l2_index(u64 addr)
|
|
|
|
{
|
|
|
|
return (addr >> 21) & 0777;
|
|
|
|
}
|
|
|
|
|
|
|
|
PageTableEntry& l2_entry(u64 addr)
|
|
|
|
{
|
|
|
|
return l2_table(addr)->entries[l2_index(addr)];
|
|
|
|
}
|
|
|
|
|
|
|
|
PageDirectory* l1_table(u64 addr)
|
|
|
|
{
|
|
|
|
u64 l4 = l4_index(addr);
|
|
|
|
u64 l3 = l3_index(addr);
|
|
|
|
u64 l2 = l2_index(addr);
|
|
|
|
u64 l1 = sign | (rindex << 39) | (l4 << 30) | (l3 << 21) | (l2 << 12);
|
|
|
|
return (PageDirectory*)l1;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 l1_index(u64 addr)
|
|
|
|
{
|
|
|
|
return (addr >> 12) & 0777;
|
|
|
|
}
|
|
|
|
|
|
|
|
PageTableEntry& l1_entry(u64 addr)
|
|
|
|
{
|
|
|
|
return l1_table(addr)->entries[l1_index(addr)];
|
|
|
|
}
|
|
|
|
|
|
|
|
void switch_page_directory(PageDirectory* dir)
|
|
|
|
{
|
|
|
|
asm volatile("mov %0, %%cr3" : : "r"(dir));
|
|
|
|
}
|
|
|
|
|
|
|
|
PageDirectory* get_page_directory()
|
|
|
|
{
|
|
|
|
PageDirectory* value;
|
|
|
|
asm volatile("mov %%cr3, %0" : "=r"(value));
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void flush_all()
|
|
|
|
{
|
|
|
|
switch_page_directory(get_page_directory());
|
|
|
|
}
|
|
|
|
|
|
|
|
void flush_page(u64 page)
|
|
|
|
{
|
|
|
|
asm volatile("invlpg (%0)" : : "r"(page) : "memory");
|
|
|
|
}
|
|
|
|
|
2022-11-13 14:33:53 +00:00
|
|
|
int arch_flags_to_mmu(PageTableEntry& entry)
|
2022-11-13 13:29:15 +00:00
|
|
|
{
|
|
|
|
int result = Flags::None;
|
|
|
|
if (entry.read_write) result |= Flags::ReadWrite;
|
|
|
|
if (entry.user) result |= Flags::User;
|
|
|
|
if (entry.no_execute) result |= Flags::NoExecute;
|
|
|
|
if (entry.write_through) result |= Flags::WriteThrough;
|
|
|
|
if (entry.cache_disabled) result |= Flags::CacheDisable;
|
2022-11-13 14:33:53 +00:00
|
|
|
return result;
|
2022-11-13 13:29:15 +00:00
|
|
|
}
|
|
|
|
|
2022-11-13 15:51:21 +00:00
|
|
|
Result<PageTableEntry*> find_entry(u64 virt)
|
|
|
|
{
|
|
|
|
auto& l4 = l4_entry(virt);
|
|
|
|
if (!l4.present) return err;
|
|
|
|
auto& l3 = l3_entry(virt);
|
|
|
|
if (!l3.present) return err;
|
|
|
|
if (l3.larger_pages) return &l3;
|
|
|
|
auto& l2 = l2_entry(virt);
|
|
|
|
if (!l2.present) return err;
|
|
|
|
if (l2.larger_pages) return &l2;
|
|
|
|
return &l1_entry(virt);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<PageTableEntry*> apply_cascading_flags(u64 virt, int flags)
|
|
|
|
{
|
|
|
|
auto& l4 = l4_entry(virt);
|
|
|
|
if (!l4.present) return err;
|
|
|
|
if (flags & Flags::ReadWrite) l4.read_write = true;
|
|
|
|
if (flags & Flags::User) l4.user = true;
|
|
|
|
auto& l3 = l3_entry(virt);
|
|
|
|
if (!l3.present) return err;
|
|
|
|
if (l3.larger_pages) return &l3;
|
|
|
|
if (flags & Flags::ReadWrite) l3.read_write = true;
|
|
|
|
if (flags & Flags::User) l3.user = true;
|
|
|
|
auto& l2 = l2_entry(virt);
|
|
|
|
if (!l2.present) return err;
|
|
|
|
if (l2.larger_pages) return &l2;
|
|
|
|
if (flags & Flags::ReadWrite) l2.read_write = true;
|
|
|
|
if (flags & Flags::User) l2.user = true;
|
|
|
|
auto& l1 = l1_entry(virt);
|
|
|
|
return &l1;
|
|
|
|
}
|
|
|
|
|
2022-11-13 14:33:53 +00:00
|
|
|
Result<void> map(u64 virt, u64 phys, int flags)
|
2022-11-13 13:29:15 +00:00
|
|
|
{
|
|
|
|
auto& l4 = l4_entry(virt);
|
|
|
|
if (!l4.present)
|
|
|
|
{
|
2022-11-19 21:27:08 +00:00
|
|
|
u64 addr = TRY(MemoryManager::alloc_frame());
|
2022-11-13 13:29:15 +00:00
|
|
|
l4.present = true;
|
2022-11-16 19:37:41 +00:00
|
|
|
l4.set_address(addr);
|
2022-11-13 14:33:53 +00:00
|
|
|
memset(l3_table(virt), 0, PAGE_SIZE);
|
2022-11-13 15:31:32 +00:00
|
|
|
l4.ignore0 = l4.ignore1 = 0;
|
2022-11-13 13:29:15 +00:00
|
|
|
}
|
2022-11-13 14:33:53 +00:00
|
|
|
if (flags & Flags::ReadWrite) l4.read_write = true;
|
|
|
|
if (flags & Flags::User) l4.user = true;
|
|
|
|
|
2022-11-13 13:29:15 +00:00
|
|
|
auto& l3 = l3_entry(virt);
|
|
|
|
if (!l3.present)
|
|
|
|
{
|
2022-11-19 21:27:08 +00:00
|
|
|
u64 addr = TRY(MemoryManager::alloc_frame());
|
2022-11-13 13:29:15 +00:00
|
|
|
l3.present = true;
|
2022-11-16 19:37:41 +00:00
|
|
|
l3.set_address(addr);
|
2022-11-13 14:33:53 +00:00
|
|
|
memset(l2_table(virt), 0, PAGE_SIZE);
|
2022-11-13 15:31:32 +00:00
|
|
|
l3.ignore0 = l3.ignore1 = 0;
|
2022-11-13 13:29:15 +00:00
|
|
|
}
|
2022-11-13 14:33:53 +00:00
|
|
|
if (flags & Flags::ReadWrite) l3.read_write = true;
|
|
|
|
if (flags & Flags::User) l3.user = true;
|
|
|
|
|
2022-11-13 13:29:15 +00:00
|
|
|
if (l3.larger_pages) return err; // FIXME: Replacing larger pages is not supported ATM
|
2022-11-13 14:33:53 +00:00
|
|
|
|
2022-11-13 13:29:15 +00:00
|
|
|
auto& l2 = l2_entry(virt);
|
|
|
|
if (!l2.present)
|
|
|
|
{
|
2022-11-19 21:27:08 +00:00
|
|
|
u64 addr = TRY(MemoryManager::alloc_frame());
|
2022-11-13 13:29:15 +00:00
|
|
|
l2.present = true;
|
2022-11-16 19:37:41 +00:00
|
|
|
l2.set_address(addr);
|
2022-11-13 14:33:53 +00:00
|
|
|
memset(l1_table(virt), 0, PAGE_SIZE);
|
2022-11-13 15:31:32 +00:00
|
|
|
l2.ignore0 = l2.ignore1 = 0;
|
2022-11-13 13:29:15 +00:00
|
|
|
}
|
2022-11-13 14:33:53 +00:00
|
|
|
if (flags & Flags::ReadWrite) l2.read_write = true;
|
|
|
|
if (flags & Flags::User) l2.user = true;
|
|
|
|
|
2022-11-13 13:29:15 +00:00
|
|
|
if (l2.larger_pages) return err; // FIXME: Replacing larger pages is not supported ATM
|
2022-11-13 14:33:53 +00:00
|
|
|
|
2022-11-13 13:29:15 +00:00
|
|
|
auto& l1 = l1_entry(virt);
|
2022-11-13 15:31:32 +00:00
|
|
|
if (l1.present) return err; // Please explicitly unmap the page before mapping it again.
|
|
|
|
l1.ignore0 = l1.ignore1 = false;
|
2022-11-13 14:33:53 +00:00
|
|
|
l1.present = true;
|
2022-11-15 18:10:19 +00:00
|
|
|
l1.read_write = (flags & Flags::ReadWrite);
|
|
|
|
l1.user = (flags & Flags::User);
|
|
|
|
l1.write_through = (flags & Flags::WriteThrough);
|
|
|
|
l1.cache_disabled = (flags & Flags::CacheDisable);
|
|
|
|
l1.no_execute = (flags & Flags::NoExecute);
|
2022-11-13 13:29:15 +00:00
|
|
|
l1.set_address(phys);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-11-13 15:51:21 +00:00
|
|
|
Result<void> remap(u64 virt, int flags)
|
|
|
|
{
|
2022-11-16 19:37:41 +00:00
|
|
|
auto& l1 = *TRY(apply_cascading_flags(virt, flags));
|
2022-11-13 15:51:21 +00:00
|
|
|
if (!l1.present) return err;
|
2022-11-15 18:10:19 +00:00
|
|
|
l1.read_write = (flags & Flags::ReadWrite);
|
|
|
|
l1.user = (flags & Flags::User);
|
|
|
|
l1.write_through = (flags & Flags::WriteThrough);
|
|
|
|
l1.cache_disabled = (flags & Flags::CacheDisable);
|
|
|
|
l1.no_execute = (flags & Flags::NoExecute);
|
2022-11-13 15:51:21 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<u64> unmap(u64 virt)
|
|
|
|
{
|
2022-11-16 19:37:41 +00:00
|
|
|
auto& l1 = *TRY(find_entry(virt));
|
2022-11-13 15:51:21 +00:00
|
|
|
if (!l1.present) return err;
|
|
|
|
u64 address = l1.get_address();
|
|
|
|
memset(&l1, 0, sizeof(l1));
|
|
|
|
flush_page(virt);
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
|
2022-11-13 13:29:15 +00:00
|
|
|
Result<u64> get_physical(u64 virt)
|
|
|
|
{
|
2022-11-16 19:37:41 +00:00
|
|
|
auto& l1 = *TRY(find_entry(virt));
|
2022-11-13 13:29:15 +00:00
|
|
|
if (!l1.present) return err;
|
|
|
|
return l1.get_address();
|
|
|
|
}
|
|
|
|
|
2022-11-13 14:33:53 +00:00
|
|
|
Result<int> get_flags(u64 virt)
|
2022-11-13 13:29:15 +00:00
|
|
|
{
|
2022-11-16 19:37:41 +00:00
|
|
|
auto& l1 = *TRY(find_entry(virt));
|
2022-11-13 13:29:15 +00:00
|
|
|
if (!l1.present) return err;
|
|
|
|
return arch_flags_to_mmu(l1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setup_initial_page_directory()
|
|
|
|
{
|
|
|
|
PageDirectory* dir = get_page_directory();
|
|
|
|
u64 paddr = (u64)dir;
|
|
|
|
PageTableEntry& recursive_entry = dir->entries[rindex];
|
|
|
|
recursive_entry.read_write = true;
|
|
|
|
recursive_entry.present = true;
|
|
|
|
recursive_entry.set_address(paddr);
|
|
|
|
flush_all();
|
|
|
|
}
|
|
|
|
}
|