From 55e176149220adb6da6a54de5868641f3328c81d Mon Sep 17 00:00:00 2001 From: Berkus Decker Date: Fri, 26 Feb 2021 00:21:52 +0200 Subject: [PATCH] [wip] directory levels traversal --- nucleus/src/arch/aarch64/memory/mmu2.rs | 87 ++++++++++++++++++++++--- 1 file changed, 79 insertions(+), 8 deletions(-) diff --git a/nucleus/src/arch/aarch64/memory/mmu2.rs b/nucleus/src/arch/aarch64/memory/mmu2.rs index c627171..fd7c150 100644 --- a/nucleus/src/arch/aarch64/memory/mmu2.rs +++ b/nucleus/src/arch/aarch64/memory/mmu2.rs @@ -46,6 +46,17 @@ use { * 48-bit virtual address space; different mappings in VBAR0 (EL0) and VBAR1 (EL1+). */ +register_bitfields! { + u64, + VA_INDEX [ + LEVEL0 OFFSET(39) NUMBITS(9) [], + LEVEL1 OFFSET(30) NUMBITS(9) [], + LEVEL2 OFFSET(21) NUMBITS(9) [], + LEVEL3 OFFSET(12) NUMBITS(9) [], + OFFSET OFFSET(0) NUMBITS(12) [], + ] +} + register_bitfields! { u64, // AArch64 Reference Manual page 2150, D5-2445 @@ -150,6 +161,7 @@ register_bitfields! { ] } +type VaIndex = register::FieldValue; type EntryFlags = register::FieldValue; // type EntryRegister = register::LocalRegisterCopy; @@ -218,17 +230,23 @@ type EntryFlags = register::FieldValue; // return enum PageTableEntry constructed from table bits in u64 enum L0Entries { - UpperDirectoryEntry, + UpperDirectoryEntry(VirtAddr), } enum L1Entries { - PageDirectoryEntry, - GiantPageFrame, + PageDirectoryEntry(VirtAddr), + GiantPageFrame(PhysFrame), } enum L2Entries { - PageTableEntry, - LargePageFrame, + PageTableEntry(VirtAddr), + LargePageFrame(PhysFrame), } enum L3Entries { + PageFrame(PhysFrame), +} + +enum Frames { + GiantPageFrame, + LargePageFrame, PageFrame, } @@ -326,6 +344,57 @@ struct Directory { level: PhantomData, } +impl Directory { + fn next(&self, address: VirtAddr) -> Option { + let va = VaIndex::new(address.to_u64()); + let index = va.read(VA_INDEX::LEVEL0); + match self.next_table_address(index.into()) { + Some(phys_addr) => { + Some(L0Entries::UpperDirectoryEntry(phys_addr.user_to_kernel())) + } + None => None, + } + } +} + +impl Directory { + fn next(&self, address: VirtAddr) -> Option { + let va = VaIndex::new(address.to_u64()); + let index = va.read(VA_INDEX::LEVEL1); + match self.next_table_address(index.into()) { + Some(phys_addr) => { + Some(L1Entries::PageDirectoryEntry(phys_addr.user_to_kernel())) + } + None => None, // @todo could be 1GiB frame + } + } +} + +impl Directory { + fn next(&self, address: VirtAddr) -> Option { + let va = VaIndex::new(address.to_u64()); + let index = va.read(VA_INDEX::LEVEL2); + match self.next_table_address(index.into()) { + Some(phys_addr) => { + Some(L2Entries::PageTableEntry(phys_addr.user_to_kernel())) + } + None => None, // @todo could be 2MiB frame + } + } +} + +impl Directory { + fn next(&self, address: VirtAddr) -> Option { + let va = VaIndex::new(address.to_u64()); + let index = va.read(VA_INDEX::LEVEL3); + match self.next_table_address(index.into()) { // @fixme wrong function + Some(phys_addr) => { + Some(L3Entries::PageFrame(phys_addr.user_to_kernel())) + } + None => None, // Nothing there + } +} + // Implementation code shared for all levels of page tables impl Directory where @@ -374,7 +443,7 @@ impl Directory where Level: HierarchicalLevel, { - fn next_table_address(&self, index: usize) -> Option { + fn next_table_address(&self, index: usize) -> Option { let entry_flags = EntryRegister::new(self[index]); // If table entry has 0b11 mask set, it is a valid table entry. // Address of the following table may be extracted from bits 47:12 @@ -387,13 +456,15 @@ where pub fn next_table(&self, index: usize) -> Option<&Table> { self.next_table_address(index) - .map(|address| unsafe { &*(address as *const _) }) + .map(|address| unsafe { &*(address.user_to_kernel() as *const _) }) } pub fn next_table_mut(&mut self, index: usize) -> Option<&mut Table> { self.next_table_address(index) - .map(|address| unsafe { &mut *(address as *mut _) }) + .map(|address| unsafe { &mut *(address.user_to_kernel() as *mut _) }) } + + pub fn translate_levels(&self, address: VirtAddr) -> Option {} } // ----