|
@@ -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))
|
|
|
+}
|