-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
implemented decoding of sensor measurement for EsfMeas
- fix unittest Signed-off-by: Andrei Gherghescu <[email protected]>
- Loading branch information
Showing
6 changed files
with
517 additions
and
167 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
[package] | ||
authors = ["Lane Kolbly <[email protected]>", "Andrei Gherghescu <[email protected]>"] | ||
edition = "2021" | ||
name = "example-basic-cli" | ||
publish = false | ||
version = "0.1.0" | ||
|
||
[dependencies] | ||
chrono = "0.4.29" | ||
clap = {version = "4.2.7", features = ["cargo"]} | ||
serde_json = "1.0.105" | ||
serialport = "4.2.2" | ||
ublox = {path = "../../ublox"} | ||
|
||
[features] | ||
alloc = ["ublox/alloc"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,302 @@ | ||
use chrono::prelude::*; | ||
use clap::{value_parser, Arg, Command}; | ||
use serialport::{ | ||
DataBits as SerialDataBits, FlowControl as SerialFlowControl, Parity as SerialParity, | ||
StopBits as SerialStopBits, | ||
}; | ||
use std::convert::TryInto; | ||
use std::time::Duration; | ||
use ublox::*; | ||
|
||
fn main() { | ||
let matches = Command::new("uBlox CLI example program") | ||
.author(clap::crate_authors!()) | ||
.about("Demonstrates usage of the Rust uBlox API") | ||
.arg_required_else_help(true) | ||
.arg( | ||
Arg::new("port") | ||
.value_name("port") | ||
.short('p') | ||
.long("port") | ||
.required(true) | ||
.help("Serial port to open"), | ||
) | ||
.arg( | ||
Arg::new("baud") | ||
.value_name("baud") | ||
.short('s') | ||
.long("baud") | ||
.required(false) | ||
.value_parser(value_parser!(u32)) | ||
.help("Baud rate of the port"), | ||
) | ||
.arg( | ||
Arg::new("stop-bits") | ||
.long("stop-bits") | ||
.help("Number of stop bits to use") | ||
.required(false) | ||
.value_parser(["1", "2"]) | ||
.default_value("1"), | ||
) | ||
.arg( | ||
Arg::new("data-bits") | ||
.long("data-bits") | ||
.help("Number of data bits to use") | ||
.required(false) | ||
.value_parser(["5", "6", "7", "8"]) | ||
.default_value("8"), | ||
) | ||
.get_matches(); | ||
|
||
let port = matches | ||
.get_one::<String>("port") | ||
.expect("Expected required 'port' cli argumnet"); | ||
let baud = matches.get_one::<u32>("baud").cloned().unwrap_or(115200); | ||
let stop_bits = match matches.get_one::<String>("stop-bits").map(|s| s.as_str()) { | ||
Some("2") => SerialStopBits::Two, | ||
_ => SerialStopBits::One, | ||
}; | ||
let data_bits = match matches.get_one::<String>("data-bits").map(|s| s.as_str()) { | ||
Some("5") => SerialDataBits::Five, | ||
Some("6") => SerialDataBits::Six, | ||
Some("7") => SerialDataBits::Seven, | ||
_ => SerialDataBits::Eight, | ||
}; | ||
|
||
let builder = serialport::new(port, baud) | ||
.stop_bits(stop_bits) | ||
.data_bits(data_bits) | ||
.timeout(Duration::from_millis(1)) | ||
.parity(SerialParity::None) | ||
.flow_control(SerialFlowControl::None); | ||
|
||
println!("{:?}", &builder); | ||
let port = builder.open().unwrap_or_else(|e| { | ||
eprintln!("Failed to open \"{}\". Error: {}", port, e); | ||
::std::process::exit(1); | ||
}); | ||
let mut device = Device::new(port); | ||
|
||
// Configure the device to talk UBX | ||
println!("Configuring UART1 port ..."); | ||
device | ||
.write_all( | ||
&CfgPrtUartBuilder { | ||
portid: UartPortId::Uart1, | ||
reserved0: 0, | ||
tx_ready: 0, | ||
mode: UartMode::new(DataBits::Eight, Parity::None, StopBits::One), | ||
baud_rate: baud, | ||
in_proto_mask: InProtoMask::UBLOX, | ||
out_proto_mask: OutProtoMask::union(OutProtoMask::NMEA, OutProtoMask::UBLOX), | ||
flags: 0, | ||
reserved5: 0, | ||
} | ||
.into_packet_bytes(), | ||
) | ||
.expect("Could not configure UBX-CFG-PRT-UART"); | ||
device | ||
.wait_for_ack::<CfgPrtUart>() | ||
.expect("Could not acknowledge UBX-CFG-PRT-UART msg"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<HnrAtt>([0, 0, 0, 0, 0, 0]).into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-HNR-ATT"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<HnrIns>([0, 0, 0, 0, 0, 0]).into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-HNR-INS"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<HnrPvt>([0, 0, 0, 0, 0, 0]).into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-HNR-PVT"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<EsfIns>([0, 1, 0, 1, 0, 0]).into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-ESF-INS"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<EsfAlg>([0, 1, 0, 1, 0, 0]).into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-ESF-INS"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<NavPvt>([0, 0, 0, 0, 0, 0]).into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-NAV-PVT"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<EsfStatus>([0, 1, 0, 1, 0, 0]) | ||
.into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-ESF-STATUS"); | ||
|
||
device | ||
.write_all( | ||
&CfgMsgAllPortsBuilder::set_rate_for::<EsfMeas>([0, 1, 0, 1, 0, 0]).into_packet_bytes(), | ||
) | ||
.expect("Could not configure ports for UBX-ESF-MEAS"); | ||
|
||
// Send a packet request for the MonVer packet | ||
device | ||
.write_all(&UbxPacketRequest::request_for::<MonVer>().into_packet_bytes()) | ||
.expect("Unable to write request/poll for UBX-MON-VER message"); | ||
|
||
// Start reading data | ||
println!("Opened uBlox device, waiting for messages..."); | ||
loop { | ||
device | ||
.update(|packet| match packet { | ||
PacketRef::MonVer(packet) => { | ||
println!( | ||
"SW version: {} HW version: {}; Extensions: {:?}", | ||
packet.software_version(), | ||
packet.hardware_version(), | ||
packet.extension().collect::<Vec<&str>>() | ||
); | ||
}, | ||
// PacketRef::NavPvt(sol) => { | ||
// let has_time = sol.fix_type() == GpsFix::Fix3D | ||
// || sol.fix_type() == GpsFix::GPSPlusDeadReckoning | ||
// || sol.fix_type() == GpsFix::TimeOnlyFix; | ||
// let has_posvel = sol.fix_type() == GpsFix::Fix3D | ||
// || sol.fix_type() == GpsFix::GPSPlusDeadReckoning; | ||
|
||
// if has_posvel { | ||
// let pos: Position = (&sol).into(); | ||
// let vel: Velocity = (&sol).into(); | ||
// println!( | ||
// "Latitude: {:.5} Longitude: {:.5} Altitude: {:.2}m", | ||
// pos.lat, pos.lon, pos.alt | ||
// ); | ||
// println!( | ||
// "Speed: {:.2} m/s Heading: {:.2} degrees", | ||
// vel.speed, vel.heading | ||
// ); | ||
// println!("Sol: {:?}", sol); | ||
// } | ||
|
||
// if has_time { | ||
// let time: DateTime<Utc> = (&sol) | ||
// .try_into() | ||
// .expect("Could not parse NAV-PVT time field to UTC"); | ||
// println!("Time: {:?}", time); | ||
// } | ||
// }, | ||
PacketRef::EsfStatus(status) => { | ||
println!( | ||
"EsfStatus: tow: {}, version: {}, {:?},{:?}, fusion_mode: {:?}, num_sens: {}", | ||
status.itow(), | ||
status.version(), | ||
status.init_status1(), | ||
status.init_status2(), | ||
status.fusion_mode(), | ||
status.num_sens(), | ||
); | ||
for s in status.data() { | ||
println!("{:?}", s); | ||
} | ||
}, | ||
|
||
PacketRef::EsfMeas(msg) => { | ||
println!("{:?}", msg); | ||
println!("time_tag: {}", msg.time_tag()); | ||
for s in msg.data() { | ||
println!("{:?}", s); | ||
println!("{:?}", s.value()); | ||
} | ||
println!("calib_tag: {:?}", msg.calib_tag()); | ||
}, | ||
|
||
_ => { | ||
println!("{:?}", packet); | ||
}, | ||
}) | ||
.unwrap(); | ||
} | ||
} | ||
|
||
struct Device { | ||
port: Box<dyn serialport::SerialPort>, | ||
parser: Parser<Vec<u8>>, | ||
} | ||
|
||
impl Device { | ||
pub fn new(port: Box<dyn serialport::SerialPort>) -> Device { | ||
let parser = Parser::default(); | ||
Device { port, parser } | ||
} | ||
|
||
pub fn write_all(&mut self, data: &[u8]) -> std::io::Result<()> { | ||
self.port.write_all(data) | ||
} | ||
|
||
pub fn update<T: FnMut(PacketRef)>(&mut self, mut cb: T) -> std::io::Result<()> { | ||
loop { | ||
const MAX_PAYLOAD_LEN: usize = 1240; | ||
let mut local_buf = [0; MAX_PAYLOAD_LEN]; | ||
let nbytes = self.read_port(&mut local_buf)?; | ||
if nbytes == 0 { | ||
break; | ||
} | ||
|
||
// parser.consume adds the buffer to its internal buffer, and | ||
// returns an iterator-like object we can use to process the packets | ||
let mut it = self.parser.consume(&local_buf[..nbytes]); | ||
loop { | ||
match it.next() { | ||
Some(Ok(packet)) => { | ||
cb(packet); | ||
}, | ||
Some(Err(_)) => { | ||
// Received a malformed packet, ignore it | ||
}, | ||
None => { | ||
// We've eaten all the packets we have | ||
break; | ||
}, | ||
} | ||
} | ||
} | ||
Ok(()) | ||
} | ||
|
||
pub fn wait_for_ack<T: UbxPacketMeta>(&mut self) -> std::io::Result<()> { | ||
let mut found_packet = false; | ||
while !found_packet { | ||
self.update(|packet| { | ||
if let PacketRef::AckAck(ack) = packet { | ||
if ack.class() == T::CLASS && ack.msg_id() == T::ID { | ||
found_packet = true; | ||
} | ||
} | ||
})?; | ||
} | ||
Ok(()) | ||
} | ||
|
||
/// Reads the serial port, converting timeouts into "no data received" | ||
fn read_port(&mut self, output: &mut [u8]) -> std::io::Result<usize> { | ||
match self.port.read(output) { | ||
Ok(b) => Ok(b), | ||
Err(e) => { | ||
if e.kind() == std::io::ErrorKind::TimedOut { | ||
Ok(0) | ||
} else { | ||
Err(e) | ||
} | ||
}, | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,7 @@ | ||
[package] | ||
authors = ["Lane Kolbly <[email protected]>", "Andrei Gherghescu <[email protected]>"] | ||
edition = "2021" | ||
name = "example-basic-cli" | ||
name = "basic-cli" | ||
publish = false | ||
version = "0.1.0" | ||
|
||
|
Oops, something went wrong.