acadia/zion/object/address_space.cpp

71 lines
1.8 KiB
C++

#include "object/address_space.h"
#include "memory/kernel_stack_manager.h"
#include "memory/paging_util.h"
#include "memory/physical_memory.h"
extern KernelStackManager* gKernelStackManager;
AddressSpace AddressSpace::ForRoot() {
uint64_t cr3 = 0;
asm volatile("mov %%cr3, %0;" : "=r"(cr3));
return {cr3};
}
AddressSpace::AddressSpace() {
cr3_ = phys_mem::AllocatePage();
InitializePml4(cr3_);
}
uint64_t AddressSpace::AllocateUserStack() {
return user_stacks_.NewUserStack();
}
uint64_t AddressSpace::GetNextMemMapAddr(uint64_t size) {
uint64_t addr = next_memmap_addr_;
next_memmap_addr_ += size;
if (next_memmap_addr_ >= 0x30'00000000) {
panic("OOM: Memmap");
}
return addr;
}
void AddressSpace::MapInMemoryObject(uint64_t vaddr,
const RefPtr<MemoryObject>& mem_obj) {
memory_mappings_.PushBack({.vaddr = vaddr, .mem_obj = mem_obj});
}
uint64_t* AddressSpace::AllocateKernelStack() {
return gKernelStackManager->AllocateKernelStack();
}
bool AddressSpace::HandlePageFault(uint64_t vaddr) {
MemoryMapping* mapping = GetMemoryMappingForAddr(vaddr);
if (mapping == nullptr) {
return false;
}
uint64_t offset = vaddr - mapping->vaddr;
uint64_t physical_addr = mapping->mem_obj->PhysicalPageAtOffset(offset);
if (physical_addr == 0) {
dbgln("WARN: Memory object returned invalid physical addr.");
return false;
}
dbgln("Mapping P(%m) at V(%m)", physical_addr, vaddr);
MapPage(cr3_, vaddr, physical_addr);
return true;
}
AddressSpace::MemoryMapping* AddressSpace::GetMemoryMappingForAddr(
uint64_t vaddr) {
auto iter = memory_mappings_.begin();
while (iter != memory_mappings_.end()) {
if ((vaddr >= (*iter).vaddr) &&
(vaddr < ((*iter).vaddr + (*iter).mem_obj->size()))) {
return &(*iter);
}
++iter;
}
return 0;
}