dynamic.rs 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. //! Frequently used first boot stage dynamic information on RISC-V.
  2. use core::ops::Range;
  3. use core::sync::atomic::{AtomicBool, Ordering};
  4. use super::BootInfo;
  5. use crate::fail;
  6. use crate::riscv_spec::current_hartid;
  7. use riscv::register::mstatus;
  8. /// Determine whether the current hart is boot hart.
  9. ///
  10. /// Return true if the current hart is boot hart.
  11. pub fn is_boot_hart(nonstandard_a2: usize) -> bool {
  12. // Track whether this is the first hart to boot
  13. static GENESIS: AtomicBool = AtomicBool::new(true);
  14. let info = read_paddr(nonstandard_a2).unwrap_or_else(fail::use_lottery);
  15. // Determine if this is the boot hart based on hart ID
  16. if info.boot_hart == usize::MAX {
  17. // If boot_hart is MAX, use atomic bool to determine first hart
  18. GENESIS.swap(false, Ordering::AcqRel)
  19. } else {
  20. // Otherwise check if current hart matches designated boot hart
  21. current_hartid() == info.boot_hart
  22. }
  23. }
  24. /// Gets boot information from nonstandard_a2 parameter.
  25. ///
  26. /// Returns BootInfo containing next stage address and privilege mode.
  27. pub fn get_boot_info(nonstandard_a2: usize) -> BootInfo {
  28. let dynamic_info = read_paddr(nonstandard_a2).unwrap_or_else(fail::no_dynamic_info_available);
  29. let (mpp, next_addr) = mpp_next_addr(&dynamic_info).unwrap_or_else(fail::invalid_dynamic_data);
  30. BootInfo {
  31. next_address: next_addr,
  32. mpp,
  33. }
  34. }
  35. /// M-mode firmware dynamic information.
  36. #[derive(Clone, Copy)]
  37. #[repr(C)]
  38. pub struct DynamicInfo {
  39. /// Dynamic information magic value.
  40. pub magic: usize,
  41. /// Version of dynamic information.
  42. pub version: usize,
  43. /// Address of the next boot-loading stage.
  44. pub next_addr: usize,
  45. /// RISC-V privilege mode of the next boot-loading stage.
  46. pub next_mode: usize,
  47. /// M-mode firmware options; its definition varies between SBI implementations.
  48. pub options: usize,
  49. /// Boot hart ID of current environment.
  50. pub boot_hart: usize,
  51. }
  52. // Definition of `boot_hart` can be found at:
  53. // https://github.com/riscv-software-src/opensbi/blob/019a8e69a1dc0c0f011fabd0372e1ba80e40dd7c/include/sbi/fw_dynamic.h#L75
  54. const DYNAMIC_INFO_INVALID_ADDRESSES: usize = 0x00000000;
  55. const NEXT_ADDR_VALID_ADDRESSES: Range<usize> = 0x80000000..0x90000000;
  56. pub(crate) const MAGIC: usize = 0x4942534f;
  57. const SUPPORTED_VERSION: Range<usize> = 0..3;
  58. /// Error type for dynamic info read failures.
  59. pub struct DynamicReadError {
  60. pub bad_paddr: Option<usize>,
  61. pub bad_magic: Option<usize>,
  62. pub bad_version: Option<usize>,
  63. }
  64. // TODO: unconstrained lifetime
  65. /// Reads dynamic info from physical address.
  66. ///
  67. /// Returns Result containing DynamicInfo or error details.
  68. pub fn read_paddr(paddr: usize) -> Result<DynamicInfo, DynamicReadError> {
  69. let mut error = DynamicReadError {
  70. bad_paddr: None,
  71. bad_magic: None,
  72. bad_version: None,
  73. };
  74. // check pointer before dereference.
  75. if DYNAMIC_INFO_INVALID_ADDRESSES == paddr {
  76. error.bad_paddr = Some(paddr);
  77. return Err(error);
  78. }
  79. let ans = unsafe { *(paddr as *const DynamicInfo) };
  80. // Validate magic number and version.
  81. if ans.magic != MAGIC {
  82. error.bad_magic = Some(ans.magic);
  83. }
  84. if !SUPPORTED_VERSION.contains(&ans.version) {
  85. error.bad_version = Some(ans.version);
  86. }
  87. if error.bad_magic.is_some() || error.bad_version.is_some() {
  88. return Err(error);
  89. }
  90. Ok(ans)
  91. }
  92. /// Error type for dynamic info validation failures.
  93. pub struct DynamicError<'a> {
  94. pub invalid_mpp: bool,
  95. pub invalid_next_addr: bool,
  96. pub bad_info: &'a DynamicInfo,
  97. }
  98. /// Validates and extracts privilege mode and next address from dynamic info.
  99. ///
  100. /// Returns Result containing tuple of (MPP, next_addr) or error details.
  101. pub fn mpp_next_addr(info: &DynamicInfo) -> Result<(mstatus::MPP, usize), DynamicError> {
  102. let mut error = DynamicError {
  103. invalid_mpp: false,
  104. invalid_next_addr: false,
  105. bad_info: info,
  106. };
  107. // fail safe, errors will be aggregated after whole checking process.
  108. let next_addr_valid = NEXT_ADDR_VALID_ADDRESSES.contains(&info.next_addr);
  109. let mpp_valid = matches!(info.next_mode, 0 | 1 | 3);
  110. if !next_addr_valid {
  111. error.invalid_next_addr = true;
  112. }
  113. if !mpp_valid {
  114. error.invalid_mpp = true;
  115. }
  116. if !next_addr_valid || !mpp_valid {
  117. return Err(error);
  118. }
  119. let mpp = match info.next_mode {
  120. 3 => mstatus::MPP::Machine,
  121. 1 => mstatus::MPP::Supervisor,
  122. // pattern `_` avoids `unreachable!`` which introduces panic handler.
  123. // pattern 0 and _
  124. _ => mstatus::MPP::User,
  125. };
  126. Ok((mpp, info.next_addr))
  127. }