hsm.rs 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. //! Chapter 9. Hart State Management Extension (EID #0x48534D "HSM")
  2. use crate::binary::{sbi_call_0, sbi_call_1, sbi_call_3};
  3. use sbi_spec::{
  4. binary::SbiRet,
  5. hsm::{EID_HSM, HART_GET_STATUS, HART_START, HART_STOP, HART_SUSPEND},
  6. };
  7. /// Start executing the given hart at specified address in supervisor-mode.
  8. ///
  9. /// This call is asynchronous - more specifically, the `hart_start()` may return before target hart starts
  10. /// executing as long as the SBI implementation is capable of ensuring the return code is accurate.
  11. ///
  12. /// It is recommended that if the SBI implementation is a platform runtime firmware executing in
  13. /// machine-mode (M-mode), then it MUST configure PMP and other the M-mode state before executing
  14. /// in supervisor-mode.
  15. ///
  16. /// # Parameters
  17. ///
  18. /// - The `hartid` parameter specifies the target hart, which is to be started.
  19. /// - The `start_addr` parameter points to a runtime-specified physical address,
  20. /// where the hart can start executing in supervisor-mode.
  21. /// - The `opaque` parameter is a `usize` value that will be set in the `a1`
  22. /// register when the hart starts executing at `start_addr`.
  23. ///
  24. /// *NOTE:* A single `usize` parameter is sufficient as `start_addr`,
  25. /// because the hart will start execution in the supervisor-mode with MMU off,
  26. /// hence the `start_addr` must be less than XLEN bits wide.
  27. ///
  28. /// # Behavior
  29. ///
  30. /// The target hart jumps to supervisor mode at the address specified by `start_addr`
  31. /// with the following values in specific registers.
  32. ///
  33. /// | Register Name | Register Value
  34. /// |:--------------|:--------------
  35. /// | `satp` | 0
  36. /// | `sstatus.SIE` | 0
  37. /// | a0 | hartid
  38. /// | a1 | `opaque` parameter
  39. ///
  40. /// # Return value
  41. ///
  42. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  43. ///
  44. /// | Return code | Description
  45. /// |:------------------------------|:----------------------------------------------
  46. /// | `SbiRet::success()` | Hart was previously in stopped state.
  47. /// It will start executing from `start_addr`.
  48. /// | `SbiRet::invalid_address()` | `start_addr` is not valid, possibly due to the following reasons:
  49. /// it is not a valid physical address,
  50. /// or the address is prohibited by PMP or H-extension G-stage to run in supervisor-mode.
  51. /// | `SbiRet::invalid_param()` | `hartid`
  52. /// is not a valid hartid as corresponding hart cannot be started in supervisor mode.
  53. /// | `SbiRet::already_available()` | The given hartid is already started.
  54. /// | `SbiRet::failed()` | The start request failed for unknown reasons.
  55. ///
  56. /// This function is defined in RISC-V SBI Specification chapter 9.1.
  57. #[inline]
  58. #[doc(alias = "sbi_hart_start")]
  59. pub fn hart_start(hartid: usize, start_addr: usize, opaque: usize) -> SbiRet {
  60. sbi_call_3(EID_HSM, HART_START, hartid, start_addr, opaque)
  61. }
  62. /// Stop executing the calling hart in supervisor-mode.
  63. ///
  64. /// This function requests the SBI implementation to stop executing the calling hart in
  65. /// supervisor-mode and return its ownership to the SBI implementation.
  66. ///
  67. /// This call is not expected to return under normal conditions.
  68. /// The `sbi_hart_stop()` must be called with the supervisor-mode interrupts disabled.
  69. ///
  70. /// # Return value
  71. ///
  72. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  73. ///
  74. /// | Error code | Description
  75. /// |:------------|:------------
  76. /// | `SbiRet::failed()` | Failed to stop execution of the current hart
  77. ///
  78. /// This function is defined in RISC-V SBI Specification chapter 9.2.
  79. #[inline]
  80. #[doc(alias = "sbi_hart_stop")]
  81. pub fn hart_stop() -> SbiRet {
  82. sbi_call_0(EID_HSM, HART_STOP)
  83. }
  84. /// Get the current status (or HSM state id) of the given hart.
  85. ///
  86. /// The harts may transition HSM states at any time due to any concurrent `hart_start()`
  87. /// or `hart_stop()` calls, the return value from this function may not represent the actual state
  88. /// of the hart at the time of return value verification.
  89. ///
  90. /// # Parameters
  91. ///
  92. /// The `hartid` parameter specifies the target hart which status is required.
  93. ///
  94. /// # Return value
  95. ///
  96. /// The possible status values returned in `SbiRet.value` are shown in the table below:
  97. ///
  98. /// | Name | Value | Description
  99. /// |:--------------|:------|:-------------------------
  100. /// | STARTED | 0 | Hart Started
  101. /// | STOPPED | 1 | Hart Stopped
  102. /// | START_PENDING | 2 | Hart start request pending
  103. /// | STOP_PENDING | 3 | Hart stop request pending
  104. ///
  105. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  106. ///
  107. /// | Error code | Description
  108. /// |:--------------------------|:------------
  109. /// | `SbiRet::invalid_param()` | The given `hartid` is not valid
  110. ///
  111. /// This function is defined in RISC-V SBI Specification chapter 9.3.
  112. #[inline]
  113. #[doc(alias = "sbi_hart_get_status")]
  114. pub fn hart_get_status(hartid: usize) -> SbiRet {
  115. sbi_call_1(EID_HSM, HART_GET_STATUS, hartid)
  116. }
  117. /// Put the calling hart into suspend or platform specific lower power states.
  118. ///
  119. /// This function requests the SBI implementation to put the calling hart in a platform specfic suspend
  120. /// (or low power) state specified by the `suspend_type` parameter.
  121. ///
  122. /// The hart will automatically come out of suspended state and resume normal execution
  123. /// when it receives an interrupt or platform specific hardware event.
  124. ///
  125. /// # Suspend behavior
  126. ///
  127. /// The platform-specific suspend states for a hart can be either retentive or non-retentive in nature.
  128. ///
  129. /// A retentive suspend state will preserve hart register and CSR values for all privilege modes,
  130. /// whereas a non-retentive suspend state will not preserve hart register and CSR values.
  131. ///
  132. /// # Resuming
  133. ///
  134. /// Resuming from a retentive suspend state is straight forward, and the supervisor-mode software
  135. /// will see SBI suspend call return without any failures.
  136. ///
  137. /// Resuming from a non-retentive suspend state is relatively more involved, and it requires software
  138. /// to restore various hart registers and CSRs for all privilege modes.
  139. /// Upon resuming from non-retentive suspend state, the hart will jump to supervisor-mode at address
  140. /// specified by `resume_addr` with specific registers values described in the table below:
  141. ///
  142. /// | Register Name | Register Value
  143. /// |:--------------|:--------------
  144. /// | `satp` | 0
  145. /// | `sstatus.SIE` | 0
  146. /// | a0 | hartid
  147. /// | a1 | `opaque` parameter
  148. ///
  149. /// # Parameters
  150. ///
  151. /// The `suspend_type` parameter is 32 bits wide, and the possible values are shown in the table below:
  152. ///
  153. /// | Value | Description
  154. /// |:------------------------|:--------------
  155. /// | 0x00000000 | Default retentive suspend
  156. /// | 0x00000001 - 0x0FFFFFFF | _Reserved for future use_
  157. /// | 0x10000000 - 0x7FFFFFFF | Platform specific retentive suspend
  158. /// | 0x80000000 | Default non-retentive suspend
  159. /// | 0x80000001 - 0x8FFFFFFF | _Reserved for future use_
  160. /// | 0x90000000 - 0xFFFFFFFF | Platform specific non-retentive suspend
  161. /// | > 0xFFFFFFFF | _Reserved_
  162. ///
  163. /// The `resume_addr` parameter points to a runtime-specified physical address,
  164. /// where the hart can resume execution in supervisor-mode after a non-retentive
  165. /// suspend.
  166. ///
  167. /// *NOTE:* A single `usize` parameter is sufficient as `resume_addr`,
  168. /// because the hart will resume execution in the supervisor-mode with MMU off,
  169. /// hence the `resume_addr` must be less than XLEN bits wide.
  170. ///
  171. /// The `opaque` parameter is an XLEN-bit value that will be set in the `a1`
  172. /// register when the hart resumes execution at `resume_addr` after a
  173. /// non-retentive suspend.
  174. ///
  175. /// # Return value
  176. ///
  177. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  178. ///
  179. /// | Error code | Description
  180. /// |:----------------------------|:------------
  181. /// | `SbiRet::success()` | Hart has been suspended and resumed back successfully from a retentive suspend state.
  182. /// | `SbiRet::invalid_param()` | `suspend_type` is not valid.
  183. /// | `SbiRet::not_supported()` | `suspend_type` is valid but not implemented.
  184. /// | `SbiRet::invalid_address()` | `resume_addr` is not valid, possibly due to the following reasons: it is not a valid physical address, or the address is prohibited by PMP or H-extension G-stage to run in supervisor-mode.
  185. /// | `SbiRet::failed()` | The suspend request failed for unknown reasons.
  186. ///
  187. /// This function is defined in RISC-V SBI Specification chapter 9.4.
  188. #[inline]
  189. #[doc(alias = "sbi_hart_suspend")]
  190. pub fn hart_suspend<T>(suspend_type: T, resume_addr: usize, opaque: usize) -> SbiRet
  191. where
  192. T: SuspendType,
  193. {
  194. sbi_call_3(
  195. EID_HSM,
  196. HART_SUSPEND,
  197. suspend_type.raw() as _,
  198. resume_addr,
  199. opaque,
  200. )
  201. }
  202. /// A valid suspend type for hart state monitor.
  203. pub trait SuspendType {
  204. /// Get a raw value to pass to SBI environment.
  205. fn raw(&self) -> u32;
  206. }
  207. #[cfg(feature = "integer-impls")]
  208. impl SuspendType for u32 {
  209. #[inline]
  210. fn raw(&self) -> u32 {
  211. *self
  212. }
  213. }
  214. macro_rules! define_suspend_type {
  215. ($($struct:ident($value:expr) #[$doc:meta])*) => {
  216. $(
  217. #[derive(Clone, Copy, Debug)]
  218. #[$doc]
  219. pub struct $struct;
  220. impl SuspendType for $struct {
  221. #[inline]
  222. fn raw(&self) -> u32 {
  223. $value
  224. }
  225. }
  226. )*
  227. };
  228. }
  229. define_suspend_type! {
  230. Retentive(sbi_spec::hsm::suspend_type::RETENTIVE) /// Default retentive hart suspension.
  231. NonRetentive(sbi_spec::hsm::suspend_type::NON_RETENTIVE) /// Default non-retentive hart suspension.
  232. }