#include "memory/physical_memory.h" #include "boot/boot_info.h" #include "debug/debug.h" #define K_PHYS_DEBUG 0 namespace phys_mem { namespace { uint64_t* PageAlign(uint64_t* addr) { return reinterpret_cast(reinterpret_cast(addr) & ~0xFFF); } void ZeroOutPage(uint64_t* ptr) { ptr = PageAlign(ptr); for (uint64_t i = 0; i < 512; i++) { ptr[i] = 0; } } struct BootstrapMemory { uint64_t init_page = 0; uint64_t next_page = 0; uint64_t max_page = 0; }; static BootstrapMemory gBootstrap; static bool gBootstrapEnabled = false; class PhysicalMemoryManager { public: // Reads the memory map and takes // control of the available regions. PhysicalMemoryManager() { const limine_memmap_response& memmap = boot::GetMemoryMap(); for (uint64_t i = 0; i < memmap.entry_count; i++) { const limine_memmap_entry& entry = *memmap.entries[i]; if (entry.type == 0) { uint64_t base = entry.base; uint64_t size = entry.length; if (base == gBootstrap.init_page) { base = gBootstrap.next_page; uint64_t bootstrap_used = gBootstrap.next_page - gBootstrap.init_page; #if K_PHYS_DEBUG dbgln("[PMM] Taking over from bootstrap, used: {x}", bootstrap_used); #endif size -= bootstrap_used; } AddMemoryRegion(base, size); } } } uint64_t AllocatePage() { if (front_ == nullptr) { panic("No available memory regions."); } if (front_->num_pages == 0) { panic("Bad state, empty memory block."); } uint64_t page = front_->base; front_->base += 0x1000; front_->num_pages--; if (front_->num_pages == 0) { MemBlock* temp = front_; front_ = front_->next; delete temp; } #if K_PHYS_DEBUG dbgln("Single {x}", page); #endif return page; } uint64_t AllocateContinuous(uint64_t num_pages) { if (front_ == nullptr) { panic("No available memory regions."); } if (front_->num_pages == 0) { panic("Bad state, empty memory block."); } MemBlock* block = front_; while (block != nullptr && block->num_pages < num_pages) { dbgln("Skipping block of size {} seeking {}", block->num_pages, num_pages); block = block->next; } if (block == nullptr) { panic("No memory regions to allocate"); } uint64_t page = front_->base; front_->base += num_pages * 0x1000; front_->num_pages -= num_pages; if (front_->num_pages == 0) { MemBlock* temp = front_; front_ = front_->next; delete temp; } #if K_PHYS_DEBUG dbgln("Continuous {x}:{}", page, num_pages); #endif return page; } void FreePage(uint64_t page) { AddMemoryRegion(page, 0x1000); } private: void AddMemoryRegion(uint64_t base, uint64_t size) { MemBlock* block = new MemBlock{ .next = front_, .base = base, .num_pages = size >> 12, }; front_ = block; } struct MemBlock { MemBlock* next = nullptr; uint64_t base = 0; uint64_t num_pages = 0; }; MemBlock* front_ = nullptr; }; static PhysicalMemoryManager* gPmm = nullptr; }; // namespace void InitBootstrapPageAllocation() { const limine_memmap_response& memmap = boot::GetMemoryMap(); for (uint64_t i = 0; i < memmap.entry_count; i++) { const limine_memmap_entry& entry = *memmap.entries[i]; // We may want to chose a high address space to not limit // the number of buffers we can allocate later but // if we limit the number of pages this should be fine. // Currently set to the minimum of 3 for one kernel heap allocation: // PageDirectory + PageTable + Page if (entry.type == 0 && entry.length >= 0x3000) { gBootstrap.init_page = entry.base; gBootstrap.next_page = entry.base; gBootstrap.max_page = entry.base + 0x3000; gBootstrapEnabled = true; return; } } } void InitPhysicalMemoryManager() { gPmm = new PhysicalMemoryManager(); } glcr::StringView MemoryRegionStr(uint64_t type) { switch (type) { case LIMINE_MEMMAP_USABLE: return "USABLE"; case LIMINE_MEMMAP_RESERVED: return "RESRVD"; case LIMINE_MEMMAP_ACPI_RECLAIMABLE: return "ACPIREC"; case LIMINE_MEMMAP_ACPI_NVS: return "ACPINVS"; case LIMINE_MEMMAP_BAD_MEMORY: return "BADMEM"; case LIMINE_MEMMAP_BOOTLOADER_RECLAIMABLE: return "LIMINE"; case LIMINE_MEMMAP_KERNEL_AND_MODULES: return "KERNEL"; case LIMINE_MEMMAP_FRAMEBUFFER: return "FRMBFR"; default: return "UNKNWN"; } } void DumpRegions() { const limine_memmap_response& memmap = boot::GetMemoryMap(); for (uint64_t i = 0; i < memmap.entry_count; i++) { const limine_memmap_entry& entry = *memmap.entries[i]; dbgln("Region({}) at {x}:{x} ({x})", MemoryRegionStr(entry.type), entry.base, entry.base + entry.length, entry.length); } } uint64_t AllocatePage() { if (gPmm != nullptr) { return gPmm->AllocatePage(); } if (!gBootstrapEnabled) { panic("No Bootstrap Memory Manager"); } #if K_PHYS_DEBUG early_dbgln("[PMM] Boostrap Alloc!"); #endif uint64_t page = gBootstrap.next_page; if (page == gBootstrap.max_page) { panic("Bootstrap Memory Manager OOM"); } gBootstrap.next_page += 0x1000; return page; } uint64_t AllocateAndZeroPage() { uint64_t paddr = AllocatePage(); ZeroOutPage( reinterpret_cast(boot::GetHigherHalfDirectMap() + paddr)); return paddr; } uint64_t AllocateContinuous(uint64_t num_continuous) { if (gPmm == nullptr) { panic("No physical memory manager!"); } return gPmm->AllocateContinuous(num_continuous); } } // namespace phys_mem