plic.rs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. //! Platform-Level Interrupt Controller (PLIC) peripheral.
  2. //!
  3. //! Specification: <https://github.com/riscv/riscv-plic-spec/blob/master/riscv-plic.adoc>
  4. pub mod claim;
  5. pub mod enables;
  6. pub mod pendings;
  7. pub mod priorities;
  8. pub mod threshold;
  9. pub use riscv_pac::{HartIdNumber, InterruptNumber, PriorityNumber}; // re-export useful riscv-pac traits
  10. /// Trait for a PLIC peripheral.
  11. ///
  12. /// # Safety
  13. ///
  14. /// * This trait must only be implemented on a PAC of a target with a PLIC peripheral.
  15. /// * The PLIC peripheral base address `BASE` must be valid for the target device.
  16. pub unsafe trait Plic: Copy {
  17. /// Base address of the PLIC peripheral.
  18. const BASE: usize;
  19. }
  20. /// Platform-Level Interrupt Controler (PLIC) peripheral.
  21. ///
  22. /// The RISC-V standard does not specify a fixed location for the PLIC.
  23. /// Thus, each platform must specify the base address of the PLIC on the platform.
  24. /// The base address, as well as all the associated types, are defined in the [`Plic`] trait.
  25. ///
  26. /// The PLIC standard allows up to 15_872 different contexts for interfacing the PLIC.
  27. /// Each context has an assigned index starting from 0 to up to 15_871.
  28. /// Usually, each HART uses a dedicated context. In this way, they do not interfere
  29. /// with each other when attending to external interruptions.
  30. #[allow(clippy::upper_case_acronyms)]
  31. #[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
  32. pub struct PLIC<P: Plic> {
  33. _marker: core::marker::PhantomData<P>,
  34. }
  35. impl<P: Plic> PLIC<P> {
  36. const PRIORITIES_OFFSET: usize = 0;
  37. const PENDINGS_OFFSET: usize = 0x1000;
  38. /// Returns the priorities register of the PLIC.
  39. /// This register allows to set the priority level of each interrupt source.
  40. /// The priority level of each interrupt source is shared among all the contexts.
  41. #[inline]
  42. pub fn priorities() -> priorities::PRIORITIES {
  43. // SAFETY: valid address
  44. unsafe { priorities::PRIORITIES::new(P::BASE + Self::PRIORITIES_OFFSET) }
  45. }
  46. /// Returns the pendings register of the PLIC.
  47. /// This register allows to check if a particular interrupt source is pending.
  48. #[inline]
  49. pub fn pendings() -> pendings::PENDINGS {
  50. // SAFETY: valid address
  51. unsafe { pendings::PENDINGS::new(P::BASE + Self::PENDINGS_OFFSET) }
  52. }
  53. /// Returns a proxy to access to all the PLIC registers of a given HART context.
  54. #[inline]
  55. pub fn ctx<H: HartIdNumber>(hart_id: H) -> CTX<P> {
  56. // SAFETY: valid context number
  57. unsafe { CTX::new(hart_id.number()) }
  58. }
  59. /// Returns the PLIC HART context for the current HART.
  60. ///
  61. /// # Note
  62. ///
  63. /// This function determines the current HART ID by reading the [`riscv::register::mhartid`] CSR.
  64. /// Thus, it can only be used in M-mode. For S-mode, use [`PLIC::ctx`] instead.
  65. #[inline]
  66. pub fn ctx_mhartid() -> CTX<P> {
  67. let hart_id = riscv::register::mhartid::read();
  68. // SAFETY: `hart_id` is valid for the target and is the current hart
  69. unsafe { CTX::new(hart_id as _) }
  70. }
  71. }
  72. /// PLIC context proxy. It provides access to the PLIC registers of a given context.
  73. #[allow(clippy::upper_case_acronyms)]
  74. #[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
  75. pub struct CTX<P: Plic> {
  76. context: usize,
  77. _marker: core::marker::PhantomData<P>,
  78. }
  79. impl<P: Plic> CTX<P> {
  80. const ENABLES_OFFSET: usize = 0x2000;
  81. const ENABLES_SEPARATION: usize = 0x80;
  82. const THRESHOLDS_OFFSET: usize = 0x20_0000;
  83. const THRESHOLDS_SEPARATION: usize = 0x1000;
  84. const CLAIMS_OFFSET: usize = 0x20_0004;
  85. const CLAIMS_SEPARATION: usize = 0x1000;
  86. /// Creates a new PLIC context proxy
  87. ///
  88. /// # Safety
  89. ///
  90. /// The context number must be valid for the target device.
  91. #[inline]
  92. pub(crate) unsafe fn new(context: u16) -> Self {
  93. Self {
  94. context: context as _,
  95. _marker: core::marker::PhantomData,
  96. }
  97. }
  98. /// Returns the context number of this proxy.
  99. #[inline]
  100. pub const fn context(self) -> u16 {
  101. self.context as _
  102. }
  103. /// Returns the interrupts enable register of the context.
  104. #[inline]
  105. pub const fn enables(self) -> enables::ENABLES {
  106. let addr = P::BASE + Self::ENABLES_OFFSET + self.context * Self::ENABLES_SEPARATION;
  107. // SAFETY: valid address
  108. unsafe { enables::ENABLES::new(addr) }
  109. }
  110. /// Returns the interrupt threshold register of the context.
  111. #[inline]
  112. pub const fn threshold(self) -> threshold::THRESHOLD {
  113. let addr = P::BASE + Self::THRESHOLDS_OFFSET + self.context * Self::THRESHOLDS_SEPARATION;
  114. // SAFETY: valid address
  115. unsafe { threshold::THRESHOLD::new(addr) }
  116. }
  117. /// Returns the interrupt claim/complete register of the context.
  118. #[inline]
  119. pub const fn claim(self) -> claim::CLAIM {
  120. let addr = P::BASE + Self::CLAIMS_OFFSET + self.context * Self::CLAIMS_SEPARATION;
  121. // SAFETY: valid address
  122. unsafe { claim::CLAIM::new(addr) }
  123. }
  124. }
  125. #[cfg(test)]
  126. pub(crate) mod test {
  127. use super::{HartIdNumber, InterruptNumber, PriorityNumber};
  128. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  129. #[repr(u16)]
  130. pub(crate) enum Interrupt {
  131. I1 = 1,
  132. I2 = 2,
  133. I3 = 3,
  134. I4 = 4,
  135. }
  136. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  137. #[repr(u8)]
  138. pub(crate) enum Priority {
  139. P0 = 0,
  140. P1 = 1,
  141. P2 = 2,
  142. P3 = 3,
  143. }
  144. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  145. #[repr(u16)]
  146. pub(crate) enum Context {
  147. C0 = 0,
  148. C1 = 1,
  149. C2 = 2,
  150. }
  151. unsafe impl InterruptNumber for Interrupt {
  152. const MAX_INTERRUPT_NUMBER: u16 = 4;
  153. #[inline]
  154. fn number(self) -> u16 {
  155. self as _
  156. }
  157. #[inline]
  158. fn from_number(number: u16) -> Result<Self, u16> {
  159. if number > Self::MAX_INTERRUPT_NUMBER || number == 0 {
  160. Err(number)
  161. } else {
  162. // SAFETY: valid interrupt number
  163. Ok(unsafe { core::mem::transmute(number) })
  164. }
  165. }
  166. }
  167. unsafe impl PriorityNumber for Priority {
  168. const MAX_PRIORITY_NUMBER: u8 = 3;
  169. #[inline]
  170. fn number(self) -> u8 {
  171. self as _
  172. }
  173. #[inline]
  174. fn from_number(number: u8) -> Result<Self, u8> {
  175. if number > Self::MAX_PRIORITY_NUMBER {
  176. Err(number)
  177. } else {
  178. // SAFETY: valid priority number
  179. Ok(unsafe { core::mem::transmute(number) })
  180. }
  181. }
  182. }
  183. unsafe impl HartIdNumber for Context {
  184. const MAX_HART_ID_NUMBER: u16 = 2;
  185. #[inline]
  186. fn number(self) -> u16 {
  187. self as _
  188. }
  189. #[inline]
  190. fn from_number(number: u16) -> Result<Self, u16> {
  191. if number > Self::MAX_HART_ID_NUMBER {
  192. Err(number)
  193. } else {
  194. // SAFETY: valid context number
  195. Ok(unsafe { core::mem::transmute(number) })
  196. }
  197. }
  198. }
  199. #[test]
  200. fn check_interrupt_enum() {
  201. assert_eq!(Interrupt::I1.number(), 1);
  202. assert_eq!(Interrupt::I2.number(), 2);
  203. assert_eq!(Interrupt::I3.number(), 3);
  204. assert_eq!(Interrupt::I4.number(), 4);
  205. assert_eq!(Interrupt::from_number(1), Ok(Interrupt::I1));
  206. assert_eq!(Interrupt::from_number(2), Ok(Interrupt::I2));
  207. assert_eq!(Interrupt::from_number(3), Ok(Interrupt::I3));
  208. assert_eq!(Interrupt::from_number(4), Ok(Interrupt::I4));
  209. assert_eq!(Interrupt::from_number(0), Err(0));
  210. assert_eq!(Interrupt::from_number(5), Err(5));
  211. }
  212. #[test]
  213. fn check_priority_enum() {
  214. assert_eq!(Priority::P0.number(), 0);
  215. assert_eq!(Priority::P1.number(), 1);
  216. assert_eq!(Priority::P2.number(), 2);
  217. assert_eq!(Priority::P3.number(), 3);
  218. assert_eq!(Priority::from_number(0), Ok(Priority::P0));
  219. assert_eq!(Priority::from_number(1), Ok(Priority::P1));
  220. assert_eq!(Priority::from_number(2), Ok(Priority::P2));
  221. assert_eq!(Priority::from_number(3), Ok(Priority::P3));
  222. assert_eq!(Priority::from_number(4), Err(4));
  223. }
  224. #[test]
  225. fn check_context_enum() {
  226. assert_eq!(Context::C0.number(), 0);
  227. assert_eq!(Context::C1.number(), 1);
  228. assert_eq!(Context::C2.number(), 2);
  229. assert_eq!(Context::from_number(0), Ok(Context::C0));
  230. assert_eq!(Context::from_number(1), Ok(Context::C1));
  231. assert_eq!(Context::from_number(2), Ok(Context::C2));
  232. assert_eq!(Context::from_number(3), Err(3));
  233. }
  234. #[allow(dead_code)]
  235. #[test]
  236. fn check_plic() {
  237. crate::plic_codegen!(
  238. base 0x0C00_0000,
  239. ctxs [ctx0 = (Context::C0, "`C0`"), ctx1 = (Context::C1, "`C1`"), ctx2 = (Context::C2, "`C2`")],
  240. );
  241. let priorities = PLIC::priorities();
  242. let pendings = PLIC::pendings();
  243. assert_eq!(priorities.address(), 0x0C00_0000);
  244. assert_eq!(pendings.address(), 0x0C00_1000);
  245. for i in 0..=Context::MAX_HART_ID_NUMBER {
  246. let context = Context::from_number(i).unwrap();
  247. let i = i as usize;
  248. let ctx = PLIC::ctx(context);
  249. assert_eq!(ctx.enables().address(), 0x0C00_0000 + 0x2000 + i * 0x80);
  250. assert_eq!(
  251. ctx.threshold().get_ptr() as usize,
  252. 0x0C00_0000 + 0x20_0000 + i * 0x1000
  253. );
  254. assert_eq!(
  255. ctx.claim().get_ptr() as usize,
  256. 0x0C00_0000 + 0x20_0004 + i * 0x1000
  257. );
  258. }
  259. assert_eq!(PLIC::ctx0(), PLIC::ctx(Context::C0));
  260. assert_eq!(PLIC::ctx1(), PLIC::ctx(Context::C1));
  261. assert_eq!(PLIC::ctx2(), PLIC::ctx(Context::C2));
  262. }
  263. }