mtimer.rs 2.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. //! Machine-level Timer Device.
  2. pub use super::HartIdNumber;
  3. use crate::common::safe_peripheral;
  4. /// MTIMER peripheral.
  5. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  6. pub struct MTIMER {
  7. /// `MTIMECMP` register for HART ID 0. In multi-HART architectures,
  8. /// use [`MTIMER::mtimecmp`] for accessing the `MTIMECMP` of other HARTs.
  9. pub mtimecmp0: MTIMECMP,
  10. /// The `MTIME` register is shared among all the HARTs.
  11. pub mtime: MTIME,
  12. }
  13. impl MTIMER {
  14. /// Creates a new `MTIMER` peripheral from a base address.
  15. ///
  16. /// # Safety
  17. ///
  18. /// The base addresses must point to valid `MTIMECMP` and `MTIME` peripherals.
  19. #[inline]
  20. pub const unsafe fn new(mtimecmp: usize, mtime: usize) -> Self {
  21. Self {
  22. mtimecmp0: MTIMECMP::new(mtimecmp),
  23. mtime: MTIME::new(mtime),
  24. }
  25. }
  26. /// Returns the `MTIMECMP` register for the HART which ID is `hart_id`.
  27. ///
  28. /// # Note
  29. ///
  30. /// For HART ID 0, you can simply use [`MTIMER::mtimecmp0`].
  31. #[inline]
  32. pub fn mtimecmp<H: HartIdNumber>(&self, hart_id: H) -> MTIMECMP {
  33. // SAFETY: `hart_id` is valid for the target
  34. unsafe { MTIMECMP::new(self.mtimecmp0.get_ptr().offset(hart_id.number() as _) as _) }
  35. }
  36. /// Returns the `MTIMECMP` register for the current HART.
  37. ///
  38. /// # Note
  39. ///
  40. /// This function determines the current HART ID by reading the [`riscv::register::mhartid`] CSR.
  41. /// Thus, it can only be used in M-mode. For S-mode, use [`MTIMER::mtimecmp`] instead.
  42. #[inline]
  43. pub fn mtimecmp_mhartid(&self) -> MTIMECMP {
  44. let hart_id = riscv::register::mhartid::read();
  45. // SAFETY: `hart_id` is valid for the target and is the current hart
  46. unsafe { MTIMECMP::new(self.mtimecmp0.get_ptr().add(hart_id) as _) }
  47. }
  48. }
  49. // MTIMECMP register.
  50. safe_peripheral!(MTIMECMP, u64, RW);
  51. // MTIME register.
  52. safe_peripheral!(MTIME, u64, RW);
  53. #[cfg(test)]
  54. mod test {
  55. use super::super::test::HartId;
  56. use super::*;
  57. #[test]
  58. fn check_mtimer() {
  59. // slice to emulate the mtimecmp registers
  60. let raw_mtimecmp = [0u64; HartId::MAX_HART_ID_NUMBER as usize + 1];
  61. let raw_mtime = 0u64;
  62. // SAFETY: valid memory addresses
  63. let mtimer =
  64. unsafe { MTIMER::new(raw_mtimecmp.as_ptr() as _, &raw_mtime as *const u64 as _) };
  65. assert_eq!(
  66. mtimer.mtimecmp(HartId::H0).get_ptr() as usize,
  67. raw_mtimecmp.as_ptr() as usize
  68. );
  69. assert_eq!(mtimer.mtimecmp(HartId::H1).get_ptr() as usize, unsafe {
  70. raw_mtimecmp.as_ptr().offset(1)
  71. }
  72. as usize);
  73. assert_eq!(mtimer.mtimecmp(HartId::H2).get_ptr() as usize, unsafe {
  74. raw_mtimecmp.as_ptr().offset(2)
  75. }
  76. as usize);
  77. assert_eq!(
  78. mtimer.mtime.get_ptr() as usize,
  79. &raw_mtime as *const u64 as _
  80. );
  81. }
  82. }