use serialport::{SerialPortBuilder, TTYPort};
use std::fmt;
use std::io::{ErrorKind as IoErrorKind, Read, Write};
pub struct Serial(pub TTYPort);
impl Serial {
pub fn open(path: String, baud_rate: u32) -> Result<Serial, serialport::Error> {
Ok(Serial(serialport::new(path, baud_rate).open_native()?))
}
pub fn open_from_builder(builder: SerialPortBuilder) -> Result<Serial, serialport::Error> {
Ok(Serial(builder.open_native()?))
}
}
fn translate_io_errors(err: std::io::Error) -> nb::Error<SerialError> {
match err.kind() {
IoErrorKind::WouldBlock | IoErrorKind::TimedOut | IoErrorKind::Interrupted => {
nb::Error::WouldBlock
}
err => nb::Error::Other(SerialError { err }),
}
}
impl embedded_hal_nb::serial::ErrorType for Serial {
type Error = SerialError;
}
impl embedded_hal_nb::serial::Read<u8> for Serial {
fn read(&mut self) -> nb::Result<u8, Self::Error> {
let mut buffer = [0; 1];
let bytes_read = self.0.read(&mut buffer).map_err(translate_io_errors)?;
if bytes_read == 1 {
Ok(buffer[0])
} else {
Err(nb::Error::WouldBlock)
}
}
}
impl embedded_hal_nb::serial::Write<u8> for Serial {
fn write(&mut self, word: u8) -> nb::Result<(), Self::Error> {
self.0.write(&[word]).map_err(translate_io_errors)?;
Ok(())
}
fn flush(&mut self) -> nb::Result<(), Self::Error> {
self.0.flush().map_err(translate_io_errors)
}
}
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
pub struct SerialError {
err: IoErrorKind,
}
impl SerialError {
pub fn inner(&self) -> &IoErrorKind {
&self.err
}
}
impl fmt::Display for SerialError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.err)
}
}
impl std::error::Error for SerialError {}
impl embedded_hal_nb::serial::Error for SerialError {
#[allow(clippy::match_single_binding)]
fn kind(&self) -> embedded_hal_nb::serial::ErrorKind {
use embedded_hal_nb::serial::ErrorKind::*;
Other
}
}
#[cfg(test)]
mod test {
use embedded_hal_nb::serial::{Read, Write};
use std::io::{Read as IoRead, Write as IoWrite};
use super::*;
fn create_pty_and_serial() -> (std::fs::File, Serial) {
let (master, _slave, name) =
openpty::openpty(None, None, None).expect("Creating pty failed");
let serial = Serial::open(name, 9600).expect("Creating TTYPort failed");
(master, serial)
}
#[test]
fn create_serial_from_builder() {
let (_master, _slave, name) =
openpty::openpty(None, None, None).expect("Creating pty failed");
let builder = serialport::new(name, 9600);
let _serial = Serial::open_from_builder(builder).expect("Creating TTYPort failed");
}
#[test]
fn test_empty_read() {
let (mut _master, mut serial) = create_pty_and_serial();
assert_eq!(Err(nb::Error::WouldBlock), serial.read());
}
#[test]
fn test_read() {
let (mut master, mut serial) = create_pty_and_serial();
master.write_all(&[1]).expect("Write failed");
assert_eq!(Ok(1), serial.read());
}
#[test]
fn test_write() {
let (mut master, mut serial) = create_pty_and_serial();
serial.write(2).expect("Write failed");
let mut buf = [0; 2];
assert_eq!(1, master.read(&mut buf).unwrap());
assert_eq!(buf, [2, 0]);
}
}