mirror of
https://github.com/phil-opp/blog_os.git
synced 2025-12-16 22:37:49 +00:00
45 lines
1.8 KiB
Rust
45 lines
1.8 KiB
Rust
use super::{VirtualAddress, PhysicalAddress, Page, PAGE_SIZE, ENTRY_COUNT};
|
|
use super::table::P4;
|
|
use super::entry::{PRESENT, HUGE_PAGE};
|
|
use memory::Frame;
|
|
|
|
pub fn translate(virtual_address: VirtualAddress) -> Option<PhysicalAddress> {
|
|
let offset = virtual_address % PAGE_SIZE;
|
|
translate_page(Page::containing_address(virtual_address))
|
|
.map(|frame| frame.number * PAGE_SIZE + offset)
|
|
}
|
|
|
|
fn translate_page(page: Page) -> Option<Frame> {
|
|
let p4 = unsafe { &*P4 };
|
|
|
|
let huge_page = || {
|
|
p4.next_table(page.p4_index())
|
|
.and_then(|p3| {
|
|
// 1GiB page?
|
|
if p3[page.p3_index()].flags().contains(HUGE_PAGE | PRESENT) {
|
|
let start_frame_number = p3[page.p3_index()].pointed_frame().number;
|
|
// address must be 1GiB aligned
|
|
assert!(start_frame_number % (ENTRY_COUNT * ENTRY_COUNT) == 0);
|
|
return Some(start_frame_number + page.p2_index() * ENTRY_COUNT + page.p1_index());
|
|
}
|
|
if let Some(p2) = p3.next_table(page.p3_index()) {
|
|
// 2MiB page?
|
|
if p2[page.p2_index()].flags().contains(HUGE_PAGE | PRESENT) {
|
|
let start_frame_number = p2[page.p2_index()].pointed_frame().number;
|
|
// address must be 2MiB aligned
|
|
assert!(start_frame_number % ENTRY_COUNT == 0);
|
|
return Some(start_frame_number + page.p1_index());
|
|
}
|
|
}
|
|
None
|
|
})
|
|
.map(|start_frame_number| Frame { number: start_frame_number })
|
|
};
|
|
|
|
p4.next_table(page.p4_index())
|
|
.and_then(|p3| p3.next_table(page.p3_index()))
|
|
.and_then(|p2| p2.next_table(page.p2_index()))
|
|
.map(|p1| p1[page.p1_index()].pointed_frame())
|
|
.or_else(huge_page)
|
|
}
|