|
@@ -32,6 +32,9 @@
|
|
|
//! ## MSRV
|
|
|
//! The MSRV is 1.56.1 stable.
|
|
|
|
|
|
+#[cfg(feature = "builder")]
|
|
|
+extern crate alloc;
|
|
|
+
|
|
|
// this crate can use std in tests only
|
|
|
#[cfg_attr(test, macro_use)]
|
|
|
#[cfg(test)]
|
|
@@ -44,6 +47,8 @@ pub use ptr_meta::Pointee;
|
|
|
|
|
|
use crate::framebuffer::UnknownFramebufferType;
|
|
|
pub use boot_loader_name::BootLoaderNameTag;
|
|
|
+#[cfg(feature = "builder")]
|
|
|
+use builder::traits::StructAsBytes;
|
|
|
pub use command_line::CommandLineTag;
|
|
|
pub use efi::{EFIImageHandle32, EFIImageHandle64, EFISdt32, EFISdt64};
|
|
|
pub use elf_sections::{
|
|
@@ -52,8 +57,8 @@ pub use elf_sections::{
|
|
|
pub use framebuffer::{FramebufferColor, FramebufferField, FramebufferTag, FramebufferType};
|
|
|
pub use image_load_addr::ImageLoadPhysAddr;
|
|
|
pub use memory_map::{
|
|
|
- BasicMemoryInfoTag, EFIMemoryAreaType, EFIMemoryDesc, EFIMemoryMapTag, MemoryArea,
|
|
|
- MemoryAreaIter, MemoryAreaType, MemoryMapTag,
|
|
|
+ BasicMemoryInfoTag, EFIBootServicesNotExited, EFIMemoryAreaType, EFIMemoryDesc,
|
|
|
+ EFIMemoryMapTag, MemoryArea, MemoryAreaIter, MemoryAreaType, MemoryMapTag,
|
|
|
};
|
|
|
pub use module::{ModuleIter, ModuleTag};
|
|
|
pub use rsdp::{RsdpV1Tag, RsdpV2Tag};
|
|
@@ -81,6 +86,9 @@ mod smbios;
|
|
|
mod tag_type;
|
|
|
mod vbe_info;
|
|
|
|
|
|
+#[cfg(feature = "builder")]
|
|
|
+pub mod builder;
|
|
|
+
|
|
|
/// Magic number that a multiboot2-compliant boot loader will store in `eax` register
|
|
|
/// right before handoff to the payload (the kernel). This value can be used to check,
|
|
|
/// that the kernel was indeed booted via multiboot2.
|
|
@@ -197,6 +205,22 @@ struct BootInformationInner {
|
|
|
_reserved: u32,
|
|
|
}
|
|
|
|
|
|
+impl BootInformationInner {
|
|
|
+ fn new(total_size: u32) -> Self {
|
|
|
+ Self {
|
|
|
+ total_size,
|
|
|
+ _reserved: 0,
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+#[cfg(feature = "builder")]
|
|
|
+impl StructAsBytes for BootInformationInner {
|
|
|
+ fn byte_size(&self) -> usize {
|
|
|
+ core::mem::size_of::<Self>()
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
impl BootInformation {
|
|
|
/// Get the start address of the boot info.
|
|
|
pub fn start_address(&self) -> usize {
|
|
@@ -225,10 +249,26 @@ impl BootInformation {
|
|
|
self.get_tag::<BasicMemoryInfoTag, _>(TagType::BasicMeminfo)
|
|
|
}
|
|
|
|
|
|
- /// Search for the ELF Sections tag.
|
|
|
- pub fn elf_sections_tag(&self) -> Option<ElfSectionsTag> {
|
|
|
- self.get_tag::<Tag, _>(TagType::ElfSections)
|
|
|
- .map(|tag| unsafe { elf_sections::elf_sections_tag(tag, self.offset) })
|
|
|
+ /// Search for the ELF Sections.
|
|
|
+ ///
|
|
|
+ /// # Examples
|
|
|
+ ///
|
|
|
+ /// ```rust,no_run
|
|
|
+ /// # let boot_info = unsafe { multiboot2::load(0xdeadbeef).unwrap() };
|
|
|
+ /// if let Some(sections) = boot_info.elf_sections() {
|
|
|
+ /// let mut total = 0;
|
|
|
+ /// for section in sections {
|
|
|
+ /// println!("Section: {:?}", section);
|
|
|
+ /// total += 1;
|
|
|
+ /// }
|
|
|
+ /// }
|
|
|
+ /// ```
|
|
|
+ pub fn elf_sections(&self) -> Option<ElfSectionIter> {
|
|
|
+ let tag = self.get_tag::<ElfSectionsTag, _>(TagType::ElfSections);
|
|
|
+ tag.map(|t| {
|
|
|
+ assert!((t.entry_size * t.shndx) <= t.size);
|
|
|
+ t.sections(self.offset)
|
|
|
+ })
|
|
|
}
|
|
|
|
|
|
/// Search for the Memory map tag.
|
|
@@ -253,9 +293,12 @@ impl BootInformation {
|
|
|
|
|
|
/// Search for the VBE framebuffer tag. The result is `Some(Err(e))`, if the
|
|
|
/// framebuffer type is unknown, while the framebuffer tag is present.
|
|
|
- pub fn framebuffer_tag(&self) -> Option<Result<FramebufferTag, UnknownFramebufferType>> {
|
|
|
- self.get_tag::<Tag, _>(TagType::Framebuffer)
|
|
|
- .map(framebuffer::framebuffer_tag)
|
|
|
+ pub fn framebuffer_tag(&self) -> Option<Result<&FramebufferTag, UnknownFramebufferType>> {
|
|
|
+ self.get_tag::<FramebufferTag, _>(TagType::Framebuffer)
|
|
|
+ .map(|tag| match tag.buffer_type() {
|
|
|
+ Ok(_) => Ok(tag),
|
|
|
+ Err(e) => Err(e),
|
|
|
+ })
|
|
|
}
|
|
|
|
|
|
/// Search for the EFI 32-bit SDT tag.
|
|
@@ -437,20 +480,14 @@ impl fmt::Debug for BootInformation {
|
|
|
.field("module_tags", &self.module_tags());
|
|
|
// usually this is REALLY big (thousands of tags) => skip it here
|
|
|
|
|
|
- let elf_sections_tag_entries_count = self
|
|
|
- .elf_sections_tag()
|
|
|
- .map(|x| x.sections().count())
|
|
|
- .unwrap_or(0);
|
|
|
+ let elf_sections_tag_entries_count = self.elf_sections().map(|x| x.count()).unwrap_or(0);
|
|
|
|
|
|
if elf_sections_tag_entries_count > ELF_SECTIONS_LIMIT {
|
|
|
debug.field("elf_sections_tags (count)", &elf_sections_tag_entries_count);
|
|
|
} else {
|
|
|
debug.field(
|
|
|
"elf_sections_tags",
|
|
|
- &self
|
|
|
- .elf_sections_tag()
|
|
|
- .map(|x| x.sections())
|
|
|
- .unwrap_or_default(),
|
|
|
+ &self.elf_sections().unwrap_or_default(),
|
|
|
);
|
|
|
}
|
|
|
|
|
@@ -574,7 +611,7 @@ mod tests {
|
|
|
assert_eq!(addr, bi.start_address());
|
|
|
assert_eq!(addr + bytes.0.len(), bi.end_address());
|
|
|
assert_eq!(bytes.0.len(), bi.total_size());
|
|
|
- assert!(bi.elf_sections_tag().is_none());
|
|
|
+ assert!(bi.elf_sections().is_none());
|
|
|
assert!(bi.memory_map_tag().is_none());
|
|
|
assert!(bi.module_tags().next().is_none());
|
|
|
assert!(bi.boot_loader_name_tag().is_none());
|
|
@@ -598,7 +635,7 @@ mod tests {
|
|
|
assert_eq!(addr, bi.start_address());
|
|
|
assert_eq!(addr + bytes.0.len(), bi.end_address());
|
|
|
assert_eq!(bytes.0.len(), bi.total_size());
|
|
|
- assert!(bi.elf_sections_tag().is_none());
|
|
|
+ assert!(bi.elf_sections().is_none());
|
|
|
assert!(bi.memory_map_tag().is_none());
|
|
|
assert!(bi.module_tags().next().is_none());
|
|
|
assert!(bi.boot_loader_name_tag().is_none());
|
|
@@ -622,7 +659,7 @@ mod tests {
|
|
|
assert_eq!(addr, bi.start_address());
|
|
|
assert_eq!(addr + bytes.0.len(), bi.end_address());
|
|
|
assert_eq!(bytes.0.len(), bi.total_size());
|
|
|
- assert!(bi.elf_sections_tag().is_none());
|
|
|
+ assert!(bi.elf_sections().is_none());
|
|
|
assert!(bi.memory_map_tag().is_none());
|
|
|
assert!(bi.module_tags().next().is_none());
|
|
|
assert!(bi.boot_loader_name_tag().is_none());
|
|
@@ -649,7 +686,7 @@ mod tests {
|
|
|
assert_eq!(addr, bi.start_address());
|
|
|
assert_eq!(addr + bytes.0.len(), bi.end_address());
|
|
|
assert_eq!(bytes.0.len(), bi.total_size());
|
|
|
- assert!(bi.elf_sections_tag().is_none());
|
|
|
+ assert!(bi.elf_sections().is_none());
|
|
|
assert!(bi.memory_map_tag().is_none());
|
|
|
assert!(bi.module_tags().next().is_none());
|
|
|
assert_eq!(
|
|
@@ -691,31 +728,34 @@ mod tests {
|
|
|
assert_eq!(addr, bi.start_address());
|
|
|
assert_eq!(addr + bytes.0.len(), bi.end_address());
|
|
|
assert_eq!(bytes.0.len(), bi.total_size());
|
|
|
- use framebuffer::{FramebufferField, FramebufferTag, FramebufferType};
|
|
|
+ use framebuffer::{FramebufferField, FramebufferType};
|
|
|
+ assert!(bi.framebuffer_tag().is_some());
|
|
|
+ let fbi = bi
|
|
|
+ .framebuffer_tag()
|
|
|
+ .expect("Framebuffer info should be available")
|
|
|
+ .expect("Framebuffer info type should be valid");
|
|
|
+ assert_eq!(fbi.address(), 4244635648);
|
|
|
+ assert_eq!(fbi.pitch(), 5120);
|
|
|
+ assert_eq!(fbi.width(), 1280);
|
|
|
+ assert_eq!(fbi.height(), 720);
|
|
|
+ assert_eq!(fbi.bpp(), 32);
|
|
|
assert_eq!(
|
|
|
- bi.framebuffer_tag(),
|
|
|
- Some(Ok(FramebufferTag {
|
|
|
- address: 4244635648,
|
|
|
- pitch: 5120,
|
|
|
- width: 1280,
|
|
|
- height: 720,
|
|
|
- bpp: 32,
|
|
|
- buffer_type: FramebufferType::RGB {
|
|
|
- red: FramebufferField {
|
|
|
- position: 16,
|
|
|
- size: 8
|
|
|
- },
|
|
|
- green: FramebufferField {
|
|
|
- position: 8,
|
|
|
- size: 8
|
|
|
- },
|
|
|
- blue: FramebufferField {
|
|
|
- position: 0,
|
|
|
- size: 8
|
|
|
- }
|
|
|
+ fbi.buffer_type().unwrap(),
|
|
|
+ FramebufferType::RGB {
|
|
|
+ red: FramebufferField {
|
|
|
+ position: 16,
|
|
|
+ size: 8
|
|
|
+ },
|
|
|
+ green: FramebufferField {
|
|
|
+ position: 8,
|
|
|
+ size: 8
|
|
|
+ },
|
|
|
+ blue: FramebufferField {
|
|
|
+ position: 0,
|
|
|
+ size: 8
|
|
|
}
|
|
|
- }))
|
|
|
- )
|
|
|
+ }
|
|
|
+ );
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
@@ -753,12 +793,12 @@ mod tests {
|
|
|
.framebuffer_tag()
|
|
|
.expect("Framebuffer info should be available")
|
|
|
.expect("Framebuffer info type should be valid");
|
|
|
- assert_eq!(fbi.address, 4244635648);
|
|
|
- assert_eq!(fbi.pitch, 5120);
|
|
|
- assert_eq!(fbi.width, 1280);
|
|
|
- assert_eq!(fbi.height, 720);
|
|
|
- assert_eq!(fbi.bpp, 32);
|
|
|
- match fbi.buffer_type {
|
|
|
+ assert_eq!(fbi.address(), 4244635648);
|
|
|
+ assert_eq!(fbi.pitch(), 5120);
|
|
|
+ assert_eq!(fbi.width(), 1280);
|
|
|
+ assert_eq!(fbi.height(), 720);
|
|
|
+ assert_eq!(fbi.bpp(), 32);
|
|
|
+ match fbi.buffer_type().unwrap() {
|
|
|
FramebufferType::Indexed { palette } => assert_eq!(
|
|
|
palette,
|
|
|
[
|
|
@@ -788,16 +828,6 @@ mod tests {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- #[test]
|
|
|
- /// Compile time test for `FramebufferTag`.
|
|
|
- fn framebuffer_tag_size() {
|
|
|
- use crate::FramebufferTag;
|
|
|
- unsafe {
|
|
|
- // 24 for the start + 24 for `FramebufferType`.
|
|
|
- core::mem::transmute::<[u8; 48], FramebufferTag>([0u8; 48]);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
#[test]
|
|
|
fn vbe_info_tag() {
|
|
|
//Taken from GRUB2 running in QEMU.
|
|
@@ -1255,16 +1285,15 @@ mod tests {
|
|
|
assert_eq!(addr, bi.start_address());
|
|
|
assert_eq!(addr + bytes.len(), bi.end_address());
|
|
|
assert_eq!(bytes.len(), bi.total_size());
|
|
|
- let es = bi.elf_sections_tag().unwrap();
|
|
|
- let mut s = es.sections();
|
|
|
- let s1 = s.next().expect("Should have one more section");
|
|
|
+ let mut es = bi.elf_sections().unwrap();
|
|
|
+ let s1 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".rodata", s1.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(0xFFFF_8000_0010_0000, s1.start_address());
|
|
|
assert_eq!(0xFFFF_8000_0010_3000, s1.end_address());
|
|
|
assert_eq!(0x0000_0000_0000_3000, s1.size());
|
|
|
assert_eq!(ElfSectionFlags::ALLOCATED, s1.flags());
|
|
|
assert_eq!(ElfSectionType::ProgramSection, s1.section_type());
|
|
|
- let s2 = s.next().expect("Should have one more section");
|
|
|
+ let s2 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".text", s2.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(0xFFFF_8000_0010_3000, s2.start_address());
|
|
|
assert_eq!(0xFFFF_8000_0010_C000, s2.end_address());
|
|
@@ -1274,7 +1303,7 @@ mod tests {
|
|
|
s2.flags()
|
|
|
);
|
|
|
assert_eq!(ElfSectionType::ProgramSection, s2.section_type());
|
|
|
- let s3 = s.next().expect("Should have one more section");
|
|
|
+ let s3 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".data", s3.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(0xFFFF_8000_0010_C000, s3.start_address());
|
|
|
assert_eq!(0xFFFF_8000_0010_E000, s3.end_address());
|
|
@@ -1284,7 +1313,7 @@ mod tests {
|
|
|
s3.flags()
|
|
|
);
|
|
|
assert_eq!(ElfSectionType::ProgramSection, s3.section_type());
|
|
|
- let s4 = s.next().expect("Should have one more section");
|
|
|
+ let s4 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".bss", s4.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(0xFFFF_8000_0010_E000, s4.start_address());
|
|
|
assert_eq!(0xFFFF_8000_0011_3000, s4.end_address());
|
|
@@ -1294,7 +1323,7 @@ mod tests {
|
|
|
s4.flags()
|
|
|
);
|
|
|
assert_eq!(ElfSectionType::Uninitialized, s4.section_type());
|
|
|
- let s5 = s.next().expect("Should have one more section");
|
|
|
+ let s5 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".data.rel.ro", s5.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(0xFFFF_8000_0011_3000, s5.start_address());
|
|
|
assert_eq!(0xFFFF_8000_0011_3000, s5.end_address());
|
|
@@ -1304,28 +1333,28 @@ mod tests {
|
|
|
s5.flags()
|
|
|
);
|
|
|
assert_eq!(ElfSectionType::ProgramSection, s5.section_type());
|
|
|
- let s6 = s.next().expect("Should have one more section");
|
|
|
+ let s6 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".symtab", s6.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(0x0000_0000_0011_3000, s6.start_address());
|
|
|
assert_eq!(0x0000_0000_0011_5BE0, s6.end_address());
|
|
|
assert_eq!(0x0000_0000_0000_2BE0, s6.size());
|
|
|
assert_eq!(ElfSectionFlags::empty(), s6.flags());
|
|
|
assert_eq!(ElfSectionType::LinkerSymbolTable, s6.section_type());
|
|
|
- let s7 = s.next().expect("Should have one more section");
|
|
|
+ let s7 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".strtab", s7.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(0x0000_0000_0011_5BE0, s7.start_address());
|
|
|
assert_eq!(0x0000_0000_0011_9371, s7.end_address());
|
|
|
assert_eq!(0x0000_0000_0000_3791, s7.size());
|
|
|
assert_eq!(ElfSectionFlags::empty(), s7.flags());
|
|
|
assert_eq!(ElfSectionType::StringTable, s7.section_type());
|
|
|
- let s8 = s.next().expect("Should have one more section");
|
|
|
+ let s8 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".shstrtab", s8.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(string_addr, s8.start_address());
|
|
|
assert_eq!(string_addr + string_bytes.len() as u64, s8.end_address());
|
|
|
assert_eq!(string_bytes.len() as u64, s8.size());
|
|
|
assert_eq!(ElfSectionFlags::empty(), s8.flags());
|
|
|
assert_eq!(ElfSectionType::StringTable, s8.section_type());
|
|
|
- assert!(s.next().is_none());
|
|
|
+ assert!(es.next().is_none());
|
|
|
let mut mm = bi.memory_map_tag().unwrap().available_memory_areas();
|
|
|
let mm1 = mm.next().unwrap();
|
|
|
assert_eq!(0x00000000, mm1.start_address());
|
|
@@ -1368,12 +1397,12 @@ mod tests {
|
|
|
.framebuffer_tag()
|
|
|
.expect("Framebuffer info should be available")
|
|
|
.expect("Framebuffer info type should be valid");
|
|
|
- assert_eq!(fbi.address, 753664);
|
|
|
- assert_eq!(fbi.pitch, 160);
|
|
|
- assert_eq!(fbi.width, 80);
|
|
|
- assert_eq!(fbi.height, 25);
|
|
|
- assert_eq!(fbi.bpp, 16);
|
|
|
- assert_eq!(fbi.buffer_type, FramebufferType::Text);
|
|
|
+ assert_eq!(fbi.address(), 753664);
|
|
|
+ assert_eq!(fbi.pitch(), 160);
|
|
|
+ assert_eq!(fbi.width(), 80);
|
|
|
+ assert_eq!(fbi.height(), 25);
|
|
|
+ assert_eq!(fbi.bpp(), 16);
|
|
|
+ assert_eq!(fbi.buffer_type(), Ok(FramebufferType::Text));
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
@@ -1440,26 +1469,15 @@ mod tests {
|
|
|
assert_eq!(addr, bi.start_address());
|
|
|
assert_eq!(addr + bytes.0.len(), bi.end_address());
|
|
|
assert_eq!(bytes.0.len(), bi.total_size());
|
|
|
- let es = bi.elf_sections_tag().unwrap();
|
|
|
- let mut s = es.sections();
|
|
|
- let s1 = s.next().expect("Should have one more section");
|
|
|
+ let mut es = bi.elf_sections().unwrap();
|
|
|
+ let s1 = es.next().expect("Should have one more section");
|
|
|
assert_eq!(".shstrtab", s1.name().expect("Should be valid utf-8"));
|
|
|
assert_eq!(string_addr, s1.start_address());
|
|
|
assert_eq!(string_addr + string_bytes.0.len() as u64, s1.end_address());
|
|
|
assert_eq!(string_bytes.0.len() as u64, s1.size());
|
|
|
assert_eq!(ElfSectionFlags::empty(), s1.flags());
|
|
|
assert_eq!(ElfSectionType::StringTable, s1.section_type());
|
|
|
- assert!(s.next().is_none());
|
|
|
- }
|
|
|
-
|
|
|
- #[test]
|
|
|
- /// Compile time test for `ElfSectionsTag`.
|
|
|
- fn elf_sections_tag_size() {
|
|
|
- use super::ElfSectionsTag;
|
|
|
- unsafe {
|
|
|
- // `ElfSectionsTagInner` is 12 bytes + 4 in the offset.
|
|
|
- core::mem::transmute::<[u8; 16], ElfSectionsTag>([0u8; 16]);
|
|
|
- }
|
|
|
+ assert!(es.next().is_none());
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
@@ -1531,26 +1549,6 @@ mod tests {
|
|
|
assert!(efi_mmap.is_none());
|
|
|
}
|
|
|
|
|
|
- #[test]
|
|
|
- /// Compile time test for `MemoryMapTag`.
|
|
|
- fn e820_memory_map_tag_size() {
|
|
|
- use super::MemoryMapTag;
|
|
|
- unsafe {
|
|
|
- // `MemoryMapTag` is 16 bytes without the 1st entry
|
|
|
- core::mem::transmute::<[u8; 16], MemoryMapTag>([0u8; 16]);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- #[test]
|
|
|
- /// Compile time test for `EFIMemoryMapTag`.
|
|
|
- fn efi_memory_map_tag_size() {
|
|
|
- use super::EFIMemoryMapTag;
|
|
|
- unsafe {
|
|
|
- // `EFIMemoryMapTag` is 16 bytes without the 1st entry
|
|
|
- core::mem::transmute::<[u8; 16], EFIMemoryMapTag>([0u8; 16]);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
#[test]
|
|
|
#[cfg(feature = "unstable")]
|
|
|
/// This test succeeds if it compiles.
|