aclint.rs 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. //! Devices for the Core Local Interruptor (CLINT) and Advanced CLINT (ACLINT) peripherals.
  2. //!
  3. //! CLINT pecification: <https://github.com/pulp-platform/clint>
  4. //! ACLINT Specification: <https://chromitem-soc.readthedocs.io/en/latest/clint.html>
  5. pub mod mswi;
  6. pub mod mtimer;
  7. pub mod sswi;
  8. pub use riscv_pac::HartIdNumber; // re-export useful riscv-pac traits
  9. /// Trait for a CLINT peripheral.
  10. ///
  11. /// # Safety
  12. ///
  13. /// * This trait must only be implemented on a PAC of a target with a CLINT peripheral.
  14. /// * The CLINT peripheral base address `BASE` must be valid for the target device.
  15. pub unsafe trait Clint: Copy {
  16. /// Base address of the CLINT peripheral.
  17. const BASE: usize;
  18. }
  19. /// Interface for a CLINT peripheral.
  20. ///
  21. /// The RISC-V standard does not specify a fixed location for the CLINT.
  22. /// Thus, each platform must specify the base address of the CLINT on the platform.
  23. /// The base address, as well as all the associated types, are defined in the [`Clint`] trait.
  24. ///
  25. /// The CLINT standard allows up to 4_095 different HARTs connected to the CLINT.
  26. /// Each HART has an assigned index starting from 0 to up to 4_094.
  27. /// In this way, each HART's timer and software interrupts can be independently configured.
  28. #[allow(clippy::upper_case_acronyms)]
  29. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  30. pub struct CLINT<C: Clint> {
  31. _marker: core::marker::PhantomData<C>,
  32. }
  33. impl<C: Clint> CLINT<C> {
  34. const MTIMECMP_OFFSET: usize = 0x4000;
  35. const MTIME_OFFSET: usize = 0xBFF8;
  36. /// Returns the `MSWI` peripheral.
  37. #[inline]
  38. pub const fn mswi() -> mswi::MSWI {
  39. // SAFETY: valid base address
  40. unsafe { mswi::MSWI::new(C::BASE) }
  41. }
  42. /// Returns the `MTIMER` peripheral.
  43. #[inline]
  44. pub const fn mtimer() -> mtimer::MTIMER {
  45. // SAFETY: valid base address
  46. unsafe {
  47. mtimer::MTIMER::new(
  48. C::BASE + Self::MTIMECMP_OFFSET,
  49. C::BASE + Self::MTIME_OFFSET,
  50. )
  51. }
  52. }
  53. }
  54. #[cfg(test)]
  55. pub(crate) mod test {
  56. use super::HartIdNumber;
  57. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  58. #[repr(u16)]
  59. pub(crate) enum HartId {
  60. H0 = 0,
  61. H1 = 1,
  62. H2 = 2,
  63. }
  64. unsafe impl HartIdNumber for HartId {
  65. const MAX_HART_ID_NUMBER: u16 = 2;
  66. #[inline]
  67. fn number(self) -> u16 {
  68. self as _
  69. }
  70. #[inline]
  71. fn from_number(number: u16) -> Result<Self, u16> {
  72. if number > Self::MAX_HART_ID_NUMBER {
  73. Err(number)
  74. } else {
  75. // SAFETY: valid context number
  76. Ok(unsafe { core::mem::transmute(number) })
  77. }
  78. }
  79. }
  80. #[test]
  81. fn check_hart_id_enum() {
  82. assert_eq!(HartId::H0.number(), 0);
  83. assert_eq!(HartId::H1.number(), 1);
  84. assert_eq!(HartId::H2.number(), 2);
  85. assert_eq!(HartId::from_number(0), Ok(HartId::H0));
  86. assert_eq!(HartId::from_number(1), Ok(HartId::H1));
  87. assert_eq!(HartId::from_number(2), Ok(HartId::H2));
  88. assert_eq!(HartId::from_number(3), Err(3));
  89. }
  90. #[allow(dead_code)]
  91. #[test]
  92. fn check_clint() {
  93. // Call CLINT macro with a base address and a list of mtimecmps for easing access to per-HART mtimecmp regs.
  94. crate::clint_codegen!(
  95. base 0x0200_0000,
  96. mtimecmps [mtimecmp0=(HartId::H0,"`H0`"), mtimecmp1=(HartId::H1,"`H1`"), mtimecmp2=(HartId::H2,"`H2`")],
  97. msips [msip0=(HartId::H0,"`H0`"), msip1=(HartId::H1,"`H1`"), msip2=(HartId::H2,"`H2`")],
  98. );
  99. let mswi = CLINT::mswi();
  100. let mtimer = CLINT::mtimer();
  101. assert_eq!(mswi.msip0.get_ptr() as usize, 0x0200_0000);
  102. assert_eq!(mtimer.mtimecmp0.get_ptr() as usize, 0x0200_4000);
  103. assert_eq!(mtimer.mtime.get_ptr() as usize, 0x0200_bff8);
  104. let mtimecmp0 = mtimer.mtimecmp(HartId::H0);
  105. let mtimecmp1 = mtimer.mtimecmp(HartId::H1);
  106. let mtimecmp2 = mtimer.mtimecmp(HartId::H2);
  107. assert_eq!(mtimecmp0.get_ptr() as usize, 0x0200_4000);
  108. assert_eq!(mtimecmp1.get_ptr() as usize, 0x0200_4000 + 8); // 8 bytes per register
  109. assert_eq!(mtimecmp2.get_ptr() as usize, 0x0200_4000 + 2 * 8);
  110. assert_eq!(CLINT::mtime(), mtimer.mtime);
  111. assert_eq!(CLINT::mtimecmp0(), mtimer.mtimecmp(HartId::H0));
  112. assert_eq!(CLINT::mtimecmp1(), mtimer.mtimecmp(HartId::H1));
  113. assert_eq!(CLINT::mtimecmp2(), mtimer.mtimecmp(HartId::H2));
  114. assert_eq!(CLINT::msip0(), mswi.msip(HartId::H0));
  115. assert_eq!(CLINT::msip1(), mswi.msip(HartId::H1));
  116. assert_eq!(CLINT::msip2(), mswi.msip(HartId::H2));
  117. }
  118. }