Merge #783
783: Reimplement BufRead for BufferedUart r=Dirbaio a=chemicstry The `AsyncBufRead` implementation for `BufferedUart` was removed in https://github.com/embassy-rs/embassy/pull/752, this PR reimplements it from `embedded-io`. This allows reading `BufferedUart` without copying slices. Co-authored-by: chemicstry <chemicstry@gmail.com>
This commit is contained in:
commit
82e873d920
@ -236,6 +236,48 @@ impl<'d, U: UarteInstance, T: TimerInstance> embedded_io::asynch::Read for Buffe
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<'d, U: UarteInstance, T: TimerInstance> embedded_io::asynch::BufRead
|
||||||
|
for BufferedUarte<'d, U, T>
|
||||||
|
{
|
||||||
|
type FillBufFuture<'a> = impl Future<Output = Result<&'a [u8], Self::Error>>
|
||||||
|
where
|
||||||
|
Self: 'a;
|
||||||
|
|
||||||
|
fn fill_buf<'a>(&'a mut self) -> Self::FillBufFuture<'a> {
|
||||||
|
poll_fn(move |cx| {
|
||||||
|
self.inner.with(|state| {
|
||||||
|
compiler_fence(Ordering::SeqCst);
|
||||||
|
trace!("fill_buf");
|
||||||
|
|
||||||
|
// We have data ready in buffer? Return it.
|
||||||
|
let buf = state.rx.pop_buf();
|
||||||
|
if !buf.is_empty() {
|
||||||
|
trace!(" got {:?} {:?}", buf.as_ptr() as u32, buf.len());
|
||||||
|
let buf: &[u8] = buf;
|
||||||
|
// Safety: buffer lives as long as uart
|
||||||
|
let buf: &[u8] = unsafe { core::mem::transmute(buf) };
|
||||||
|
return Poll::Ready(Ok(buf));
|
||||||
|
}
|
||||||
|
|
||||||
|
trace!(" empty");
|
||||||
|
state.rx_waker.register(cx.waker());
|
||||||
|
Poll::<Result<&[u8], Self::Error>>::Pending
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn consume(&mut self, amt: usize) {
|
||||||
|
let signal = self.inner.with(|state| {
|
||||||
|
let full = state.rx.is_full();
|
||||||
|
state.rx.pop(amt);
|
||||||
|
full
|
||||||
|
});
|
||||||
|
if signal {
|
||||||
|
self.inner.pend();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<'d, U: UarteInstance, T: TimerInstance> embedded_io::asynch::Write
|
impl<'d, U: UarteInstance, T: TimerInstance> embedded_io::asynch::Write
|
||||||
for BufferedUarte<'d, U, T>
|
for BufferedUarte<'d, U, T>
|
||||||
{
|
{
|
||||||
|
@ -191,6 +191,43 @@ impl<'d, T: Instance> embedded_io::asynch::Read for BufferedUart<'d, T> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<'d, T: Instance> embedded_io::asynch::BufRead for BufferedUart<'d, T> {
|
||||||
|
type FillBufFuture<'a> = impl Future<Output = Result<&'a [u8], Self::Error>>
|
||||||
|
where
|
||||||
|
Self: 'a;
|
||||||
|
|
||||||
|
fn fill_buf<'a>(&'a mut self) -> Self::FillBufFuture<'a> {
|
||||||
|
poll_fn(move |cx| {
|
||||||
|
self.inner.with(|state| {
|
||||||
|
compiler_fence(Ordering::SeqCst);
|
||||||
|
|
||||||
|
// We have data ready in buffer? Return it.
|
||||||
|
let buf = state.rx.pop_buf();
|
||||||
|
if !buf.is_empty() {
|
||||||
|
let buf: &[u8] = buf;
|
||||||
|
// Safety: buffer lives as long as uart
|
||||||
|
let buf: &[u8] = unsafe { core::mem::transmute(buf) };
|
||||||
|
return Poll::Ready(Ok(buf));
|
||||||
|
}
|
||||||
|
|
||||||
|
state.rx_waker.register(cx.waker());
|
||||||
|
Poll::<Result<&[u8], Self::Error>>::Pending
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn consume(&mut self, amt: usize) {
|
||||||
|
let signal = self.inner.with(|state| {
|
||||||
|
let full = state.rx.is_full();
|
||||||
|
state.rx.pop(amt);
|
||||||
|
full
|
||||||
|
});
|
||||||
|
if signal {
|
||||||
|
self.inner.pend();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<'d, T: Instance> embedded_io::asynch::Write for BufferedUart<'d, T> {
|
impl<'d, T: Instance> embedded_io::asynch::Write for BufferedUart<'d, T> {
|
||||||
type WriteFuture<'a> = impl Future<Output = Result<usize, Self::Error>>
|
type WriteFuture<'a> = impl Future<Output = Result<usize, Self::Error>>
|
||||||
where
|
where
|
||||||
|
@ -6,7 +6,7 @@ use defmt::*;
|
|||||||
use embassy::executor::Spawner;
|
use embassy::executor::Spawner;
|
||||||
use embassy_nrf::buffered_uarte::State;
|
use embassy_nrf::buffered_uarte::State;
|
||||||
use embassy_nrf::{buffered_uarte::BufferedUarte, interrupt, uarte, Peripherals};
|
use embassy_nrf::{buffered_uarte::BufferedUarte, interrupt, uarte, Peripherals};
|
||||||
use embedded_io::asynch::{Read, Write};
|
use embedded_io::asynch::{BufRead, Write};
|
||||||
use futures::pin_mut;
|
use futures::pin_mut;
|
||||||
|
|
||||||
use defmt_rtt as _; // global logger
|
use defmt_rtt as _; // global logger
|
||||||
@ -46,23 +46,13 @@ async fn main(_spawner: Spawner, p: Peripherals) {
|
|||||||
unwrap!(u.write_all(b"Hello!\r\n").await);
|
unwrap!(u.write_all(b"Hello!\r\n").await);
|
||||||
info!("wrote hello in uart!");
|
info!("wrote hello in uart!");
|
||||||
|
|
||||||
// Simple demo, reading 8-char chunks and echoing them back reversed.
|
|
||||||
loop {
|
loop {
|
||||||
info!("reading...");
|
info!("reading...");
|
||||||
let mut buf = [0u8; 8];
|
let buf = unwrap!(u.fill_buf().await);
|
||||||
unwrap!(u.read_exact(&mut buf).await);
|
|
||||||
info!("read done, got {}", buf);
|
info!("read done, got {}", buf);
|
||||||
|
|
||||||
// Reverse buf
|
// Read bytes have to be explicitly consumed, otherwise fill_buf() will return them again
|
||||||
for i in 0..4 {
|
let n = buf.len();
|
||||||
buf.swap(i, 7 - i);
|
u.consume(n);
|
||||||
}
|
|
||||||
|
|
||||||
info!("writing...");
|
|
||||||
unwrap!(u.write_all(&buf).await);
|
|
||||||
info!("write done");
|
|
||||||
|
|
||||||
// Wait until the bytes are actually finished being transmitted
|
|
||||||
unwrap!(u.flush().await);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,7 @@ defmt-rtt = "0.3"
|
|||||||
cortex-m = "0.7.3"
|
cortex-m = "0.7.3"
|
||||||
cortex-m-rt = "0.7.0"
|
cortex-m-rt = "0.7.0"
|
||||||
embedded-hal = "0.2.6"
|
embedded-hal = "0.2.6"
|
||||||
|
embedded-io = "0.3.0"
|
||||||
panic-probe = { version = "0.3", features = ["print-defmt"] }
|
panic-probe = { version = "0.3", features = ["print-defmt"] }
|
||||||
futures = { version = "0.3.17", default-features = false, features = ["async-await"] }
|
futures = { version = "0.3.17", default-features = false, features = ["async-await"] }
|
||||||
heapless = { version = "0.7.5", default-features = false }
|
heapless = { version = "0.7.5", default-features = false }
|
||||||
|
36
examples/stm32f4/src/bin/usart_buffered.rs
Normal file
36
examples/stm32f4/src/bin/usart_buffered.rs
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
#![no_std]
|
||||||
|
#![no_main]
|
||||||
|
#![feature(type_alias_impl_trait)]
|
||||||
|
|
||||||
|
use defmt::*;
|
||||||
|
use defmt_rtt as _; // global logger
|
||||||
|
use embassy::executor::Spawner;
|
||||||
|
use embassy_stm32::dma::NoDma;
|
||||||
|
use embassy_stm32::usart::{BufferedUart, Config, State, Uart};
|
||||||
|
use embassy_stm32::{interrupt, Peripherals};
|
||||||
|
use embedded_io::asynch::BufRead;
|
||||||
|
use panic_probe as _;
|
||||||
|
|
||||||
|
#[embassy::main]
|
||||||
|
async fn main(_spawner: Spawner, p: Peripherals) {
|
||||||
|
info!("Hello World!");
|
||||||
|
|
||||||
|
let config = Config::default();
|
||||||
|
let usart = Uart::new(p.USART3, p.PD9, p.PD8, NoDma, NoDma, config);
|
||||||
|
|
||||||
|
let mut state = State::new();
|
||||||
|
let irq = interrupt::take!(USART3);
|
||||||
|
let mut tx_buf = [0u8; 32];
|
||||||
|
let mut rx_buf = [0u8; 32];
|
||||||
|
let mut buf_usart =
|
||||||
|
unsafe { BufferedUart::new(&mut state, usart, irq, &mut tx_buf, &mut rx_buf) };
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let buf = buf_usart.fill_buf().await.unwrap();
|
||||||
|
info!("Received: {}", buf);
|
||||||
|
|
||||||
|
// Read bytes have to be explicitly consumed, otherwise fill_buf() will return them again
|
||||||
|
let n = buf.len();
|
||||||
|
buf_usart.consume(n);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user