2024-05-05 21:58:54 -04:00
|
|
|
//! Input capture driver.
|
|
|
|
|
2024-05-06 02:47:42 -04:00
|
|
|
use core::future::Future;
|
2024-05-05 21:58:54 -04:00
|
|
|
use core::marker::PhantomData;
|
2024-05-06 02:47:42 -04:00
|
|
|
use core::pin::Pin;
|
|
|
|
use core::task::{Context, Poll};
|
2024-05-05 21:58:54 -04:00
|
|
|
|
|
|
|
use embassy_hal_internal::{into_ref, PeripheralRef};
|
|
|
|
|
2024-05-30 17:43:38 -04:00
|
|
|
use super::low_level::{CountingMode, FilterValue, InputCaptureMode, InputTISelection, Timer};
|
2024-05-06 02:52:22 -04:00
|
|
|
use super::{
|
|
|
|
CaptureCompareInterruptHandler, Channel, Channel1Pin, Channel2Pin, Channel3Pin, Channel4Pin,
|
|
|
|
GeneralInstance4Channel,
|
|
|
|
};
|
2024-06-01 18:16:40 +02:00
|
|
|
use crate::gpio::{AfType, AnyPin, Pull};
|
2024-05-06 02:47:42 -04:00
|
|
|
use crate::interrupt::typelevel::{Binding, Interrupt};
|
2024-05-05 21:58:54 -04:00
|
|
|
use crate::time::Hertz;
|
|
|
|
use crate::Peripheral;
|
|
|
|
|
|
|
|
/// Channel 1 marker type.
|
|
|
|
pub enum Ch1 {}
|
|
|
|
/// Channel 2 marker type.
|
|
|
|
pub enum Ch2 {}
|
|
|
|
/// Channel 3 marker type.
|
|
|
|
pub enum Ch3 {}
|
|
|
|
/// Channel 4 marker type.
|
|
|
|
pub enum Ch4 {}
|
|
|
|
|
|
|
|
/// Capture pin wrapper.
|
|
|
|
///
|
|
|
|
/// This wraps a pin to make it usable with capture.
|
|
|
|
pub struct CapturePin<'d, T, C> {
|
|
|
|
_pin: PeripheralRef<'d, AnyPin>,
|
|
|
|
phantom: PhantomData<(T, C)>,
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! channel_impl {
|
|
|
|
($new_chx:ident, $channel:ident, $pin_trait:ident) => {
|
|
|
|
impl<'d, T: GeneralInstance4Channel> CapturePin<'d, T, $channel> {
|
|
|
|
#[doc = concat!("Create a new ", stringify!($channel), " capture pin instance.")]
|
2024-06-01 18:16:40 +02:00
|
|
|
pub fn $new_chx(pin: impl Peripheral<P = impl $pin_trait<T>> + 'd, pull: Pull) -> Self {
|
2024-05-05 21:58:54 -04:00
|
|
|
into_ref!(pin);
|
2024-06-01 18:16:40 +02:00
|
|
|
pin.set_as_af(pin.af_num(), AfType::input(pull));
|
2024-05-05 21:58:54 -04:00
|
|
|
CapturePin {
|
|
|
|
_pin: pin.map_into(),
|
|
|
|
phantom: PhantomData,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
channel_impl!(new_ch1, Ch1, Channel1Pin);
|
|
|
|
channel_impl!(new_ch2, Ch2, Channel2Pin);
|
|
|
|
channel_impl!(new_ch3, Ch3, Channel3Pin);
|
|
|
|
channel_impl!(new_ch4, Ch4, Channel4Pin);
|
|
|
|
|
|
|
|
/// Input capture driver.
|
|
|
|
pub struct InputCapture<'d, T: GeneralInstance4Channel> {
|
|
|
|
inner: Timer<'d, T>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'d, T: GeneralInstance4Channel> InputCapture<'d, T> {
|
|
|
|
/// Create a new input capture driver.
|
|
|
|
pub fn new(
|
|
|
|
tim: impl Peripheral<P = T> + 'd,
|
|
|
|
_ch1: Option<CapturePin<'d, T, Ch1>>,
|
|
|
|
_ch2: Option<CapturePin<'d, T, Ch2>>,
|
|
|
|
_ch3: Option<CapturePin<'d, T, Ch3>>,
|
|
|
|
_ch4: Option<CapturePin<'d, T, Ch4>>,
|
2024-05-06 02:47:42 -04:00
|
|
|
_irq: impl Binding<T::CaptureCompareInterrupt, CaptureCompareInterruptHandler<T>> + 'd,
|
2024-05-05 21:58:54 -04:00
|
|
|
freq: Hertz,
|
|
|
|
counting_mode: CountingMode,
|
|
|
|
) -> Self {
|
2024-05-28 22:43:23 -04:00
|
|
|
Self::new_inner(tim, freq, counting_mode)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn new_inner(tim: impl Peripheral<P = T> + 'd, freq: Hertz, counting_mode: CountingMode) -> Self {
|
2024-05-30 17:49:20 -04:00
|
|
|
let mut this = Self { inner: Timer::new(tim) };
|
2024-05-05 21:58:54 -04:00
|
|
|
|
2024-05-30 17:49:20 -04:00
|
|
|
this.inner.set_counting_mode(counting_mode);
|
|
|
|
this.inner.set_tick_freq(freq);
|
|
|
|
this.inner.enable_outputs(); // Required for advanced timers, see GeneralInstance4Channel for details
|
|
|
|
this.inner.start();
|
2024-05-05 21:58:54 -04:00
|
|
|
|
2024-05-06 02:47:42 -04:00
|
|
|
// enable NVIC interrupt
|
|
|
|
T::CaptureCompareInterrupt::unpend();
|
|
|
|
unsafe { T::CaptureCompareInterrupt::enable() };
|
2024-05-05 21:58:54 -04:00
|
|
|
|
2024-05-30 17:49:20 -04:00
|
|
|
this
|
2024-05-05 21:58:54 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Enable the given channel.
|
|
|
|
pub fn enable(&mut self, channel: Channel) {
|
|
|
|
self.inner.enable_channel(channel, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Disable the given channel.
|
|
|
|
pub fn disable(&mut self, channel: Channel) {
|
|
|
|
self.inner.enable_channel(channel, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check whether given channel is enabled
|
|
|
|
pub fn is_enabled(&self, channel: Channel) -> bool {
|
|
|
|
self.inner.get_channel_enable_state(channel)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set the input capture mode for a given channel.
|
|
|
|
pub fn set_input_capture_mode(&mut self, channel: Channel, mode: InputCaptureMode) {
|
|
|
|
self.inner.set_input_capture_mode(channel, mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set input TI selection.
|
|
|
|
pub fn set_input_ti_selection(&mut self, channel: Channel, tisel: InputTISelection) {
|
|
|
|
self.inner.set_input_ti_selection(channel, tisel)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get capture value for a channel.
|
|
|
|
pub fn get_capture_value(&self, channel: Channel) -> u32 {
|
|
|
|
self.inner.get_capture_value(channel)
|
|
|
|
}
|
2024-05-05 23:00:48 -04:00
|
|
|
|
|
|
|
/// Get input interrupt.
|
|
|
|
pub fn get_input_interrupt(&self, channel: Channel) -> bool {
|
|
|
|
self.inner.get_input_interrupt(channel)
|
|
|
|
}
|
2024-05-06 02:47:42 -04:00
|
|
|
|
|
|
|
fn new_future(&self, channel: Channel, mode: InputCaptureMode, tisel: InputTISelection) -> InputCaptureFuture<T> {
|
2024-05-29 00:28:26 -04:00
|
|
|
// Configuration steps from ST RM0390 (STM32F446) chapter 17.3.5
|
|
|
|
// or ST RM0008 (STM32F103) chapter 15.3.5 Input capture mode
|
2024-05-28 22:38:08 -04:00
|
|
|
self.inner.set_input_ti_selection(channel, tisel);
|
|
|
|
self.inner.set_input_capture_filter(channel, FilterValue::NOFILTER);
|
|
|
|
self.inner.set_input_capture_mode(channel, mode);
|
|
|
|
self.inner.set_input_capture_prescaler(channel, 0);
|
|
|
|
self.inner.enable_channel(channel, true);
|
|
|
|
self.inner.enable_input_interrupt(channel, true);
|
2024-05-06 02:47:42 -04:00
|
|
|
|
|
|
|
InputCaptureFuture {
|
|
|
|
channel,
|
|
|
|
phantom: PhantomData,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Asynchronously wait until the pin sees a rising edge.
|
|
|
|
pub async fn wait_for_rising_edge(&mut self, channel: Channel) -> u32 {
|
|
|
|
self.new_future(channel, InputCaptureMode::Rising, InputTISelection::Normal)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Asynchronously wait until the pin sees a falling edge.
|
|
|
|
pub async fn wait_for_falling_edge(&mut self, channel: Channel) -> u32 {
|
|
|
|
self.new_future(channel, InputCaptureMode::Falling, InputTISelection::Normal)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Asynchronously wait until the pin sees any edge.
|
|
|
|
pub async fn wait_for_any_edge(&mut self, channel: Channel) -> u32 {
|
|
|
|
self.new_future(channel, InputCaptureMode::BothEdges, InputTISelection::Normal)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Asynchronously wait until the (alternate) pin sees a rising edge.
|
|
|
|
pub async fn wait_for_rising_edge_alternate(&mut self, channel: Channel) -> u32 {
|
|
|
|
self.new_future(channel, InputCaptureMode::Rising, InputTISelection::Alternate)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Asynchronously wait until the (alternate) pin sees a falling edge.
|
|
|
|
pub async fn wait_for_falling_edge_alternate(&mut self, channel: Channel) -> u32 {
|
|
|
|
self.new_future(channel, InputCaptureMode::Falling, InputTISelection::Alternate)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Asynchronously wait until the (alternate) pin sees any edge.
|
|
|
|
pub async fn wait_for_any_edge_alternate(&mut self, channel: Channel) -> u32 {
|
|
|
|
self.new_future(channel, InputCaptureMode::BothEdges, InputTISelection::Alternate)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[must_use = "futures do nothing unless you `.await` or poll them"]
|
|
|
|
struct InputCaptureFuture<T: GeneralInstance4Channel> {
|
|
|
|
channel: Channel,
|
|
|
|
phantom: PhantomData<T>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: GeneralInstance4Channel> Drop for InputCaptureFuture<T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
critical_section::with(|_| {
|
2024-05-30 17:43:38 -04:00
|
|
|
let regs = unsafe { crate::pac::timer::TimGp16::from_ptr(T::regs()) };
|
2024-05-06 02:47:42 -04:00
|
|
|
|
|
|
|
// disable interrupt enable
|
|
|
|
regs.dier().modify(|w| w.set_ccie(self.channel.index(), false));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: GeneralInstance4Channel> Future for InputCaptureFuture<T> {
|
|
|
|
type Output = u32;
|
|
|
|
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
|
|
T::state().cc_waker[self.channel.index()].register(cx.waker());
|
|
|
|
|
2024-05-30 17:43:38 -04:00
|
|
|
let regs = unsafe { crate::pac::timer::TimGp16::from_ptr(T::regs()) };
|
2024-05-06 02:47:42 -04:00
|
|
|
|
|
|
|
let dier = regs.dier().read();
|
|
|
|
if !dier.ccie(self.channel.index()) {
|
|
|
|
let val = regs.ccr(self.channel.index()).read().0;
|
|
|
|
Poll::Ready(val)
|
|
|
|
} else {
|
|
|
|
Poll::Pending
|
|
|
|
}
|
|
|
|
}
|
2024-05-05 21:58:54 -04:00
|
|
|
}
|