ipi.rs 1.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142
  1. use crate::ecall::SbiRet;
  2. use crate::hart_mask::HartMask;
  3. use crate::util::AmoOncePtr;
  4. /// Inter-processor interrupt support
  5. pub trait Ipi: Send + Sync {
  6. /// Send an inter-processor interrupt to all the harts defined in `hart_mask`.
  7. ///
  8. /// Inter-processor interrupts manifest at the receiving harts as the supervisor software interrupts.
  9. ///
  10. /// # Return value
  11. ///
  12. /// Should return error code `SBI_SUCCESS` if IPI was sent to all the targeted harts successfully.
  13. fn send_ipi_many(&self, hart_mask: HartMask) -> SbiRet;
  14. #[doc(hidden)]
  15. /// Get the maximum hart id available by this IPI support module
  16. fn max_hart_id(&self) -> usize {
  17. unimplemented!("remained for compatibility, should remove in 0.3.0")
  18. }
  19. }
  20. static IPI: AmoOncePtr<dyn Ipi> = AmoOncePtr::new();
  21. #[doc(hidden)] // use through a macro
  22. pub fn init_ipi(ipi: &'static dyn Ipi) {
  23. if IPI.try_call_once(ipi) {
  24. panic!("load sbi module when already loaded")
  25. }
  26. }
  27. #[inline]
  28. pub(crate) fn probe_ipi() -> bool {
  29. IPI.get().is_some()
  30. }
  31. pub(crate) fn send_ipi_many(hart_mask: HartMask) -> SbiRet {
  32. if let Some(ipi) = IPI.get() {
  33. ipi.send_ipi_many(hart_mask)
  34. } else {
  35. SbiRet::not_supported()
  36. }
  37. }