2021-05-06 01:59:16 +00:00
|
|
|
#![macro_use]
|
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
use core::future::poll_fn;
|
2021-12-08 04:12:48 +00:00
|
|
|
use core::marker::PhantomData;
|
2022-10-26 16:58:22 +00:00
|
|
|
use core::task::Poll;
|
2022-06-12 20:15:44 +00:00
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
use atomic_polyfill::{compiler_fence, Ordering};
|
|
|
|
use embassy_cortex_m::interrupt::InterruptExt;
|
2022-07-23 12:00:19 +00:00
|
|
|
use embassy_hal_common::{into_ref, PeripheralRef};
|
2021-04-14 13:34:58 +00:00
|
|
|
|
2021-12-08 04:12:48 +00:00
|
|
|
use crate::dma::NoDma;
|
2022-02-23 23:20:43 +00:00
|
|
|
use crate::gpio::sealed::AFType;
|
2022-08-19 13:27:11 +00:00
|
|
|
#[cfg(any(lpuart_v1, lpuart_v2))]
|
|
|
|
use crate::pac::lpuart::{regs, vals, Lpuart as Regs};
|
|
|
|
#[cfg(not(any(lpuart_v1, lpuart_v2)))]
|
|
|
|
use crate::pac::usart::{regs, vals, Usart as Regs};
|
2022-10-26 16:58:22 +00:00
|
|
|
use crate::time::Hertz;
|
2022-07-23 12:00:19 +00:00
|
|
|
use crate::{peripherals, Peripheral};
|
2021-04-14 13:34:58 +00:00
|
|
|
|
2021-06-30 18:37:35 +00:00
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
|
|
|
pub enum DataBits {
|
|
|
|
DataBits8,
|
|
|
|
DataBits9,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
|
|
|
pub enum Parity {
|
|
|
|
ParityNone,
|
|
|
|
ParityEven,
|
|
|
|
ParityOdd,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
|
|
|
pub enum StopBits {
|
|
|
|
#[doc = "1 stop bit"]
|
|
|
|
STOP1,
|
|
|
|
#[doc = "0.5 stop bits"]
|
|
|
|
STOP0P5,
|
|
|
|
#[doc = "2 stop bits"]
|
|
|
|
STOP2,
|
|
|
|
#[doc = "1.5 stop bits"]
|
|
|
|
STOP1P5,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[non_exhaustive]
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
|
|
|
pub struct Config {
|
|
|
|
pub baudrate: u32,
|
|
|
|
pub data_bits: DataBits,
|
|
|
|
pub stop_bits: StopBits,
|
|
|
|
pub parity: Parity,
|
2022-10-26 16:58:22 +00:00
|
|
|
/// if true, on read-like method, if there is a latent error pending,
|
|
|
|
/// read will abort, the error reported and cleared
|
|
|
|
/// if false, the error is ignored and cleared
|
|
|
|
pub detect_previous_overrun: bool,
|
2021-06-30 18:37:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Config {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
baudrate: 115200,
|
|
|
|
data_bits: DataBits::DataBits8,
|
|
|
|
stop_bits: StopBits::STOP1,
|
|
|
|
parity: Parity::ParityNone,
|
2022-10-26 16:58:22 +00:00
|
|
|
// historical behavior
|
|
|
|
detect_previous_overrun: false,
|
2021-06-30 18:37:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-15 01:52:58 +00:00
|
|
|
/// Serial error
|
|
|
|
#[derive(Debug, Eq, PartialEq, Copy, Clone)]
|
2021-07-04 21:34:37 +00:00
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
2021-04-14 13:34:58 +00:00
|
|
|
#[non_exhaustive]
|
2021-05-15 01:52:58 +00:00
|
|
|
pub enum Error {
|
|
|
|
/// Framing error
|
|
|
|
Framing,
|
|
|
|
/// Noise error
|
|
|
|
Noise,
|
|
|
|
/// RX buffer overrun
|
|
|
|
Overrun,
|
|
|
|
/// Parity check error
|
|
|
|
Parity,
|
2022-10-26 16:58:22 +00:00
|
|
|
/// Buffer too large for DMA
|
|
|
|
BufferTooLong,
|
2021-04-14 13:34:58 +00:00
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
pub struct Uart<'d, T: BasicInstance, TxDma = NoDma, RxDma = NoDma> {
|
2022-03-14 21:03:35 +00:00
|
|
|
tx: UartTx<'d, T, TxDma>,
|
|
|
|
rx: UartRx<'d, T, RxDma>,
|
2021-12-08 04:12:48 +00:00
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
pub struct UartTx<'d, T: BasicInstance, TxDma = NoDma> {
|
2022-03-14 21:03:35 +00:00
|
|
|
phantom: PhantomData<&'d mut T>,
|
2022-07-23 12:00:19 +00:00
|
|
|
tx_dma: PeripheralRef<'d, TxDma>,
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
2021-12-08 04:12:48 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
pub struct UartRx<'d, T: BasicInstance, RxDma = NoDma> {
|
2022-10-26 16:58:22 +00:00
|
|
|
_peri: PeripheralRef<'d, T>,
|
2022-07-23 12:00:19 +00:00
|
|
|
rx_dma: PeripheralRef<'d, RxDma>,
|
2022-10-26 16:58:22 +00:00
|
|
|
detect_previous_overrun: bool,
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
2021-12-08 04:12:48 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma> UartTx<'d, T, TxDma> {
|
2022-10-27 08:52:21 +00:00
|
|
|
/// usefull if you only want Uart Tx. It saves 1 pin and consumes a little less power
|
|
|
|
pub fn new(
|
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
|
|
|
tx: impl Peripheral<P = impl TxPin<T>> + 'd,
|
|
|
|
tx_dma: impl Peripheral<P = TxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
|
|
|
T::enable();
|
|
|
|
T::reset();
|
|
|
|
|
|
|
|
Self::new_inner(peri, tx, tx_dma, config)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_with_cts(
|
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
|
|
|
tx: impl Peripheral<P = impl TxPin<T>> + 'd,
|
|
|
|
cts: impl Peripheral<P = impl CtsPin<T>> + 'd,
|
|
|
|
tx_dma: impl Peripheral<P = TxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
|
|
|
into_ref!(cts);
|
|
|
|
|
|
|
|
T::enable();
|
|
|
|
T::reset();
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
cts.set_as_af(cts.af_num(), AFType::Input);
|
|
|
|
T::regs().cr3().write(|w| {
|
|
|
|
w.set_ctse(true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
Self::new_inner(peri, tx, tx_dma, config)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn new_inner(
|
|
|
|
_peri: impl Peripheral<P = T> + 'd,
|
|
|
|
tx: impl Peripheral<P = impl TxPin<T>> + 'd,
|
|
|
|
tx_dma: impl Peripheral<P = TxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
|
|
|
into_ref!(_peri, tx, tx_dma);
|
|
|
|
|
|
|
|
let r = T::regs();
|
|
|
|
|
|
|
|
configure(r, &config, T::frequency(), T::MULTIPLIER);
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
tx.set_as_af(tx.af_num(), AFType::OutputPushPull);
|
|
|
|
|
|
|
|
r.cr2().write(|_w| {});
|
|
|
|
r.cr1().write(|w| {
|
|
|
|
// enable uart
|
|
|
|
w.set_ue(true);
|
|
|
|
// enable transceiver
|
|
|
|
w.set_te(true);
|
|
|
|
// configure word size
|
|
|
|
w.set_m0(if config.parity != Parity::ParityNone {
|
|
|
|
vals::M0::BIT9
|
|
|
|
} else {
|
|
|
|
vals::M0::BIT8
|
|
|
|
});
|
|
|
|
// configure parity
|
|
|
|
w.set_pce(config.parity != Parity::ParityNone);
|
|
|
|
w.set_ps(match config.parity {
|
|
|
|
Parity::ParityOdd => vals::Ps::ODD,
|
|
|
|
Parity::ParityEven => vals::Ps::EVEN,
|
|
|
|
_ => vals::Ps::EVEN,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// create state once!
|
|
|
|
let _s = T::state();
|
|
|
|
|
2021-12-08 04:12:48 +00:00
|
|
|
Self {
|
|
|
|
tx_dma,
|
2022-03-14 21:03:35 +00:00
|
|
|
phantom: PhantomData,
|
2021-12-08 04:12:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-14 21:43:40 +00:00
|
|
|
pub async fn write(&mut self, buffer: &[u8]) -> Result<(), Error>
|
2021-12-08 04:12:48 +00:00
|
|
|
where
|
|
|
|
TxDma: crate::usart::TxDma<T>,
|
|
|
|
{
|
|
|
|
let ch = &mut self.tx_dma;
|
|
|
|
let request = ch.request();
|
|
|
|
unsafe {
|
2022-03-14 21:03:35 +00:00
|
|
|
T::regs().cr3().modify(|reg| {
|
2021-12-08 04:12:48 +00:00
|
|
|
reg.set_dmat(true);
|
|
|
|
});
|
|
|
|
}
|
2022-03-17 17:41:44 +00:00
|
|
|
// If we don't assign future to a variable, the data register pointer
|
|
|
|
// is held across an await and makes the future non-Send.
|
2022-03-17 16:23:47 +00:00
|
|
|
let transfer = crate::dma::write(ch, request, buffer, tdr(T::regs()));
|
|
|
|
transfer.await;
|
2021-12-08 04:12:48 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2022-03-14 21:03:35 +00:00
|
|
|
pub fn blocking_write(&mut self, buffer: &[u8]) -> Result<(), Error> {
|
|
|
|
unsafe {
|
|
|
|
let r = T::regs();
|
|
|
|
for &b in buffer {
|
|
|
|
while !sr(r).read().txe() {}
|
|
|
|
tdr(r).write_volatile(b);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn blocking_flush(&mut self) -> Result<(), Error> {
|
|
|
|
unsafe {
|
|
|
|
let r = T::regs();
|
|
|
|
while !sr(r).read().tc() {}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, RxDma> UartRx<'d, T, RxDma> {
|
2022-10-26 16:58:22 +00:00
|
|
|
/// usefull if you only want Uart Rx. It saves 1 pin and consumes a little less power
|
|
|
|
pub fn new(
|
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
|
|
|
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
|
|
|
rx: impl Peripheral<P = impl RxPin<T>> + 'd,
|
|
|
|
rx_dma: impl Peripheral<P = RxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
2022-10-27 08:52:21 +00:00
|
|
|
T::enable();
|
|
|
|
T::reset();
|
|
|
|
|
|
|
|
Self::new_inner(peri, irq, rx, rx_dma, config)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_with_rts(
|
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
|
|
|
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
|
|
|
rx: impl Peripheral<P = impl RxPin<T>> + 'd,
|
|
|
|
rts: impl Peripheral<P = impl RtsPin<T>> + 'd,
|
|
|
|
rx_dma: impl Peripheral<P = RxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
|
|
|
into_ref!(rts);
|
2022-10-26 16:58:22 +00:00
|
|
|
|
|
|
|
T::enable();
|
|
|
|
T::reset();
|
|
|
|
|
2022-10-27 08:52:21 +00:00
|
|
|
unsafe {
|
|
|
|
rts.set_as_af(rts.af_num(), AFType::OutputPushPull);
|
|
|
|
T::regs().cr3().write(|w| {
|
|
|
|
w.set_rtse(true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Self::new_inner(peri, irq, rx, rx_dma, config)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn new_inner(
|
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
|
|
|
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
|
|
|
rx: impl Peripheral<P = impl RxPin<T>> + 'd,
|
|
|
|
rx_dma: impl Peripheral<P = RxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
|
|
|
into_ref!(peri, irq, rx, rx_dma);
|
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
let r = T::regs();
|
|
|
|
|
|
|
|
configure(r, &config, T::frequency(), T::MULTIPLIER);
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
rx.set_as_af(rx.af_num(), AFType::Input);
|
|
|
|
|
|
|
|
r.cr2().write(|_w| {});
|
2022-10-27 08:52:21 +00:00
|
|
|
r.cr3().modify(|w| {
|
2022-10-26 16:58:22 +00:00
|
|
|
// enable Error Interrupt: (Frame error, Noise error, Overrun error)
|
|
|
|
w.set_eie(true);
|
|
|
|
});
|
|
|
|
r.cr1().write(|w| {
|
|
|
|
// enable uart
|
|
|
|
w.set_ue(true);
|
|
|
|
// enable receiver
|
|
|
|
w.set_re(true);
|
|
|
|
// configure word size
|
|
|
|
w.set_m0(if config.parity != Parity::ParityNone {
|
|
|
|
vals::M0::BIT9
|
|
|
|
} else {
|
|
|
|
vals::M0::BIT8
|
|
|
|
});
|
|
|
|
// configure parity
|
|
|
|
w.set_pce(config.parity != Parity::ParityNone);
|
|
|
|
w.set_ps(match config.parity {
|
|
|
|
Parity::ParityOdd => vals::Ps::ODD,
|
|
|
|
Parity::ParityEven => vals::Ps::EVEN,
|
|
|
|
_ => vals::Ps::EVEN,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
irq.set_handler(Self::on_interrupt);
|
|
|
|
irq.unpend();
|
|
|
|
irq.enable();
|
|
|
|
|
|
|
|
// create state once!
|
|
|
|
let _s = T::state();
|
|
|
|
|
2022-03-14 21:03:35 +00:00
|
|
|
Self {
|
2022-10-26 16:58:22 +00:00
|
|
|
_peri: peri,
|
2022-03-14 21:03:35 +00:00
|
|
|
rx_dma,
|
2022-10-26 16:58:22 +00:00
|
|
|
detect_previous_overrun: config.detect_previous_overrun,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn on_interrupt(_: *mut ()) {
|
|
|
|
let r = T::regs();
|
|
|
|
let s = T::state();
|
|
|
|
|
|
|
|
let (sr, cr1, cr3) = unsafe { (sr(r).read(), r.cr1().read(), r.cr3().read()) };
|
|
|
|
|
|
|
|
let has_errors = (sr.pe() && cr1.peie()) || ((sr.fe() || sr.ne() || sr.ore()) && cr3.eie());
|
|
|
|
|
|
|
|
if has_errors {
|
|
|
|
// clear all interrupts and DMA Rx Request
|
|
|
|
unsafe {
|
|
|
|
r.cr1().modify(|w| {
|
|
|
|
// disable RXNE interrupt
|
|
|
|
w.set_rxneie(false);
|
|
|
|
// disable parity interrupt
|
|
|
|
w.set_peie(false);
|
|
|
|
// disable idle line interrupt
|
|
|
|
w.set_idleie(false);
|
|
|
|
});
|
|
|
|
r.cr3().modify(|w| {
|
|
|
|
// disable Error Interrupt: (Frame error, Noise error, Overrun error)
|
|
|
|
w.set_eie(false);
|
|
|
|
// disable DMA Rx Request
|
|
|
|
w.set_dmar(false);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
compiler_fence(Ordering::SeqCst);
|
|
|
|
|
|
|
|
s.rx_waker.wake();
|
|
|
|
} else if cr1.idleie() && sr.idle() {
|
|
|
|
// IDLE detected: no more data will come
|
|
|
|
unsafe {
|
|
|
|
r.cr1().modify(|w| {
|
|
|
|
// disable idle line detection
|
|
|
|
w.set_idleie(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
r.cr3().modify(|w| {
|
|
|
|
// disable DMA Rx Request
|
|
|
|
w.set_dmar(false);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
compiler_fence(Ordering::SeqCst);
|
|
|
|
|
|
|
|
s.rx_waker.wake();
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-14 21:43:40 +00:00
|
|
|
pub async fn read(&mut self, buffer: &mut [u8]) -> Result<(), Error>
|
2021-12-08 04:12:48 +00:00
|
|
|
where
|
|
|
|
RxDma: crate::usart::RxDma<T>,
|
|
|
|
{
|
2022-10-26 16:58:22 +00:00
|
|
|
self.inner_read(buffer, false).await?;
|
|
|
|
|
2021-12-08 04:12:48 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2022-09-29 05:58:11 +00:00
|
|
|
pub fn nb_read(&mut self) -> Result<u8, nb::Error<Error>> {
|
|
|
|
let r = T::regs();
|
|
|
|
unsafe {
|
|
|
|
let sr = sr(r).read();
|
|
|
|
if sr.pe() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
Err(nb::Error::Other(Error::Parity))
|
|
|
|
} else if sr.fe() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
Err(nb::Error::Other(Error::Framing))
|
|
|
|
} else if sr.ne() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
Err(nb::Error::Other(Error::Noise))
|
|
|
|
} else if sr.ore() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
Err(nb::Error::Other(Error::Overrun))
|
|
|
|
} else if sr.rxne() {
|
|
|
|
Ok(rdr(r).read_volatile())
|
|
|
|
} else {
|
|
|
|
Err(nb::Error::WouldBlock)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-14 21:43:40 +00:00
|
|
|
pub fn blocking_read(&mut self, buffer: &mut [u8]) -> Result<(), Error> {
|
2021-12-08 04:12:48 +00:00
|
|
|
unsafe {
|
2022-03-14 21:03:35 +00:00
|
|
|
let r = T::regs();
|
2021-12-08 04:12:48 +00:00
|
|
|
for b in buffer {
|
|
|
|
loop {
|
|
|
|
let sr = sr(r).read();
|
|
|
|
if sr.pe() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
return Err(Error::Parity);
|
|
|
|
} else if sr.fe() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
return Err(Error::Framing);
|
|
|
|
} else if sr.ne() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
return Err(Error::Noise);
|
|
|
|
} else if sr.ore() {
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
return Err(Error::Overrun);
|
|
|
|
} else if sr.rxne() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*b = rdr(r).read_volatile();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2022-10-26 16:58:22 +00:00
|
|
|
|
|
|
|
pub async fn read_until_idle(&mut self, buffer: &mut [u8]) -> Result<usize, Error>
|
|
|
|
where
|
|
|
|
RxDma: crate::usart::RxDma<T>,
|
|
|
|
{
|
|
|
|
self.inner_read(buffer, true).await
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn inner_read(&mut self, buffer: &mut [u8], enable_idle_line_detection: bool) -> Result<usize, Error>
|
|
|
|
where
|
|
|
|
RxDma: crate::usart::RxDma<T>,
|
|
|
|
{
|
|
|
|
if buffer.is_empty() {
|
|
|
|
return Ok(0);
|
|
|
|
} else if buffer.len() > 0xFFFF {
|
|
|
|
return Err(Error::BufferTooLong);
|
|
|
|
}
|
|
|
|
|
|
|
|
let r = T::regs();
|
|
|
|
|
|
|
|
let buffer_len = buffer.len();
|
|
|
|
|
|
|
|
let ch = &mut self.rx_dma;
|
|
|
|
let request = ch.request();
|
|
|
|
|
|
|
|
// SAFETY: The only way we might have a problem is using split rx and tx
|
|
|
|
// here we only modify or read Rx related flags, interrupts and DMA channel
|
|
|
|
unsafe {
|
|
|
|
// Start USART DMA
|
|
|
|
// will not do anything yet because DMAR is not yet set
|
|
|
|
ch.start_read(request, rdr(r), buffer, Default::default());
|
|
|
|
|
|
|
|
// clear ORE flag just before enabling DMA Rx Request: can be mandatory for the second transfer
|
|
|
|
if !self.detect_previous_overrun {
|
|
|
|
let sr = sr(r).read();
|
|
|
|
// This read also clears the error and idle interrupt flags on v1.
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
clear_interrupt_flags(r, sr);
|
|
|
|
}
|
|
|
|
|
|
|
|
r.cr1().modify(|w| {
|
|
|
|
// disable RXNE interrupt
|
|
|
|
w.set_rxneie(false);
|
|
|
|
// enable parity interrupt if not ParityNone
|
|
|
|
w.set_peie(w.pce());
|
|
|
|
});
|
|
|
|
|
|
|
|
r.cr3().modify(|w| {
|
|
|
|
// enable Error Interrupt: (Frame error, Noise error, Overrun error)
|
|
|
|
w.set_eie(true);
|
|
|
|
// enable DMA Rx Request
|
|
|
|
w.set_dmar(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
compiler_fence(Ordering::SeqCst);
|
|
|
|
|
|
|
|
// In case of errors already pending when reception started, interrupts may have already been raised
|
|
|
|
// and lead to reception abortion (Overrun error for instance). In such a case, all interrupts
|
|
|
|
// have been disabled in interrupt handler and DMA Rx Request has been disabled.
|
|
|
|
|
|
|
|
let cr3 = r.cr3().read();
|
|
|
|
|
|
|
|
if !cr3.dmar() {
|
|
|
|
// something went wrong
|
|
|
|
// because the only way to get this flag cleared is to have an interrupt
|
|
|
|
|
|
|
|
// abort DMA transfer
|
|
|
|
ch.request_stop();
|
|
|
|
while ch.is_running() {}
|
|
|
|
|
|
|
|
let sr = sr(r).read();
|
|
|
|
// This read also clears the error and idle interrupt flags on v1.
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
clear_interrupt_flags(r, sr);
|
|
|
|
|
|
|
|
if sr.pe() {
|
|
|
|
return Err(Error::Parity);
|
|
|
|
}
|
|
|
|
if sr.fe() {
|
|
|
|
return Err(Error::Framing);
|
|
|
|
}
|
|
|
|
if sr.ne() {
|
|
|
|
return Err(Error::Noise);
|
|
|
|
}
|
|
|
|
if sr.ore() {
|
|
|
|
return Err(Error::Overrun);
|
|
|
|
}
|
|
|
|
|
|
|
|
unreachable!();
|
|
|
|
}
|
|
|
|
|
|
|
|
// clear idle flag
|
|
|
|
if enable_idle_line_detection {
|
|
|
|
let sr = sr(r).read();
|
|
|
|
// This read also clears the error and idle interrupt flags on v1.
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
clear_interrupt_flags(r, sr);
|
|
|
|
|
|
|
|
// enable idle interrupt
|
|
|
|
r.cr1().modify(|w| {
|
|
|
|
w.set_idleie(true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
compiler_fence(Ordering::SeqCst);
|
|
|
|
|
|
|
|
let res = poll_fn(move |cx| {
|
|
|
|
let s = T::state();
|
|
|
|
|
|
|
|
ch.set_waker(cx.waker());
|
|
|
|
s.rx_waker.register(cx.waker());
|
|
|
|
|
|
|
|
// SAFETY: read only and we only use Rx related flags
|
|
|
|
let sr = unsafe { sr(r).read() };
|
|
|
|
|
|
|
|
// SAFETY: only clears Rx related flags
|
|
|
|
unsafe {
|
|
|
|
// This read also clears the error and idle interrupt flags on v1.
|
|
|
|
rdr(r).read_volatile();
|
|
|
|
clear_interrupt_flags(r, sr);
|
|
|
|
}
|
|
|
|
|
|
|
|
compiler_fence(Ordering::SeqCst);
|
|
|
|
|
|
|
|
let has_errors = sr.pe() || sr.fe() || sr.ne() || sr.ore();
|
|
|
|
|
|
|
|
if has_errors {
|
|
|
|
// all Rx interrupts and Rx DMA Request have already been cleared in interrupt handler
|
|
|
|
|
|
|
|
// stop dma transfer
|
|
|
|
ch.request_stop();
|
|
|
|
while ch.is_running() {}
|
|
|
|
|
|
|
|
if sr.pe() {
|
|
|
|
return Poll::Ready(Err(Error::Parity));
|
|
|
|
}
|
|
|
|
if sr.fe() {
|
|
|
|
return Poll::Ready(Err(Error::Framing));
|
|
|
|
}
|
|
|
|
if sr.ne() {
|
|
|
|
return Poll::Ready(Err(Error::Noise));
|
|
|
|
}
|
|
|
|
if sr.ore() {
|
|
|
|
return Poll::Ready(Err(Error::Overrun));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if sr.idle() {
|
|
|
|
// Idle line
|
|
|
|
|
|
|
|
// stop dma transfer
|
|
|
|
ch.request_stop();
|
|
|
|
while ch.is_running() {}
|
|
|
|
|
|
|
|
let n = buffer_len - (ch.remaining_transfers() as usize);
|
|
|
|
|
|
|
|
return Poll::Ready(Ok(n));
|
|
|
|
} else if !ch.is_running() {
|
|
|
|
// DMA complete
|
|
|
|
return Poll::Ready(Ok(buffer_len));
|
|
|
|
}
|
|
|
|
|
|
|
|
Poll::Pending
|
|
|
|
})
|
|
|
|
.await;
|
|
|
|
|
|
|
|
// clear all interrupts and DMA Rx Request
|
|
|
|
// SAFETY: only clears Rx related flags
|
|
|
|
unsafe {
|
|
|
|
r.cr1().modify(|w| {
|
|
|
|
// disable RXNE interrupt
|
|
|
|
w.set_rxneie(false);
|
|
|
|
// disable parity interrupt
|
|
|
|
w.set_peie(false);
|
|
|
|
// disable idle line interrupt
|
|
|
|
w.set_idleie(false);
|
|
|
|
});
|
|
|
|
r.cr3().modify(|w| {
|
|
|
|
// disable Error Interrupt: (Frame error, Noise error, Overrun error)
|
|
|
|
w.set_eie(false);
|
|
|
|
// disable DMA Rx Request
|
|
|
|
w.set_dmar(false);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
res
|
|
|
|
}
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> Uart<'d, T, TxDma, RxDma> {
|
2022-03-14 21:03:35 +00:00
|
|
|
pub fn new(
|
2022-10-26 16:58:22 +00:00
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
2022-07-23 12:00:19 +00:00
|
|
|
rx: impl Peripheral<P = impl RxPin<T>> + 'd,
|
|
|
|
tx: impl Peripheral<P = impl TxPin<T>> + 'd,
|
2022-10-26 16:58:22 +00:00
|
|
|
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
2022-07-23 12:00:19 +00:00
|
|
|
tx_dma: impl Peripheral<P = TxDma> + 'd,
|
|
|
|
rx_dma: impl Peripheral<P = RxDma> + 'd,
|
2022-03-14 21:03:35 +00:00
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
2022-10-26 09:51:37 +00:00
|
|
|
T::enable();
|
|
|
|
T::reset();
|
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
Self::new_inner(peri, rx, tx, irq, tx_dma, rx_dma, config)
|
2022-10-26 09:51:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_with_rtscts(
|
2022-10-26 16:58:22 +00:00
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
2022-10-26 09:51:37 +00:00
|
|
|
rx: impl Peripheral<P = impl RxPin<T>> + 'd,
|
|
|
|
tx: impl Peripheral<P = impl TxPin<T>> + 'd,
|
2022-10-26 16:58:22 +00:00
|
|
|
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
2022-10-26 09:51:37 +00:00
|
|
|
rts: impl Peripheral<P = impl RtsPin<T>> + 'd,
|
|
|
|
cts: impl Peripheral<P = impl CtsPin<T>> + 'd,
|
|
|
|
tx_dma: impl Peripheral<P = TxDma> + 'd,
|
|
|
|
rx_dma: impl Peripheral<P = RxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
|
|
|
into_ref!(cts, rts);
|
2022-03-14 21:03:35 +00:00
|
|
|
|
|
|
|
T::enable();
|
2022-03-17 22:46:46 +00:00
|
|
|
T::reset();
|
2022-10-26 09:51:37 +00:00
|
|
|
|
|
|
|
unsafe {
|
|
|
|
rts.set_as_af(rts.af_num(), AFType::OutputPushPull);
|
|
|
|
cts.set_as_af(cts.af_num(), AFType::Input);
|
|
|
|
T::regs().cr3().write(|w| {
|
|
|
|
w.set_rtse(true);
|
|
|
|
w.set_ctse(true);
|
|
|
|
});
|
|
|
|
}
|
2022-10-26 16:58:22 +00:00
|
|
|
Self::new_inner(peri, rx, tx, irq, tx_dma, rx_dma, config)
|
2022-10-26 09:51:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn new_inner(
|
2022-10-26 16:58:22 +00:00
|
|
|
peri: impl Peripheral<P = T> + 'd,
|
2022-10-26 09:51:37 +00:00
|
|
|
rx: impl Peripheral<P = impl RxPin<T>> + 'd,
|
|
|
|
tx: impl Peripheral<P = impl TxPin<T>> + 'd,
|
2022-10-26 16:58:22 +00:00
|
|
|
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
2022-10-26 09:51:37 +00:00
|
|
|
tx_dma: impl Peripheral<P = TxDma> + 'd,
|
|
|
|
rx_dma: impl Peripheral<P = RxDma> + 'd,
|
|
|
|
config: Config,
|
|
|
|
) -> Self {
|
2022-10-26 16:58:22 +00:00
|
|
|
into_ref!(peri, rx, tx, irq, tx_dma, rx_dma);
|
2022-03-14 21:03:35 +00:00
|
|
|
|
|
|
|
let r = T::regs();
|
2022-01-14 21:43:40 +00:00
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
configure(r, &config, T::frequency(), T::MULTIPLIER);
|
|
|
|
|
2022-01-14 21:43:40 +00:00
|
|
|
unsafe {
|
2022-03-14 21:03:35 +00:00
|
|
|
rx.set_as_af(rx.af_num(), AFType::Input);
|
|
|
|
tx.set_as_af(tx.af_num(), AFType::OutputPushPull);
|
|
|
|
|
|
|
|
r.cr2().write(|_w| {});
|
|
|
|
r.cr1().write(|w| {
|
|
|
|
w.set_ue(true);
|
|
|
|
w.set_te(true);
|
|
|
|
w.set_re(true);
|
2022-05-11 17:57:56 +00:00
|
|
|
w.set_m0(if config.parity != Parity::ParityNone {
|
|
|
|
vals::M0::BIT9
|
|
|
|
} else {
|
|
|
|
vals::M0::BIT8
|
|
|
|
});
|
2022-03-14 21:03:35 +00:00
|
|
|
w.set_pce(config.parity != Parity::ParityNone);
|
|
|
|
w.set_ps(match config.parity {
|
|
|
|
Parity::ParityOdd => vals::Ps::ODD,
|
|
|
|
Parity::ParityEven => vals::Ps::EVEN,
|
|
|
|
_ => vals::Ps::EVEN,
|
|
|
|
});
|
|
|
|
});
|
2022-01-14 21:43:40 +00:00
|
|
|
}
|
2022-03-14 21:03:35 +00:00
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
irq.set_handler(UartRx::<T, RxDma>::on_interrupt);
|
|
|
|
irq.unpend();
|
|
|
|
irq.enable();
|
|
|
|
|
|
|
|
// create state once!
|
|
|
|
let _s = T::state();
|
|
|
|
|
2022-03-14 21:03:35 +00:00
|
|
|
Self {
|
2022-10-27 08:52:21 +00:00
|
|
|
tx: UartTx {
|
|
|
|
tx_dma,
|
|
|
|
phantom: PhantomData,
|
|
|
|
},
|
2022-10-26 16:58:22 +00:00
|
|
|
rx: UartRx {
|
|
|
|
_peri: peri,
|
|
|
|
rx_dma,
|
|
|
|
detect_previous_overrun: config.detect_previous_overrun,
|
|
|
|
},
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn write(&mut self, buffer: &[u8]) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
TxDma: crate::usart::TxDma<T>,
|
|
|
|
{
|
|
|
|
self.tx.write(buffer).await
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn blocking_write(&mut self, buffer: &[u8]) -> Result<(), Error> {
|
|
|
|
self.tx.blocking_write(buffer)
|
2022-01-14 21:43:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn blocking_flush(&mut self) -> Result<(), Error> {
|
2022-03-14 21:03:35 +00:00
|
|
|
self.tx.blocking_flush()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn read(&mut self, buffer: &mut [u8]) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
RxDma: crate::usart::RxDma<T>,
|
|
|
|
{
|
|
|
|
self.rx.read(buffer).await
|
|
|
|
}
|
|
|
|
|
2022-09-29 05:58:11 +00:00
|
|
|
pub fn nb_read(&mut self) -> Result<u8, nb::Error<Error>> {
|
|
|
|
self.rx.nb_read()
|
|
|
|
}
|
|
|
|
|
2022-03-14 21:03:35 +00:00
|
|
|
pub fn blocking_read(&mut self, buffer: &mut [u8]) -> Result<(), Error> {
|
|
|
|
self.rx.blocking_read(buffer)
|
|
|
|
}
|
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
pub async fn read_until_idle(&mut self, buffer: &mut [u8]) -> Result<usize, Error>
|
|
|
|
where
|
|
|
|
RxDma: crate::usart::RxDma<T>,
|
|
|
|
{
|
|
|
|
self.rx.read_until_idle(buffer).await
|
|
|
|
}
|
|
|
|
|
2022-03-14 21:03:35 +00:00
|
|
|
/// Split the Uart into a transmitter and receiver, which is
|
|
|
|
/// particuarly useful when having two tasks correlating to
|
|
|
|
/// transmitting and receiving.
|
|
|
|
pub fn split(self) -> (UartTx<'d, T, TxDma>, UartRx<'d, T, RxDma>) {
|
|
|
|
(self.tx, self.rx)
|
2022-01-14 21:43:40 +00:00
|
|
|
}
|
2021-12-08 04:12:48 +00:00
|
|
|
}
|
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
fn configure(r: Regs, config: &Config, pclk_freq: Hertz, multiplier: u32) {
|
|
|
|
// TODO: better calculation, including error checking and OVER8 if possible.
|
|
|
|
let div = (pclk_freq.0 + (config.baudrate / 2)) / config.baudrate * multiplier;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
r.brr().write_value(regs::Brr(div));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-26 21:39:06 +00:00
|
|
|
mod eh02 {
|
|
|
|
use super::*;
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, RxDma> embedded_hal_02::serial::Read<u8> for UartRx<'d, T, RxDma> {
|
2022-01-26 21:39:06 +00:00
|
|
|
type Error = Error;
|
|
|
|
fn read(&mut self) -> Result<u8, nb::Error<Self::Error>> {
|
2022-09-29 05:58:11 +00:00
|
|
|
self.nb_read()
|
2021-12-17 11:50:48 +00:00
|
|
|
}
|
|
|
|
}
|
2022-01-26 21:39:06 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma> embedded_hal_02::blocking::serial::Write<u8> for UartTx<'d, T, TxDma> {
|
2022-03-14 21:03:35 +00:00
|
|
|
type Error = Error;
|
|
|
|
fn bwrite_all(&mut self, buffer: &[u8]) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_write(buffer)
|
|
|
|
}
|
|
|
|
fn bflush(&mut self) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_flush()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_02::serial::Read<u8> for Uart<'d, T, TxDma, RxDma> {
|
2022-03-14 21:03:35 +00:00
|
|
|
type Error = Error;
|
|
|
|
fn read(&mut self) -> Result<u8, nb::Error<Self::Error>> {
|
2022-09-29 05:58:11 +00:00
|
|
|
self.nb_read()
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_02::blocking::serial::Write<u8> for Uart<'d, T, TxDma, RxDma> {
|
2022-01-26 21:39:06 +00:00
|
|
|
type Error = Error;
|
|
|
|
fn bwrite_all(&mut self, buffer: &[u8]) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_write(buffer)
|
|
|
|
}
|
|
|
|
fn bflush(&mut self) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_flush()
|
|
|
|
}
|
|
|
|
}
|
2021-12-17 11:50:48 +00:00
|
|
|
}
|
|
|
|
|
2022-01-26 21:39:06 +00:00
|
|
|
#[cfg(feature = "unstable-traits")]
|
|
|
|
mod eh1 {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
impl embedded_hal_1::serial::Error for Error {
|
|
|
|
fn kind(&self) -> embedded_hal_1::serial::ErrorKind {
|
|
|
|
match *self {
|
|
|
|
Self::Framing => embedded_hal_1::serial::ErrorKind::FrameFormat,
|
|
|
|
Self::Noise => embedded_hal_1::serial::ErrorKind::Noise,
|
|
|
|
Self::Overrun => embedded_hal_1::serial::ErrorKind::Overrun,
|
|
|
|
Self::Parity => embedded_hal_1::serial::ErrorKind::Parity,
|
2022-10-26 16:58:22 +00:00
|
|
|
Self::BufferTooLong => embedded_hal_1::serial::ErrorKind::Other,
|
2022-01-26 21:39:06 +00:00
|
|
|
}
|
|
|
|
}
|
2021-12-08 04:12:48 +00:00
|
|
|
}
|
2022-01-26 21:39:06 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_1::serial::ErrorType for Uart<'d, T, TxDma, RxDma> {
|
2022-01-26 21:39:06 +00:00
|
|
|
type Error = Error;
|
2021-12-08 04:12:48 +00:00
|
|
|
}
|
2022-03-14 21:03:35 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma> embedded_hal_1::serial::ErrorType for UartTx<'d, T, TxDma> {
|
2022-03-14 21:03:35 +00:00
|
|
|
type Error = Error;
|
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl<'d, T: BasicInstance, RxDma> embedded_hal_1::serial::ErrorType for UartRx<'d, T, RxDma> {
|
2022-03-14 21:03:35 +00:00
|
|
|
type Error = Error;
|
|
|
|
}
|
2022-09-28 03:19:43 +00:00
|
|
|
|
2022-09-30 03:35:46 +00:00
|
|
|
impl<'d, T: BasicInstance, RxDma> embedded_hal_nb::serial::Read for UartRx<'d, T, RxDma> {
|
2022-09-28 03:19:43 +00:00
|
|
|
fn read(&mut self) -> nb::Result<u8, Self::Error> {
|
2022-09-29 05:58:11 +00:00
|
|
|
self.nb_read()
|
2022-09-28 03:19:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-30 03:35:46 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma> embedded_hal_1::serial::Write for UartTx<'d, T, TxDma> {
|
2022-09-28 03:19:43 +00:00
|
|
|
fn write(&mut self, buffer: &[u8]) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_write(buffer)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flush(&mut self) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_flush()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-30 03:35:46 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma> embedded_hal_nb::serial::Write for UartTx<'d, T, TxDma> {
|
2022-09-28 03:19:43 +00:00
|
|
|
fn write(&mut self, char: u8) -> nb::Result<(), Self::Error> {
|
|
|
|
self.blocking_write(&[char]).map_err(nb::Error::Other)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flush(&mut self) -> nb::Result<(), Self::Error> {
|
|
|
|
self.blocking_flush().map_err(nb::Error::Other)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-30 03:35:46 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_nb::serial::Read for Uart<'d, T, TxDma, RxDma> {
|
2022-09-28 03:19:43 +00:00
|
|
|
fn read(&mut self) -> Result<u8, nb::Error<Self::Error>> {
|
2022-09-29 05:58:11 +00:00
|
|
|
self.nb_read()
|
2022-09-28 03:19:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-30 03:35:46 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_1::serial::Write for Uart<'d, T, TxDma, RxDma> {
|
2022-09-28 03:19:43 +00:00
|
|
|
fn write(&mut self, buffer: &[u8]) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_write(buffer)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flush(&mut self) -> Result<(), Self::Error> {
|
|
|
|
self.blocking_flush()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-30 03:35:46 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_nb::serial::Write for Uart<'d, T, TxDma, RxDma> {
|
2022-09-28 03:19:43 +00:00
|
|
|
fn write(&mut self, char: u8) -> nb::Result<(), Self::Error> {
|
|
|
|
self.blocking_write(&[char]).map_err(nb::Error::Other)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flush(&mut self) -> nb::Result<(), Self::Error> {
|
|
|
|
self.blocking_flush().map_err(nb::Error::Other)
|
|
|
|
}
|
|
|
|
}
|
2022-02-12 01:26:15 +00:00
|
|
|
}
|
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
#[cfg(all(
|
|
|
|
feature = "unstable-traits",
|
|
|
|
feature = "nightly",
|
|
|
|
feature = "_todo_embedded_hal_serial"
|
|
|
|
))]
|
|
|
|
mod eha {
|
|
|
|
use core::future::Future;
|
2022-03-14 21:03:35 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
use super::*;
|
2022-03-14 21:03:35 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma> embedded_hal_async::serial::Write for UartTx<'d, T, TxDma>
|
|
|
|
where
|
|
|
|
TxDma: crate::usart::TxDma<T>,
|
|
|
|
{
|
|
|
|
type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a;
|
|
|
|
|
|
|
|
fn write<'a>(&'a mut self, buf: &'a [u8]) -> Self::WriteFuture<'a> {
|
|
|
|
self.write(buf)
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
type FlushFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a;
|
2022-03-14 21:03:35 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
fn flush<'a>(&'a mut self) -> Self::FlushFuture<'a> {
|
|
|
|
async move { Ok(()) }
|
2022-03-14 21:03:35 +00:00
|
|
|
}
|
2022-08-31 00:46:52 +00:00
|
|
|
}
|
2022-03-14 21:03:35 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
impl<'d, T: BasicInstance, RxDma> embedded_hal_async::serial::Read for UartRx<'d, T, RxDma>
|
|
|
|
where
|
|
|
|
RxDma: crate::usart::RxDma<T>,
|
|
|
|
{
|
|
|
|
type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a;
|
2022-01-26 21:39:06 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
fn read<'a>(&'a mut self, buf: &'a mut [u8]) -> Self::ReadFuture<'a> {
|
|
|
|
self.read(buf)
|
|
|
|
}
|
|
|
|
}
|
2021-12-08 04:12:48 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_async::serial::Write for Uart<'d, T, TxDma, RxDma>
|
|
|
|
where
|
|
|
|
TxDma: crate::usart::TxDma<T>,
|
|
|
|
{
|
|
|
|
type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a;
|
2022-01-26 21:39:06 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
fn write<'a>(&'a mut self, buf: &'a [u8]) -> Self::WriteFuture<'a> {
|
|
|
|
self.write(buf)
|
2022-01-26 21:39:06 +00:00
|
|
|
}
|
2021-12-08 04:12:48 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
type FlushFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a;
|
2021-12-08 04:12:48 +00:00
|
|
|
|
2022-08-31 00:46:52 +00:00
|
|
|
fn flush<'a>(&'a mut self) -> Self::FlushFuture<'a> {
|
|
|
|
async move { Ok(()) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'d, T: BasicInstance, TxDma, RxDma> embedded_hal_async::serial::Read for Uart<'d, T, TxDma, RxDma>
|
|
|
|
where
|
|
|
|
RxDma: crate::usart::RxDma<T>,
|
|
|
|
{
|
|
|
|
type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a;
|
|
|
|
|
|
|
|
fn read<'a>(&'a mut self, buf: &'a mut [u8]) -> Self::ReadFuture<'a> {
|
|
|
|
self.read(buf)
|
2022-01-26 21:39:06 +00:00
|
|
|
}
|
2021-12-08 04:12:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-04 18:48:37 +00:00
|
|
|
#[cfg(feature = "nightly")]
|
2021-12-08 04:12:48 +00:00
|
|
|
pub use buffered::*;
|
2022-05-04 18:48:37 +00:00
|
|
|
#[cfg(feature = "nightly")]
|
|
|
|
mod buffered;
|
2021-12-08 04:12:48 +00:00
|
|
|
|
|
|
|
#[cfg(usart_v1)]
|
|
|
|
fn tdr(r: crate::pac::usart::Usart) -> *mut u8 {
|
|
|
|
r.dr().ptr() as _
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(usart_v1)]
|
|
|
|
fn rdr(r: crate::pac::usart::Usart) -> *mut u8 {
|
|
|
|
r.dr().ptr() as _
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(usart_v1)]
|
|
|
|
fn sr(r: crate::pac::usart::Usart) -> crate::pac::common::Reg<regs::Sr, crate::pac::common::RW> {
|
|
|
|
r.sr()
|
|
|
|
}
|
|
|
|
|
2021-12-26 15:04:07 +00:00
|
|
|
#[cfg(usart_v1)]
|
2022-05-04 18:48:37 +00:00
|
|
|
#[allow(unused)]
|
2022-08-19 13:27:11 +00:00
|
|
|
unsafe fn clear_interrupt_flags(_r: Regs, _sr: regs::Sr) {
|
2022-02-25 12:19:51 +00:00
|
|
|
// On v1 the flags are cleared implicitly by reads and writes to DR.
|
2021-12-26 15:04:07 +00:00
|
|
|
}
|
|
|
|
|
2021-12-08 04:12:48 +00:00
|
|
|
#[cfg(usart_v2)]
|
2022-08-19 13:27:11 +00:00
|
|
|
fn tdr(r: Regs) -> *mut u8 {
|
2021-12-08 04:12:48 +00:00
|
|
|
r.tdr().ptr() as _
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(usart_v2)]
|
2022-08-19 13:27:11 +00:00
|
|
|
fn rdr(r: Regs) -> *mut u8 {
|
2021-12-08 04:12:48 +00:00
|
|
|
r.rdr().ptr() as _
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(usart_v2)]
|
2022-08-19 13:27:11 +00:00
|
|
|
fn sr(r: Regs) -> crate::pac::common::Reg<regs::Isr, crate::pac::common::R> {
|
2021-12-08 04:12:48 +00:00
|
|
|
r.isr()
|
|
|
|
}
|
|
|
|
|
2021-12-26 15:04:07 +00:00
|
|
|
#[cfg(usart_v2)]
|
2022-05-04 18:48:37 +00:00
|
|
|
#[allow(unused)]
|
2022-08-19 13:27:11 +00:00
|
|
|
unsafe fn clear_interrupt_flags(r: Regs, sr: regs::Isr) {
|
2022-06-09 13:17:03 +00:00
|
|
|
r.icr().write(|w| *w = regs::Icr(sr.0));
|
2021-12-26 15:04:07 +00:00
|
|
|
}
|
|
|
|
|
2021-04-25 20:35:51 +00:00
|
|
|
pub(crate) mod sealed {
|
2022-10-26 16:58:22 +00:00
|
|
|
use embassy_sync::waitqueue::AtomicWaker;
|
|
|
|
|
2022-08-19 13:27:11 +00:00
|
|
|
use super::*;
|
2022-06-09 13:17:03 +00:00
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
pub struct State {
|
|
|
|
pub rx_waker: AtomicWaker,
|
|
|
|
pub tx_waker: AtomicWaker,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl State {
|
|
|
|
pub const fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
rx_waker: AtomicWaker::new(),
|
|
|
|
tx_waker: AtomicWaker::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
pub trait BasicInstance: crate::rcc::RccPeripheral {
|
|
|
|
const MULTIPLIER: u32;
|
|
|
|
type Interrupt: crate::interrupt::Interrupt;
|
|
|
|
|
2022-08-19 13:27:11 +00:00
|
|
|
fn regs() -> Regs;
|
2022-10-26 16:58:22 +00:00
|
|
|
fn state() -> &'static State;
|
2021-04-25 20:35:51 +00:00
|
|
|
}
|
2021-06-25 18:00:11 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
pub trait FullInstance: BasicInstance {
|
|
|
|
fn regs_uart() -> crate::pac::usart::Usart;
|
|
|
|
}
|
2021-08-16 15:16:49 +00:00
|
|
|
}
|
2022-02-10 20:38:03 +00:00
|
|
|
|
2022-10-26 16:58:22 +00:00
|
|
|
pub trait BasicInstance: Peripheral<P = Self> + sealed::BasicInstance + 'static + Send {}
|
2022-02-10 20:38:03 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
pub trait FullInstance: sealed::FullInstance {}
|
|
|
|
|
|
|
|
pin_trait!(RxPin, BasicInstance);
|
|
|
|
pin_trait!(TxPin, BasicInstance);
|
|
|
|
pin_trait!(CtsPin, BasicInstance);
|
|
|
|
pin_trait!(RtsPin, BasicInstance);
|
|
|
|
pin_trait!(CkPin, BasicInstance);
|
|
|
|
|
|
|
|
dma_trait!(TxDma, BasicInstance);
|
|
|
|
dma_trait!(RxDma, BasicInstance);
|
|
|
|
|
|
|
|
macro_rules! impl_lpuart {
|
|
|
|
($inst:ident, $irq:ident, $mul:expr) => {
|
|
|
|
impl sealed::BasicInstance for crate::peripherals::$inst {
|
|
|
|
const MULTIPLIER: u32 = $mul;
|
|
|
|
type Interrupt = crate::interrupt::$irq;
|
|
|
|
|
2022-08-19 13:27:11 +00:00
|
|
|
fn regs() -> Regs {
|
|
|
|
Regs(crate::pac::$inst.0)
|
2022-06-09 13:17:03 +00:00
|
|
|
}
|
2022-10-26 16:58:22 +00:00
|
|
|
|
|
|
|
fn state() -> &'static crate::usart::sealed::State {
|
|
|
|
static STATE: crate::usart::sealed::State = crate::usart::sealed::State::new();
|
|
|
|
&STATE
|
|
|
|
}
|
2022-06-09 13:17:03 +00:00
|
|
|
}
|
2022-08-19 13:27:11 +00:00
|
|
|
|
|
|
|
impl BasicInstance for peripherals::$inst {}
|
2022-06-09 13:17:03 +00:00
|
|
|
};
|
|
|
|
}
|
2021-06-25 18:00:11 +00:00
|
|
|
|
2022-02-26 00:40:43 +00:00
|
|
|
foreach_interrupt!(
|
2022-08-19 13:27:11 +00:00
|
|
|
($inst:ident, lpuart, $block:ident, $signal_name:ident, $irq:ident) => {
|
2022-06-09 13:17:03 +00:00
|
|
|
impl_lpuart!($inst, $irq, 255);
|
|
|
|
};
|
|
|
|
|
2022-08-19 13:27:11 +00:00
|
|
|
($inst:ident, usart, $block:ident, $signal_name:ident, $irq:ident) => {
|
2022-06-09 13:17:03 +00:00
|
|
|
impl_lpuart!($inst, $irq, 1);
|
|
|
|
|
|
|
|
impl sealed::FullInstance for peripherals::$inst {
|
|
|
|
|
|
|
|
fn regs_uart() -> crate::pac::usart::Usart {
|
2021-05-06 01:43:46 +00:00
|
|
|
crate::pac::$inst
|
2021-04-14 13:34:58 +00:00
|
|
|
}
|
|
|
|
}
|
2021-06-03 15:27:17 +00:00
|
|
|
|
2022-06-09 13:17:03 +00:00
|
|
|
impl FullInstance for peripherals::$inst {
|
2021-08-16 15:16:49 +00:00
|
|
|
}
|
2021-04-14 13:34:58 +00:00
|
|
|
};
|
2021-06-03 15:27:17 +00:00
|
|
|
);
|