sswi.rs 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. //! Supervisor-level Software Interrupt Device.
  2. pub use super::HartIdNumber;
  3. use crate::common::unsafe_peripheral;
  4. /// SSWI peripheral.
  5. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  6. #[repr(transparent)]
  7. pub struct SSWI {
  8. /// `SETSSIP` register for HART ID 0. In multi-HART architectures,
  9. /// use [`SSWI::setssip`] for accessing the `SETSSIP` of other HARTs.
  10. pub setssip0: SETSSIP,
  11. }
  12. impl SSWI {
  13. /// Creates a new `SSWI` peripheral from a base address.
  14. ///
  15. /// # Safety
  16. ///
  17. /// The base address must point to a valid `SSWI` peripheral.
  18. #[inline]
  19. pub const unsafe fn new(address: usize) -> Self {
  20. Self {
  21. setssip0: SETSSIP::new(address),
  22. }
  23. }
  24. /// Returns `true` if a supervisor software interrupt is pending.
  25. #[inline]
  26. pub fn is_interrupting() -> bool {
  27. riscv::register::sip::read().ssoft()
  28. }
  29. /// Returns `true` if Supervisor Software Interrupts are enabled.
  30. #[inline]
  31. pub fn is_enabled() -> bool {
  32. riscv::register::mie::read().ssoft()
  33. }
  34. /// Sets the Supervisor Software Interrupt bit of the `mie` CSR.
  35. /// This bit must be set for the `SSWI` to trigger supervisor software interrupts.
  36. ///
  37. /// # Safety
  38. ///
  39. /// Enabling the `SSWI` may break mask-based critical sections.
  40. #[inline]
  41. pub unsafe fn enable() {
  42. riscv::register::mie::set_ssoft();
  43. }
  44. /// Clears the Supervisor Software Interrupt bit of the `mie` CSR.
  45. /// When cleared, the `SSWI` cannot trigger supervisor software interrupts.
  46. #[inline]
  47. pub fn disable() {
  48. // SAFETY: it is safe to disable interrupts
  49. unsafe { riscv::register::mie::clear_ssoft() };
  50. }
  51. /// Returns the `SETSSIP` register for the HART which ID is `hart_id`.
  52. ///
  53. /// # Note
  54. ///
  55. /// For HART ID 0, you can simply use [`SSWI::setssip0`].
  56. #[inline]
  57. pub fn setssip<H: HartIdNumber>(&self, hart_id: H) -> SETSSIP {
  58. // SAFETY: `hart_id` is valid for the target
  59. unsafe { SETSSIP::new(self.setssip0.get_ptr().offset(hart_id.number() as _) as _) }
  60. }
  61. }
  62. unsafe_peripheral!(SETSSIP, u32, RW);
  63. impl SETSSIP {
  64. /// Returns `true` if a supervisor software interrupt is pending.
  65. #[inline]
  66. pub fn is_pending(self) -> bool {
  67. self.register.read() != 0
  68. }
  69. /// Writes to the register to trigger a supervisor software interrupt.
  70. #[inline]
  71. pub fn pend(self) {
  72. self.register.write(1);
  73. }
  74. /// Clears the register to unpend a supervisor software interrupt.
  75. #[inline]
  76. pub fn unpend(self) {
  77. self.register.write(0);
  78. }
  79. }
  80. #[cfg(test)]
  81. mod test {
  82. use super::super::test::HartId;
  83. use super::*;
  84. #[test]
  85. fn test_sswi() {
  86. // slice to emulate the interrupt pendings register
  87. let raw_reg = [0u32; HartId::MAX_HART_ID_NUMBER as usize + 1];
  88. // SAFETY: valid memory address
  89. let mswi = unsafe { SSWI::new(raw_reg.as_ptr() as _) };
  90. for i in 0..=HartId::MAX_HART_ID_NUMBER {
  91. let hart_id = HartId::from_number(i).unwrap();
  92. let setssip = mswi.setssip(hart_id);
  93. assert!(!setssip.is_pending());
  94. assert_eq!(raw_reg[i as usize], 0);
  95. setssip.pend();
  96. assert!(setssip.is_pending());
  97. assert_ne!(raw_reg[i as usize], 0);
  98. setssip.unpend();
  99. assert!(!setssip.is_pending());
  100. assert_eq!(raw_reg[i as usize], 0);
  101. }
  102. }
  103. }