Browse Source

Add WIP support for multiboot2.

Caduser2020 4 years ago
parent
commit
da825f4e0e
3 changed files with 246 additions and 3 deletions
  1. 19 2
      src/lib.rs
  2. 195 1
      src/memory_map.rs
  3. 32 0
      src/rsdp.rs

+ 19 - 2
src/lib.rs

@@ -30,9 +30,9 @@ pub use elf_sections::{
 };
 pub use framebuffer::{FramebufferColor, FramebufferField, FramebufferTag, FramebufferType};
 use header::{Tag, TagIter};
-pub use memory_map::{MemoryArea, MemoryAreaIter, MemoryAreaType, MemoryMapTag};
+pub use memory_map::{EFIMemoryAreaType, EFIMemoryMapTag, EFIMemoryDesc, MemoryArea, MemoryAreaIter, MemoryAreaType, MemoryMapTag};
 pub use module::{ModuleIter, ModuleTag};
-pub use rsdp::{RsdpV1Tag, RsdpV2Tag};
+pub use rsdp::{EFISdt32, EFISdt64, RsdpV1Tag, RsdpV2Tag};
 pub use vbe_info::{
     VBECapabilities, VBEControlInfo, VBEDirectColorAttributes, VBEField, VBEInfoTag,
     VBEMemoryModel, VBEModeAttributes, VBEModeInfo, VBEWindowAttributes,
@@ -170,6 +170,12 @@ impl BootInformation {
         self.get_tag(8).map(|tag| framebuffer::framebuffer_tag(tag))
     }
 
+    /// Search for the EFI 32-bit SDT tag.
+    pub fn efi_sdt_32_tag<'a>(&self) -> Option<&'a EFISdt32> {
+        self.get_tag(12)
+            .map(|tag| unsafe { &*(tag as *const Tag as *const EFISdt32) })
+    }
+
     /// Search for the (ACPI 1.0) RSDP tag.
     pub fn rsdp_v1_tag<'a>(&self) -> Option<&'a RsdpV1Tag> {
         self.get_tag(14)
@@ -182,6 +188,17 @@ impl BootInformation {
             .map(|tag| unsafe { &*(tag as *const Tag as *const RsdpV2Tag) })
     }
 
+    /// Search for the EFI Memory map tag.
+    pub fn efi_memory_map_tag<'a>(&'a self) -> Option<&'a EFIMemoryMapTag> {
+        match self.get_tag(18) {
+            Some(_) => {
+                self.get_tag(17)
+                    .map(|tag| unsafe { &*(tag as *const Tag as *const EFIMemoryMapTag) })
+            }, 
+            None => None
+        }
+    }
+
     /// Search for the VBE information tag.
     pub fn vbe_info_tag(&self) -> Option<&'static VBEInfoTag> {
         self.get_tag(7)

+ 195 - 1
src/memory_map.rs

@@ -52,7 +52,7 @@ impl MemoryArea {
 
     /// The end address of the memory region.
     pub fn end_address(&self) -> u64 {
-        (self.base_addr + self.length)
+        self.base_addr + self.length
     }
 
     /// The size, in bytes, of the memory region.
@@ -114,3 +114,197 @@ impl<'a> Iterator for MemoryAreaIter<'a> {
         }
     }
 }
+
+/// EFI memory map as per EFI specification.
+#[derive(Debug)]
+#[repr(C)]
+pub struct EFIMemoryMapTag {
+    typ: u32,
+    size: u32,
+    desc_size: u32,
+    desc_version: u32,
+    first_desc: EFIMemoryDesc,
+}
+
+impl EFIMemoryMapTag {
+    /// Return an iterator over ALL marked memory areas.
+    ///
+    /// This differs from `MemoryMapTag` as for UEFI, the OS needs some non-
+    /// available memory areas for tables and such.
+    pub fn memory_areas(&self) -> EFIMemoryAreaIter {
+        let self_ptr = self as *const EFIMemoryMapTag;
+        let start_area = (&self.first_desc) as *const EFIMemoryDesc;
+        EFIMemoryAreaIter {
+            current_area: start_area as u64,
+            last_area: (self_ptr as u64 + self.size as u64),
+            entry_size: self.desc_size,
+            phantom: PhantomData,
+        }
+    }
+}
+
+/// EFI Boot Memory Map Descriptor
+#[derive(Debug)]
+#[repr(C)]
+pub struct EFIMemoryDesc {
+    typ: u32,
+    _padding: u32,
+    phys_addr: u64,
+    virt_addr: u64,
+    num_pages: u64,
+    attr: u64,
+}
+
+/// An enum of possible reported region types.
+#[derive(Debug, PartialEq, Eq)]
+pub enum EFIMemoryAreaType {
+    /// Unusable.
+    EfiReservedMemoryType,
+    /// Code area of a UEFI application.
+    EfiLoaderCode,
+    /// Data area of a UEFI application.
+    EfiLoaderData,
+    /// Code area of a UEFI Boot Service Driver.
+    EfiBootServicesCode,
+    /// Data area of a UEFI Boot Service Driver.
+    EfiBootServicesData,
+    /// Code area of a UEFI Runtime Driver.
+    ///
+    /// Must be preserved in working and ACPI S1-S3 states.
+    EfiRuntimeServicesCode,
+    /// Data area of a UEFI Runtime Driver.
+    ///
+    /// Must be preserved in working and ACPI S1-S3 states.
+    EfiRuntimeServicesData,
+    /// Available memory.
+    EfiConventionalMemory,
+    /// Memory with errors, treat as unusable.
+    EfiUnusableMemory,
+    /// Memory containing the ACPI tables.
+    ///
+    /// Must be preserved in working and ACPI S1-S3 states.
+    EfiACPIReclaimMemory,
+    /// Memory reserved by firmware.
+    ///
+    /// Must be preserved in working and ACPI S1-S3 states.
+    EfiACPIMemoryNVS,
+    /// Memory used by firmware for requesting memory mapping of IO.
+    ///
+    /// Should not be used by the OS. Use the ACPI tables for memory mapped IO
+    /// information.
+    EfiMemoryMappedIO,
+    /// Memory used to translate memory cycles to IO cycles.
+    ///
+    /// Should not be used by the OS. Use the ACPI tables for memory mapped IO
+    /// information.
+    EfiMemoryMappedIOPortSpace,
+    /// Memory used by the processor.
+    ///
+    /// Must be preserved in working and ACPI S1-S4 states. Processor defined
+    /// otherwise.
+    EfiPalCode,
+    /// Available memory supporting byte-addressable non-volatility.
+    EfiPersistentMemory,
+    /// Unknown region type, treat as unusable.
+    EfiUnknown,
+}
+
+impl EFIMemoryDesc {
+    /// The physical address of the memory region.
+    pub fn physical_address(&self) -> u64 {
+        self.phys_addr
+    }
+
+    /// The virtual address of the memory region.
+    pub fn virtual_address(&self) -> u64 {
+        self.virt_addr
+    }
+
+    /// The size in bytes of the memory region.
+    pub fn size(&self, page_size: u64) -> usize {
+        (self.num_pages * page_size) as usize
+    }
+
+    /// The type of the memory region.
+    pub fn typ(&self) -> EFIMemoryAreaType {
+        match self.typ {
+            0 => EFIMemoryAreaType::EfiReservedMemoryType,
+            1 => EFIMemoryAreaType::EfiLoaderCode,
+            2 => EFIMemoryAreaType::EfiLoaderData,
+            3 => EFIMemoryAreaType::EfiBootServicesCode,
+            4 => EFIMemoryAreaType::EfiBootServicesData,
+            5 => EFIMemoryAreaType::EfiRuntimeServicesCode,
+            6 => EFIMemoryAreaType::EfiRuntimeServicesData,
+            7 => EFIMemoryAreaType::EfiConventionalMemory,
+            8 => EFIMemoryAreaType::EfiUnusableMemory,
+            9 => EFIMemoryAreaType::EfiACPIReclaimMemory,
+            10 => EFIMemoryAreaType::EfiACPIMemoryNVS,
+            11 => EFIMemoryAreaType::EfiMemoryMappedIO,
+            12 => EFIMemoryAreaType::EfiMemoryMappedIOPortSpace,
+            13 => EFIMemoryAreaType::EfiPalCode,
+            14 => EFIMemoryAreaType::EfiPersistentMemory,
+            _ => EFIMemoryAreaType::EfiUnknown,
+        }
+    }
+}
+
+/// EFI ExitBootServices was not called
+#[derive(Debug)]
+#[repr(C)]
+pub struct EFIBootServicesNotExited {
+    typ: u32,
+    size: u32,
+}
+
+/// Contains pointer to boot loader image handle.
+#[derive(Debug)]
+#[repr(C)]
+pub struct EFIImageHandle32 {
+    typ: u32,
+    size: u32,
+    pointer: u32,
+}
+
+/// Contains pointer to boot loader image handle.
+#[derive(Debug)]
+#[repr(C)]
+pub struct EFIImageHandle64 {
+    typ: u32,
+    size: u32,
+    pointer: u64,
+}
+
+/// If the image has relocatable header tag, this tag contains the image's 
+/// base physical address.
+#[derive(Debug)]
+#[repr(C)]
+pub struct ImageLoadPhysAddr {
+    typ: u32,
+    size: u32,
+    load_base_addr: u32,
+}
+
+/// An iterator over All EFI memory areas.
+#[derive(Clone, Debug)]
+pub struct EFIMemoryAreaIter<'a> {
+    current_area: u64,
+    last_area: u64,
+    entry_size: u32,
+    phantom: PhantomData<&'a EFIMemoryDesc>,
+}
+
+impl<'a> Iterator for EFIMemoryAreaIter<'a> {
+    type Item = &'a EFIMemoryDesc;
+    fn next(&mut self) -> Option<&'a EFIMemoryDesc> {
+        if self.current_area > self.last_area {
+            None
+        } else {
+            let area = unsafe{&*(self.current_area as *const EFIMemoryDesc)};
+            self.current_area = self.current_area + (self.entry_size as u64);
+            if area.typ() == EFIMemoryAreaType::EfiConventionalMemory {
+                Some(area)
+            } else {self.next()}
+        }
+    }
+}
+

+ 32 - 0
src/rsdp.rs

@@ -11,6 +11,38 @@ use core::str;
 
 const RSDPV1_LENGTH: usize = 20;
 
+/// EFI system table in 32 bit mode
+#[derive(Clone, Copy, Debug)]
+#[repr(C, packed)]
+pub struct EFISdt32 {
+    typ: u32,
+    size: u32,
+    pointer: u32,
+}
+
+impl EFISdt32 {
+    /// The Physical address of a i386 EFI system table.
+    pub fn sdt_address(&self) -> usize {
+        self.pointer as usize
+    }
+}
+
+/// EFI system table in 64 bit mode
+#[derive(Clone, Copy, Debug)]
+#[repr(C, packed)]
+pub struct EFISdt64 {
+    typ: u32,
+    size: u32,
+    pointer: u64,
+}
+
+impl EFISdt64 {
+    /// The Physical address of a x86_64 EFI system table.
+    pub fn sdt_address(&self) -> usize {
+        self.pointer as usize
+    }
+}
+
 /// This tag contains a copy of RSDP as defined per ACPI 1.0 specification. 
 #[derive(Clone, Copy, Debug)]
 #[repr(C, packed)]