atsamd_hal/interrupt.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179
//! Primitives for manipulating interrupts
use core::sync::atomic::{compiler_fence, Ordering};
use crate::pac::NVIC;
use atsamd_hal_macros::{hal_cfg, hal_macro_helper};
pub use crate::pac::NVIC_PRIO_BITS;
/// Logical interrupt priority level.
///
/// P4 is the most urgent, and P1 is the least urgent priority.
#[hal_cfg(any("nvic-d11", "nvic-d21"))]
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
#[repr(u8)]
#[allow(missing_docs)]
pub enum Priority {
P1 = 1,
P2 = 2,
P3 = 3,
P4 = 4,
}
/// Logical interrupt priority level.
///
/// P8 is the most urgent, and P1 is the least urgent priority.
#[hal_cfg("nvic-d5x")]
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
#[repr(u8)]
#[allow(missing_docs)]
pub enum Priority {
P1 = 1,
P2 = 2,
P3 = 3,
P4 = 4,
P5 = 5,
P6 = 6,
P7 = 7,
P8 = 8,
}
impl Priority {
/// Creates the `Priority` from a numeric priority if possible.
pub const fn from_numeric(prio: u8) -> Option<Self> {
if prio >= 1 && prio <= 8 {
Some(unsafe { core::mem::transmute::<u8, Self>(prio) })
} else {
None
}
}
/// Convert a logical priority (where higher priority number = higher
/// priority level) to a hardware priority level (where lower priority
/// number = higher priority level).
///
/// Taken from [`cortex-m-interrupt`]
///
/// See LICENSE-MIT for the license.
///
/// [`cortex-m-interrupt`]: https://github.com/datdenkikniet/cortex-m-interrupt
#[inline]
#[must_use]
pub const fn logical2hw(self) -> u8 {
((1 << NVIC_PRIO_BITS) - self as u8) << (8 - NVIC_PRIO_BITS)
}
/// Convert a hardware priority level (where lower priority number = higher
/// priority level) to a logical priority (where a higher priority number =
/// higher priority level).
///
/// # Panics
///
/// This method may only be used with allowed hardware priority levels. Ie,
/// * 0x00,
/// * 0x20,
/// * 0x40,
/// * and so on.
///
/// Any other value will cause a panic. To save yourself some
/// trouble, use this method only with hardware priority values gotten
/// directly from the NVIC.
#[inline]
#[must_use]
pub const fn hw2logical(prio: u8) -> Self {
assert!(prio % 0x20 == 0);
unsafe { core::mem::transmute((1 << NVIC_PRIO_BITS) - (prio >> (8 - NVIC_PRIO_BITS))) }
}
}
/// An interrupt type that can be configured by the HAL to handle
/// interrupts.
///
/// The PAC defined enum-level interrupts implement this trait.
pub trait InterruptExt: cortex_m::interrupt::InterruptNumber + Copy {
/// Enable the interrupt.
///
/// # Safety
///
/// Do not enable any interrupt inside a critical section.
#[inline]
unsafe fn enable(self) {
compiler_fence(Ordering::SeqCst);
NVIC::unmask(self)
}
/// Disable the interrupt.
#[inline]
fn disable(self) {
NVIC::mask(self);
compiler_fence(Ordering::SeqCst);
}
/// Check if interrupt is enabled.
#[inline]
fn is_enabled(self) -> bool {
NVIC::is_enabled(self)
}
/// Check if interrupt is pending.
#[inline]
fn is_pending(self) -> bool {
NVIC::is_pending(self)
}
/// Set interrupt pending.
#[inline]
fn pend(self) {
NVIC::pend(self)
}
/// Unset interrupt pending.
#[inline]
fn unpend(self) {
NVIC::unpend(self)
}
/// Get the priority of the interrupt.
#[inline]
fn get_priority(self) -> Priority {
Priority::hw2logical(NVIC::get_priority(self))
}
/// Set the interrupt priority.
#[inline]
#[hal_macro_helper]
fn set_priority(self, prio: Priority) {
unsafe {
let mut nvic = steal_nvic();
// On thumbv6, set_priority must do a RMW to change 8bit in a 32bit reg.
#[hal_cfg(any("nvic-d11", "nvic-d21"))]
critical_section::with(|_| nvic.set_priority(self, prio.logical2hw()));
// On thumbv7+, set_priority does an atomic 8bit write, so no CS needed.
#[hal_cfg("nvic-d5x")]
nvic.set_priority(self, prio.logical2hw());
}
}
/// Set the interrupt priority with an already-acquired critical section.
///
/// Equivalent to [`set_priority`](Self::set_priority), except you pass a
/// [`CriticalSection`] to prove you've already acquired a critical
/// section. This prevents acquiring another one, which saves code size.
#[inline]
fn set_priority_with_cs(self, _cs: critical_section::CriticalSection, prio: Priority) {
unsafe {
let mut nvic = steal_nvic();
nvic.set_priority(self, prio.logical2hw());
}
}
}
impl<T: cortex_m::interrupt::InterruptNumber + Copy> InterruptExt for T {}
unsafe fn steal_nvic() -> NVIC {
cortex_m::peripheral::Peripherals::steal().NVIC
}