This is a port of the adafruit python module (https://github.com/adafruit/AdafruitCircuitPythonGPS) that reads and parses the NMEA data sentences from their Ultimate GPS and Adafruit Mini GPS PA1010D. Most of this readme is also pulled directly from the adafruit library.
The GPS module provides a serial byte signal providing longitude, latitude and metadata.
This module has only been tested using an Adafruit Mini GPS PA1010D on a raspberry pi zero.
Serialport: https://docs.rs/serialport/3.2.0/serialport/. Serialport is used for reading the byte information provided by the GPS as well as sending it commands.
``` let mut gps = Gps { port: openport("/dev/serial0") }; let mut gpsvalues = GpsArgValues::default();
// Turn on the basic GGA and RMC info (what you typically want)
gps.send_command("PMTK314,0,1,0,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0");
// Turn on just minimum info (RMC only, location):
// gps.send_command("PMTK314,0,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0");
// Turn off everything:
// gps.send_command("PMTK314,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0");
//Then on everything (not all of it is parsed!)
// gps.send_command("PMTK314,1,1,1,1,1,1,0,0,0,0,0,0,0,0,0,0,0,0,0");
// Set update rate to once a second (1hz) which is what you typically want.
gps.send_command("PMTK220,1000");
// Or decrease to once every two seconds by doubling the millisecond value.
// Be sure to also increase your UART timeout above!
// gps.send_command("PMTK220,2000");
// You can also speed up the rate, but don't go too fast or else you can lose
// data during parsing. This would be twice a second (2hz, 500ms delay):
// gps.send_command("PMTK220,500");
let mut last_print = SystemTime::now();
loop {
gps_values = gps.update(gps_values);
if last_print.elapsed().unwrap().as_secs() >= 1 {
last_print = SystemTime::now();
if (gps_values.fix_quality < Some(1)) | (gps_values.fix_quality == None) {
println!("Waiting for fix...");
continue;
} else {
println!("=========================================");
println!("{:?}", gps_values.timestamp);
println!("Latitude ----{:?} degrees", gps_values.latitude);
println!("Longitude ---{:?} degrees", gps_values.longitude);
println!("Fix quality -{:?}", gps_values.fix_quality);
println!("Satellites --{:?}", gps_values.satellites);
println!("Altitude (m) {:?}", gps_values.altitude_m);
println!("Speed (knots){:?}", gps_values.speed_knots);
println!("Track angle {:?}", gps_values.track_angle_deg);
println!("HODP --------{:?}", gps_values.horizontal_dilution);
println!("Geod height -{:?}", gps_values.height_geoid);
}
}
}
```
Note: Sending multiple PMTK314 packets with gps.sendcommand() will not work unless there is a substantial amount of time in-between each time gps.sendcommand() is called. A time.sleep() of 1 second or more should fix this.
The GPS uses NMEA 0183 protocol.
The data is therefore formatted by the GPS in two ways: GGA and RMC.
11
1 2 3 4 5 6 7 8 9 10 | 12 13 14 15
| | | | | | | | | | | | | | |
$--GGA,hhmmss.ss,llll.ll,a,yyyyy.yy,a,x,xx,x.x,x.x,M,x.x,M,x.x,xxxx*hh
GPS Quality Indicator,
12
1 2 3 4 5 6 7 8 9 10 11|
| | | | | | | | | | | |
$--RMC,hhmmss.ss,A,llll.ll,a,yyyyy.yy,a,x.x,x.x,xxxx,x.x,a*hh
Info about NMEA taken from here
This crate library has mostly been made as a personal challenge and to fill a narrow gap so all contributions are welcome. That said, this code is likely to need improvement, all of which is welcome.
RMC and GGA are both documented but the source code includes GPGLL and GNGGL, neither of which have been tested but have been included in the lib.