rp-hal-boards/rp2040-hal/src/uart.rs

521 lines
15 KiB
Rust
Raw Normal View History

2021-04-25 10:51:46 +02:00
//! Universal Asynchronous Receiver Transmitter (UART)
//!
//! See [Chapter 4 Section 2](https://datasheets.raspberrypi.org/rp2040/rp2040_datasheet.pdf) of the datasheet for more details
//!
//! ## Usage
//!
//! See [examples/uart.rs](https://github.com/rp-rs/rp-hal/tree/main/rp2040-hal/examples/uart.rs) for a more complete example
//! ```no_run
//! use rp2040_hal::{clocks::init_clocks_and_plls, gpio::{Pins, FunctionUart}, pac, sio::Sio, uart::{self, UartPeripheral}, watchdog::Watchdog};
//!
//! const XOSC_CRYSTAL_FREQ: u32 = 12_000_000; // Typically found in BSP crates
//!
//! let mut peripherals = pac::Peripherals::take().unwrap();
//! let sio = Sio::new(peripherals.SIO);
//! let pins = Pins::new(peripherals.IO_BANK0, peripherals.PADS_BANK0, sio.gpio_bank0, &mut peripherals.RESETS);
//! let mut watchdog = Watchdog::new(peripherals.WATCHDOG);
//! let mut clocks = init_clocks_and_plls(XOSC_CRYSTAL_FREQ, peripherals.XOSC, peripherals.CLOCKS, peripherals.PLL_SYS, peripherals.PLL_USB, &mut peripherals.RESETS, &mut watchdog).ok().unwrap();
//!
//! // Need to perform clock init before using UART or it will freeze.
//! let uart = UartPeripheral::<_, _>::enable(
//! peripherals.UART0,
//! &mut peripherals.RESETS,
//! uart::common_configs::_9600_8_N_1,
//! clocks.peripheral_clock.into(),
//! ).unwrap();
//!
//! // Set up UART on GP0 and GP1 (Pico pins 1 and 2)
//! let _tx_pin = pins.gpio0.into_mode::<FunctionUart>();
//! let _rx_pin = pins.gpio1.into_mode::<FunctionUart>();
//! uart.write_full_blocking(b"Hello World!\r\n");
//! ```
2021-04-25 10:51:46 +02:00
use core::convert::Infallible;
use core::fmt;
2021-04-25 10:51:46 +02:00
use core::ops::Deref;
2021-05-02 08:42:51 +02:00
use embedded_time::fixed_point::FixedPoint;
2021-04-25 10:51:46 +02:00
use embedded_time::rate::Baud;
use embedded_time::rate::Hertz;
2021-05-02 08:42:51 +02:00
use embedded_hal::serial::{Read, Write};
2021-05-02 08:42:51 +02:00
use nb::Error::{Other, WouldBlock};
2021-04-25 10:51:46 +02:00
2021-04-25 20:24:56 +02:00
use crate::pac::{
2021-05-02 08:42:51 +02:00
uart0::{uartlcr_h::W as UART_LCR_H_Writer, RegisterBlock},
UART0, UART1,
2021-04-25 20:24:56 +02:00
};
2021-04-25 10:51:46 +02:00
use crate::resets::SubsystemReset;
/// Error type for UART operations.
#[derive(Debug)]
pub enum Error {
/// Bad argument : when things overflow, ...
2021-05-02 08:42:51 +02:00
BadArgument,
}
2021-05-02 08:27:29 +02:00
/// When there's a read error.
pub struct ReadError<'err> {
/// The type of error
pub err_type: ReadErrorType,
/// Reference to the data that was read but eventually discared because of the error.
2021-05-02 08:42:51 +02:00
pub discared: &'err [u8],
2021-05-02 08:27:29 +02:00
}
/// Possible types of read errors. See Chapter 4, Section 2 §8 - Table 436: "UARTDR Register"
pub enum ReadErrorType {
/// Triggered when the FIFO (or shift-register) is overflowed.
Overrun,
/// Triggered when a break is received
Break,
/// Triggered when there is a parity mismatch between what's received and our settings.
Parity,
/// Triggered when the received character didn't have a valid stop bit.
2021-05-02 08:42:51 +02:00
Framing,
2021-05-02 08:27:29 +02:00
}
2021-04-25 10:51:46 +02:00
/// State of the UART Peripheral.
pub trait State {}
/// Trait to handle both underlying devices (UART0 & UART1)
pub trait UartDevice: Deref<Target = RegisterBlock> + SubsystemReset {}
2021-04-25 10:51:46 +02:00
2021-05-09 09:53:22 +02:00
impl UartDevice for UART0 {}
impl UartDevice for UART1 {}
2021-04-25 10:51:46 +02:00
/// UART is enabled.
pub struct Enabled;
/// UART is disabled.
pub struct Disabled;
impl State for Enabled {}
impl State for Disabled {}
/// Data bits
pub enum DataBits {
/// 5 bits
Five,
/// 6 bits
Six,
/// 7 bits
Seven,
/// 8 bits
2021-05-02 08:42:51 +02:00
Eight,
2021-04-25 10:51:46 +02:00
}
/// Stop bits
pub enum StopBits {
/// 1 bit
One,
/// 2 bits
2021-05-02 08:42:51 +02:00
Two,
2021-04-25 10:51:46 +02:00
}
/// Parity
/// The "none" state of parity is represented with the Option type (None).
pub enum Parity {
/// Odd parity
Odd,
/// Even parity
2021-05-02 08:42:51 +02:00
Even,
2021-04-25 10:51:46 +02:00
}
/// A struct holding the configuration for an UART device.
2021-05-09 09:53:22 +02:00
pub struct UartConfig {
2021-04-25 10:51:46 +02:00
baudrate: Baud,
data_bits: DataBits,
stop_bits: StopBits,
2021-05-02 08:42:51 +02:00
parity: Option<Parity>,
2021-04-25 10:51:46 +02:00
}
/// Common configurations for UART.
pub mod common_configs {
2021-05-09 09:53:22 +02:00
use super::{DataBits, StopBits, UartConfig};
2021-04-25 10:51:46 +02:00
use embedded_time::rate::Baud;
/// 9600 baud, 8 data bits, no parity, 1 stop bit
2021-05-09 09:53:22 +02:00
pub const _9600_8_N_1: UartConfig = UartConfig {
2021-04-25 10:51:46 +02:00
baudrate: Baud(9600),
data_bits: DataBits::Eight,
stop_bits: StopBits::One,
2021-05-02 08:42:51 +02:00
parity: None,
2021-04-25 10:51:46 +02:00
};
/// 19200 baud, 8 data bits, no parity, 1 stop bit
2021-05-09 09:53:22 +02:00
pub const _19200_8_N_1: UartConfig = UartConfig {
2021-04-25 10:51:46 +02:00
baudrate: Baud(19200),
data_bits: DataBits::Eight,
stop_bits: StopBits::One,
2021-05-02 08:42:51 +02:00
parity: None,
2021-04-25 10:51:46 +02:00
};
/// 38400 baud, 8 data bits, no parity, 1 stop bit
2021-05-09 09:53:22 +02:00
pub const _38400_8_N_1: UartConfig = UartConfig {
2021-04-25 10:51:46 +02:00
baudrate: Baud(38400),
data_bits: DataBits::Eight,
stop_bits: StopBits::One,
2021-05-02 08:42:51 +02:00
parity: None,
2021-04-25 10:51:46 +02:00
};
/// 57600 baud, 8 data bits, no parity, 1 stop bit
2021-05-09 09:53:22 +02:00
pub const _57600_8_N_1: UartConfig = UartConfig {
2021-04-25 10:51:46 +02:00
baudrate: Baud(57600),
data_bits: DataBits::Eight,
stop_bits: StopBits::One,
2021-05-02 08:42:51 +02:00
parity: None,
2021-04-25 10:51:46 +02:00
};
/// 115200 baud, 8 data bits, no parity, 1 stop bit
2021-05-09 09:53:22 +02:00
pub const _115200_8_N_1: UartConfig = UartConfig {
2021-04-25 10:51:46 +02:00
baudrate: Baud(115200),
data_bits: DataBits::Eight,
stop_bits: StopBits::One,
2021-05-02 08:42:51 +02:00
parity: None,
2021-04-25 10:51:46 +02:00
};
}
/// An UART Peripheral based on an underlying UART device.
2021-05-09 09:53:22 +02:00
pub struct UartPeripheral<S: State, D: UartDevice> {
2021-04-25 10:51:46 +02:00
device: D,
_state: S,
2021-05-09 09:53:22 +02:00
config: UartConfig,
2021-05-02 08:42:51 +02:00
effective_baudrate: Baud,
2021-04-25 10:51:46 +02:00
}
2021-05-09 09:53:22 +02:00
impl<S: State, D: UartDevice> UartPeripheral<S, D> {
fn transition<To: State>(self, state: To) -> UartPeripheral<To, D> {
UartPeripheral {
2021-04-25 10:51:46 +02:00
device: self.device,
config: self.config,
effective_baudrate: self.effective_baudrate,
2021-05-02 08:42:51 +02:00
_state: state,
2021-04-25 10:51:46 +02:00
}
}
/// Releases the underlying device.
2021-05-02 08:42:51 +02:00
pub fn free(self) -> D {
2021-04-25 10:51:46 +02:00
self.device
}
}
2021-05-09 09:53:22 +02:00
impl<D: UartDevice> UartPeripheral<Disabled, D> {
2021-04-25 10:51:46 +02:00
/// Enables the provided UART device with the given configuration.
2021-05-02 08:42:51 +02:00
pub fn enable(
mut device: D,
resets: &mut pac::RESETS,
2021-05-09 09:53:22 +02:00
config: UartConfig,
2021-05-02 08:42:51 +02:00
frequency: Hertz,
2021-05-09 09:53:22 +02:00
) -> Result<UartPeripheral<Enabled, D>, Error> {
device.reset_bring_up(resets);
let effective_baudrate = configure_baudrate(&mut device, &config.baudrate, &frequency)?;
2021-04-25 10:51:46 +02:00
// Enable the UART, both TX and RX
device.uartcr.write(|w| {
w.uarten().set_bit();
w.txe().set_bit();
w.rxe().set_bit();
w
});
device.uartlcr_h.write(|w| {
w.fen().set_bit();
set_format(w, &config.data_bits, &config.stop_bits, &config.parity);
w
});
device.uartdmacr.write(|w| {
w.txdmae().set_bit();
w.rxdmae().set_bit();
w
});
2021-05-09 09:53:22 +02:00
Ok(UartPeripheral {
2021-05-02 08:42:51 +02:00
device,
config,
effective_baudrate,
_state: Enabled,
})
2021-04-25 10:51:46 +02:00
}
}
2021-05-09 09:53:22 +02:00
impl<D: UartDevice> UartPeripheral<Enabled, D> {
2021-04-25 10:51:46 +02:00
/// Disable this UART Peripheral, falling back to the Disabled state.
2021-05-09 09:53:22 +02:00
pub fn disable(self) -> UartPeripheral<Disabled, D> {
// Disable the UART, both TX and RX
self.device.uartcr.write(|w| {
w.uarten().clear_bit();
w.txe().clear_bit();
w.rxe().clear_bit();
w
});
2021-04-25 10:51:46 +02:00
self.transition(Disabled)
}
pub(crate) fn transmit_flushed(&self) -> nb::Result<(), Infallible> {
if self.device.uartfr.read().txfe().bit_is_set() {
Ok(())
2021-05-02 08:42:51 +02:00
} else {
Err(WouldBlock)
}
}
2021-04-25 10:51:46 +02:00
fn uart_is_writable(&self) -> bool {
self.device.uartfr.read().txff().bit_is_clear()
}
fn uart_is_readable(&self) -> bool {
self.device.uartfr.read().rxfe().bit_is_clear()
}
/// Writes bytes to the UART.
/// This function writes as long as it can. As soon that the FIFO is full, if :
/// - 0 bytes were written, a WouldBlock Error is returned
/// - some bytes were written, it is deemed to be a success
/// Upon success, the remaining slice is returned.
2021-05-02 08:42:51 +02:00
pub fn write_raw<'d>(&self, data: &'d [u8]) -> nb::Result<&'d [u8], Infallible> {
2021-04-25 10:51:46 +02:00
let mut bytes_written = 0;
for c in data {
if !self.uart_is_writable() {
if bytes_written == 0 {
2021-05-02 08:42:51 +02:00
return Err(WouldBlock);
} else {
return Ok(&data[bytes_written..]);
2021-04-25 10:51:46 +02:00
}
}
self.device.uartdr.write(|w| unsafe {
w.data().bits(*c);
w
});
bytes_written += 1;
2021-04-25 10:51:46 +02:00
}
2021-05-05 08:06:47 +02:00
Ok(&data[bytes_written..])
2021-04-25 10:51:46 +02:00
}
/// Reads bytes from the UART.
/// This function reads as long as it can. As soon that the FIFO is empty, if :
/// - 0 bytes were read, a WouldBlock Error is returned
/// - some bytes were read, it is deemed to be a success
/// Upon success, the remaining slice is returned.
2021-05-02 08:27:29 +02:00
pub fn read_raw<'b>(&self, buffer: &'b mut [u8]) -> nb::Result<&'b mut [u8], ReadError<'b>> {
2021-04-25 10:51:46 +02:00
let mut bytes_read = 0;
Ok(loop {
if !self.uart_is_readable() {
if bytes_read == 0 {
2021-05-02 08:42:51 +02:00
return Err(WouldBlock);
} else {
break &mut buffer[bytes_read..];
2021-04-25 10:51:46 +02:00
}
}
if bytes_read < buffer.len() {
2021-05-02 08:27:29 +02:00
let mut error: Option<ReadErrorType> = None;
if self.device.uartdr.read().oe().bit_is_set() {
error = Some(ReadErrorType::Overrun);
}
if self.device.uartdr.read().be().bit_is_set() {
error = Some(ReadErrorType::Break);
}
if self.device.uartdr.read().pe().bit_is_set() {
error = Some(ReadErrorType::Parity);
}
if self.device.uartdr.read().fe().bit_is_set() {
error = Some(ReadErrorType::Framing);
}
if let Some(err_type) = error {
return Err(Other(ReadError {
2021-05-02 08:42:51 +02:00
err_type,
discared: buffer,
2021-05-02 08:27:29 +02:00
}));
}
2021-04-25 10:51:46 +02:00
buffer[bytes_read] = self.device.uartdr.read().data().bits();
bytes_read += 1;
2021-05-02 08:42:51 +02:00
} else {
break &mut buffer[bytes_read..];
2021-04-25 10:51:46 +02:00
}
})
}
/// Writes bytes to the UART.
/// This function blocks until the full buffer has been sent.
pub fn write_full_blocking(&self, data: &[u8]) {
let mut temp = data;
2021-04-25 10:51:46 +02:00
while !temp.is_empty() {
temp = match self.write_raw(temp) {
Ok(remaining) => remaining,
2021-04-25 10:51:46 +02:00
Err(WouldBlock) => continue,
2021-05-02 08:42:51 +02:00
Err(_) => unreachable!(),
}
2021-04-25 10:51:46 +02:00
}
}
/// Reads bytes from the UART.
/// This function blocks until the full buffer has been received.
2021-05-02 09:04:05 +02:00
pub fn read_full_blocking(&self, buffer: &mut [u8]) -> Result<(), ReadErrorType> {
2021-04-25 10:51:46 +02:00
let mut offset = 0;
while offset != buffer.len() {
offset += match self.read_raw(&mut buffer[offset..]) {
2021-05-02 08:42:51 +02:00
Ok(remaining) => remaining.len(),
2021-05-02 09:04:05 +02:00
Err(e) => match e {
Other(inner) => return Err(inner.err_type),
WouldBlock => continue,
},
}
2021-04-25 10:51:46 +02:00
}
2021-05-02 09:04:05 +02:00
Ok(())
2021-04-25 10:51:46 +02:00
}
}
/// The PL011 (PrimeCell UART) supports a fractional baud rate divider
/// From the wanted baudrate, we calculate the divider's two parts: integer and fractional parts.
/// Code inspired from the C SDK.
2021-05-02 08:42:51 +02:00
fn calculate_baudrate_dividers(
wanted_baudrate: &Baud,
frequency: &Hertz,
) -> Result<(u16, u16), Error> {
2021-05-05 08:02:53 +02:00
// See Chapter 4, Section 2 §7.1 from the datasheet for an explanation of how baudrate is
// calculated
2021-05-05 08:06:47 +02:00
let baudrate_div = frequency
.integer()
.checked_mul(8)
2021-07-26 21:16:09 +02:00
.and_then(|r| r.checked_div(wanted_baudrate.integer()))
2021-05-02 08:42:51 +02:00
.ok_or(Error::BadArgument)?;
2021-04-25 10:51:46 +02:00
Ok(match (baudrate_div >> 7, ((baudrate_div & 0x7F) + 1) / 2) {
2021-04-25 10:51:46 +02:00
(0, _) => (1, 0),
(int_part, _) if int_part >= 65535 => (65535, 0),
2021-04-25 10:51:46 +02:00
(int_part, frac_part) => (int_part as u16, frac_part as u16),
})
}
/// Baudrate configuration. Code loosely inspired from the C SDK.
2021-05-02 08:42:51 +02:00
fn configure_baudrate(
2021-05-09 09:53:22 +02:00
device: &mut dyn UartDevice,
2021-05-02 08:42:51 +02:00
wanted_baudrate: &Baud,
frequency: &Hertz,
) -> Result<Baud, Error> {
let (baud_div_int, baud_div_frac) = calculate_baudrate_dividers(wanted_baudrate, frequency)?;
2021-04-25 10:51:46 +02:00
// First we load the integer part of the divider.
2021-04-25 10:51:46 +02:00
device.uartibrd.write(|w| unsafe {
w.baud_divint().bits(baud_div_int as u16);
2021-04-25 10:51:46 +02:00
w
});
// Then we load the fractional part of the divider.
2021-04-25 10:51:46 +02:00
device.uartfbrd.write(|w| unsafe {
w.baud_divfrac().bits(baud_div_frac as u8);
2021-04-25 10:51:46 +02:00
w
});
// PL011 needs a (dummy) line control register write to latch in the
// divisors. We don't want to actually change LCR contents here.
2021-05-02 08:42:51 +02:00
device.uartlcr_h.modify(|_, w| w);
2021-04-25 10:51:46 +02:00
2021-05-02 08:42:51 +02:00
Ok(Baud(
2021-07-26 21:16:09 +02:00
(4 * frequency.integer()) / (64 * baud_div_int + baud_div_frac) as u32,
2021-05-02 08:42:51 +02:00
))
2021-04-25 10:51:46 +02:00
}
/// Format configuration. Code loosely inspired from the C SDK.
2021-05-02 08:42:51 +02:00
fn set_format<'w>(
w: &'w mut UART_LCR_H_Writer,
data_bits: &DataBits,
stop_bits: &StopBits,
parity: &Option<Parity>,
) -> &'w mut UART_LCR_H_Writer {
2021-04-25 10:51:46 +02:00
match parity {
Some(p) => {
w.pen().set_bit();
match p {
2021-05-04 19:56:36 +02:00
Parity::Odd => w.eps().clear_bit(),
2021-05-02 08:42:51 +02:00
Parity::Even => w.eps().set_bit(),
2021-04-25 10:51:46 +02:00
};
2021-05-02 08:42:51 +02:00
}
None => {
w.pen().bit(false);
}
2021-04-25 10:51:46 +02:00
};
2021-05-02 08:42:51 +02:00
unsafe {
w.wlen().bits(match data_bits {
DataBits::Five => 0b00,
DataBits::Six => 0b01,
DataBits::Seven => 0b10,
DataBits::Eight => 0b11,
})
2021-04-25 10:51:46 +02:00
};
match stop_bits {
2021-05-04 19:56:36 +02:00
StopBits::One => w.stp2().clear_bit(),
2021-05-02 08:42:51 +02:00
StopBits::Two => w.stp2().set_bit(),
2021-04-25 10:51:46 +02:00
};
w
}
2021-05-09 09:53:22 +02:00
impl<D: UartDevice> Read<u8> for UartPeripheral<Enabled, D> {
2021-05-02 09:04:05 +02:00
type Error = ReadErrorType;
fn read(&mut self) -> nb::Result<u8, Self::Error> {
let byte: &mut [u8] = &mut [0; 1];
2021-05-02 09:04:05 +02:00
match self.read_raw(byte) {
Ok(_) => Ok(byte[0]),
Err(e) => match e {
2021-05-05 08:06:47 +02:00
Other(inner) => Err(Other(inner.err_type)),
WouldBlock => Err(WouldBlock),
2021-05-02 09:04:05 +02:00
},
}
}
}
2021-05-09 09:53:22 +02:00
impl<D: UartDevice> Write<u8> for UartPeripheral<Enabled, D> {
type Error = Infallible;
fn write(&mut self, word: u8) -> nb::Result<(), Self::Error> {
2021-05-05 08:06:47 +02:00
if self.write_raw(&[word]).is_err() {
Err(WouldBlock)
2021-05-02 08:42:51 +02:00
} else {
Ok(())
}
}
fn flush(&mut self) -> nb::Result<(), Self::Error> {
self.transmit_flushed()
}
}
impl<D: UartDevice> fmt::Write for UartPeripheral<Enabled, D> {
fn write_str(&mut self, s: &str) -> fmt::Result {
s.bytes()
.try_for_each(|c| nb::block!(self.write(c)))
.map_err(|_| fmt::Error)
}
}