debug and fix a bunch of modem read/write errors
This commit is contained in:
parent
018af71262
commit
98d0a66cf3
5 changed files with 121 additions and 84 deletions
11
src/accel.rs
11
src/accel.rs
|
@ -6,12 +6,13 @@ use std::time::Duration;
|
||||||
|
|
||||||
use crate::types::*;
|
use crate::types::*;
|
||||||
|
|
||||||
pub fn main(sender: SyncSender<Msg>) -> Result<(), anyhow::Error> {
|
pub fn main(sender: SyncSender<Msg>) -> anyhow::Result<()> {
|
||||||
|
let mut c = 1_usize;
|
||||||
println!("entering ACCELERATOR sender loop ...");
|
println!("entering ACCELERATOR sender loop ...");
|
||||||
for i in 0..20 {
|
loop {
|
||||||
println!("sending ACCELERATOR message ({}) of 20 ...", i);
|
println!("sending ACCELERATOR message No. {}", c);
|
||||||
let _ = sender.send(Msg::Accelerometer("{\"velocity\": 21.43, \"altitude\": 367}".to_string()))?;
|
let _ = sender.send(Msg::Accelerometer("{\"velocity\": 21.43, \"altitude\": 367}".to_string()))?;
|
||||||
thread::sleep(Duration::from_millis(2000));
|
thread::sleep(Duration::from_secs(5));
|
||||||
|
c += 1;
|
||||||
}
|
}
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -268,7 +268,7 @@ impl Command {
|
||||||
Command {
|
Command {
|
||||||
text: format!("AT+CIPSTART=\"TCP\",\"{}\",\"{}\"", addr, port),
|
text: format!("AT+CIPSTART=\"TCP\",\"{}\",\"{}\"", addr, port),
|
||||||
timeout: Duration::from_millis(5000),
|
timeout: Duration::from_millis(5000),
|
||||||
contains: Some("CONNECT OK".to_string()),
|
contains: Some("OK".to_string()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
18
src/main.rs
18
src/main.rs
|
@ -8,11 +8,7 @@ mod serial;
|
||||||
mod types;
|
mod types;
|
||||||
|
|
||||||
use anyhow;
|
use anyhow;
|
||||||
use std::{
|
use std::{thread::{self, JoinHandle}, time::Duration};
|
||||||
thread::{self, JoinHandle},
|
|
||||||
time::Duration,
|
|
||||||
};
|
|
||||||
|
|
||||||
use esp_idf_hal::peripherals::Peripherals;
|
use esp_idf_hal::peripherals::Peripherals;
|
||||||
use types::*;
|
use types::*;
|
||||||
|
|
||||||
|
@ -35,11 +31,11 @@ fn main() -> anyhow::Result<()> {
|
||||||
|
|
||||||
let mut threads: Vec<JoinHandle<anyhow::Result<_>>> = vec![];
|
let mut threads: Vec<JoinHandle<anyhow::Result<_>>> = vec![];
|
||||||
|
|
||||||
// Rx/Tx pins for the GPS modem
|
// // Rx/Tx pins for the GPS modem
|
||||||
let gps_rx = dp.pins.gpio13;
|
// let gps_rx = dp.pins.gpio13;
|
||||||
let gps_tx = dp.pins.gpio12;
|
// let gps_tx = dp.pins.gpio12;
|
||||||
// UART interface for the GPS modem
|
// // UART interface for the GPS modem
|
||||||
let gps_uart = dp.uart2;
|
// let gps_uart = dp.uart2;
|
||||||
|
|
||||||
|
|
||||||
let (gps_sender, receiver) = std::sync::mpsc::sync_channel::<Msg>(1);
|
let (gps_sender, receiver) = std::sync::mpsc::sync_channel::<Msg>(1);
|
||||||
|
@ -49,7 +45,7 @@ fn main() -> anyhow::Result<()> {
|
||||||
// threads.push(thread::spawn(move || gps::main(gps_rx, gps_tx, gps_uart, gps_sender)));
|
// threads.push(thread::spawn(move || gps::main(gps_rx, gps_tx, gps_uart, gps_sender)));
|
||||||
//thread::sleep(Duration::from_millis(1000));
|
//thread::sleep(Duration::from_millis(1000));
|
||||||
threads.push(thread::spawn(move || accel::main(accel_sender)));
|
threads.push(thread::spawn(move || accel::main(accel_sender)));
|
||||||
//thread::sleep(Duration::from_millis(1000));
|
thread::sleep(Duration::from_millis(3000));
|
||||||
|
|
||||||
let _ = modem::main(modem_rx, modem_tx, modem_uart, modem_pwrkey, modem_rst, modem_power, receiver)?;
|
let _ = modem::main(modem_rx, modem_tx, modem_uart, modem_pwrkey, modem_rst, modem_power, receiver)?;
|
||||||
|
|
||||||
|
|
154
src/modem.rs
154
src/modem.rs
|
@ -16,10 +16,8 @@ use esp_idf_hal::serial::{self, Rx, Tx};
|
||||||
|
|
||||||
use embedded_hal::digital::v2::OutputPin;
|
use embedded_hal::digital::v2::OutputPin;
|
||||||
|
|
||||||
use mqtt::packet::{ConnectPacket, PublishPacket, QoSWithPacketIdentifier};
|
use mqtt::packet::{ConnectPacket, PublishPacket, QoSWithPacketIdentifier, VariablePacket};
|
||||||
use mqtt::{Encodable, TopicName};
|
use mqtt::{Encodable, Decodable, TopicName};
|
||||||
|
|
||||||
const MAX_TCP_MANUAL_REPLY_SIZE: usize = 300;
|
|
||||||
|
|
||||||
pub type Result<T> = std::result::Result<T, ModemError>;
|
pub type Result<T> = std::result::Result<T, ModemError>;
|
||||||
|
|
||||||
|
@ -32,7 +30,7 @@ pub enum ModemError {
|
||||||
CommandError(String),
|
CommandError(String),
|
||||||
SetupError(String),
|
SetupError(String),
|
||||||
SendDataError(String),
|
SendDataError(String),
|
||||||
ReadError,
|
ReadError(String),
|
||||||
TimeoutError,
|
TimeoutError,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -77,10 +75,14 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
thread::sleep(Duration::from_millis(1000));
|
thread::sleep(Duration::from_millis(1000));
|
||||||
pwrkey.set_high().map_err(|_| ModemError::SetupError("Error setting PWRKEY to high.".to_string()))?;
|
pwrkey.set_high().map_err(|_| ModemError::SetupError("Error setting PWRKEY to high.".to_string()))?;
|
||||||
println!("Waiting for sim module to come online ...");
|
println!("Waiting for sim module to come online ...");
|
||||||
|
thread::sleep(Duration::from_millis(3000));
|
||||||
loop {
|
loop {
|
||||||
match self.send_command(Command::probe()) {
|
match self.send_command(Command::probe()) {
|
||||||
Ok(_) => break,
|
Ok(_) => break,
|
||||||
_ => continue,
|
_ => {
|
||||||
|
thread::sleep(Duration::from_millis(2000));
|
||||||
|
continue
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -90,16 +92,17 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
/// None, then the first line is returned. If a timeout is reached. The timeout is provided on
|
/// None, then the first line is returned. If a timeout is reached. The timeout is provided on
|
||||||
/// input via the `timeout` argument. The first argument `contains` is checked against every
|
/// input via the `timeout` argument. The first argument `contains` is checked against every
|
||||||
/// line in the response.
|
/// line in the response.
|
||||||
fn command_read_response(&mut self) -> Result<String> {
|
fn command_read_response(&mut self, contains: Option<String>) -> Result<String> {
|
||||||
let mut response = String::new();
|
let mut response = String::new();
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let mut buf = vec![0; 1024];
|
let mut buf = vec![0; 1024];
|
||||||
let num_bytes = self.serial
|
let num_bytes = self.serial
|
||||||
.read(buf.as_mut_slice())
|
.read(buf.as_mut_slice())
|
||||||
.map_err(|_| ModemError::ReadError)?;
|
.map_err(|err| ModemError::ReadError(format!("Error in serial.read(buf) ({:?})", err)))?;
|
||||||
|
|
||||||
response.push_str(std::str::from_utf8(&buf[0..num_bytes]).map_err(|_| ModemError::ReadError)?);
|
response.push_str(std::str::from_utf8(&buf[0..num_bytes])
|
||||||
|
.map_err(|err| ModemError::ReadError(format!("Error in str::from_utf8 ({:?})", err)))?);
|
||||||
|
|
||||||
if num_bytes < buf.len() {
|
if num_bytes < buf.len() {
|
||||||
break
|
break
|
||||||
|
@ -107,12 +110,20 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
}
|
}
|
||||||
|
|
||||||
print!("Read {} bytes from serial: {}", response.len(), response);
|
print!("Read {} bytes from serial: {}", response.len(), response);
|
||||||
|
if let Some(c) = contains {
|
||||||
|
if response.contains(&c) {
|
||||||
Ok(response)
|
Ok(response)
|
||||||
|
} else {
|
||||||
|
Err(ModemError::CommandError(format!("Didn't get expected ({}) from modem.", c)))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Ok(response)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn send_command(&mut self, cmd: Command) -> Result<String> {
|
fn send_command(&mut self, cmd: Command) -> Result<String> {
|
||||||
println!("-----------------------------------------------------------");
|
println!("-----------------------------------------------------------");
|
||||||
println!("Sending to TX ({}) ...", cmd.text);
|
println!("Sending {} ...", cmd.text);
|
||||||
|
|
||||||
let _ = self.serial
|
let _ = self.serial
|
||||||
.write_bytes(cmd.text.as_bytes())
|
.write_bytes(cmd.text.as_bytes())
|
||||||
|
@ -122,16 +133,23 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
.write(&['\r' as u8])
|
.write(&['\r' as u8])
|
||||||
.map_err(|_| ModemError::SendDataError(format!("Error in send_command({})", cmd.text)))?;
|
.map_err(|_| ModemError::SendDataError(format!("Error in send_command({})", cmd.text)))?;
|
||||||
|
|
||||||
self.command_read_response()
|
self.command_read_response(cmd.contains)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_prompt(&mut self) -> Result<()> {
|
fn handle_prompt(&mut self) -> Result<()> {
|
||||||
let mut prompt_buf = vec![0; 3];
|
let mut prompt_buf = vec![0; 256];
|
||||||
let prompt_len = self.serial.read(&mut prompt_buf).map_err(|_| ModemError::ReadError)?;
|
let prompt_len = self.serial.read(&mut prompt_buf)
|
||||||
let prompt = std::str::from_utf8(prompt_buf.as_slice()).unwrap_or("");
|
.map_err(|err| ModemError::ReadError(format!("Error in handle_prompt() ({:?})", err)))?;
|
||||||
|
|
||||||
if prompt_len != 3 && prompt != "\r\n>" {
|
let prompt = String::from_utf8(prompt_buf[0..prompt_len].to_vec())
|
||||||
let msg = format!("Prompt error, expected \\r\\n>, got {:?}", prompt);
|
.unwrap_or("".to_string())
|
||||||
|
.trim()
|
||||||
|
.to_string();
|
||||||
|
|
||||||
|
println!("Prompt is: ({})", prompt);
|
||||||
|
|
||||||
|
if prompt != ">" {
|
||||||
|
let msg = format!("Prompt error, expected (>), got ({})", prompt);
|
||||||
Err(ModemError::SendDataError(msg))
|
Err(ModemError::SendDataError(msg))
|
||||||
} else {
|
} else {
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -139,19 +157,34 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn tcp_manual_send_data(&mut self, buf: &[u8]) -> Result<String> {
|
fn tcp_manual_send_data(&mut self, buf: &[u8]) -> Result<String> {
|
||||||
|
println!("Sending AT+CIPSEND to serial TX!");
|
||||||
let _ = self.serial
|
let _ = self.serial
|
||||||
.write("AT+CIPSEND\r".as_bytes())
|
.write("AT+CIPSEND\r".as_bytes())
|
||||||
.map_err(|_| ModemError::SendDataError("Error in tcp_manual_send_data ... AT_CIPSEND\\r".to_string()))?;
|
.map_err(|_| ModemError::SendDataError("Error in tcp_manual_send_data ... AT_CIPSEND\\r".to_string()))?;
|
||||||
|
|
||||||
let _ = self.handle_prompt()?;
|
let _ = self.handle_prompt()?;
|
||||||
|
println!("Handled prompt OK!!");
|
||||||
|
|
||||||
|
println!("Writing bytes in serial TX! ({:?})", String::from_utf8(buf.to_vec()));
|
||||||
self.serial
|
self.serial
|
||||||
.write_bytes(buf)
|
.write_bytes(buf)
|
||||||
.map_err(|err| ModemError::SendDataError(format!("{:?}", err)))?;
|
.map_err(|err| ModemError::SendDataError(format!("{:?}", err)))?;
|
||||||
self.serial
|
self.serial
|
||||||
.write(&[26_u8]) // 26_u8 = Ctrl+z - to end sending data
|
.write(&[26_u8]) // 26_u8 = Ctrl+z - to end sending data
|
||||||
.map_err(|err| ModemError::SendDataError(format!("{:?}", err)))?;
|
.map_err(|err| ModemError::SendDataError(format!("{:?}", err)))?;
|
||||||
self.command_read_response()
|
println!("DONE Writing bytes in serial TX!");
|
||||||
|
|
||||||
|
thread::sleep(Duration::from_millis(500));
|
||||||
|
|
||||||
|
println!("Reading bytes in serial RX!");
|
||||||
|
for _ in 0..3 {
|
||||||
|
let res = self.command_read_response(Some("SEND OK".into()));
|
||||||
|
if res.is_ok() {
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
thread::sleep(Duration::from_millis(1000))
|
||||||
|
}
|
||||||
|
Err(ModemError::ReadError(format!("ReadError: cannot read serial RX!")))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn gprs_status(&mut self) -> Result<String> {
|
pub fn gprs_status(&mut self) -> Result<String> {
|
||||||
|
@ -218,8 +251,15 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn tcp_connect(&mut self, addr: &str, port: u16) -> Result<()> {
|
pub fn tcp_connect(&mut self, addr: &str, port: u16) -> Result<()> {
|
||||||
self.send_command(Command::tcp_connect(addr, port))
|
let _ = self.send_command(Command::tcp_connect(addr, port))?;
|
||||||
.map(|_| ())
|
for _ in 0..3 {
|
||||||
|
if let Ok(reply) = self.command_read_response(Some("CONNECT_OK".to_string())) {
|
||||||
|
println!("TCP connect replied with {}", reply);
|
||||||
|
break
|
||||||
|
}
|
||||||
|
thread::sleep(Duration::from_millis(500));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn tcp_set_quick_mode(&mut self, mode: bool) -> Result<()> {
|
pub fn tcp_set_quick_mode(&mut self, mode: bool) -> Result<()> {
|
||||||
|
@ -248,18 +288,21 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
|
|
||||||
pub fn tcp_receive_reply_len(&mut self) -> Result<usize> {
|
pub fn tcp_receive_reply_len(&mut self) -> Result<usize> {
|
||||||
let reply = self.send_command(Command::tcp_receive_reply_len())?;
|
let reply = self.send_command(Command::tcp_receive_reply_len())?;
|
||||||
reply.lines()
|
println!("Receiving TCP reply length!");
|
||||||
|
let res = reply.lines()
|
||||||
.filter(|line| line.contains("+CIPRXGET: 4"))
|
.filter(|line| line.contains("+CIPRXGET: 4"))
|
||||||
.next()
|
.next()
|
||||||
.ok_or(ModemError::CommandError("reply not found :/".to_string()))
|
.ok_or(ModemError::CommandError("reply body missing :/".to_string()))
|
||||||
.map(|line| self.tcp_parse_response_size(line))
|
.and_then(|line| self.tcp_parse_response_size(line))
|
||||||
.unwrap_or(Err(ModemError::CommandError(format!("received 0 elements from parsing"))))
|
.map_err(|_| ModemError::CommandError(format!("received 0 elements from parsing")));
|
||||||
|
println!("Received ({:?})", res);
|
||||||
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn tcp_receive(&mut self, buf: &mut [u8]) -> Result<usize> {
|
pub fn tcp_receive(&mut self, buf: &mut [u8]) -> Result<usize> {
|
||||||
let mut size = 0;
|
let mut size = 0;
|
||||||
loop {
|
loop {
|
||||||
let reply = self.send_command(Command::tcp_receive(MAX_TCP_MANUAL_REPLY_SIZE))
|
let reply = self.send_command(Command::tcp_receive(buf.len()))
|
||||||
.map(|reply| {
|
.map(|reply| {
|
||||||
// TODO: parse the response properly
|
// TODO: parse the response properly
|
||||||
// 1. the first line is \r\n
|
// 1. the first line is \r\n
|
||||||
|
@ -349,7 +392,7 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
self.serial
|
self.serial
|
||||||
.write(buf)
|
.write(buf)
|
||||||
.map_err(|err| ModemError::SendDataError(format!("Error sending bytes via serial ({:?})", err)))?;
|
.map_err(|err| ModemError::SendDataError(format!("Error sending bytes via serial ({:?})", err)))?;
|
||||||
let _ = self.command_read_response();
|
let _ = self.command_read_response(None);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -380,33 +423,44 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
.map(|_| ())
|
.map(|_| ())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn mqtt_receive_reply(&mut self) -> std::result::Result<(), anyhow::Error> {
|
fn mqtt_receive_reply(&mut self) -> Result<VariablePacket> {
|
||||||
println!("receiving mqtt reply from modem ...");
|
for _ in 0..3 {
|
||||||
let size = self.tcp_receive_reply_len()?;
|
let size = self.tcp_receive_reply_len()?;
|
||||||
println!("receiving reply len({}) ...", size);
|
println!("received reply len({}) ...", size);
|
||||||
|
if size == 0 {
|
||||||
|
println!("retrying ...");
|
||||||
|
continue
|
||||||
|
} else {
|
||||||
let mut reply = vec![0 as u8; size];
|
let mut reply = vec![0 as u8; size];
|
||||||
println!("receiving tcp reply ...");
|
println!("receiving mqtt reply ...");
|
||||||
let _ = self.tcp_receive(&mut reply);
|
let _ = self.tcp_receive(&mut reply);
|
||||||
println!("received tcp reply ...");
|
let reply = std::str::from_utf8(&reply).unwrap_or("");
|
||||||
Ok(())
|
println!("received mqtt reply ({})", reply);
|
||||||
|
return VariablePacket::decode(&mut reply.as_bytes())
|
||||||
|
.map_err(|err| ModemError::CommandError(format!("Undecodable MQTT message. ({:?})", err)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(ModemError::ReadError("TCP server didn't respond!".into()))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn mqtt_connect(&mut self, device_id: &str) -> std::result::Result<(), anyhow::Error> {
|
fn mqtt_connect(&mut self, device_id: &str) -> anyhow::Result<()> {
|
||||||
let mut buf = Vec::new();
|
let mut buf = Vec::new();
|
||||||
let mut conn = ConnectPacket::new(device_id);
|
let mut conn = ConnectPacket::new(device_id);
|
||||||
conn.set_clean_session(true);
|
conn.set_clean_session(true);
|
||||||
conn.set_keep_alive(100);
|
conn.set_keep_alive(100);
|
||||||
let _ = conn.encode(&mut buf)?;
|
let _ = conn.encode(&mut buf)?;
|
||||||
let _ = self.tcp_manual_send(&mut buf)?;
|
self.tcp_manual_send(&mut buf).ok();
|
||||||
|
|
||||||
thread::sleep(Duration::from_millis(2000));
|
let reply = self.mqtt_receive_reply()?;
|
||||||
drop(buf);
|
println!("mqtt decoded packet: ({:?})", reply);
|
||||||
|
|
||||||
let _ = self.mqtt_receive_reply()?;
|
match reply {
|
||||||
Ok(())
|
VariablePacket::ConnackPacket(_) => Ok(()),
|
||||||
|
_ => Err(anyhow::Error::msg("Invalid MQTT reply ... expected CONNACK!"))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn mqtt_publish(&mut self, _device_id: &str, message: &str) -> std::result::Result<(), anyhow::Error> {
|
fn mqtt_publish(&mut self, _device_id: &str, message: &str) -> anyhow::Result<()> {
|
||||||
println!("entered mqtt publish ...");
|
println!("entered mqtt publish ...");
|
||||||
let mut buf = Vec::new();
|
let mut buf = Vec::new();
|
||||||
let packet = PublishPacket::new(
|
let packet = PublishPacket::new(
|
||||||
|
@ -414,16 +468,10 @@ impl<UART: serial::Uart> Modem<UART> {
|
||||||
QoSWithPacketIdentifier::Level0,
|
QoSWithPacketIdentifier::Level0,
|
||||||
message.as_bytes(),
|
message.as_bytes(),
|
||||||
);
|
);
|
||||||
println!("created mqtt publish packet ...");
|
|
||||||
let _ = packet.encode(&mut buf)?;
|
let _ = packet.encode(&mut buf)?;
|
||||||
println!("modem tcp send publish pakage ...");
|
println!("created mqtt publish packet ... ({})",
|
||||||
let _ = self.tcp_manual_send(&mut buf)?;
|
std::str::from_utf8(buf.as_slice()).unwrap_or(""));
|
||||||
|
self.tcp_manual_send(&mut buf).ok();
|
||||||
thread::sleep(Duration::from_millis(2000));
|
|
||||||
drop(buf);
|
|
||||||
|
|
||||||
println!("receiving modem publish reply ...");
|
|
||||||
let _ = self.mqtt_receive_reply()?;
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -434,18 +482,6 @@ impl<UART: serial::Uart> std::io::Read for Modem<UART> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<UART: serial::Uart> std::io::Write for Modem<UART> {
|
|
||||||
fn write(&mut self, buf: &[u8]) -> std::io::Result<usize> {
|
|
||||||
self.tcp_manual_send(buf)
|
|
||||||
.map_err(|_| std::io::Error::from(std::io::ErrorKind::ConnectionAborted))?;
|
|
||||||
Ok(buf.len())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn flush(&mut self) -> std::io::Result<()> {
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn main<T: Sync + Send>(
|
pub fn main<T: Sync + Send>(
|
||||||
rx: esp_idf_hal::gpio::Gpio26<T>,
|
rx: esp_idf_hal::gpio::Gpio26<T>,
|
||||||
tx: esp_idf_hal::gpio::Gpio27<T>,
|
tx: esp_idf_hal::gpio::Gpio27<T>,
|
||||||
|
|
|
@ -77,7 +77,11 @@ impl<UART: serial::Uart> SerialIO<UART> {
|
||||||
Err(nb::Error::Other(err)) => println!("Serial read error :: {:?}", err),
|
Err(nb::Error::Other(err)) => println!("Serial read error :: {:?}", err),
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
if count > 0 {
|
||||||
Ok(count)
|
Ok(count)
|
||||||
|
} else {
|
||||||
|
Err(nb::Error::Other(SerialError::ReadError("Rx buffer empty.".to_string())))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -87,7 +91,7 @@ const READ_WAIT_TIME: u64 = 50;
|
||||||
impl<UART: serial::Uart> io::Read for SerialIO<UART> {
|
impl<UART: serial::Uart> io::Read for SerialIO<UART> {
|
||||||
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
||||||
let count = nb::block!(self.read_bytes(buf))
|
let count = nb::block!(self.read_bytes(buf))
|
||||||
.map_err(|_| io::Error::from(io::ErrorKind::Other))?;
|
.map_err(|err| io::Error::new(io::ErrorKind::Other, err))?;
|
||||||
Ok(count)
|
Ok(count)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -95,11 +99,11 @@ impl<UART: serial::Uart> io::Read for SerialIO<UART> {
|
||||||
impl<UART: serial::Uart> io::Write for SerialIO<UART> {
|
impl<UART: serial::Uart> io::Write for SerialIO<UART> {
|
||||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
||||||
nb::block!(self.write_bytes(buf))
|
nb::block!(self.write_bytes(buf))
|
||||||
.map_err(|_| io::Error::from(io::ErrorKind::Other))
|
.map_err(|err| io::Error::new(io::ErrorKind::Other, err))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn flush(&mut self) -> io::Result<()> {
|
fn flush(&mut self) -> io::Result<()> {
|
||||||
self.tx.flush()
|
self.tx.flush()
|
||||||
.map_err(|_| io::Error::from(io::ErrorKind::Other))
|
.map_err(|err| io::Error::new(io::ErrorKind::Other, SerialError::ReadError(format!("Flush error ({:?})", err))))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue