srst.rs 3.2 KB

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