use crate::mem::paging::EntryFlags; use crate::mem::FrameAllocator; use super::tables::{Level1, Table}; use super::{ActivePageTable, VirtualAddress, Frame}; use super::Page; pub struct TemporaryPage { page: Page, allocator: TinyAllocator } struct TinyAllocator([Option; 3]); impl TemporaryPage { pub fn new(page: Page, allocator: &mut A) -> TemporaryPage where A: FrameAllocator { TemporaryPage { page: page, allocator: TinyAllocator::new(allocator), } } /// Maps the temporary page to the given frame in the active table. /// Returns the start address of the temporary page. pub fn map(&mut self, frame: Frame, active_table: &mut ActivePageTable) -> VirtualAddress { assert!(active_table.translate_page(self.page).is_none(), "temporary page is already mapped"); active_table.map_to(self.page, frame, EntryFlags::WRITABLE, &mut self.allocator); self.page.start_address() } /// Unmaps the temporary page in the active table. pub fn unmap(&mut self, active_table: &mut ActivePageTable) { active_table.unmap(self.page, &mut self.allocator) } /// Maps the temporary page to the given page table frame in the active /// table. Returns a reference to the now mapped table. pub fn map_table_frame(&mut self, frame: Frame, active_table: &mut ActivePageTable) -> &mut Table { unsafe { &mut *(self.map(frame, active_table) as *mut Table) } } } impl FrameAllocator for TinyAllocator { fn allocate_frame(&mut self) -> Option { for frame_option in &mut self.0 { if frame_option.is_some() { return frame_option.take(); } } None } fn deallocate_frame(&mut self, frame: Frame) { for frame_option in &mut self.0 { if frame_option.is_none() { *frame_option = Some(frame); return; } } panic!("Tiny allocator can hold only 3 frames."); } } impl TinyAllocator { fn new(allocator: &mut A) -> TinyAllocator where A: FrameAllocator { let mut f = || allocator.allocate_frame(); let frames = [f(), f(), f()]; TinyAllocator(frames) } }