pendings.rs 1.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364
  1. //! Interrupt pending bits register.
  2. use crate::{
  3. common::{Reg, RO},
  4. plic::InterruptNumber,
  5. };
  6. /// Interrupts pending bits register.
  7. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  8. #[repr(transparent)]
  9. pub struct PENDINGS {
  10. ptr: *mut u32,
  11. }
  12. impl PENDINGS {
  13. /// Creates a new Interrupts pending bits register from a base address.
  14. ///
  15. /// # Safety
  16. ///
  17. /// The base address must point to a valid Interrupts pending bits register.
  18. #[inline]
  19. pub(crate) const unsafe fn new(address: usize) -> Self {
  20. Self { ptr: address as _ }
  21. }
  22. #[cfg(test)]
  23. #[inline]
  24. pub(crate) fn address(self) -> usize {
  25. self.ptr as _
  26. }
  27. /// Checks if an interrupt triggered by a given source is pending.
  28. #[inline]
  29. pub fn is_pending<I: InterruptNumber>(self, source: I) -> bool {
  30. let source = source.number() as usize;
  31. let offset = (source / u32::BITS as usize) as _;
  32. // SAFETY: valid interrupt number
  33. let reg: Reg<u32, RO> = unsafe { Reg::new(self.ptr.offset(offset)) };
  34. reg.read_bit(source % u32::BITS as usize)
  35. }
  36. }
  37. #[cfg(test)]
  38. mod test {
  39. use super::super::test::Interrupt;
  40. use super::*;
  41. #[test]
  42. fn test_pendings() {
  43. // slice to emulate the interrupt pendings register
  44. let mut raw_reg = [0u32; 32];
  45. // SAFETY: valid memory address
  46. let pendings = unsafe { PENDINGS::new(raw_reg.as_mut_ptr() as _) };
  47. for i in 0..255 {
  48. // SAFETY: valid memory address
  49. unsafe { raw_reg.as_mut_ptr().write_volatile(i) };
  50. assert_eq!(pendings.is_pending(Interrupt::I1), i & 0x2 != 0);
  51. assert_eq!(pendings.is_pending(Interrupt::I2), i & 0x4 != 0);
  52. assert_eq!(pendings.is_pending(Interrupt::I3), i & 0x8 != 0);
  53. assert_eq!(pendings.is_pending(Interrupt::I4), i & 0x10 != 0);
  54. }
  55. }
  56. }