hsm.rs 9.4 KB

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