From 059566e698773ff87841c02bbe465b2620c7d778 Mon Sep 17 00:00:00 2001 From: Berkus Decker Date: Sun, 11 Jul 2021 22:31:15 +0300 Subject: [PATCH] [wip] Add wrapper for device_tree crate, bootup with DTB This is to easily add necessary functions without touching the original crate - this will probably need to be upstreamed eventually! --- Cargo.lock | 36 +++++++++++++++++++++++++ nucleus/Cargo.toml | 2 ++ nucleus/src/arch/aarch64/boot.rs | 46 +++++++++++++++++++++----------- nucleus/src/device_tree.rs | 32 ++++++++++++++++++++++ nucleus/src/main.rs | 36 ++++++++++++++++++++++++- 5 files changed, 136 insertions(+), 16 deletions(-) create mode 100644 nucleus/src/device_tree.rs diff --git a/Cargo.lock b/Cargo.lock index 545dca4..2a35c5f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -35,6 +35,27 @@ version = "0.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fea41bba32d969b513997752735605054bc0dfa92b4c56bf1189f2e174be7a10" +[[package]] +name = "dtb" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be6c6e1672a044abd9652045ba11beb1c7fbec7e5262dcba54795e82aa577971" + +[[package]] +name = "either" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e78d4f1cc4ae33bbfc157ed5d5a5ef3bc29227303d595861deb238fcec4e9457" + +[[package]] +name = "itertools" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f56a2d0bc861f9165be4eb3442afd3c236d8a98afd426f65d92324ae1091a484" +dependencies = [ + "either", +] + [[package]] name = "proc-macro2" version = "1.0.27" @@ -65,6 +86,19 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bd7a31eed1591dcbc95d92ad7161908e72f4677f8fabf2a32ca49b4237cbf211" +[[package]] +name = "shrinkwraprs" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e63e6744142336dfb606fe2b068afa2e1cca1ee6a5d8377277a92945d81fa331" +dependencies = [ + "bitflags", + "itertools", + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "snafu" version = "0.7.0-beta.0" @@ -129,8 +163,10 @@ dependencies = [ "bitflags", "cfg-if", "cortex-a", + "dtb", "qemu-exit", "r0", + "shrinkwraprs", "snafu", "tock-registers", "usize_conversions", diff --git a/nucleus/Cargo.toml b/nucleus/Cargo.toml index 3eb0c5c..62d4681 100644 --- a/nucleus/Cargo.toml +++ b/nucleus/Cargo.toml @@ -34,3 +34,5 @@ bit_field = "0.10.1" bitflags = "1.2" cfg-if = "1.0" snafu = { version = "0.7.0-beta.0", default-features = false } +shrinkwraprs = { version = "0.3", default-features = false } +dtb = "0.2" diff --git a/nucleus/src/arch/aarch64/boot.rs b/nucleus/src/arch/aarch64/boot.rs index 7af95ed..0d397c5 100644 --- a/nucleus/src/arch/aarch64/boot.rs +++ b/nucleus/src/arch/aarch64/boot.rs @@ -8,6 +8,14 @@ //! Low-level boot of the Raspberry's processor //! +//! Raspi kernel boot helper: https://github.com/raspberrypi/tools/blob/master/armstubs/armstub8.S +//! In particular, see dtb_ptr32 + +//! To get memory size from DTB: +//! 1. Find nodes with unit-names `/memory` +//! 2. From those read reg entries, using `/#address-cells` and `/#size-cells` as units +//! 3. Union of all these reg entries will be the available memory. Enter it as mem-regions. + use { crate::endless_sleep, cortex_a::{asm, registers::*}, @@ -24,11 +32,11 @@ macro_rules! entry { /// # Safety /// Only type-checks! #[export_name = "main"] - pub unsafe fn __main() -> ! { + pub unsafe fn __main(dtb: u32) -> ! { // type check the given path - let f: fn() -> ! = $path; + let f: fn(u32) -> ! = $path; - f() + f(dtb) } }; } @@ -41,7 +49,7 @@ macro_rules! entry { /// /// Totally unsafe! We're in the hardware land. #[link_section = ".text.boot"] -unsafe fn reset() -> ! { +unsafe fn reset(dtb: u32) -> ! { extern "C" { // Boundaries of the .bss section, provided by the linker script static mut __BSS_START: u64; @@ -52,10 +60,10 @@ unsafe fn reset() -> ! { r0::zero_bss(&mut __BSS_START, &mut __BSS_END); extern "Rust" { - fn main() -> !; + fn main(dtb: u32) -> !; } - main() + main(dtb) } // [ARMv6 unaligned data access restrictions](https://developer.arm.com/documentation/ddi0333/h/unaligned-and-mixed-endian-data-access-support/unaligned-access-support/armv6-unaligned-data-access-restrictions?lang=en) @@ -108,11 +116,16 @@ fn shared_setup_and_enter_pre() { #[link_section = ".text.boot"] #[inline] -fn shared_setup_and_enter_post() -> ! { +fn shared_setup_and_enter_post(dtb: u32) -> ! { // Set up SP_EL1 (stack pointer), which will be used by EL1 once // we "return" to it. SP_EL1.set(STACK_START); + unsafe { + asm!("mov {dtb:w}, w0", dtb = in(reg) dtb); + // @todo How to enforce dtb being in w0 at this point? -- must be an arg to eret() + } + // Use `eret` to "return" to EL1. This will result in execution of // `reset()` in EL1. asm::eret() @@ -123,7 +136,7 @@ fn shared_setup_and_enter_post() -> ! { /// Prepare and execute transition from EL2 to EL1. #[link_section = ".text.boot"] #[inline] -fn setup_and_enter_el1_from_el2() -> ! { +fn setup_and_enter_el1_from_el2(dtb: u32) -> ! { // Set Saved Program Status Register (EL2) // Set up a simulated exception return. // @@ -140,7 +153,7 @@ fn setup_and_enter_el1_from_el2() -> ! { // Make the Exception Link Register (EL2) point to reset(). ELR_EL2.set(reset as *const () as u64); - shared_setup_and_enter_post() + shared_setup_and_enter_post(dtb) } /// QEMU boot-up sequence. @@ -156,7 +169,7 @@ fn setup_and_enter_el1_from_el2() -> ! { #[cfg(qemu)] #[link_section = ".text.boot"] #[inline] -fn setup_and_enter_el1_from_el3() -> ! { +fn setup_and_enter_el1_from_el3(dtb: u32) -> ! { // Set Secure Configuration Register (EL3) SCR_EL3.write(SCR_EL3::RW::NextELIsAarch64 + SCR_EL3::NS::NonSecure); @@ -176,7 +189,7 @@ fn setup_and_enter_el1_from_el3() -> ! { // Make the Exception Link Register (EL3) point to reset(). ELR_EL3.set(reset as *const () as u64); - shared_setup_and_enter_post() + shared_setup_and_enter_post(dtb) } /// Entrypoint of the processor. @@ -192,7 +205,7 @@ fn setup_and_enter_el1_from_el3() -> ! { /// #[no_mangle] #[link_section = ".text.boot.entry"] -pub unsafe extern "C" fn _boot_cores() -> ! { +pub unsafe extern "C" fn _boot_cores(dtb: u32) -> ! { const CORE_0: u64 = 0; const CORE_MASK: u64 = 0x3; // Can't match values with dots in match, so use intermediate consts. @@ -207,11 +220,14 @@ pub unsafe extern "C" fn _boot_cores() -> ! { shared_setup_and_enter_pre(); if CORE_0 == MPIDR_EL1.get() & CORE_MASK { + // @todo On entry, w0 should contain the dtb address. + // For non-primary cores it however contains 0. + match CurrentEL.get() { #[cfg(qemu)] - EL3 => setup_and_enter_el1_from_el3(), - EL2 => setup_and_enter_el1_from_el2(), - EL1 => reset(), + EL3 => setup_and_enter_el1_from_el3(dtb), + EL2 => setup_and_enter_el1_from_el2(dtb), + EL1 => reset(dtb), _ => endless_sleep(), } } diff --git a/nucleus/src/device_tree.rs b/nucleus/src/device_tree.rs new file mode 100644 index 0000000..0dd9b3d --- /dev/null +++ b/nucleus/src/device_tree.rs @@ -0,0 +1,32 @@ +use shrinkwraprs::Shrinkwrap; + +#[derive(Shrinkwrap)] +pub struct DeadTree<'a>(dtb::Reader<'a>); + +impl<'a> DeadTree<'a> { + pub fn new(reader: dtb::Reader<'a>) -> Self { + Self(reader) + } + + pub fn try_struct_u32_value<'s, P: Into<&'s str>>(&self, path: P) -> Result { + let mut buf = [0u8; 4]; + Ok(self + .0 + .struct_items() + .path_struct_items(path.into()) + .next() + .ok_or(dtb::Error::BadPropertyName)? + .0 + .value_u32_list(&mut buf)?[0]) + } + + pub fn try_struct_str_value<'s, P: Into<&'s str>>(&self, path: P) -> Result<&str, dtb::Error> { + self.0 + .struct_items() + .path_struct_items(path.into()) + .next() + .ok_or(dtb::Error::BadPropertyName)? + .0 + .value_str() + } +} diff --git a/nucleus/src/main.rs b/nucleus/src/main.rs index 57d396b..e31dbe1 100644 --- a/nucleus/src/main.rs +++ b/nucleus/src/main.rs @@ -32,6 +32,7 @@ use architecture_not_supported_sorry; #[macro_use] pub mod arch; pub use arch::*; +mod device_tree; mod devices; mod macros; mod mm; @@ -151,7 +152,7 @@ fn init_uart_serial() { /// Kernel entry point. /// `arch` crate is responsible for calling it. #[inline] -pub fn kmain() -> ! { +pub fn kmain(dtb: u32) -> ! { #[cfg(feature = "jtag")] jtag::wait_debugger(); @@ -164,6 +165,39 @@ pub fn kmain() -> ! { #[cfg(test)] test_main(); + println!("DTB loaded at {:x}", dtb); + + // Safety: we got the address from the bootloader, if it lied - well, we're screwed! + let device_tree = crate::device_tree::DeadTree::new(unsafe { + dtb::Reader::read_from_address(dtb as usize).expect("DeviceTree not found") + }); + + // List unusable memory, and remove it from the memory regions for the allocator. + for entry in device_tree.reserved_mem_entries() { + println!("reserved: {:?} bytes at {:?}", entry.size, entry.address); + } + // Also, remove the DTB memory region. + + // To init memory allocation we need to parse memory regions from dtb and add the regions to + // available memory regions list. Then initial BootRegionAllocator will get memory from these + // regions and record their usage into some OTHER structures, removing these allocations from + // the free regions list. + // memory allocation is described by reg attribute of /memory block. + // /#address-cells and /#size-cells specify the sizes of address and size attributes in reg. + + let address_cells = device_tree.try_struct_u32_value("/#address-cells"); + let size_cells = device_tree.try_struct_u32_value("/#size-cells"); + let board = device_tree.try_struct_str_value("/model"); + + if board.is_ok() { + println!("Running on {}", board.unwrap()); + } + + println!( + "Memory DTB info: address-cells {:?}, size-cells {:?}", + address_cells, size_cells + ); + command_prompt(); reboot()