srst.rs 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. //! Chapter 10. System Reset Extension (EID #0x53525354 "SRST")
  2. use crate::binary::sbi_call_2;
  3. use sbi_spec::{
  4. binary::SbiRet,
  5. srst::{
  6. EID_SRST, RESET_REASON_NO_REASON, RESET_REASON_SYSTEM_FAILURE, RESET_TYPE_COLD_REBOOT,
  7. RESET_TYPE_SHUTDOWN, RESET_TYPE_WARM_REBOOT, SYSTEM_RESET,
  8. },
  9. };
  10. /// Reset the system based on provided `reset_type` and `reset_reason`.
  11. ///
  12. /// This is a synchronous call and does not return if it succeeds.
  13. ///
  14. /// # Warm reboot and cold reboot
  15. ///
  16. /// When supervisor software is running natively, the SBI implementation is machine mode firmware.
  17. /// In this case, shutdown is equivalent to physical power down of the entire system, and
  18. /// cold reboot is equivalent to a physical power cycle of the entire system.
  19. /// Further, warm reboot is equivalent to a power cycle of the main processor and parts of the system
  20. /// 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. /// This function is defined in RISC-V SBI Specification chapter 10.1.
  30. #[inline]
  31. #[doc(alias = "sbi_system_reset")]
  32. pub fn system_reset<T, R>(reset_type: T, reset_reason: R) -> SbiRet
  33. where
  34. T: ResetType,
  35. R: ResetReason,
  36. {
  37. sbi_call_2(
  38. EID_SRST,
  39. SYSTEM_RESET,
  40. reset_type.raw() as _,
  41. reset_reason.raw() as _,
  42. )
  43. }
  44. /// A valid type for system reset.
  45. pub trait ResetType {
  46. /// Get a raw value to pass to SBI environment.
  47. fn raw(&self) -> u32;
  48. }
  49. #[cfg(feature = "integer-impls")]
  50. impl ResetType for u32 {
  51. #[inline]
  52. fn raw(&self) -> u32 {
  53. *self
  54. }
  55. }
  56. #[cfg(feature = "integer-impls")]
  57. impl ResetType for i32 {
  58. #[inline]
  59. fn raw(&self) -> u32 {
  60. u32::from_ne_bytes(i32::to_ne_bytes(*self))
  61. }
  62. }
  63. /// A valid reason for system reset.
  64. pub trait ResetReason {
  65. /// Get a raw value to pass to SBI environment.
  66. fn raw(&self) -> u32;
  67. }
  68. #[cfg(feature = "integer-impls")]
  69. impl ResetReason for u32 {
  70. #[inline]
  71. fn raw(&self) -> u32 {
  72. *self
  73. }
  74. }
  75. #[cfg(feature = "integer-impls")]
  76. impl ResetReason for i32 {
  77. #[inline]
  78. fn raw(&self) -> u32 {
  79. u32::from_ne_bytes(i32::to_ne_bytes(*self))
  80. }
  81. }
  82. macro_rules! define_reset_param {
  83. ($($struct:ident($value:expr): $trait:ident #[$doc:meta])*) => {
  84. $(
  85. #[derive(Clone, Copy, Debug)]
  86. #[$doc]
  87. pub struct $struct;
  88. impl $trait for $struct {
  89. #[inline]
  90. fn raw(&self) -> u32 {
  91. $value
  92. }
  93. }
  94. )*
  95. };
  96. }
  97. define_reset_param! {
  98. Shutdown(RESET_TYPE_SHUTDOWN): ResetType /// Shutdown as a reset type.
  99. ColdReboot(RESET_TYPE_COLD_REBOOT): ResetType /// Cold reboot as a reset type.
  100. WarmReboot(RESET_TYPE_WARM_REBOOT): ResetType /// Warm reboot as a reset type.
  101. NoReason(RESET_REASON_NO_REASON): ResetReason /// No reason as a reset reason.
  102. SystemFailure(RESET_REASON_SYSTEM_FAILURE): ResetReason /// System failure as a reset reason.
  103. }