407 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			407 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| //! External Interrupts (EXTI)
 | |
| use core::convert::Infallible;
 | |
| use core::future::Future;
 | |
| use core::marker::PhantomData;
 | |
| use core::pin::Pin;
 | |
| use core::task::{Context, Poll};
 | |
| 
 | |
| use embassy_hal_internal::{impl_peripheral, into_ref};
 | |
| use embassy_sync::waitqueue::AtomicWaker;
 | |
| 
 | |
| use crate::gpio::{AnyPin, Input, Level, Pin as GpioPin, Pull};
 | |
| use crate::pac::exti::regs::Lines;
 | |
| use crate::pac::EXTI;
 | |
| use crate::{interrupt, pac, peripherals, Peripheral};
 | |
| 
 | |
| const EXTI_COUNT: usize = 16;
 | |
| static EXTI_WAKERS: [AtomicWaker; EXTI_COUNT] = [const { AtomicWaker::new() }; EXTI_COUNT];
 | |
| 
 | |
| #[cfg(exti_w)]
 | |
| fn cpu_regs() -> pac::exti::Cpu {
 | |
|     EXTI.cpu(crate::pac::CORE_INDEX)
 | |
| }
 | |
| 
 | |
| #[cfg(not(exti_w))]
 | |
| fn cpu_regs() -> pac::exti::Exti {
 | |
|     EXTI
 | |
| }
 | |
| 
 | |
| #[cfg(not(any(exti_c0, exti_g0, exti_u0, exti_l5, gpio_v1, exti_u5, exti_h5, exti_h50)))]
 | |
| fn exticr_regs() -> pac::syscfg::Syscfg {
 | |
|     pac::SYSCFG
 | |
| }
 | |
| #[cfg(any(exti_c0, exti_g0, exti_u0, exti_l5, exti_u5, exti_h5, exti_h50))]
 | |
| fn exticr_regs() -> pac::exti::Exti {
 | |
|     EXTI
 | |
| }
 | |
| #[cfg(gpio_v1)]
 | |
| fn exticr_regs() -> pac::afio::Afio {
 | |
|     pac::AFIO
 | |
| }
 | |
| 
 | |
| unsafe fn on_irq() {
 | |
|     #[cfg(not(any(exti_c0, exti_g0, exti_u0, exti_l5, exti_u5, exti_h5, exti_h50)))]
 | |
|     let bits = EXTI.pr(0).read().0;
 | |
|     #[cfg(any(exti_c0, exti_g0, exti_u0, exti_l5, exti_u5, exti_h5, exti_h50))]
 | |
|     let bits = EXTI.rpr(0).read().0 | EXTI.fpr(0).read().0;
 | |
| 
 | |
|     // We don't handle or change any EXTI lines above 16.
 | |
|     let bits = bits & 0x0000FFFF;
 | |
| 
 | |
|     // Mask all the channels that fired.
 | |
|     cpu_regs().imr(0).modify(|w| w.0 &= !bits);
 | |
| 
 | |
|     // Wake the tasks
 | |
|     for pin in BitIter(bits) {
 | |
|         EXTI_WAKERS[pin as usize].wake();
 | |
|     }
 | |
| 
 | |
|     // Clear pending
 | |
|     #[cfg(not(any(exti_c0, exti_g0, exti_u0, exti_l5, exti_u5, exti_h5, exti_h50)))]
 | |
|     EXTI.pr(0).write_value(Lines(bits));
 | |
|     #[cfg(any(exti_c0, exti_g0, exti_u0, exti_l5, exti_u5, exti_h5, exti_h50))]
 | |
|     {
 | |
|         EXTI.rpr(0).write_value(Lines(bits));
 | |
|         EXTI.fpr(0).write_value(Lines(bits));
 | |
|     }
 | |
| 
 | |
|     #[cfg(feature = "low-power")]
 | |
|     crate::low_power::on_wakeup_irq();
 | |
| }
 | |
| 
 | |
| struct BitIter(u32);
 | |
| 
 | |
| impl Iterator for BitIter {
 | |
|     type Item = u32;
 | |
| 
 | |
|     fn next(&mut self) -> Option<Self::Item> {
 | |
|         match self.0.trailing_zeros() {
 | |
|             32 => None,
 | |
|             b => {
 | |
|                 self.0 &= !(1 << b);
 | |
|                 Some(b)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| /// EXTI input driver.
 | |
| ///
 | |
| /// This driver augments a GPIO `Input` with EXTI functionality. EXTI is not
 | |
| /// built into `Input` itself because it needs to take ownership of the corresponding
 | |
| /// EXTI channel, which is a limited resource.
 | |
| ///
 | |
| /// Pins PA5, PB5, PC5... all use EXTI channel 5, so you can't use EXTI on, say, PA5 and PC5 at the same time.
 | |
| pub struct ExtiInput<'d> {
 | |
|     pin: Input<'d>,
 | |
| }
 | |
| 
 | |
| impl<'d> Unpin for ExtiInput<'d> {}
 | |
| 
 | |
| impl<'d> ExtiInput<'d> {
 | |
|     /// Create an EXTI input.
 | |
|     pub fn new<T: GpioPin>(
 | |
|         pin: impl Peripheral<P = T> + 'd,
 | |
|         ch: impl Peripheral<P = T::ExtiChannel> + 'd,
 | |
|         pull: Pull,
 | |
|     ) -> Self {
 | |
|         into_ref!(pin, ch);
 | |
| 
 | |
|         // Needed if using AnyPin+AnyChannel.
 | |
|         assert_eq!(pin.pin(), ch.number());
 | |
| 
 | |
|         Self {
 | |
|             pin: Input::new(pin, pull),
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// Get whether the pin is high.
 | |
|     pub fn is_high(&self) -> bool {
 | |
|         self.pin.is_high()
 | |
|     }
 | |
| 
 | |
|     /// Get whether the pin is low.
 | |
|     pub fn is_low(&self) -> bool {
 | |
|         self.pin.is_low()
 | |
|     }
 | |
| 
 | |
|     /// Get the pin level.
 | |
|     pub fn get_level(&self) -> Level {
 | |
|         self.pin.get_level()
 | |
|     }
 | |
| 
 | |
|     /// Asynchronously wait until the pin is high.
 | |
|     ///
 | |
|     /// This returns immediately if the pin is already high.
 | |
|     pub async fn wait_for_high(&mut self) {
 | |
|         let fut = ExtiInputFuture::new(self.pin.pin.pin.pin(), self.pin.pin.pin.port(), true, false);
 | |
|         if self.is_high() {
 | |
|             return;
 | |
|         }
 | |
|         fut.await
 | |
|     }
 | |
| 
 | |
|     /// Asynchronously wait until the pin is low.
 | |
|     ///
 | |
|     /// This returns immediately if the pin is already low.
 | |
|     pub async fn wait_for_low(&mut self) {
 | |
|         let fut = ExtiInputFuture::new(self.pin.pin.pin.pin(), self.pin.pin.pin.port(), false, true);
 | |
|         if self.is_low() {
 | |
|             return;
 | |
|         }
 | |
|         fut.await
 | |
|     }
 | |
| 
 | |
|     /// Asynchronously wait until the pin sees a rising edge.
 | |
|     ///
 | |
|     /// If the pin is already high, it will wait for it to go low then back high.
 | |
|     pub async fn wait_for_rising_edge(&mut self) {
 | |
|         ExtiInputFuture::new(self.pin.pin.pin.pin(), self.pin.pin.pin.port(), true, false).await
 | |
|     }
 | |
| 
 | |
|     /// Asynchronously wait until the pin sees a falling edge.
 | |
|     ///
 | |
|     /// If the pin is already low, it will wait for it to go high then back low.
 | |
|     pub async fn wait_for_falling_edge(&mut self) {
 | |
|         ExtiInputFuture::new(self.pin.pin.pin.pin(), self.pin.pin.pin.port(), false, true).await
 | |
|     }
 | |
| 
 | |
|     /// Asynchronously wait until the pin sees any edge (either rising or falling).
 | |
|     pub async fn wait_for_any_edge(&mut self) {
 | |
|         ExtiInputFuture::new(self.pin.pin.pin.pin(), self.pin.pin.pin.port(), true, true).await
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<'d> embedded_hal_02::digital::v2::InputPin for ExtiInput<'d> {
 | |
|     type Error = Infallible;
 | |
| 
 | |
|     fn is_high(&self) -> Result<bool, Self::Error> {
 | |
|         Ok(self.is_high())
 | |
|     }
 | |
| 
 | |
|     fn is_low(&self) -> Result<bool, Self::Error> {
 | |
|         Ok(self.is_low())
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<'d> embedded_hal_1::digital::ErrorType for ExtiInput<'d> {
 | |
|     type Error = Infallible;
 | |
| }
 | |
| 
 | |
| impl<'d> embedded_hal_1::digital::InputPin for ExtiInput<'d> {
 | |
|     fn is_high(&mut self) -> Result<bool, Self::Error> {
 | |
|         Ok((*self).is_high())
 | |
|     }
 | |
| 
 | |
|     fn is_low(&mut self) -> Result<bool, Self::Error> {
 | |
|         Ok((*self).is_low())
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<'d> embedded_hal_async::digital::Wait for ExtiInput<'d> {
 | |
|     async fn wait_for_high(&mut self) -> Result<(), Self::Error> {
 | |
|         self.wait_for_high().await;
 | |
|         Ok(())
 | |
|     }
 | |
| 
 | |
|     async fn wait_for_low(&mut self) -> Result<(), Self::Error> {
 | |
|         self.wait_for_low().await;
 | |
|         Ok(())
 | |
|     }
 | |
| 
 | |
|     async fn wait_for_rising_edge(&mut self) -> Result<(), Self::Error> {
 | |
|         self.wait_for_rising_edge().await;
 | |
|         Ok(())
 | |
|     }
 | |
| 
 | |
|     async fn wait_for_falling_edge(&mut self) -> Result<(), Self::Error> {
 | |
|         self.wait_for_falling_edge().await;
 | |
|         Ok(())
 | |
|     }
 | |
| 
 | |
|     async fn wait_for_any_edge(&mut self) -> Result<(), Self::Error> {
 | |
|         self.wait_for_any_edge().await;
 | |
|         Ok(())
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[must_use = "futures do nothing unless you `.await` or poll them"]
 | |
| struct ExtiInputFuture<'a> {
 | |
|     pin: u8,
 | |
|     phantom: PhantomData<&'a mut AnyPin>,
 | |
| }
 | |
| 
 | |
| impl<'a> ExtiInputFuture<'a> {
 | |
|     fn new(pin: u8, port: u8, rising: bool, falling: bool) -> Self {
 | |
|         critical_section::with(|_| {
 | |
|             let pin = pin as usize;
 | |
|             exticr_regs().exticr(pin / 4).modify(|w| w.set_exti(pin % 4, port));
 | |
|             EXTI.rtsr(0).modify(|w| w.set_line(pin, rising));
 | |
|             EXTI.ftsr(0).modify(|w| w.set_line(pin, falling));
 | |
| 
 | |
|             // clear pending bit
 | |
|             #[cfg(not(any(exti_c0, exti_g0, exti_u0, exti_l5, exti_u5, exti_h5, exti_h50)))]
 | |
|             EXTI.pr(0).write(|w| w.set_line(pin, true));
 | |
|             #[cfg(any(exti_c0, exti_g0, exti_u0, exti_l5, exti_u5, exti_h5, exti_h50))]
 | |
|             {
 | |
|                 EXTI.rpr(0).write(|w| w.set_line(pin, true));
 | |
|                 EXTI.fpr(0).write(|w| w.set_line(pin, true));
 | |
|             }
 | |
| 
 | |
|             cpu_regs().imr(0).modify(|w| w.set_line(pin, true));
 | |
|         });
 | |
| 
 | |
|         Self {
 | |
|             pin,
 | |
|             phantom: PhantomData,
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<'a> Drop for ExtiInputFuture<'a> {
 | |
|     fn drop(&mut self) {
 | |
|         critical_section::with(|_| {
 | |
|             let pin = self.pin as _;
 | |
|             cpu_regs().imr(0).modify(|w| w.set_line(pin, false));
 | |
|         });
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<'a> Future for ExtiInputFuture<'a> {
 | |
|     type Output = ();
 | |
| 
 | |
|     fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
 | |
|         EXTI_WAKERS[self.pin as usize].register(cx.waker());
 | |
| 
 | |
|         let imr = cpu_regs().imr(0).read();
 | |
|         if !imr.line(self.pin as _) {
 | |
|             Poll::Ready(())
 | |
|         } else {
 | |
|             Poll::Pending
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| macro_rules! foreach_exti_irq {
 | |
|     ($action:ident) => {
 | |
|         foreach_interrupt!(
 | |
|             (EXTI0)  => { $action!(EXTI0); };
 | |
|             (EXTI1)  => { $action!(EXTI1); };
 | |
|             (EXTI2)  => { $action!(EXTI2); };
 | |
|             (EXTI3)  => { $action!(EXTI3); };
 | |
|             (EXTI4)  => { $action!(EXTI4); };
 | |
|             (EXTI5)  => { $action!(EXTI5); };
 | |
|             (EXTI6)  => { $action!(EXTI6); };
 | |
|             (EXTI7)  => { $action!(EXTI7); };
 | |
|             (EXTI8)  => { $action!(EXTI8); };
 | |
|             (EXTI9)  => { $action!(EXTI9); };
 | |
|             (EXTI10) => { $action!(EXTI10); };
 | |
|             (EXTI11) => { $action!(EXTI11); };
 | |
|             (EXTI12) => { $action!(EXTI12); };
 | |
|             (EXTI13) => { $action!(EXTI13); };
 | |
|             (EXTI14) => { $action!(EXTI14); };
 | |
|             (EXTI15) => { $action!(EXTI15); };
 | |
| 
 | |
|             // plus the weird ones
 | |
|             (EXTI0_1)   => { $action!( EXTI0_1 ); };
 | |
|             (EXTI15_10) => { $action!(EXTI15_10); };
 | |
|             (EXTI15_4)  => { $action!(EXTI15_4); };
 | |
|             (EXTI1_0)   => { $action!(EXTI1_0); };
 | |
|             (EXTI2_3)   => { $action!(EXTI2_3); };
 | |
|             (EXTI2_TSC) => { $action!(EXTI2_TSC); };
 | |
|             (EXTI3_2)   => { $action!(EXTI3_2); };
 | |
|             (EXTI4_15)  => { $action!(EXTI4_15); };
 | |
|             (EXTI9_5)   => { $action!(EXTI9_5); };
 | |
|         );
 | |
|     };
 | |
| }
 | |
| 
 | |
| macro_rules! impl_irq {
 | |
|     ($e:ident) => {
 | |
|         #[allow(non_snake_case)]
 | |
|         #[cfg(feature = "rt")]
 | |
|         #[interrupt]
 | |
|         unsafe fn $e() {
 | |
|             on_irq()
 | |
|         }
 | |
|     };
 | |
| }
 | |
| 
 | |
| foreach_exti_irq!(impl_irq);
 | |
| 
 | |
| trait SealedChannel {}
 | |
| 
 | |
| /// EXTI channel trait.
 | |
| #[allow(private_bounds)]
 | |
| pub trait Channel: SealedChannel + Sized {
 | |
|     /// Get the EXTI channel number.
 | |
|     fn number(&self) -> u8;
 | |
| 
 | |
|     /// Type-erase (degrade) this channel into an `AnyChannel`.
 | |
|     ///
 | |
|     /// This converts EXTI channel singletons (`EXTI0`, `EXTI1`, ...), which
 | |
|     /// are all different types, into the same type. It is useful for
 | |
|     /// creating arrays of channels, or avoiding generics.
 | |
|     fn degrade(self) -> AnyChannel {
 | |
|         AnyChannel {
 | |
|             number: self.number() as u8,
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| /// Type-erased (degraded) EXTI channel.
 | |
| ///
 | |
| /// This represents ownership over any EXTI channel, known at runtime.
 | |
| pub struct AnyChannel {
 | |
|     number: u8,
 | |
| }
 | |
| 
 | |
| impl_peripheral!(AnyChannel);
 | |
| impl SealedChannel for AnyChannel {}
 | |
| impl Channel for AnyChannel {
 | |
|     fn number(&self) -> u8 {
 | |
|         self.number
 | |
|     }
 | |
| }
 | |
| 
 | |
| macro_rules! impl_exti {
 | |
|     ($type:ident, $number:expr) => {
 | |
|         impl SealedChannel for peripherals::$type {}
 | |
|         impl Channel for peripherals::$type {
 | |
|             fn number(&self) -> u8 {
 | |
|                 $number
 | |
|             }
 | |
|         }
 | |
|     };
 | |
| }
 | |
| 
 | |
| impl_exti!(EXTI0, 0);
 | |
| impl_exti!(EXTI1, 1);
 | |
| impl_exti!(EXTI2, 2);
 | |
| impl_exti!(EXTI3, 3);
 | |
| impl_exti!(EXTI4, 4);
 | |
| impl_exti!(EXTI5, 5);
 | |
| impl_exti!(EXTI6, 6);
 | |
| impl_exti!(EXTI7, 7);
 | |
| impl_exti!(EXTI8, 8);
 | |
| impl_exti!(EXTI9, 9);
 | |
| impl_exti!(EXTI10, 10);
 | |
| impl_exti!(EXTI11, 11);
 | |
| impl_exti!(EXTI12, 12);
 | |
| impl_exti!(EXTI13, 13);
 | |
| impl_exti!(EXTI14, 14);
 | |
| impl_exti!(EXTI15, 15);
 | |
| 
 | |
| macro_rules! enable_irq {
 | |
|     ($e:ident) => {
 | |
|         crate::interrupt::typelevel::$e::enable();
 | |
|     };
 | |
| }
 | |
| 
 | |
| /// safety: must be called only once
 | |
| pub(crate) unsafe fn init(_cs: critical_section::CriticalSection) {
 | |
|     use crate::interrupt::typelevel::Interrupt;
 | |
| 
 | |
|     foreach_exti_irq!(enable_irq);
 | |
| }
 |