use core::alloc::AllocError;
use hermit_sync::InterruptTicketMutex;
use crate::arch::aarch64::mm::paging::{BasePageSize, PageSize};
use crate::arch::aarch64::mm::VirtAddr;
use crate::mm;
use crate::mm::freelist::{FreeList, FreeListEntry};
static KERNEL_FREE_LIST: InterruptTicketMutex<FreeList> =
InterruptTicketMutex::new(FreeList::new());
const KERNEL_VIRTUAL_MEMORY_END: VirtAddr = VirtAddr(0x1_0000_0000);
pub fn init() {
let entry = FreeListEntry {
start: mm::kernel_end_address().as_usize(),
end: KERNEL_VIRTUAL_MEMORY_END.as_usize(),
};
KERNEL_FREE_LIST.lock().list.push_back(entry);
}
pub fn allocate(size: usize) -> Result<VirtAddr, AllocError> {
assert!(size > 0);
assert_eq!(
size % BasePageSize::SIZE as usize,
0,
"Size {:#X} is not a multiple of {:#X}",
size,
BasePageSize::SIZE
);
Ok(VirtAddr(
KERNEL_FREE_LIST
.lock()
.allocate(size, None)?
.try_into()
.unwrap(),
))
}
pub fn allocate_aligned(size: usize, alignment: usize) -> Result<VirtAddr, AllocError> {
assert!(size > 0);
assert!(alignment > 0);
assert_eq!(
size % alignment,
0,
"Size {size:#X} is not a multiple of the given alignment {alignment:#X}"
);
assert_eq!(
alignment % BasePageSize::SIZE as usize,
0,
"Alignment {:#X} is not a multiple of {:#X}",
alignment,
BasePageSize::SIZE
);
Ok(VirtAddr(
KERNEL_FREE_LIST
.lock()
.allocate(size, Some(alignment))?
.try_into()
.unwrap(),
))
}
pub fn deallocate(virtual_address: VirtAddr, size: usize) {
assert!(
virtual_address >= mm::kernel_end_address() || virtual_address < mm::kernel_start_address(),
"Virtual address {virtual_address:p} belongs to the kernel"
);
assert!(
virtual_address < KERNEL_VIRTUAL_MEMORY_END,
"Virtual address {virtual_address:p} is not < KERNEL_VIRTUAL_MEMORY_END"
);
assert_eq!(
virtual_address % BasePageSize::SIZE,
0,
"Virtual address {:p} is not a multiple of {:#X}",
virtual_address,
BasePageSize::SIZE
);
assert!(size > 0);
assert_eq!(
size % BasePageSize::SIZE as usize,
0,
"Size {:#X} is not a multiple of {:#X}",
size,
BasePageSize::SIZE
);
KERNEL_FREE_LIST
.lock()
.deallocate(virtual_address.as_usize(), size);
}
pub fn print_information() {
KERNEL_FREE_LIST
.lock()
.print_information(" KERNEL VIRTUAL MEMORY FREE LIST ");
}