Skip to content

Commit

Permalink
copy more PML4 entries
Browse files Browse the repository at this point in the history
If there is more than 512 GiB of memory, we need to copy more than the
first PML4 entry to access it. Similarly if the frame buffer is not in
the address range covered by the first PML4 entry, we need to copy more
entries in order to access it.
  • Loading branch information
Freax13 committed Oct 26, 2024
1 parent ea3f61a commit 7dcc963
Showing 1 changed file with 19 additions and 4 deletions.
23 changes: 19 additions & 4 deletions uefi/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -144,7 +144,8 @@ fn main_inner(image: Handle, mut st: SystemTable<Boot>) -> Status {
let mut frame_allocator =
LegacyFrameAllocator::new(memory_map.entries().copied().map(UefiMemoryDescriptor));

let page_tables = create_page_tables(&mut frame_allocator);
let max_phys_addr = frame_allocator.max_phys_addr();
let page_tables = create_page_tables(&mut frame_allocator, max_phys_addr, framebuffer.as_ref());
let mut ramdisk_len = 0u64;
let ramdisk_addr = if let Some(rd) = ramdisk {
ramdisk_len = rd.len() as u64;
Expand Down Expand Up @@ -385,6 +386,8 @@ fn load_file_from_tftp_boot_server(
/// Creates page table abstraction types for both the bootloader and kernel page tables.
fn create_page_tables(
frame_allocator: &mut impl FrameAllocator<Size4KiB>,
max_phys_addr: PhysAddr,
frame_buffer: Option<&RawFrameBufferInfo>,
) -> bootloader_x86_64_common::PageTables {
// UEFI identity-maps all memory, so the offset between physical and virtual addresses is 0
let phys_offset = VirtAddr::new(0);
Expand All @@ -410,9 +413,21 @@ fn create_page_tables(
}
};

// copy the first entry (we don't need to access more than 512 GiB; also, some UEFI
// implementations seem to create an level 4 table entry 0 in all slots)
new_table[0] = old_table[0].clone();
// copy the pml4 entries for all identity mapped memory.
let end_addr = VirtAddr::new(max_phys_addr.as_u64() - 1);
for p4 in 0..=usize::from(end_addr.p4_index()) {
new_table[p4] = old_table[p4].clone();
}

// copy the pml4 entry for the frame buffer (the frame buffer is not
// necessarily part of the identity mapping).
if let Some(frame_buffer) = frame_buffer {
let start_addr = VirtAddr::new(frame_buffer.addr.as_u64());
let end_addr = start_addr + frame_buffer.info.byte_len;
for p4 in usize::from(start_addr.p4_index())..=usize::from(end_addr.p4_index()) {
new_table[p4] = old_table[p4].clone();
}
}

// the first level 4 table entry is now identical, so we can just load the new one
unsafe {
Expand Down

0 comments on commit 7dcc963

Please sign in to comment.