Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

UART: Remove blocking version of read_bytes and rename drain_fifo to read_bytes instead #2895

Merged
merged 6 commits into from
Jan 7, 2025
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions esp-hal/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- UART: Make `AtCmdConfig` and `ConfigError` non-exhaustive (#2851)
- UART: Make `AtCmdConfig` use builder-lite pattern (#2851)
- UART: Fix naming violations for `DataBits`, `Parity`, and `StopBits` enum variants (#2893)
- UART: Remove blocking version of `read_bytes` and rename `drain_fifo` to `read_bytes` instead (#2895)

### Fixed

Expand Down
4 changes: 4 additions & 0 deletions esp-hal/MIGRATING-0.22.md
Original file line number Diff line number Diff line change
Expand Up @@ -382,3 +382,7 @@ e.g.)
- StopBits::Stop1
+ StopBits::_1
```

The previous blocking implementation of `read_bytes` has been removed, and the non-blocking `drain_fifo` has instead been renamed to `read_bytes` in its place.

Any code which was previously using `read_bytes` to fill a buffer in a blocking manner will now need to implement the necessary logic to block until the buffer is filled in their application instead.
50 changes: 9 additions & 41 deletions esp-hal/src/uart.rs
Original file line number Diff line number Diff line change
Expand Up @@ -848,40 +848,6 @@ where
self.uart.info().apply_config(config)
}

/// Fill a buffer with received bytes
pub fn read_bytes(&mut self, buf: &mut [u8]) -> Result<(), Error> {
cfg_if::cfg_if! {
if #[cfg(esp32s2)] {
// On the ESP32-S2 we need to use PeriBus2 to read the FIFO:
let fifo = unsafe {
&*((self.register_block().fifo().as_ptr() as *mut u8).add(0x20C00000)
as *mut crate::peripherals::uart0::FIFO)
};
} else {
let fifo = self.register_block().fifo();
}
}

for byte in buf.iter_mut() {
while self.rx_fifo_count() == 0 {
// Block until we received at least one byte
}

cfg_if::cfg_if! {
if #[cfg(esp32)] {
// https://docs.espressif.com/projects/esp-chip-errata/en/latest/esp32/03-errata-description/esp32/cpu-subsequent-access-halted-when-get-interrupted.html
jessebraham marked this conversation as resolved.
Show resolved Hide resolved
crate::interrupt::free(|| {
*byte = fifo.read().rxfifo_rd_byte().bits();
});
} else {
*byte = fifo.read().rxfifo_rd_byte().bits();
}
}
}

Ok(())
}

/// Read a byte from the UART
pub fn read_byte(&mut self) -> nb::Result<u8, Error> {
cfg_if::cfg_if! {
Expand All @@ -897,15 +863,16 @@ where
}

if self.rx_fifo_count() > 0 {
Ok(fifo.read().rxfifo_rd_byte().bits())
let byte = crate::interrupt::free(|| fifo.read().rxfifo_rd_byte().bits());
jessebraham marked this conversation as resolved.
Show resolved Hide resolved
Ok(byte)
} else {
Err(nb::Error::WouldBlock)
}
}

/// Read all available bytes from the RX FIFO into the provided buffer and
/// returns the number of read bytes. Never blocks
pub fn drain_fifo(&mut self, buf: &mut [u8]) -> usize {
/// returns the number of read bytes without blocking.
pub fn read_bytes(&mut self, buf: &mut [u8]) -> usize {
jessebraham marked this conversation as resolved.
Show resolved Hide resolved
let mut count = 0;
while count < buf.len() {
if let Ok(byte) = self.read_byte() {
Expand Down Expand Up @@ -1143,8 +1110,9 @@ where
self.tx.write_bytes(data)
}

/// Fill a buffer with received bytes
pub fn read_bytes(&mut self, buf: &mut [u8]) -> Result<(), Error> {
/// Read all available bytes from the RX FIFO into the provided buffer and
/// returns the number of read bytes without blocking.
pub fn read_bytes(&mut self, buf: &mut [u8]) -> usize {
self.rx.read_bytes(buf)
}

Expand Down Expand Up @@ -1481,7 +1449,7 @@ where
// Block until we received at least one byte
}

Ok(self.drain_fifo(buf))
Ok(self.read_bytes(buf))
}
}

Expand Down Expand Up @@ -1858,7 +1826,7 @@ where

let events_happened = UartRxFuture::new(self.uart.reborrow(), events).await;
// always drain the fifo, if an error has occurred the data is lost
let read_bytes = self.drain_fifo(buf);
let read_bytes = self.read_bytes(buf);
// check error events
for event_happened in events_happened {
match event_happened {
Expand Down
11 changes: 9 additions & 2 deletions hil-test/tests/uart_tx_rx.rs
Original file line number Diff line number Diff line change
Expand Up @@ -53,8 +53,15 @@ mod tests {
ctx.tx.flush().unwrap();
ctx.tx.write_bytes(&bytes).unwrap();

ctx.rx.read_bytes(&mut buf).unwrap();

let mut bytes_read = 0;
loop {
bytes_read += ctx.rx.read_bytes(&mut buf[bytes_read..]);
if bytes_read == 3 {
break;
}
}

assert_eq!(bytes_read, 3);
assert_eq!(buf, bytes);
}
}
Loading