reset.rs 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. use sbi_spec::binary::SbiRet;
  2. /// System Reset Extension
  3. ///
  4. /// Provides a function that allow the supervisor software to request system-level reboot or shutdown.
  5. ///
  6. /// The term "system" refers to the world-view of supervisor software and the underlying SBI implementation
  7. /// could be machine mode firmware or hypervisor.
  8. ///
  9. /// Ref: [Section 9, RISC-V Supervisor Binary Interface Specification](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/riscv-sbi.adoc#9-system-reset-extension-eid-0x53525354-srst)
  10. pub trait Reset: Send + Sync {
  11. /// Reset the system based on provided `reset_type` and `reset_reason`.
  12. ///
  13. /// This is a synchronous call and does not return if it succeeds.
  14. ///
  15. /// # Warm reboot and cold reboot
  16. ///
  17. /// When supervisor software is running natively, the SBI implementation is machine mode firmware.
  18. /// In this case, shutdown is equivalent to physical power down of the entire system and
  19. /// cold reboot is equivalent to physical power cycle of the entire system. Further, warm reboot
  20. /// is equivalent to a power cycle of main processor and parts of the system but not the entire system.
  21. ///
  22. /// For example, on a server class system with a BMC (board management controller),
  23. /// a warm reboot will not power cycle the BMC whereas a cold reboot will definitely power cycle the BMC.
  24. ///
  25. /// When supervisor software is running inside a virtual machine, the SBI implementation is a hypervisor.
  26. /// The shutdown, cold reboot and warm reboot will behave functionally the same as the native case but might
  27. /// not result in any physical power changes.
  28. ///
  29. /// # Return value
  30. ///
  31. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  32. ///
  33. /// | Error code | Description
  34. /// |:----------------------|:---------------
  35. /// | SBI_ERR_INVALID_PARAM | `reset_type` or `reset_reason` is not valid.
  36. /// | SBI_ERR_NOT_SUPPORTED | `reset_type` is valid but not implemented.
  37. /// | SBI_ERR_FAILED | Reset request failed for unknown reasons.
  38. fn system_reset(&self, reset_type: u32, reset_reason: u32) -> SbiRet;
  39. /// Legacy extension's reset function
  40. ///
  41. /// Puts all the harts to shut down state from supervisor point of view. This SBI call doesn’t return.
  42. #[cfg(feature = "legacy")]
  43. fn legacy_reset(&self) -> ! {
  44. use sbi_spec::srst::{RESET_REASON_NO_REASON, RESET_TYPE_SHUTDOWN};
  45. // By default, this function redirects to `system_reset`.
  46. self.system_reset(RESET_TYPE_SHUTDOWN, RESET_REASON_NO_REASON);
  47. unreachable!()
  48. }
  49. }
  50. use crate::util::AmoOnceRef;
  51. static RESET: AmoOnceRef<dyn Reset> = AmoOnceRef::new();
  52. /// Init SRST module
  53. pub fn init_reset(reset: &'static dyn Reset) {
  54. if !RESET.try_call_once(reset) {
  55. panic!("load sbi module when already loaded")
  56. }
  57. }
  58. #[inline]
  59. pub(crate) fn probe_reset() -> bool {
  60. RESET.get().is_some()
  61. }
  62. #[inline]
  63. pub(crate) fn system_reset(reset_type: u32, reset_reason: u32) -> SbiRet {
  64. if let Some(obj) = RESET.get() {
  65. return obj.system_reset(reset_type, reset_reason);
  66. }
  67. SbiRet::not_supported()
  68. }
  69. #[cfg(feature = "legacy")]
  70. #[inline]
  71. pub(crate) fn legacy_reset() -> ! {
  72. if let Some(obj) = RESET.get() {
  73. obj.legacy_reset()
  74. }
  75. unreachable!("no reset handler available; this is okay if your platform didn't declare a legacy reset handler")
  76. }