pub use cortex_m::{
Peripherals,
asm::wfi,
interrupt,
peripheral::{DWT, NVIC, SCB, SYST, scb::SystemHandler},
};
#[cfg(not(any(feature = "thumbv6-backend", feature = "thumbv8base-backend")))]
compile_error!(
"Building for Cortex-M with source masking, but 'thumbv6-backend' or 'thumbv8base-backend' backend not selected"
);
#[derive(Copy, Clone)]
pub struct Mask<const M: usize>([u32; M]);
pub const fn create_mask<const N: usize, const M: usize>(list_of_shifts: [u32; N]) -> Mask<M> {
let mut mask = Mask([0; M]);
let mut i = 0;
while i < N {
let shift = list_of_shifts[i];
i += 1;
mask = mask.set_bit(shift);
}
mask
}
pub const fn compute_mask_chunks<const L: usize>(ids: [u32; L]) -> usize {
let mut max: usize = 0;
let mut i = 0;
while i < L {
let id = ids[i] as usize;
i += 1;
if id > max {
max = id;
}
}
(max + 32) / 32
}
impl<const M: usize> Mask<M> {
const fn set_bit(mut self, bit: u32) -> Self {
let block = bit / 32;
if block as usize >= M {
panic!(
"Generating masks for thumbv6/thumbv8m.base failed! Are you compiling for thumbv6 on an thumbv7 MCU or using an unsupported thumbv8m.base MCU?"
);
}
let offset = bit - (block * 32);
self.0[block as usize] |= 1 << offset;
self
}
}
#[inline(always)]
pub fn run<F>(_priority: u8, f: F)
where
F: FnOnce(),
{
f();
}
#[inline(always)]
pub unsafe fn lock<T, R, const M: usize>(
ptr: *mut T,
ceiling: u8,
masks: &[Mask<M>; 3],
f: impl FnOnce(&mut T) -> R,
) -> R {
unsafe {
if ceiling >= 4 {
critical_section::with(|_| f(&mut *ptr))
} else {
let mask = compute_mask(0, ceiling, masks);
let old_mask = read_mask(mask);
clear_enable_mask(mask);
let r = f(&mut *ptr);
set_enable_mask(mask, old_mask);
r
}
}
}
#[inline(always)]
pub const fn compute_mask<const M: usize>(
from_prio: u8,
to_prio: u8,
masks: &[Mask<M>; 3],
) -> Mask<M> {
let mut res = Mask([0; M]);
let mut idx = from_prio as usize;
while idx < to_prio as usize {
let mut i = 0;
while i < M {
res.0[i] |= masks[idx].0[i];
i += 1;
}
idx += 1;
}
res
}
#[inline(always)]
unsafe fn read_mask<const M: usize>(mask: Mask<M>) -> Mask<M> {
let mut out = Mask([0; M]);
for i in 0..M {
if mask.0[i] != 0 {
out.0[i] = unsafe { (*NVIC::PTR).iser[i].read() };
}
}
out
}
#[inline(always)]
unsafe fn set_enable_mask<const M: usize>(mask: Mask<M>, old_mask: Mask<M>) {
for i in 0..M {
if mask.0[i] != 0 {
unsafe {
(*NVIC::PTR).iser[i].write(old_mask.0[i]);
}
}
}
}
#[inline(always)]
unsafe fn clear_enable_mask<const M: usize>(mask: Mask<M>) {
for i in 0..M {
if mask.0[i] != 0 {
unsafe {
(*NVIC::PTR).icer[i].write(mask.0[i]);
}
}
}
}