diff options
| -rw-r--r-- | embassy-stm32/src/dma_traits.rs | 11 | ||||
| -rw-r--r-- | embassy-stm32/src/usart/v3.rs | 40 | ||||
| -rw-r--r-- | examples/stm32h7/src/bin/usart.rs | 3 | ||||
| -rw-r--r-- | examples/stm32l4/src/bin/usart.rs | 3 | ||||
| -rw-r--r-- | examples/stm32l4/src/bin/usart_dma.rs | 12 |
5 files changed, 55 insertions, 14 deletions
diff --git a/embassy-stm32/src/dma_traits.rs b/embassy-stm32/src/dma_traits.rs index 8f1a9f40e..6733d911a 100644 --- a/embassy-stm32/src/dma_traits.rs +++ b/embassy-stm32/src/dma_traits.rs | |||
| @@ -1,4 +1,5 @@ | |||
| 1 | use core::future::Future; | 1 | use core::future::Future; |
| 2 | use embassy::util::Unborrow; | ||
| 2 | 3 | ||
| 3 | pub trait WriteDma<T> { | 4 | pub trait WriteDma<T> { |
| 4 | type WriteDmaFuture<'a>: Future<Output = ()> + 'a | 5 | type WriteDmaFuture<'a>: Future<Output = ()> + 'a |
| @@ -19,3 +20,13 @@ pub trait ReadDma<T> { | |||
| 19 | where | 20 | where |
| 20 | T: 'a; | 21 | T: 'a; |
| 21 | } | 22 | } |
| 23 | |||
| 24 | pub struct NoDma; | ||
| 25 | |||
| 26 | unsafe impl Unborrow for NoDma { | ||
| 27 | type Target = NoDma; | ||
| 28 | |||
| 29 | unsafe fn unborrow(self) -> Self::Target { | ||
| 30 | self | ||
| 31 | } | ||
| 32 | } | ||
diff --git a/embassy-stm32/src/usart/v3.rs b/embassy-stm32/src/usart/v3.rs index 0071c597a..4d78abb24 100644 --- a/embassy-stm32/src/usart/v3.rs +++ b/embassy-stm32/src/usart/v3.rs | |||
| @@ -6,20 +6,29 @@ use embassy_extras::unborrow; | |||
| 6 | use crate::pac::usart::{regs, vals}; | 6 | use crate::pac::usart::{regs, vals}; |
| 7 | 7 | ||
| 8 | use super::*; | 8 | use super::*; |
| 9 | use core::future::Future; | ||
| 10 | use futures::TryFutureExt; | ||
| 9 | 11 | ||
| 10 | pub struct Uart<'d, T: Instance> { | 12 | use crate::dma_traits::NoDma; |
| 13 | |||
| 14 | #[allow(dead_code)] | ||
| 15 | pub struct Uart<'d, T: Instance, TxDma = NoDma, RxDma = NoDma> { | ||
| 11 | inner: T, | 16 | inner: T, |
| 12 | phantom: PhantomData<&'d mut T>, | 17 | phantom: PhantomData<&'d mut T>, |
| 18 | tx_dma: TxDma, | ||
| 19 | rx_dma: RxDma, | ||
| 13 | } | 20 | } |
| 14 | 21 | ||
| 15 | impl<'d, T: Instance> Uart<'d, T> { | 22 | impl<'d, T: Instance, TxDma, RxDma> Uart<'d, T, TxDma, RxDma> { |
| 16 | pub fn new( | 23 | pub fn new( |
| 17 | inner: impl Unborrow<Target = T>, | 24 | inner: impl Unborrow<Target = T>, |
| 18 | rx: impl Unborrow<Target = impl RxPin<T>>, | 25 | rx: impl Unborrow<Target = impl RxPin<T>>, |
| 19 | tx: impl Unborrow<Target = impl TxPin<T>>, | 26 | tx: impl Unborrow<Target = impl TxPin<T>>, |
| 27 | tx_dma: impl Unborrow<Target = TxDma>, | ||
| 28 | rx_dma: impl Unborrow<Target = RxDma>, | ||
| 20 | config: Config, | 29 | config: Config, |
| 21 | ) -> Self { | 30 | ) -> Self { |
| 22 | unborrow!(inner, rx, tx); | 31 | unborrow!(inner, rx, tx, tx_dma, rx_dma); |
| 23 | 32 | ||
| 24 | // Uncomment once we find all of the H7's UART clocks. | 33 | // Uncomment once we find all of the H7's UART clocks. |
| 25 | T::enable(); | 34 | T::enable(); |
| @@ -54,11 +63,16 @@ impl<'d, T: Instance> Uart<'d, T> { | |||
| 54 | Self { | 63 | Self { |
| 55 | inner, | 64 | inner, |
| 56 | phantom: PhantomData, | 65 | phantom: PhantomData, |
| 66 | tx_dma, | ||
| 67 | rx_dma, | ||
| 57 | } | 68 | } |
| 58 | } | 69 | } |
| 59 | 70 | ||
| 60 | #[cfg(any(dma, dmamux))] | 71 | async fn write_dma(&mut self, buffer: &[u8]) -> Result<(), Error> |
| 61 | pub async fn write_dma(&mut self, ch: &mut impl TxDma<T>, buffer: &[u8]) -> Result<(), Error> { | 72 | where |
| 73 | TxDma: crate::usart::TxDma<T>, | ||
| 74 | { | ||
| 75 | let ch = &mut self.tx_dma; | ||
| 62 | unsafe { | 76 | unsafe { |
| 63 | self.inner.regs().cr3().modify(|reg| { | 77 | self.inner.regs().cr3().modify(|reg| { |
| 64 | reg.set_dmat(true); | 78 | reg.set_dmat(true); |
| @@ -99,7 +113,9 @@ impl<'d, T: Instance> Uart<'d, T> { | |||
| 99 | } | 113 | } |
| 100 | } | 114 | } |
| 101 | 115 | ||
| 102 | impl<'d, T: Instance> embedded_hal::blocking::serial::Write<u8> for Uart<'d, T> { | 116 | impl<'d, T: Instance, RxDma> embedded_hal::blocking::serial::Write<u8> |
| 117 | for Uart<'d, T, NoDma, RxDma> | ||
| 118 | { | ||
| 103 | type Error = Error; | 119 | type Error = Error; |
| 104 | fn bwrite_all(&mut self, buffer: &[u8]) -> Result<(), Self::Error> { | 120 | fn bwrite_all(&mut self, buffer: &[u8]) -> Result<(), Self::Error> { |
| 105 | unsafe { | 121 | unsafe { |
| @@ -119,3 +135,15 @@ impl<'d, T: Instance> embedded_hal::blocking::serial::Write<u8> for Uart<'d, T> | |||
| 119 | Ok(()) | 135 | Ok(()) |
| 120 | } | 136 | } |
| 121 | } | 137 | } |
| 138 | |||
| 139 | // rustfmt::skip because intellij removes the 'where' claus on the associated type. | ||
| 140 | #[rustfmt::skip] | ||
| 141 | impl<'d, T: Instance, TxDma, RxDma> embassy_traits::uart::Write for Uart<'d, T, TxDma, RxDma> | ||
| 142 | where TxDma: crate::usart::TxDma<T> | ||
| 143 | { | ||
| 144 | type WriteFuture<'a> where Self: 'a = impl Future<Output = Result<(), embassy_traits::uart::Error>>; | ||
| 145 | |||
| 146 | fn write<'a>(&'a mut self, buf: &'a [u8]) -> Self::WriteFuture<'a> { | ||
| 147 | self.write_dma(buf).map_err(|_| embassy_traits::uart::Error::Other) | ||
| 148 | } | ||
| 149 | } | ||
diff --git a/examples/stm32h7/src/bin/usart.rs b/examples/stm32h7/src/bin/usart.rs index b44d712b5..d8e60158b 100644 --- a/examples/stm32h7/src/bin/usart.rs +++ b/examples/stm32h7/src/bin/usart.rs | |||
| @@ -14,6 +14,7 @@ use embassy::executor::Executor; | |||
| 14 | use embassy::time::Clock; | 14 | use embassy::time::Clock; |
| 15 | use embassy::util::Forever; | 15 | use embassy::util::Forever; |
| 16 | use embassy_stm32::usart::{Config, Uart}; | 16 | use embassy_stm32::usart::{Config, Uart}; |
| 17 | use embassy_stm32::dma_traits::NoDma; | ||
| 17 | use example_common::*; | 18 | use example_common::*; |
| 18 | 19 | ||
| 19 | use stm32h7xx_hal as hal; | 20 | use stm32h7xx_hal as hal; |
| @@ -27,7 +28,7 @@ async fn main_task() { | |||
| 27 | let p = embassy_stm32::init(Default::default()); | 28 | let p = embassy_stm32::init(Default::default()); |
| 28 | 29 | ||
| 29 | let config = Config::default(); | 30 | let config = Config::default(); |
| 30 | let mut usart = Uart::new(p.UART7, p.PF6, p.PF7, config); | 31 | let mut usart = Uart::new(p.UART7, p.PF6, p.PF7, NoDma, NoDma, config); |
| 31 | 32 | ||
| 32 | usart.bwrite_all(b"Hello Embassy World!\r\n").unwrap(); | 33 | usart.bwrite_all(b"Hello Embassy World!\r\n").unwrap(); |
| 33 | info!("wrote Hello, starting echo"); | 34 | info!("wrote Hello, starting echo"); |
diff --git a/examples/stm32l4/src/bin/usart.rs b/examples/stm32l4/src/bin/usart.rs index 0b14eeb59..dc0f649b7 100644 --- a/examples/stm32l4/src/bin/usart.rs +++ b/examples/stm32l4/src/bin/usart.rs | |||
| @@ -18,13 +18,14 @@ use example_common::*; | |||
| 18 | 18 | ||
| 19 | use cortex_m_rt::entry; | 19 | use cortex_m_rt::entry; |
| 20 | use stm32l4::stm32l4x5 as pac; | 20 | use stm32l4::stm32l4x5 as pac; |
| 21 | use embassy_stm32::dma_traits::NoDma; | ||
| 21 | 22 | ||
| 22 | #[embassy::task] | 23 | #[embassy::task] |
| 23 | async fn main_task() { | 24 | async fn main_task() { |
| 24 | let p = embassy_stm32::init(Default::default()); | 25 | let p = embassy_stm32::init(Default::default()); |
| 25 | 26 | ||
| 26 | let config = Config::default(); | 27 | let config = Config::default(); |
| 27 | let mut usart = Uart::new(p.UART4, p.PA1, p.PA0, config); | 28 | let mut usart = Uart::new(p.UART4, p.PA1, p.PA0, NoDma, NoDma, config); |
| 28 | 29 | ||
| 29 | usart.bwrite_all(b"Hello Embassy World!\r\n").unwrap(); | 30 | usart.bwrite_all(b"Hello Embassy World!\r\n").unwrap(); |
| 30 | info!("wrote Hello, starting echo"); | 31 | info!("wrote Hello, starting echo"); |
diff --git a/examples/stm32l4/src/bin/usart_dma.rs b/examples/stm32l4/src/bin/usart_dma.rs index cc630e0df..6a2341859 100644 --- a/examples/stm32l4/src/bin/usart_dma.rs +++ b/examples/stm32l4/src/bin/usart_dma.rs | |||
| @@ -17,22 +17,22 @@ use embassy_stm32::usart::{Config, Uart}; | |||
| 17 | use example_common::*; | 17 | use example_common::*; |
| 18 | use heapless::String; | 18 | use heapless::String; |
| 19 | use stm32l4::stm32l4x5 as pac; | 19 | use stm32l4::stm32l4x5 as pac; |
| 20 | use embassy_stm32::dma_traits::NoDma; | ||
| 21 | use embassy_traits::uart::Write as AsyncWrite; | ||
| 20 | 22 | ||
| 21 | #[embassy::task] | 23 | #[embassy::task] |
| 22 | async fn main_task() { | 24 | async fn main_task() { |
| 23 | let mut p = embassy_stm32::init(Default::default()); | 25 | let p = embassy_stm32::init(Default::default()); |
| 24 | 26 | ||
| 25 | let config = Config::default(); | 27 | let config = Config::default(); |
| 26 | let mut usart = Uart::new(p.UART4, p.PA1, p.PA0, config); | 28 | let mut usart = Uart::new(p.UART4, p.PA1, p.PA0, p.DMA1_3, NoDma, config); |
| 27 | 29 | ||
| 28 | for n in 0u32.. { | 30 | for n in 0u32.. { |
| 29 | let mut s: String<128> = String::new(); | 31 | let mut s: String<128> = String::new(); |
| 30 | core::write!(&mut s, "Hello DMA World {}!\r\n", n).unwrap(); | 32 | core::write!(&mut s, "Hello DMA World {}!\r\n", n).unwrap(); |
| 31 | 33 | ||
| 32 | usart | 34 | usart.write( s.as_bytes() ).await.ok(); |
| 33 | .write_dma(&mut p.DMA1_3, s.as_bytes()) | 35 | |
| 34 | .await | ||
| 35 | .unwrap(); | ||
| 36 | info!("wrote DMA"); | 36 | info!("wrote DMA"); |
| 37 | } | 37 | } |
| 38 | } | 38 | } |
