use alloc::alloc::{GlobalAlloc, Layout}; use core::ptr::null_mut; use fixed_size_block::FixedSizeBlockAllocator; use x86_64::{ structures::paging::{ mapper::MapToError, FrameAllocator, Mapper, Page, PageTableFlags, Size4KiB, }, VirtAddr, }; pub mod bump; pub mod fixed_size_block; pub mod linked_list; pub const HEAP_START: usize = 0x_4444_4444_0000; pub const HEAP_SIZE: usize = 4 * 1024 * 1024; // 4 MiB #[global_allocator] static ALLOCATOR: Locked = Locked::new(FixedSizeBlockAllocator::new()); pub fn init_heap( mapper: &mut impl Mapper, frame_allocator: &mut impl FrameAllocator, ) -> Result<(), MapToError> { let page_range = { let heap_start = VirtAddr::new(HEAP_START as u64); let heap_end = heap_start + HEAP_SIZE - 1u64; let heap_start_page = Page::containing_address(heap_start); let heap_end_page = Page::containing_address(heap_end); Page::range_inclusive(heap_start_page, heap_end_page) }; for page in page_range { let frame = frame_allocator .allocate_frame() .ok_or(MapToError::FrameAllocationFailed)?; let flags = PageTableFlags::PRESENT | PageTableFlags::WRITABLE; mapper.map_to(page, frame, flags, frame_allocator)?.flush(); } unsafe { ALLOCATOR.lock().init(HEAP_START, HEAP_SIZE); } Ok(()) } pub struct Dummy; unsafe impl GlobalAlloc for Dummy { unsafe fn alloc(&self, _layout: Layout) -> *mut u8 { null_mut() } unsafe fn dealloc(&self, _ptr: *mut u8, _layout: Layout) { panic!("dealloc should be never called") } } /// A wrapper around spin::Mutex to permit trait implementations. pub struct Locked { inner: spin::Mutex, } impl Locked { pub const fn new(inner: A) -> Self { Locked { inner: spin::Mutex::new(inner), } } pub fn lock(&self) -> spin::MutexGuard { self.inner.lock() } } /// Align the given address `addr` upwards to alignment `align`. /// /// Requires that `align` is a power of two. fn align_up(addr: usize, align: usize) -> usize { (addr + align - 1) & !(align - 1) }