macros.rs 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. /// Macro to create a mutable reference to a statically allocated value
  2. ///
  3. /// This macro returns a value with type `Option<&'static mut $ty>`. `Some($expr)` will be returned
  4. /// the first time the macro is executed; further calls will return `None`. To avoid `unwrap`ping a
  5. /// `None` variant the caller must ensure that the macro is called from a function that's executed
  6. /// at most once in the whole lifetime of the program.
  7. ///
  8. /// # Note
  9. ///
  10. /// This macro requires a `critical-section` implementation to be set. For most single-hart systems,
  11. /// you can enable the `critical-section-single-hart` feature for this crate. For other systems, you
  12. /// have to provide one from elsewhere, typically your chip's HAL crate.
  13. ///
  14. /// # Example
  15. ///
  16. /// ``` no_run
  17. /// use riscv::singleton;
  18. ///
  19. /// fn main() {
  20. /// // OK if `main` is executed only once
  21. /// let x: &'static mut bool = singleton!(: bool = false).unwrap();
  22. ///
  23. /// let y = alias();
  24. /// // BAD this second call to `alias` will definitively `panic!`
  25. /// let y_alias = alias();
  26. /// }
  27. ///
  28. /// fn alias() -> &'static mut bool {
  29. /// singleton!(: bool = false).unwrap()
  30. /// }
  31. /// ```
  32. #[macro_export]
  33. macro_rules! singleton {
  34. (: $ty:ty = $expr:expr) => {
  35. $crate::_export::critical_section::with(|_| {
  36. static mut VAR: Option<$ty> = None;
  37. #[allow(unsafe_code)]
  38. let used = unsafe { VAR.is_some() };
  39. if used {
  40. None
  41. } else {
  42. let expr = $expr;
  43. #[allow(unsafe_code)]
  44. unsafe {
  45. VAR = Some(expr)
  46. }
  47. #[allow(unsafe_code)]
  48. unsafe {
  49. VAR.as_mut()
  50. }
  51. }
  52. })
  53. };
  54. }
  55. /// Macro to create interfaces to PLIC contexts in PACs.
  56. ///
  57. /// This macro expects 5 arguments:
  58. ///
  59. /// - `PLIC`: name of the PLIC context interface structure to be created.
  60. /// We recommend to leave `PLIC` for context 0 and `PLICx` for the remaining contexts.
  61. ///
  62. /// - `BASE`: base address of the PLIC peripheral of the target.
  63. ///
  64. /// - `CONTEXT`: context number assigned to the PLIC interface.
  65. ///
  66. /// - `INTERRUPT`: enum type of the external interruptions of the target.
  67. /// This type must implement the [`crate::peripheral::plic::InterruptNumber`] trait.
  68. ///
  69. /// - `PRIORITY`: enum type of the priority levels supported by the target.
  70. /// This type must implement the [`crate::peripheral::plic::PriorityNumber`] trait.
  71. ///
  72. /// # Note
  73. ///
  74. /// This macro requires the `plic` feature to be active.
  75. #[cfg(feature = "plic")]
  76. #[macro_export]
  77. macro_rules! plic_context {
  78. ($PLIC:ident, $BASE:literal, $CONTEXT:literal, $INTERRUPT:ident, $PRIORITY:ident) => {
  79. /// Platform-Level Interrupt Controller (PLIC) context.
  80. #[repr(transparent)]
  81. pub struct $PLIC {
  82. context: $crate::peripheral::PLIC<$BASE, $CONTEXT>,
  83. }
  84. impl $PLIC {
  85. /// Creates a new PLIC context interface.
  86. pub const fn new() -> Self {
  87. Self {
  88. context: $crate::peripheral::PLIC::new(),
  89. }
  90. }
  91. /// Enables machine external interrupts.
  92. #[inline(always)]
  93. pub fn enable() {
  94. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::enable();
  95. }
  96. /// Disables machine external interrupts.
  97. #[inline(always)]
  98. pub fn disable() {
  99. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::disable();
  100. }
  101. /// Returns the priority level associated to a given interrupt source.
  102. #[inline(always)]
  103. pub fn priority(source: $INTERRUPT) -> $PRIORITY {
  104. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::priority(source)
  105. }
  106. /// Getter method for the priority level associated to a given interrupt source.
  107. #[inline(always)]
  108. pub fn get_priority(&self, source: $INTERRUPT) -> $PRIORITY {
  109. Self::priority(source)
  110. }
  111. /// Sets the priority level of a given interrupt source.
  112. ///
  113. /// # Note
  114. ///
  115. /// Interrupt source priorities are shared among all the contexts of the PLIC.
  116. /// Thus, changing the priority of sources may affect other PLIC contexts.
  117. ///
  118. /// # Safety
  119. ///
  120. /// Changing priority levels can break priority-based critical sections and compromise memory safety.
  121. #[inline(always)]
  122. pub unsafe fn set_priority(&mut self, source: $INTERRUPT, priority: $PRIORITY) {
  123. self.context.set_priority(source, priority);
  124. }
  125. /// Checks if an interrupt triggered by a given source is pending.
  126. #[inline(always)]
  127. pub fn is_interrupt_pending(source: $INTERRUPT) -> bool {
  128. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::is_interrupt_pending(source)
  129. }
  130. /// Checks if an interrupt source is enabled for the PLIC context.
  131. #[inline(always)]
  132. pub fn is_interrupt_enabled(source: $INTERRUPT) -> bool {
  133. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::is_interrupt_enabled(source)
  134. }
  135. /// Enables an interrupt source for the PLIC context.
  136. ///
  137. /// # Safety
  138. ///
  139. /// It performs non-atomic read-modify-write operations, which may lead to undefined behavior.
  140. /// Additionally, Enabling an interrupt source can break mask-based critical sections.
  141. #[inline(always)]
  142. pub unsafe fn enable_interrupt(&mut self, source: $INTERRUPT) {
  143. self.context.enable_interrupt(source);
  144. }
  145. /// Disables an interrupt source for the PLIC context.
  146. ///
  147. /// # Safety
  148. ///
  149. /// It performs non-atomic read-modify-write operations, which may lead to undefined behavior.
  150. #[inline(always)]
  151. pub unsafe fn disable_interrupt(&mut self, source: $INTERRUPT) {
  152. self.context.disable_interrupt(source);
  153. }
  154. /// Returns the priority threshold of the PLIC context.
  155. #[inline(always)]
  156. pub fn threshold() -> $PRIORITY {
  157. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::threshold()
  158. }
  159. /// Getter method for the priority threshold of the PLIC context.
  160. #[inline(always)]
  161. pub fn get_threshold(&self) -> $PRIORITY {
  162. Self::threshold()
  163. }
  164. /// Sets the priority threshold for for the PLIC context.
  165. ///
  166. /// # Safety
  167. ///
  168. /// Unmasking an interrupt source can break mask-based critical sections.
  169. #[inline(always)]
  170. pub unsafe fn set_threshold(&mut self, priority: $PRIORITY) {
  171. self.context.set_threshold(priority);
  172. }
  173. /// Claims the number of a pending interrupt for for the PLIC context.
  174. /// If no interrupt is pending for this context, it returns [`None`].
  175. #[inline(always)]
  176. pub fn claim() -> Option<$INTERRUPT> {
  177. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::claim()
  178. }
  179. /// Marks a pending interrupt as complete from for the PLIC context.
  180. #[inline(always)]
  181. pub fn complete(source: $INTERRUPT) {
  182. $crate::peripheral::PLIC::<$BASE, $CONTEXT>::complete(source);
  183. }
  184. /// Resets the PLIC peripherals.
  185. ///
  186. /// # Safety
  187. ///
  188. /// It performs non-atomic read-modify-write operations, which may lead to undefined behavior.
  189. #[inline(always)]
  190. pub unsafe fn reset(&mut self) {
  191. self.context.reset::<$INTERRUPT, $PRIORITY>();
  192. }
  193. }
  194. };
  195. }