rtic/src/export.rs

192 lines
4 KiB
Rust
Raw Normal View History

use core::{
cell::Cell,
sync::atomic::{AtomicBool, Ordering},
};
2018-11-03 17:02:41 +01:00
2020-12-08 20:49:13 +01:00
// pub use crate::tq::{NotReady, TimerQueue};
pub use bare_metal::CriticalSection;
2018-11-03 17:02:41 +01:00
#[cfg(armv7m)]
pub use cortex_m::register::basepri;
2018-11-03 17:02:41 +01:00
pub use cortex_m::{
asm::wfi,
interrupt,
2019-09-15 18:36:00 +02:00
peripheral::{scb::SystemHandler, syst::SystClkSource, DWT, NVIC},
Peripherals,
2018-11-03 17:02:41 +01:00
};
2020-08-27 13:21:56 +02:00
use heapless::spsc::SingleCore;
pub use heapless::{consts, i::Queue as iQueue, spsc::Queue};
pub use heapless::{i::BinaryHeap as iBinaryHeap, BinaryHeap};
2020-12-10 20:33:13 +01:00
pub use rtic_core::monotonic::Monotonic;
2018-11-03 17:02:41 +01:00
pub type SCFQ<N> = Queue<u8, N, u8, SingleCore>;
pub type SCRQ<T, N> = Queue<(T, u8), N, u8, SingleCore>;
2018-11-03 17:02:41 +01:00
#[cfg(armv7m)]
#[inline(always)]
pub fn run<F>(priority: u8, f: F)
2018-11-03 17:02:41 +01:00
where
F: FnOnce(),
{
if priority == 1 {
2020-09-01 19:04:55 +02:00
// If the priority of this interrupt is `1` then BASEPRI can only be `0`
f();
unsafe { basepri::write(0) }
} else {
let initial = basepri::read();
f();
unsafe { basepri::write(initial) }
}
2018-11-03 17:02:41 +01:00
}
#[cfg(not(armv7m))]
#[inline(always)]
pub fn run<F>(_priority: u8, f: F)
2018-11-03 17:02:41 +01:00
where
F: FnOnce(),
{
f();
}
pub struct Barrier {
inner: AtomicBool,
}
impl Barrier {
pub const fn new() -> Self {
Barrier {
inner: AtomicBool::new(false),
}
}
pub fn release(&self) {
self.inner.store(true, Ordering::Release)
}
pub fn wait(&self) {
while !self.inner.load(Ordering::Acquire) {}
}
}
2019-02-16 00:22:00 +01:00
// Newtype over `Cell` that forbids mutation through a shared reference
pub struct Priority {
inner: Cell<u8>,
}
impl Priority {
2020-10-13 16:16:33 +02:00
/// Create a new Priority
///
/// # Safety
///
/// Will overwrite the current Priority
2019-02-16 00:22:00 +01:00
#[inline(always)]
pub unsafe fn new(value: u8) -> Self {
2019-02-16 00:26:07 +01:00
Priority {
inner: Cell::new(value),
}
2019-02-16 00:22:00 +01:00
}
2020-10-13 16:16:33 +02:00
/// Change the current priority to `value`
2020-09-01 19:04:55 +02:00
// These two methods are used by `lock` (see below) but can't be used from the RTIC application
2019-02-16 00:22:00 +01:00
#[inline(always)]
fn set(&self, value: u8) {
self.inner.set(value)
}
2020-10-13 16:16:33 +02:00
/// Get the current priority
2019-02-16 00:22:00 +01:00
#[inline(always)]
fn get(&self) -> u8 {
self.inner.get()
}
}
2018-11-03 17:02:41 +01:00
#[inline(always)]
pub fn assert_send<T>()
where
T: Send,
{
}
#[inline(always)]
pub fn assert_sync<T>()
where
T: Sync,
{
}
2020-12-10 20:33:13 +01:00
#[inline(always)]
pub fn assert_monotonic<T>()
where
T: Monotonic,
{
}
2020-10-13 16:16:33 +02:00
/// Lock the resource proxy by setting the BASEPRI
/// and running the closure with interrupt::free
///
/// # Safety
///
/// Writing to the BASEPRI
/// Dereferencing a raw pointer
#[cfg(armv7m)]
#[inline(always)]
pub unsafe fn lock<T, R>(
ptr: *mut T,
2019-02-16 00:22:00 +01:00
priority: &Priority,
ceiling: u8,
nvic_prio_bits: u8,
f: impl FnOnce(&mut T) -> R,
) -> R {
let current = priority.get();
if current < ceiling {
if ceiling == (1 << nvic_prio_bits) {
priority.set(u8::max_value());
let r = interrupt::free(|_| f(&mut *ptr));
priority.set(current);
r
} else {
priority.set(ceiling);
basepri::write(logical2hw(ceiling, nvic_prio_bits));
let r = f(&mut *ptr);
basepri::write(logical2hw(current, nvic_prio_bits));
priority.set(current);
r
}
} else {
f(&mut *ptr)
}
}
2020-10-13 16:16:33 +02:00
/// Lock the resource proxy by setting the PRIMASK
/// and running the closure with interrupt::free
///
/// # Safety
///
/// Writing to the PRIMASK
/// Dereferencing a raw pointer
#[cfg(not(armv7m))]
#[inline(always)]
pub unsafe fn lock<T, R>(
ptr: *mut T,
2019-02-16 00:22:00 +01:00
priority: &Priority,
ceiling: u8,
_nvic_prio_bits: u8,
f: impl FnOnce(&mut T) -> R,
) -> R {
let current = priority.get();
if current < ceiling {
priority.set(u8::max_value());
let r = interrupt::free(|_| f(&mut *ptr));
priority.set(current);
r
} else {
f(&mut *ptr)
}
}
#[inline]
pub fn logical2hw(logical: u8, nvic_prio_bits: u8) -> u8 {
((1 << nvic_prio_bits) - logical) << (8 - nvic_prio_bits)
}