parent
e6d6e82e54
commit
02781ed744
@ -15,7 +15,9 @@ use crate::gpio::sealed::Pin as _;
|
|||||||
use crate::gpio::{OptionalPin as GpioOptionalPin, Pin as GpioPin};
|
use crate::gpio::{OptionalPin as GpioOptionalPin, Pin as GpioPin};
|
||||||
use crate::pac;
|
use crate::pac;
|
||||||
use crate::ppi::{AnyConfigurableChannel, ConfigurableChannel, Event, Ppi, Task};
|
use crate::ppi::{AnyConfigurableChannel, ConfigurableChannel, Event, Ppi, Task};
|
||||||
|
use crate::timer::Frequency;
|
||||||
use crate::timer::Instance as TimerInstance;
|
use crate::timer::Instance as TimerInstance;
|
||||||
|
use crate::timer::Timer;
|
||||||
use crate::uarte::{Config, Instance as UarteInstance};
|
use crate::uarte::{Config, Instance as UarteInstance};
|
||||||
|
|
||||||
// Re-export SVD variants to allow user to directly set values
|
// Re-export SVD variants to allow user to directly set values
|
||||||
@ -35,7 +37,7 @@ enum TxState {
|
|||||||
|
|
||||||
struct State<'d, U: UarteInstance, T: TimerInstance> {
|
struct State<'d, U: UarteInstance, T: TimerInstance> {
|
||||||
phantom: PhantomData<&'d mut U>,
|
phantom: PhantomData<&'d mut U>,
|
||||||
timer: T,
|
timer: Timer<'d, T>,
|
||||||
_ppi_ch1: Ppi<'d, AnyConfigurableChannel>,
|
_ppi_ch1: Ppi<'d, AnyConfigurableChannel>,
|
||||||
_ppi_ch2: Ppi<'d, AnyConfigurableChannel>,
|
_ppi_ch2: Ppi<'d, AnyConfigurableChannel>,
|
||||||
|
|
||||||
@ -76,10 +78,11 @@ impl<'d, U: UarteInstance, T: TimerInstance> BufferedUarte<'d, U, T> {
|
|||||||
rx_buffer: &'d mut [u8],
|
rx_buffer: &'d mut [u8],
|
||||||
tx_buffer: &'d mut [u8],
|
tx_buffer: &'d mut [u8],
|
||||||
) -> Self {
|
) -> Self {
|
||||||
unborrow!(timer, ppi_ch1, ppi_ch2, irq, rxd, txd, cts, rts);
|
unborrow!(ppi_ch1, ppi_ch2, irq, rxd, txd, cts, rts);
|
||||||
|
|
||||||
let r = U::regs();
|
let r = U::regs();
|
||||||
let rt = timer.regs();
|
|
||||||
|
let timer = Timer::new_irqless(timer);
|
||||||
|
|
||||||
rxd.conf().write(|w| w.input().connect().drive().h0h1());
|
rxd.conf().write(|w| w.input().connect().drive().h0h1());
|
||||||
r.psel.rxd.write(|w| unsafe { w.bits(rxd.psel_bits()) });
|
r.psel.rxd.write(|w| unsafe { w.bits(rxd.psel_bits()) });
|
||||||
@ -133,25 +136,19 @@ impl<'d, U: UarteInstance, T: TimerInstance> BufferedUarte<'d, U, T> {
|
|||||||
// This gives us the amount of 16M ticks for 20 bits.
|
// This gives us the amount of 16M ticks for 20 bits.
|
||||||
let timeout = 0x8000_0000 / (config.baudrate as u32 / 40);
|
let timeout = 0x8000_0000 / (config.baudrate as u32 / 40);
|
||||||
|
|
||||||
rt.tasks_stop.write(|w| unsafe { w.bits(1) });
|
timer.set_frequency(Frequency::F16MHz);
|
||||||
rt.bitmode.write(|w| w.bitmode()._32bit());
|
timer.cc0().set(timeout);
|
||||||
rt.prescaler.write(|w| unsafe { w.prescaler().bits(0) });
|
timer.cc0().short_compare_clear();
|
||||||
rt.cc[0].write(|w| unsafe { w.bits(timeout) });
|
timer.cc0().short_compare_stop();
|
||||||
rt.mode.write(|w| w.mode().timer());
|
|
||||||
rt.shorts.write(|w| {
|
|
||||||
w.compare0_clear().set_bit();
|
|
||||||
w.compare0_stop().set_bit();
|
|
||||||
w
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut ppi_ch1 = Ppi::new(ppi_ch1.degrade_configurable());
|
let mut ppi_ch1 = Ppi::new(ppi_ch1.degrade_configurable());
|
||||||
ppi_ch1.set_event(Event::from_reg(&r.events_rxdrdy));
|
ppi_ch1.set_event(Event::from_reg(&r.events_rxdrdy));
|
||||||
ppi_ch1.set_task(Task::from_reg(&rt.tasks_clear));
|
ppi_ch1.set_task(timer.task_clear());
|
||||||
ppi_ch1.set_fork_task(Task::from_reg(&rt.tasks_start));
|
ppi_ch1.set_fork_task(timer.task_start());
|
||||||
ppi_ch1.enable();
|
ppi_ch1.enable();
|
||||||
|
|
||||||
let mut ppi_ch2 = Ppi::new(ppi_ch2.degrade_configurable());
|
let mut ppi_ch2 = Ppi::new(ppi_ch2.degrade_configurable());
|
||||||
ppi_ch2.set_event(Event::from_reg(&rt.events_compare[0]));
|
ppi_ch2.set_event(timer.cc0().event_compare());
|
||||||
ppi_ch2.set_task(Task::from_reg(&r.tasks_stoprx));
|
ppi_ch2.set_task(Task::from_reg(&r.tasks_stoprx));
|
||||||
ppi_ch2.enable();
|
ppi_ch2.enable();
|
||||||
|
|
||||||
@ -181,11 +178,10 @@ impl<'d, U: UarteInstance, T: TimerInstance> BufferedUarte<'d, U, T> {
|
|||||||
inner.as_mut().register_interrupt();
|
inner.as_mut().register_interrupt();
|
||||||
inner.with(|state, _irq| {
|
inner.with(|state, _irq| {
|
||||||
let r = U::regs();
|
let r = U::regs();
|
||||||
let rt = state.timer.regs();
|
|
||||||
|
|
||||||
let timeout = 0x8000_0000 / (baudrate as u32 / 40);
|
let timeout = 0x8000_0000 / (baudrate as u32 / 40);
|
||||||
rt.cc[0].write(|w| unsafe { w.bits(timeout) });
|
state.timer.cc0().set(timeout);
|
||||||
rt.tasks_clear.write(|w| unsafe { w.bits(1) });
|
state.timer.clear();
|
||||||
|
|
||||||
r.baudrate.write(|w| w.baudrate().variant(baudrate));
|
r.baudrate.write(|w| w.baudrate().variant(baudrate));
|
||||||
});
|
});
|
||||||
@ -268,11 +264,10 @@ impl<'d, U: UarteInstance, T: TimerInstance> AsyncWrite for BufferedUarte<'d, U,
|
|||||||
impl<'a, U: UarteInstance, T: TimerInstance> Drop for State<'a, U, T> {
|
impl<'a, U: UarteInstance, T: TimerInstance> Drop for State<'a, U, T> {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
let r = U::regs();
|
let r = U::regs();
|
||||||
let rt = self.timer.regs();
|
|
||||||
|
|
||||||
// TODO this probably deadlocks. do like Uarte instead.
|
// TODO this probably deadlocks. do like Uarte instead.
|
||||||
|
|
||||||
rt.tasks_stop.write(|w| unsafe { w.bits(1) });
|
self.timer.stop();
|
||||||
if let RxState::Receiving = self.rx_state {
|
if let RxState::Receiving = self.rx_state {
|
||||||
r.tasks_stoprx.write(|w| unsafe { w.bits(1) });
|
r.tasks_stoprx.write(|w| unsafe { w.bits(1) });
|
||||||
}
|
}
|
||||||
@ -293,7 +288,6 @@ impl<'a, U: UarteInstance, T: TimerInstance> PeripheralState for State<'a, U, T>
|
|||||||
fn on_interrupt(&mut self) {
|
fn on_interrupt(&mut self) {
|
||||||
trace!("irq: start");
|
trace!("irq: start");
|
||||||
let r = U::regs();
|
let r = U::regs();
|
||||||
let rt = self.timer.regs();
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
match self.rx_state {
|
match self.rx_state {
|
||||||
@ -330,7 +324,7 @@ impl<'a, U: UarteInstance, T: TimerInstance> PeripheralState for State<'a, U, T>
|
|||||||
RxState::Receiving => {
|
RxState::Receiving => {
|
||||||
trace!(" irq_rx: in state receiving");
|
trace!(" irq_rx: in state receiving");
|
||||||
if r.events_endrx.read().bits() != 0 {
|
if r.events_endrx.read().bits() != 0 {
|
||||||
rt.tasks_stop.write(|w| unsafe { w.bits(1) });
|
self.timer.stop();
|
||||||
|
|
||||||
let n: usize = r.rxd.amount.read().amount().bits() as usize;
|
let n: usize = r.rxd.amount.read().amount().bits() as usize;
|
||||||
trace!(" irq_rx: endrx {:?}", n);
|
trace!(" irq_rx: endrx {:?}", n);
|
||||||
|
@ -1,15 +1,30 @@
|
|||||||
#![macro_use]
|
#![macro_use]
|
||||||
|
|
||||||
|
use core::marker::PhantomData;
|
||||||
|
use core::task::Poll;
|
||||||
|
|
||||||
use embassy::interrupt::Interrupt;
|
use embassy::interrupt::Interrupt;
|
||||||
|
use embassy::interrupt::InterruptExt;
|
||||||
|
use embassy::util::OnDrop;
|
||||||
use embassy::util::Unborrow;
|
use embassy::util::Unborrow;
|
||||||
|
use embassy_extras::unborrow;
|
||||||
|
use futures::future::poll_fn;
|
||||||
|
|
||||||
use crate::pac;
|
use crate::pac;
|
||||||
|
use crate::ppi::Event;
|
||||||
|
use crate::ppi::Task;
|
||||||
|
|
||||||
pub(crate) mod sealed {
|
pub(crate) mod sealed {
|
||||||
|
use embassy::util::AtomicWaker;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
pub trait Instance {
|
pub trait Instance {
|
||||||
fn regs(&self) -> &pac::timer0::RegisterBlock;
|
/// The number of CC registers this instance has.
|
||||||
|
const CCS: usize;
|
||||||
|
fn regs() -> &'static pac::timer0::RegisterBlock;
|
||||||
|
/// Storage for the waker for CC register `n`.
|
||||||
|
fn waker(n: usize) -> &'static AtomicWaker;
|
||||||
}
|
}
|
||||||
pub trait ExtendedInstance {}
|
pub trait ExtendedInstance {}
|
||||||
}
|
}
|
||||||
@ -20,19 +35,373 @@ pub trait Instance: Unborrow<Target = Self> + sealed::Instance + 'static {
|
|||||||
pub trait ExtendedInstance: Instance + sealed::ExtendedInstance {}
|
pub trait ExtendedInstance: Instance + sealed::ExtendedInstance {}
|
||||||
|
|
||||||
macro_rules! impl_timer {
|
macro_rules! impl_timer {
|
||||||
($type:ident, $pac_type:ident, $irq:ident) => {
|
($type:ident, $pac_type:ident, $irq:ident, $ccs:literal) => {
|
||||||
impl crate::timer::sealed::Instance for peripherals::$type {
|
impl crate::timer::sealed::Instance for peripherals::$type {
|
||||||
fn regs(&self) -> &pac::timer0::RegisterBlock {
|
const CCS: usize = $ccs;
|
||||||
|
fn regs() -> &'static pac::timer0::RegisterBlock {
|
||||||
unsafe { &*(pac::$pac_type::ptr() as *const pac::timer0::RegisterBlock) }
|
unsafe { &*(pac::$pac_type::ptr() as *const pac::timer0::RegisterBlock) }
|
||||||
}
|
}
|
||||||
|
fn waker(n: usize) -> &'static ::embassy::util::AtomicWaker {
|
||||||
|
use ::embassy::util::AtomicWaker;
|
||||||
|
const NEW_AW: AtomicWaker = AtomicWaker::new();
|
||||||
|
static WAKERS: [AtomicWaker; $ccs] = [NEW_AW; $ccs];
|
||||||
|
&WAKERS[n]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
impl crate::timer::Instance for peripherals::$type {
|
impl crate::timer::Instance for peripherals::$type {
|
||||||
type Interrupt = crate::interrupt::$irq;
|
type Interrupt = crate::interrupt::$irq;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
($type:ident, $pac_type:ident, $irq:ident) => {
|
||||||
|
impl_timer!($type, $pac_type, $irq, 4);
|
||||||
|
};
|
||||||
($type:ident, $pac_type:ident, $irq:ident, extended) => {
|
($type:ident, $pac_type:ident, $irq:ident, extended) => {
|
||||||
impl_timer!($type, $pac_type, $irq);
|
impl_timer!($type, $pac_type, $irq, 6);
|
||||||
impl crate::timer::sealed::ExtendedInstance for peripherals::$type {}
|
impl crate::timer::sealed::ExtendedInstance for peripherals::$type {}
|
||||||
impl crate::timer::ExtendedInstance for peripherals::$type {}
|
impl crate::timer::ExtendedInstance for peripherals::$type {}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[repr(u8)]
|
||||||
|
pub enum Frequency {
|
||||||
|
// TODO: These variant names are terrible, what should they be?
|
||||||
|
F16MHz = 0,
|
||||||
|
F8MHz = 1,
|
||||||
|
F4MHz = 2,
|
||||||
|
F2MHz = 3,
|
||||||
|
F1MHz = 4,
|
||||||
|
F500kHz = 5,
|
||||||
|
F250kHz = 6,
|
||||||
|
F125kHz = 7,
|
||||||
|
F62500Hz = 8,
|
||||||
|
F31250Hz = 9,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// nRF Timer driver.
|
||||||
|
///
|
||||||
|
/// The timer has an internal counter, which is incremented for every tick of the timer.
|
||||||
|
/// The counter is 32-bit, so it wraps back to 0 at 4294967296.
|
||||||
|
///
|
||||||
|
/// It has either 4 or 6 Capture/Compare registers, which can be used to capture the current state of the counter
|
||||||
|
/// or trigger an event when the counter reaches a certain value.
|
||||||
|
pub struct Timer<'d, T: Instance> {
|
||||||
|
phantom: PhantomData<&'d mut T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'d, T: Instance> Timer<'d, T> {
|
||||||
|
pub fn new(
|
||||||
|
timer: impl Unborrow<Target = T> + 'd,
|
||||||
|
irq: impl Unborrow<Target = T::Interrupt> + 'd,
|
||||||
|
) -> Self {
|
||||||
|
unborrow!(irq);
|
||||||
|
|
||||||
|
irq.set_handler(Self::on_interrupt);
|
||||||
|
irq.unpend();
|
||||||
|
irq.enable();
|
||||||
|
|
||||||
|
Self::new_irqless(timer)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a `Timer` without an interrupt, meaning `Cc::wait` won't work.
|
||||||
|
///
|
||||||
|
/// This is used by `Uarte` internally.
|
||||||
|
pub(crate) fn new_irqless(_timer: impl Unborrow<Target = T> + 'd) -> Self {
|
||||||
|
let regs = T::regs();
|
||||||
|
|
||||||
|
// Set the instance to timer mode.
|
||||||
|
regs.mode.write(|w| w.mode().timer());
|
||||||
|
|
||||||
|
// Make the counter's max value as high as possible.
|
||||||
|
// TODO: is there a reason someone would want to set this lower?
|
||||||
|
regs.bitmode.write(|w| w.bitmode()._32bit());
|
||||||
|
|
||||||
|
let this = Self {
|
||||||
|
phantom: PhantomData,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Initialize the timer as stopped.
|
||||||
|
this.stop();
|
||||||
|
|
||||||
|
// Initialize the counter at 0.
|
||||||
|
this.clear();
|
||||||
|
|
||||||
|
// Initialize all the shorts as disabled.
|
||||||
|
for n in 0..T::CCS {
|
||||||
|
let cc = Cc::<T> {
|
||||||
|
n,
|
||||||
|
phantom: PhantomData,
|
||||||
|
};
|
||||||
|
cc.unshort_compare_clear();
|
||||||
|
cc.unshort_compare_stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
this
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Starts the timer.
|
||||||
|
pub fn start(&self) {
|
||||||
|
T::regs().tasks_start.write(|w| w.tasks_start().trigger())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stops the timer.
|
||||||
|
pub fn stop(&self) {
|
||||||
|
T::regs().tasks_stop.write(|w| w.tasks_stop().trigger())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Reset the timer's counter to 0.
|
||||||
|
pub fn clear(&self) {
|
||||||
|
T::regs().tasks_clear.write(|w| w.tasks_clear().trigger())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the START task, for use with PPI.
|
||||||
|
///
|
||||||
|
/// When triggered, this task starts the timer.
|
||||||
|
pub fn task_start(&self) -> Task {
|
||||||
|
Task::from_reg(&T::regs().tasks_start)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the STOP task, for use with PPI.
|
||||||
|
///
|
||||||
|
/// When triggered, this task stops the timer.
|
||||||
|
pub fn task_stop(&self) -> Task {
|
||||||
|
Task::from_reg(&T::regs().tasks_stop)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the CLEAR task, for use with PPI.
|
||||||
|
///
|
||||||
|
/// When triggered, this task resets the timer's counter to 0.
|
||||||
|
pub fn task_clear(&self) -> Task {
|
||||||
|
Task::from_reg(&T::regs().tasks_clear)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Change the timer's frequency.
|
||||||
|
///
|
||||||
|
/// This will stop the timer if it isn't already stopped,
|
||||||
|
/// because the timer may exhibit 'unpredictable behaviour' if it's frequency is changed while it's running.
|
||||||
|
pub fn set_frequency(&self, frequency: Frequency) {
|
||||||
|
self.stop();
|
||||||
|
|
||||||
|
T::regs()
|
||||||
|
.prescaler
|
||||||
|
// SAFETY: `frequency` is a variant of `Frequency`,
|
||||||
|
// whose values are all in the range of 0-9 (the valid range of `prescaler`).
|
||||||
|
.write(|w| unsafe { w.prescaler().bits(frequency as u8) })
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_interrupt(_: *mut ()) {
|
||||||
|
let regs = T::regs();
|
||||||
|
for n in 0..T::CCS {
|
||||||
|
if regs.events_compare[n]
|
||||||
|
.read()
|
||||||
|
.events_compare()
|
||||||
|
.is_generated()
|
||||||
|
{
|
||||||
|
T::waker(n).wake();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the 0th CC register.
|
||||||
|
pub fn cc0<'a>(&'a self) -> Cc<'a, T> {
|
||||||
|
Cc {
|
||||||
|
n: 0,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the 1st CC register.
|
||||||
|
pub fn cc1<'a>(&'a self) -> Cc<'a, T> {
|
||||||
|
Cc {
|
||||||
|
n: 1,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the 2nd CC register.
|
||||||
|
pub fn cc2<'a>(&'a self) -> Cc<'a, T> {
|
||||||
|
Cc {
|
||||||
|
n: 2,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the 3rd CC register.
|
||||||
|
pub fn cc3<'a>(&'a self) -> Cc<'a, T> {
|
||||||
|
Cc {
|
||||||
|
n: 3,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'d, T: ExtendedInstance> Timer<'d, T> {
|
||||||
|
/// Returns the 4th CC register.
|
||||||
|
pub fn cc4<'a>(&'a self) -> Cc<'a, T> {
|
||||||
|
Cc {
|
||||||
|
n: 4,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the 5th CC register.
|
||||||
|
pub fn cc5<'a>(&'a self) -> Cc<'a, T> {
|
||||||
|
Cc {
|
||||||
|
n: 5,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A representation of a timer's Capture/Compare (CC) register.
|
||||||
|
///
|
||||||
|
/// A CC register holds a 32-bit value.
|
||||||
|
/// This is used either to store a capture of the timer's current count, or to specify the value for the timer to compare against.
|
||||||
|
///
|
||||||
|
/// The timer will fire the register's COMPARE event when its counter reaches the value stored in the register.
|
||||||
|
/// When the register's CAPTURE task is triggered, the timer will store the current value of its counter in the register
|
||||||
|
pub struct Cc<'a, T: Instance> {
|
||||||
|
n: usize,
|
||||||
|
phantom: PhantomData<&'a mut T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T: Instance> Cc<'a, T> {
|
||||||
|
/// Get the current value stored in the register.
|
||||||
|
pub fn value(&self) -> u32 {
|
||||||
|
T::regs().cc[self.n].read().cc().bits()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set the value stored in the register.
|
||||||
|
///
|
||||||
|
/// `event_compare` will fire when the timer's counter reaches this value.
|
||||||
|
pub fn set(&self, value: u32) {
|
||||||
|
// SAFETY: there are no invalid values for the CC register.
|
||||||
|
T::regs().cc[self.n].write(|w| unsafe { w.cc().bits(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Capture the current value of the timer's counter in this register, and return it.
|
||||||
|
pub fn capture(&self) -> u32 {
|
||||||
|
T::regs().tasks_capture[self.n].write(|w| w.tasks_capture().trigger());
|
||||||
|
self.value()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns this CC register's CAPTURE task, for use with PPI.
|
||||||
|
///
|
||||||
|
/// When triggered, this task will capture the current value of the timer's counter in this register.
|
||||||
|
pub fn task_capture(&self) -> Task {
|
||||||
|
Task::from_reg(&T::regs().tasks_capture[self.n])
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns this CC register's COMPARE event, for use with PPI.
|
||||||
|
///
|
||||||
|
/// This event will fire when the timer's counter reaches the value in this CC register.
|
||||||
|
pub fn event_compare(&self) -> Event {
|
||||||
|
Event::from_reg(&T::regs().events_compare)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Enable the shortcut between this CC register's COMPARE event and the timer's CLEAR task.
|
||||||
|
///
|
||||||
|
/// This means that when the COMPARE event is fired, the CLEAR task will be triggered.
|
||||||
|
///
|
||||||
|
/// So, when the timer's counter reaches the value stored in this register, the timer's counter will be reset to 0.
|
||||||
|
pub fn short_compare_clear(&self) {
|
||||||
|
T::regs().shorts.write(|w| match self.n {
|
||||||
|
0 => w.compare0_clear().enabled(),
|
||||||
|
1 => w.compare1_clear().enabled(),
|
||||||
|
2 => w.compare2_clear().enabled(),
|
||||||
|
3 => w.compare3_clear().enabled(),
|
||||||
|
4 => w.compare4_clear().enabled(),
|
||||||
|
5 => w.compare5_clear().enabled(),
|
||||||
|
_ => unreachable!("a `Cc` cannot be created with `n > 5`"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Disable the shortcut between this CC register's COMPARE event and the timer's CLEAR task.
|
||||||
|
pub fn unshort_compare_clear(&self) {
|
||||||
|
T::regs().shorts.write(|w| match self.n {
|
||||||
|
0 => w.compare0_clear().disabled(),
|
||||||
|
1 => w.compare1_clear().disabled(),
|
||||||
|
2 => w.compare2_clear().disabled(),
|
||||||
|
3 => w.compare3_clear().disabled(),
|
||||||
|
4 => w.compare4_clear().disabled(),
|
||||||
|
5 => w.compare5_clear().disabled(),
|
||||||
|
_ => unreachable!("a `Cc` cannot be created with `n > 5`"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Enable the shortcut between this CC register's COMPARE event and the timer's STOP task.
|
||||||
|
///
|
||||||
|
/// This means that when the COMPARE event is fired, the STOP task will be triggered.
|
||||||
|
///
|
||||||
|
/// So, when the timer's counter reaches the value stored in this register, the timer will stop counting up.
|
||||||
|
pub fn short_compare_stop(&self) {
|
||||||
|
T::regs().shorts.write(|w| match self.n {
|
||||||
|
0 => w.compare0_stop().enabled(),
|
||||||
|
1 => w.compare1_stop().enabled(),
|
||||||
|
2 => w.compare2_stop().enabled(),
|
||||||
|
3 => w.compare3_stop().enabled(),
|
||||||
|
4 => w.compare4_stop().enabled(),
|
||||||
|
5 => w.compare5_stop().enabled(),
|
||||||
|
_ => unreachable!("a `Cc` cannot be created with `n > 5`"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Disable the shortcut between this CC register's COMPARE event and the timer's STOP task.
|
||||||
|
pub fn unshort_compare_stop(&self) {
|
||||||
|
T::regs().shorts.write(|w| match self.n {
|
||||||
|
0 => w.compare0_stop().disabled(),
|
||||||
|
1 => w.compare1_stop().disabled(),
|
||||||
|
2 => w.compare2_stop().disabled(),
|
||||||
|
3 => w.compare3_stop().disabled(),
|
||||||
|
4 => w.compare4_stop().disabled(),
|
||||||
|
5 => w.compare5_stop().disabled(),
|
||||||
|
_ => unreachable!("a `Cc` cannot be created with `n > 5`"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Wait until the timer's counter reaches the value stored in this register.
|
||||||
|
pub async fn wait(&self) {
|
||||||
|
let regs = T::regs();
|
||||||
|
|
||||||
|
// Enable the interrupt for this CC's COMPARE event.
|
||||||
|
regs.intenset.write(|w| match self.n {
|
||||||
|
0 => w.compare0().set(),
|
||||||
|
1 => w.compare1().set(),
|
||||||
|
2 => w.compare2().set(),
|
||||||
|
3 => w.compare3().set(),
|
||||||
|
4 => w.compare4().set(),
|
||||||
|
5 => w.compare5().set(),
|
||||||
|
_ => unreachable!("a `Cc` cannot be created with `n > 5`"),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Disable the interrupt if the future is dropped.
|
||||||
|
let on_drop = OnDrop::new(|| {
|
||||||
|
regs.intenclr.write(|w| match self.n {
|
||||||
|
0 => w.compare0().clear(),
|
||||||
|
1 => w.compare1().clear(),
|
||||||
|
2 => w.compare2().clear(),
|
||||||
|
3 => w.compare3().clear(),
|
||||||
|
4 => w.compare4().clear(),
|
||||||
|
5 => w.compare5().clear(),
|
||||||
|
_ => unreachable!("a `Cc` cannot be created with `n > 5`"),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
poll_fn(|cx| {
|
||||||
|
T::waker(self.n).register(cx.waker());
|
||||||
|
|
||||||
|
if regs.events_compare[self.n]
|
||||||
|
.read()
|
||||||
|
.events_compare()
|
||||||
|
.is_generated()
|
||||||
|
{
|
||||||
|
Poll::Ready(())
|
||||||
|
} else {
|
||||||
|
Poll::Pending
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
// Trigger the interrupt to be disabled.
|
||||||
|
drop(on_drop);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -18,7 +18,9 @@ use crate::gpio::{self, OptionalPin as GpioOptionalPin, Pin as GpioPin};
|
|||||||
use crate::interrupt::Interrupt;
|
use crate::interrupt::Interrupt;
|
||||||
use crate::pac;
|
use crate::pac;
|
||||||
use crate::ppi::{AnyConfigurableChannel, ConfigurableChannel, Event, Ppi, Task};
|
use crate::ppi::{AnyConfigurableChannel, ConfigurableChannel, Event, Ppi, Task};
|
||||||
|
use crate::timer::Frequency;
|
||||||
use crate::timer::Instance as TimerInstance;
|
use crate::timer::Instance as TimerInstance;
|
||||||
|
use crate::timer::Timer;
|
||||||
|
|
||||||
// Re-export SVD variants to allow user to directly set values.
|
// Re-export SVD variants to allow user to directly set values.
|
||||||
pub use pac::uarte0::{baudrate::BAUDRATE_A as Baudrate, config::PARITY_A as Parity};
|
pub use pac::uarte0::{baudrate::BAUDRATE_A as Baudrate, config::PARITY_A as Parity};
|
||||||
@ -287,7 +289,7 @@ impl<'d, T: Instance> Write for Uarte<'d, T> {
|
|||||||
/// allowing it to implement the ReadUntilIdle trait.
|
/// allowing it to implement the ReadUntilIdle trait.
|
||||||
pub struct UarteWithIdle<'d, U: Instance, T: TimerInstance> {
|
pub struct UarteWithIdle<'d, U: Instance, T: TimerInstance> {
|
||||||
uarte: Uarte<'d, U>,
|
uarte: Uarte<'d, U>,
|
||||||
timer: T,
|
timer: Timer<'d, T>,
|
||||||
ppi_ch1: Ppi<'d, AnyConfigurableChannel>,
|
ppi_ch1: Ppi<'d, AnyConfigurableChannel>,
|
||||||
_ppi_ch2: Ppi<'d, AnyConfigurableChannel>,
|
_ppi_ch2: Ppi<'d, AnyConfigurableChannel>,
|
||||||
}
|
}
|
||||||
@ -316,11 +318,11 @@ impl<'d, U: Instance, T: TimerInstance> UarteWithIdle<'d, U, T> {
|
|||||||
) -> Self {
|
) -> Self {
|
||||||
let baudrate = config.baudrate;
|
let baudrate = config.baudrate;
|
||||||
let uarte = Uarte::new(uarte, irq, rxd, txd, cts, rts, config);
|
let uarte = Uarte::new(uarte, irq, rxd, txd, cts, rts, config);
|
||||||
|
let timer = Timer::new_irqless(timer);
|
||||||
|
|
||||||
unborrow!(timer, ppi_ch1, ppi_ch2);
|
unborrow!(ppi_ch1, ppi_ch2);
|
||||||
|
|
||||||
let r = U::regs();
|
let r = U::regs();
|
||||||
let rt = timer.regs();
|
|
||||||
|
|
||||||
// BAUDRATE register values are `baudrate * 2^32 / 16000000`
|
// BAUDRATE register values are `baudrate * 2^32 / 16000000`
|
||||||
// source: https://devzone.nordicsemi.com/f/nordic-q-a/391/uart-baudrate-register-values
|
// source: https://devzone.nordicsemi.com/f/nordic-q-a/391/uart-baudrate-register-values
|
||||||
@ -330,25 +332,19 @@ impl<'d, U: Instance, T: TimerInstance> UarteWithIdle<'d, U, T> {
|
|||||||
// This gives us the amount of 16M ticks for 20 bits.
|
// This gives us the amount of 16M ticks for 20 bits.
|
||||||
let timeout = 0x8000_0000 / (baudrate as u32 / 40);
|
let timeout = 0x8000_0000 / (baudrate as u32 / 40);
|
||||||
|
|
||||||
rt.tasks_stop.write(|w| unsafe { w.bits(1) });
|
timer.set_frequency(Frequency::F16MHz);
|
||||||
rt.bitmode.write(|w| w.bitmode()._32bit());
|
timer.cc0().set(timeout);
|
||||||
rt.prescaler.write(|w| unsafe { w.prescaler().bits(0) });
|
timer.cc0().short_compare_clear();
|
||||||
rt.cc[0].write(|w| unsafe { w.bits(timeout) });
|
timer.cc0().short_compare_stop();
|
||||||
rt.mode.write(|w| w.mode().timer());
|
|
||||||
rt.shorts.write(|w| {
|
|
||||||
w.compare0_clear().set_bit();
|
|
||||||
w.compare0_stop().set_bit();
|
|
||||||
w
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut ppi_ch1 = Ppi::new(ppi_ch1.degrade_configurable());
|
let mut ppi_ch1 = Ppi::new(ppi_ch1.degrade_configurable());
|
||||||
ppi_ch1.set_event(Event::from_reg(&r.events_rxdrdy));
|
ppi_ch1.set_event(Event::from_reg(&r.events_rxdrdy));
|
||||||
ppi_ch1.set_task(Task::from_reg(&rt.tasks_clear));
|
ppi_ch1.set_task(timer.task_clear());
|
||||||
ppi_ch1.set_fork_task(Task::from_reg(&rt.tasks_start));
|
ppi_ch1.set_fork_task(timer.task_start());
|
||||||
ppi_ch1.enable();
|
ppi_ch1.enable();
|
||||||
|
|
||||||
let mut ppi_ch2 = Ppi::new(ppi_ch2.degrade_configurable());
|
let mut ppi_ch2 = Ppi::new(ppi_ch2.degrade_configurable());
|
||||||
ppi_ch2.set_event(Event::from_reg(&rt.events_compare[0]));
|
ppi_ch2.set_event(timer.cc0().event_compare());
|
||||||
ppi_ch2.set_task(Task::from_reg(&r.tasks_stoprx));
|
ppi_ch2.set_task(Task::from_reg(&r.tasks_stoprx));
|
||||||
ppi_ch2.enable();
|
ppi_ch2.enable();
|
||||||
|
|
||||||
@ -373,12 +369,10 @@ impl<'d, U: Instance, T: TimerInstance> ReadUntilIdle for UarteWithIdle<'d, U, T
|
|||||||
let r = U::regs();
|
let r = U::regs();
|
||||||
let s = U::state();
|
let s = U::state();
|
||||||
|
|
||||||
let rt = self.timer.regs();
|
let drop = OnDrop::new(|| {
|
||||||
|
|
||||||
let drop = OnDrop::new(move || {
|
|
||||||
info!("read drop: stopping");
|
info!("read drop: stopping");
|
||||||
|
|
||||||
rt.tasks_stop.write(|w| unsafe { w.bits(1) });
|
self.timer.stop();
|
||||||
|
|
||||||
r.intenclr.write(|w| w.endrx().clear());
|
r.intenclr.write(|w| w.endrx().clear());
|
||||||
r.events_rxto.reset();
|
r.events_rxto.reset();
|
||||||
@ -413,7 +407,7 @@ impl<'d, U: Instance, T: TimerInstance> ReadUntilIdle for UarteWithIdle<'d, U, T
|
|||||||
let n = r.rxd.amount.read().amount().bits() as usize;
|
let n = r.rxd.amount.read().amount().bits() as usize;
|
||||||
|
|
||||||
// Stop timer
|
// Stop timer
|
||||||
rt.tasks_stop.write(|w| unsafe { w.bits(1) });
|
self.timer.stop();
|
||||||
r.events_rxstarted.reset();
|
r.events_rxstarted.reset();
|
||||||
|
|
||||||
drop.defuse();
|
drop.defuse();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user