Browse Source

refactor(prototyper): add platform mod

Signed-off-by: Woshiluo Luo <woshiluo.luo@outlook.com>
Woshiluo Luo 5 months ago
parent
commit
e418dd8093

+ 0 - 101
prototyper/src/dynamic.rs

@@ -1,102 +1 @@
-//! Frequently used first boot stage dynamic information on RISC-V.
 
-use core::ops::Range;
-
-use riscv::register::mstatus;
-
-/// M-mode firmware dynamic information.
-#[derive(Clone, Copy)]
-#[repr(C)]
-pub struct DynamicInfo {
-    /// Dynamic information magic value.
-    pub magic: usize,
-    /// Version of dynamic information.
-    pub version: usize,
-    /// Address of the next boot-loading stage.
-    pub next_addr: usize,
-    /// RISC-V privilege mode of the next boot-loading stage.
-    pub next_mode: usize,
-    /// M-mode firmware options; its definition varies between SBI implementations.
-    pub options: usize,
-    /// Boot hart ID of current environment.
-    pub boot_hart: usize,
-}
-
-// Definition of `boot_hart` can be found at:
-// https://github.com/riscv-software-src/opensbi/blob/019a8e69a1dc0c0f011fabd0372e1ba80e40dd7c/include/sbi/fw_dynamic.h#L75
-
-const DYNAMIC_INFO_INVALID_ADDRESSES: usize = 0x00000000;
-const NEXT_ADDR_VALID_ADDRESSES: Range<usize> = 0x80000000..0x90000000;
-pub(crate) const MAGIC: usize = 0x4942534f;
-const SUPPORTED_VERSION: Range<usize> = 2..3;
-
-pub struct DynamicReadError {
-    pub bad_paddr: Option<usize>,
-    pub bad_magic: Option<usize>,
-    pub bad_version: Option<usize>,
-}
-
-// TODO unconstrained lifetime
-pub fn read_paddr(paddr: usize) -> Result<DynamicInfo, DynamicReadError> {
-    let mut error = DynamicReadError {
-        bad_paddr: None,
-        bad_magic: None,
-        bad_version: None,
-    };
-    // check pointer before dereference
-    if DYNAMIC_INFO_INVALID_ADDRESSES == paddr {
-        error.bad_paddr = Some(paddr);
-        return Err(error);
-    }
-    let ans = unsafe { *(paddr as *const DynamicInfo) };
-
-    if ans.magic != MAGIC {
-        error.bad_magic = Some(ans.magic);
-    }
-    if !SUPPORTED_VERSION.contains(&ans.version) {
-        error.bad_version = Some(ans.version);
-    }
-    if error.bad_magic.is_some() || error.bad_version.is_some() {
-        return Err(error);
-    }
-    Ok(ans)
-}
-
-pub struct DynamicError<'a> {
-    pub invalid_mpp: bool,
-    pub invalid_next_addr: bool,
-    pub bad_info: &'a DynamicInfo,
-}
-
-pub fn mpp_next_addr(info: &DynamicInfo) -> Result<(mstatus::MPP, usize), DynamicError> {
-    let mut error = DynamicError {
-        invalid_mpp: false,
-        invalid_next_addr: false,
-        bad_info: info,
-    };
-
-    // fail safe, errors will be aggregated after whole checking process.
-    let next_addr_valid = NEXT_ADDR_VALID_ADDRESSES.contains(&info.next_addr);
-    let mpp_valid = matches!(info.next_mode, 0 | 1 | 3);
-
-    if !next_addr_valid {
-        error.invalid_next_addr = true;
-    }
-    if !mpp_valid {
-        error.invalid_mpp = true;
-    }
-
-    if !next_addr_valid || !mpp_valid {
-        return Err(error);
-    }
-
-    let mpp = match info.next_mode {
-        3 => mstatus::MPP::Machine,
-        1 => mstatus::MPP::Supervisor,
-        // pattern `_` avoids `unreachable!`` which introduces panic handler.
-        // pattern 0 and _
-        _ => mstatus::MPP::User,
-    };
-
-    Ok((mpp, info.next_addr))
-}

+ 3 - 2
prototyper/src/fail.rs

@@ -1,11 +1,12 @@
-use riscv::register::mstatus;
 use serde_device_tree::Dtb;
 
 use crate::dt::{self, ParseDeviceTreeError, Tree};
 use crate::sbi::reset;
 
 #[cfg(not(feature = "payload"))]
-use crate::dynamic;
+use crate::platform::dynamic;
+#[cfg(not(feature = "payload"))]
+use riscv::register::mstatus;
 
 #[cold]
 pub fn device_tree_format(err: dt::ParseDeviceTreeError) -> Dtb {

+ 10 - 28
prototyper/src/main.rs

@@ -10,11 +10,8 @@ mod macros;
 
 mod board;
 mod dt;
-#[cfg(not(feature = "payload"))]
-mod dynamic;
 mod fail;
-#[cfg(feature = "payload")]
-mod payload;
+mod platform;
 mod riscv_spec;
 mod sbi;
 
@@ -22,6 +19,8 @@ use core::sync::atomic::{AtomicBool, Ordering};
 use core::{arch::asm, mem::MaybeUninit};
 
 use crate::board::{SBI_IMPL, SIFIVECLINT, SIFIVETEST, UART};
+#[cfg(not(feature = "payload"))]
+use crate::platform::dynamic;
 use crate::riscv_spec::{current_hartid, menvcfg};
 use crate::sbi::console::SbiConsole;
 use crate::sbi::hart_context::NextStage;
@@ -37,35 +36,18 @@ use crate::sbi::SBI;
 #[no_mangle]
 extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
     // parse dynamic information
-    #[cfg(not(feature = "payload"))]
-    let info = dynamic::read_paddr(nonstandard_a2).unwrap_or_else(fail::no_dynamic_info_available);
-    static GENESIS: AtomicBool = AtomicBool::new(true);
     static SBI_READY: AtomicBool = AtomicBool::new(false);
 
-    #[cfg(not(feature = "payload"))]
-    let is_boot_hart = if info.boot_hart == usize::MAX {
-        GENESIS.swap(false, Ordering::AcqRel)
-    } else {
-        current_hartid() == info.boot_hart
-    };
-    #[cfg(feature = "payload")]
-    let is_boot_hart = true;
-
-    #[cfg(feature = "payload")]
-    let fdt_address = payload::get_fdt_address();
-    #[cfg(not(feature = "payload"))]
-    let fdt_address = opaque;
+    let boot_info = platform::get_boot_info(opaque, nonstandard_a2);
 
-    if is_boot_hart {
-        #[cfg(feature = "payload")]
-        let (mpp, next_addr) = (mstatus::MPP::Supervisor, payload::get_image_address());
-        #[cfg(not(feature = "payload"))]
-        let (mpp, next_addr) =
-            dynamic::mpp_next_addr(&info).unwrap_or_else(fail::invalid_dynamic_data);
+    if boot_info.is_boot_hart {
+        let mpp = boot_info.mpp;
+        let next_addr = boot_info.next_address;
+        let fdt_addr = boot_info.fdt_address;
 
         // 1. Init FDT
         // parse the device tree
-        let dtb = dt::parse_device_tree(fdt_address).unwrap_or_else(fail::device_tree_format);
+        let dtb = dt::parse_device_tree(fdt_addr).unwrap_or_else(fail::device_tree_format);
         let dtb = dtb.share();
         let tree =
             serde_device_tree::from_raw_mut(&dtb).unwrap_or_else(fail::device_tree_deserialize);
@@ -151,7 +133,7 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
         local_remote_hsm().start(NextStage {
             start_addr: next_addr,
             next_mode: mpp,
-            opaque: fdt_address,
+            opaque: fdt_addr,
         });
 
         info!(

+ 0 - 28
prototyper/src/payload.rs

@@ -1,29 +1 @@
-use core::arch::asm;
 
-#[naked]
-#[link_section = ".fw_fdt"]
-pub unsafe extern "C" fn raw_fdt() {
-    asm!(
-        concat!(".incbin \"", env!("PROTOTYPER_FDT"), "\""),
-        options(noreturn)
-    );
-}
-
-#[naked]
-#[link_section = ".payload"]
-pub unsafe extern "C" fn payload_image() {
-    asm!(
-        concat!(".incbin \"", env!("PROTOTYPER_IMAGE"), "\""),
-        options(noreturn)
-    );
-}
-
-#[inline]
-pub fn get_fdt_address() -> usize {
-    raw_fdt as usize
-}
-
-#[inline]
-pub fn get_image_address() -> usize {
-    payload_image as usize
-}

+ 124 - 0
prototyper/src/platform/dynamic.rs

@@ -0,0 +1,124 @@
+//! Frequently used first boot stage dynamic information on RISC-V.
+
+use core::ops::Range;
+use core::sync::atomic::{AtomicBool, Ordering};
+
+use super::BootInfo;
+use crate::fail;
+use crate::riscv_spec::current_hartid;
+
+use riscv::register::mstatus;
+
+pub fn get_boot_info(opaque: usize, nonstandard_a2: usize) -> BootInfo {
+    static GENESIS: AtomicBool = AtomicBool::new(true);
+    let info = read_paddr(nonstandard_a2).unwrap_or_else(fail::no_dynamic_info_available);
+    let is_boot_hart = if info.boot_hart == usize::MAX {
+        GENESIS.swap(false, Ordering::AcqRel)
+    } else {
+        current_hartid() == info.boot_hart
+    };
+    let (mpp, next_addr) = mpp_next_addr(&info).unwrap_or_else(fail::invalid_dynamic_data);
+    BootInfo {
+        fdt_address: opaque,
+        next_address: next_addr,
+        mpp,
+        is_boot_hart,
+    }
+}
+
+/// M-mode firmware dynamic information.
+#[derive(Clone, Copy)]
+#[repr(C)]
+pub struct DynamicInfo {
+    /// Dynamic information magic value.
+    pub magic: usize,
+    /// Version of dynamic information.
+    pub version: usize,
+    /// Address of the next boot-loading stage.
+    pub next_addr: usize,
+    /// RISC-V privilege mode of the next boot-loading stage.
+    pub next_mode: usize,
+    /// M-mode firmware options; its definition varies between SBI implementations.
+    pub options: usize,
+    /// Boot hart ID of current environment.
+    pub boot_hart: usize,
+}
+
+// Definition of `boot_hart` can be found at:
+// https://github.com/riscv-software-src/opensbi/blob/019a8e69a1dc0c0f011fabd0372e1ba80e40dd7c/include/sbi/fw_dynamic.h#L75
+
+const DYNAMIC_INFO_INVALID_ADDRESSES: usize = 0x00000000;
+const NEXT_ADDR_VALID_ADDRESSES: Range<usize> = 0x80000000..0x90000000;
+pub(crate) const MAGIC: usize = 0x4942534f;
+const SUPPORTED_VERSION: Range<usize> = 2..3;
+
+pub struct DynamicReadError {
+    pub bad_paddr: Option<usize>,
+    pub bad_magic: Option<usize>,
+    pub bad_version: Option<usize>,
+}
+
+// TODO unconstrained lifetime
+pub fn read_paddr(paddr: usize) -> Result<DynamicInfo, DynamicReadError> {
+    let mut error = DynamicReadError {
+        bad_paddr: None,
+        bad_magic: None,
+        bad_version: None,
+    };
+    // check pointer before dereference
+    if DYNAMIC_INFO_INVALID_ADDRESSES == paddr {
+        error.bad_paddr = Some(paddr);
+        return Err(error);
+    }
+    let ans = unsafe { *(paddr as *const DynamicInfo) };
+
+    if ans.magic != MAGIC {
+        error.bad_magic = Some(ans.magic);
+    }
+    if !SUPPORTED_VERSION.contains(&ans.version) {
+        error.bad_version = Some(ans.version);
+    }
+    if error.bad_magic.is_some() || error.bad_version.is_some() {
+        return Err(error);
+    }
+    Ok(ans)
+}
+
+pub struct DynamicError<'a> {
+    pub invalid_mpp: bool,
+    pub invalid_next_addr: bool,
+    pub bad_info: &'a DynamicInfo,
+}
+
+pub fn mpp_next_addr(info: &DynamicInfo) -> Result<(mstatus::MPP, usize), DynamicError> {
+    let mut error = DynamicError {
+        invalid_mpp: false,
+        invalid_next_addr: false,
+        bad_info: info,
+    };
+
+    // fail safe, errors will be aggregated after whole checking process.
+    let next_addr_valid = NEXT_ADDR_VALID_ADDRESSES.contains(&info.next_addr);
+    let mpp_valid = matches!(info.next_mode, 0 | 1 | 3);
+
+    if !next_addr_valid {
+        error.invalid_next_addr = true;
+    }
+    if !mpp_valid {
+        error.invalid_mpp = true;
+    }
+
+    if !next_addr_valid || !mpp_valid {
+        return Err(error);
+    }
+
+    let mpp = match info.next_mode {
+        3 => mstatus::MPP::Machine,
+        1 => mstatus::MPP::Supervisor,
+        // pattern `_` avoids `unreachable!`` which introduces panic handler.
+        // pattern 0 and _
+        _ => mstatus::MPP::User,
+    };
+
+    Ok((mpp, info.next_addr))
+}

+ 18 - 0
prototyper/src/platform/mod.rs

@@ -0,0 +1,18 @@
+#[cfg(not(feature = "payload"))]
+pub mod dynamic;
+#[cfg(feature = "payload")]
+pub mod payload;
+
+use riscv::register::mstatus;
+
+pub struct BootInfo {
+    pub next_address: usize,
+    pub fdt_address: usize,
+    pub is_boot_hart: bool,
+    pub mpp: mstatus::MPP,
+}
+
+#[cfg(not(feature = "payload"))]
+pub use dynamic::get_boot_info;
+#[cfg(feature = "payload")]
+pub use payload::get_boot_info;

+ 41 - 0
prototyper/src/platform/payload.rs

@@ -0,0 +1,41 @@
+use super::BootInfo;
+use core::arch::asm;
+
+use riscv::register::mstatus;
+
+pub fn get_boot_info(_opaque: usize, _nonstandard_a2: usize) -> BootInfo {
+    BootInfo {
+        fdt_address: get_fdt_address(),
+        next_address: get_image_address(),
+        mpp: mstatus::MPP::Supervisor,
+        is_boot_hart: true,
+    }
+}
+
+#[naked]
+#[link_section = ".fw_fdt"]
+pub unsafe extern "C" fn raw_fdt() {
+    asm!(
+        concat!(".incbin \"", env!("PROTOTYPER_FDT"), "\""),
+        options(noreturn)
+    );
+}
+
+#[naked]
+#[link_section = ".payload"]
+pub unsafe extern "C" fn payload_image() {
+    asm!(
+        concat!(".incbin \"", env!("PROTOTYPER_IMAGE"), "\""),
+        options(noreturn)
+    );
+}
+
+#[inline]
+fn get_fdt_address() -> usize {
+    raw_fdt as usize
+}
+
+#[inline]
+fn get_image_address() -> usize {
+    payload_image as usize
+}