//! Async low power Serial. //! //! The peripheral is autmatically enabled and disabled as required to save power. //! Lowest power consumption can only be guaranteed if the send receive futures //! are dropped correctly (e.g. not using `mem::forget()`). use core::future::Future; use core::task::{Context, Poll}; use embassy::interrupt::OwnedInterrupt; use embassy::util::Signal; use embedded_dma::{StaticReadBuffer, StaticWriteBuffer, WriteBuffer}; use crate::hal::dma::config::DmaConfig; use crate::hal::dma::traits::{PeriAddress, Stream}; use crate::hal::dma::{ Channel4, MemoryToPeripheral, PeripheralToMemory, Stream2, Stream7, StreamsTuple, Transfer, }; use crate::hal::gpio::gpioa::{PA10, PA9}; use crate::hal::gpio::{Alternate, AF7}; use crate::hal::prelude::*; use crate::hal::rcc::Clocks; use crate::hal::serial::config::{ Config as SerialConfig, DmaConfig as SerialDmaConfig, Parity, StopBits, WordLength, }; use crate::hal::serial::{Event as SerialEvent, Serial as HalSerial}; use crate::hal::time::Bps; use crate::interrupt; use crate::pac::Interrupt; use crate::pac::{DMA2, USART1}; /// Interface to the Serial peripheral pub struct Serial, TSTREAM: Stream, RSTREAM: Stream> { // tx_transfer: Transfer, Channel4, USART1, MemoryToPeripheral, &mut [u8; 20]>, // rx_transfer: Transfer, Channel4, USART1, PeripheralToMemory, &mut [u8; 20]>, tx_stream: Option, rx_stream: Option, usart: Option, } struct State { tx_done: Signal<()>, rx_done: Signal<()>, } static STATE: State = State { tx_done: Signal::new(), rx_done: Signal::new(), }; impl Serial, Stream2> { pub fn new( txd: PA9>, rxd: PA10>, tx_int: interrupt::DMA2_STREAM2Interrupt, rx_int: interrupt::DMA2_STREAM7Interrupt, dma: DMA2, usart: USART1, parity: Parity, baudrate: Bps, clocks: Clocks, ) -> Self { let serial = HalSerial::usart1( usart, (txd, rxd), SerialConfig { baudrate: baudrate, wordlength: WordLength::DataBits8, parity: Parity::ParityNone, stopbits: StopBits::STOP1, dma: SerialDmaConfig::TxRx, }, clocks, ) .unwrap(); let (usart, _) = serial.release(); // serial.listen(SerialEvent::Idle); // Register ISR tx_int.set_handler(Self::on_tx_irq); tx_int.unpend(); tx_int.enable(); rx_int.set_handler(Self::on_rx_irq); rx_int.unpend(); rx_int.enable(); let streams = StreamsTuple::new(dma); Serial { tx_stream: Some(streams.7), rx_stream: Some(streams.2), usart: Some(usart), } } unsafe fn on_tx_irq() { STATE.tx_done.signal(()); } unsafe fn on_rx_irq() { STATE.rx_done.signal(()); } /// Sends serial data. /// /// `tx_buffer` is marked as static as per `embedded-dma` requirements. /// It it safe to use a buffer with a non static lifetime if memory is not /// reused until the future has finished. pub fn send<'a, B>( &'a mut self, tx_buffer: B, ) -> SendFuture<'a, B, USART1, Stream7, Stream2, Channel4> where B: WriteBuffer + 'static, { let tx_stream = self.tx_stream.take().unwrap(); let usart = self.usart.take().unwrap(); let mut tx_transfer = Transfer::init( tx_stream, usart, tx_buffer, None, DmaConfig::default() .transfer_complete_interrupt(true) .memory_increment(true) .double_buffer(false), ); STATE.tx_done.reset(); SendFuture { Serial: self, tx_transfer: Some(tx_transfer), // tx_stream: Some(tx_stream), // usart: Some(usart), } } /// Receives serial data. /// /// The future is pending until the buffer is completely filled. /// A common pattern is to use [`stop()`](ReceiveFuture::stop) to cancel /// unfinished transfers after a timeout to prevent lockup when no more data /// is incoming. /// /// `rx_buffer` is marked as static as per `embedded-dma` requirements. /// It it safe to use a buffer with a non static lifetime if memory is not /// reused until the future has finished. pub fn receive<'a, B>( &'a mut self, rx_buffer: B, ) -> ReceiveFuture<'a, B, USART1, Stream7, Stream2, Channel4> where B: WriteBuffer + 'static, { let rx_stream = self.rx_stream.take().unwrap(); let usart = self.usart.take().unwrap(); let mut rx_transfer = Transfer::init( rx_stream, usart, rx_buffer, None, DmaConfig::default() .transfer_complete_interrupt(true) .half_transfer_interrupt(true) .memory_increment(true) .double_buffer(false), ); STATE.rx_done.reset(); ReceiveFuture { Serial: self, rx_transfer: Some(rx_transfer), } } } /// Future for the [`LowPowerSerial::send()`] method. pub struct SendFuture< 'a, B: WriteBuffer + 'static, USART: PeriAddress, TSTREAM: Stream, RSTREAM: Stream, CHANNEL, > { Serial: &'a mut Serial, tx_transfer: Option>, } // impl<'a, B> Drop for SendFuture<'a, B> // where // B: WriteBuffer + 'static, // { // fn drop(self: &mut Self) {} // } impl<'a, B> Future for SendFuture<'a, B, USART1, Stream7, Stream2, Channel4> where B: WriteBuffer + 'static, { type Output = (); fn poll(self: core::pin::Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<()> { let Self { Serial, tx_transfer, } = unsafe { self.get_unchecked_mut() }; let mut taken = tx_transfer.take().unwrap(); if Stream7::::get_transfer_complete_flag() { let (tx_stream, usart, buf, _) = taken.free(); Serial.tx_stream.replace(tx_stream); Serial.usart.replace(usart); Poll::Ready(()) } else { // waker_interrupt!(DMA2_STREAM7, cx.waker().clone()); taken.start(|_usart| {}); tx_transfer.replace(taken); // Poll::Pending STATE.tx_done.poll_wait(cx) } } } /// Future for the [`Serial::receive()`] method. pub struct ReceiveFuture< 'a, B: WriteBuffer + 'static, USART: PeriAddress, TSTREAM: Stream, RSTREAM: Stream, CHANNEL, > { Serial: &'a mut Serial, rx_transfer: Option>, } // impl<'a, B> Drop for ReceiveFuture<'a, B, USART1, Stream7, Channel4> // where // B: WriteBuffer + 'static, // { // fn drop(self: &mut Self) {} // } impl<'a, B> Future for ReceiveFuture<'a, B, USART1, Stream7, Stream2, Channel4> where B: WriteBuffer + 'static + Unpin, { type Output = B; fn poll(self: core::pin::Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { let Self { Serial, rx_transfer, } = unsafe { self.get_unchecked_mut() }; let mut taken = rx_transfer.take().unwrap(); if Stream7::::get_transfer_complete_flag() { let (rx_stream, usart, buf, _) = rx_transfer.take().unwrap().free(); Serial.rx_stream.replace(rx_stream); Serial.usart.replace(usart); Poll::Ready(buf) } else { // waker_interrupt!(DMA2_STREAM2, cx.waker().clone()); taken.start(|_usart| {}); rx_transfer.replace(taken); STATE.rx_done.poll_wait(cx); /* Note: we have to do this because rx_transfer owns the buffer and we can't access it until the transfer is completed. Therefore we can't pass the buffer to poll_wait, but we still need to be woken. */ Poll::Pending } } }