adapt to changes in cortex-m, verify the input of the priority function

This commit is contained in:
Jorge Aparicio 2017-03-08 08:10:58 -05:00
parent e56f8156bc
commit d88a15fb25
2 changed files with 63 additions and 105 deletions

View file

@ -4,5 +4,5 @@ version = "0.1.0"
authors = ["Jorge Aparicio <japaricious@gmail.com>"] authors = ["Jorge Aparicio <japaricious@gmail.com>"]
[dependencies.cortex-m] [dependencies.cortex-m]
branch = "svd2rust" branch = "ng"
git = "https://github.com/japaric/cortex-m" git = "https://github.com/japaric/cortex-m"

View file

@ -2,6 +2,8 @@
//! //!
//! NOTE ARMv6-M is not supported at the moment. //! NOTE ARMv6-M is not supported at the moment.
#![deny(missing_docs)]
#![deny(warnings)]
#![feature(asm)] #![feature(asm)]
#![feature(const_fn)] #![feature(const_fn)]
#![no_std] #![no_std]
@ -12,146 +14,95 @@ const PRIORITY_BITS: u8 = 4;
extern crate cortex_m; extern crate cortex_m;
use cortex_m::interrupt::CsToken; use cortex_m::interrupt::CsCtxt;
use cortex_m::peripheral::Peripheral;
use cortex_m::register::{basepri, basepri_max}; use cortex_m::register::{basepri, basepri_max};
use core::cell::UnsafeCell; use core::cell::UnsafeCell;
use core::marker::PhantomData;
// XXX why is this needed? // XXX Do we need memory / instruction / compiler barriers here?
#[inline(always)] #[inline(always)]
fn compiler_barrier() { unsafe fn claim<T, R, F>(f: F, res: *const T, ceiling: u8) -> R
unsafe { where F: FnOnce(&T) -> R
asm!("" {
: let old_basepri = basepri::read();
: basepri_max::write(ceiling);
: "memory" let ret = f(&*res);
: "volatile") basepri::write(old_basepri);
} ret
} }
// XXX why is this needed? /// A peripheral as a resource
#[inline(always)] pub struct ResourceP<T>
fn memory_barrier() {
unsafe {
asm!("dsb
isb"
:
:
: "memory"
: "volatile")
}
}
pub struct Peripheral<T>
where T: 'static where T: 'static
{ {
_ty: PhantomData<&'static mut T>, peripheral: Peripheral<T>,
address: usize, // NOTE NVIC-style priority ceiling
ceiling: u8, ceiling: u8,
} }
impl<T> Peripheral<T> { impl<T> ResourceP<T> {
pub const unsafe fn new(address: usize, ceiling: u8) -> Self { /// Wraps a `peripheral` into a `Resource`
Peripheral { ///
_ty: PhantomData, /// NOTE `ceiling` must be in the range `[1, 15]` (inclusive)
address: address, ///
ceiling: ceiling, /// # Unsafety
///
/// - Do not create two resources that point to the same peripheral
pub const unsafe fn new(p: Peripheral<T>, ceiling: u8) -> Self {
ResourceP {
peripheral: p,
// NOTE elements 1 and 2 of the tuple are a poor man's const context
// range checker
ceiling: (priority(ceiling), ceiling - 1, ceiling + 240).0,
} }
} }
pub fn claim<F, R>(&self, f: F) -> R /// Claims the resource, blocking tasks with priority lower than `ceiling`
pub fn claim<R, F>(&'static self, f: F) -> R
where F: FnOnce(&T) -> R where F: FnOnce(&T) -> R
{ {
unsafe { unsafe { claim(f, self.peripheral.get(), self.ceiling) }
let old_basepri = basepri::read(); }
basepri_max::write(priority(self.ceiling));
memory_barrier(); /// Borrows the resource for the duration of a critical section
let ret = f(&*(self.address as *const T)); pub fn borrow<'a>(&'static self, _ctxt: &'a CsCtxt) -> &'a T {
compiler_barrier(); unsafe { &*self.peripheral.get() }
basepri::write(old_basepri);
ret
} }
} }
pub fn claim_mut<F, R>(&self, f: F) -> R unsafe impl<T> Sync for ResourceP<T> {}
where F: FnOnce(&mut T) -> R
{
unsafe {
let old_basepri = basepri::read();
basepri_max::write(priority(self.ceiling));
memory_barrier();
let ret = f(&mut *(self.address as *mut T));
compiler_barrier();
basepri::write(old_basepri);
ret
}
}
pub fn take<'a>(&self, _token: &'a CsToken) -> &'a T {
unsafe {
&*(self.address as *const T)
}
}
pub fn take_mut<'a>(&self, _token: &'a CsToken) -> &'a mut T {
unsafe {
&mut *(self.address as *mut T)
}
}
}
/// A resource
pub struct Resource<T> { pub struct Resource<T> {
// NOTE NVIC-style priority ceiling
ceiling: u8, ceiling: u8,
data: UnsafeCell<T>, data: UnsafeCell<T>,
} }
impl<T> Resource<T> { impl<T> Resource<T> {
/// Initializes a resource
///
/// NOTE `ceiling` must be in the range `[1, 15]`
pub const fn new(data: T, ceiling: u8) -> Self { pub const fn new(data: T, ceiling: u8) -> Self {
Resource { Resource {
ceiling: ceiling, // NOTE elements 1 and 2 of the tuple are a poor man's const context
// range checker
ceiling: (priority(ceiling), ceiling - 1, ceiling + 240).0,
data: UnsafeCell::new(data), data: UnsafeCell::new(data),
} }
} }
pub fn claim<F, R>(&self, f: F) -> R /// Claims the resource, blocking tasks with priority lower than `ceiling`
pub fn claim<F, R>(&'static self, f: F) -> R
where F: FnOnce(&T) -> R where F: FnOnce(&T) -> R
{ {
unsafe { unsafe { claim(f, self.data.get(), self.ceiling) }
let old_basepri = basepri::read();
basepri_max::write(priority(self.ceiling));
memory_barrier();
let ret = f(&*self.data.get());
compiler_barrier();
basepri::write(old_basepri);
ret
}
} }
pub fn claim_mut<F, R>(&self, f: F) -> R /// Borrows the resource for the duration of a critical section
where F: FnOnce(&mut T) -> R pub fn borrow<'cs>(&self, _ctxt: &'cs CsCtxt) -> &'cs T {
{ unsafe { &*self.data.get() }
unsafe {
let old_basepri = basepri::read();
basepri_max::write(priority(self.ceiling));
memory_barrier();
let ret = f(&mut *self.data.get());
compiler_barrier();
basepri::write(old_basepri);
ret
}
}
pub fn take<'a>(&self, _token: &'a CsToken) -> &'a T {
unsafe {
&*self.data.get()
}
}
pub fn take_mut<'a>(&self, _token: &'a CsToken) -> &'a mut T {
unsafe {
&mut *self.data.get()
}
} }
} }
@ -164,7 +115,14 @@ unsafe impl<T> Sync for Resource<T> {}
/// With NVIC priorities, `32` has LOWER priority than `16`. (Also, NVIC /// With NVIC priorities, `32` has LOWER priority than `16`. (Also, NVIC
/// priorities encode the actual priority in the highest bits of a byte so /// priorities encode the actual priority in the highest bits of a byte so
/// priorities like `1` and `2` aren't actually different) /// priorities like `1` and `2` aren't actually different)
// TODO review the handling of extreme values ///
/// NOTE `logical` must be in the range `[1, 15]` (inclusive)
pub const fn priority(logical: u8) -> u8 { pub const fn priority(logical: u8) -> u8 {
((1 << PRIORITY_BITS) - logical) << (8 - PRIORITY_BITS) // NOTE elements 1 and 2 of the tuple are a poor man's const context range
// checker
(((1 << PRIORITY_BITS) - logical) << (8 - PRIORITY_BITS),
logical - 1,
logical + 240)
.0
} }