debug and fix a bunch of modem read/write errors

This commit is contained in:
Vladan Popovic 2022-11-29 21:34:38 +01:00
parent 018af71262
commit 98d0a66cf3
5 changed files with 121 additions and 84 deletions

View File

@ -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(())
} }

View File

@ -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()),
} }
} }

View File

@ -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)?;

View File

@ -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);
Ok(response) if let Some(c) = contains {
if response.contains(&c) {
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);
let mut reply = vec![0 as u8; size]; if size == 0 {
println!("receiving tcp reply ..."); println!("retrying ...");
let _ = self.tcp_receive(&mut reply); continue
println!("received tcp reply ..."); } else {
Ok(()) let mut reply = vec![0 as u8; size];
println!("receiving mqtt reply ...");
let _ = self.tcp_receive(&mut reply);
let reply = std::str::from_utf8(&reply).unwrap_or("");
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>,

View File

@ -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),
} }
}; };
Ok(count) if count > 0 {
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))))
} }
} }