Fastest GPS library?

I am using a 15210 GPS on a UNO R4 Minima via I2C.

What is the fastest library that will give me Date, Time, Longitude and Latitude, and speed? I don’t care if its all filtered nicely from the NMEA sentences or not. I’ll take the raw NMEA sentences (and post process them later) if that is faster.

Seems the one I am using takes 2-3 seconds to report those parameters to the serial monitor - way too long. It has a good fix so I would have thought the parameters would be stored and it would simply report them quickly.

Thanks

Johnny

Hi Johnny,

I think you might be misunderstanding how the u-blox modules work. The NMEA and UBX messages are generated each navigation interval - the default is once per second. UART (serial) messages are output each second. I2C messages are added to the module’s internal buffer once per second. The library will read and return those messages as they are generated. The position etc. aren’t stored as such, and you can’t read them ‘immediately’. Depending on when you try to read the position, it could be a full second before it is returned. If you are using our u-blox GNSS library, have a look at this example: https://github.com/sparkfun/SparkFun_u- … AGPGGA.ino

I hope this helps,

Paul

Also, please be careful with your voltages. The R4 Minima is a 5V board. All the IO pins are 5V. Our u-blox GNSS boards are all 3.3V; Qwiic I2C is 3.3V.

The R4 WiFi has a level-shifted Qwiic connector on it. For the Minima, you need a separate level-shifter.

If the messages are enabled for periodic output you certainly should be able to recover the solution within the second the measurements were made.

Thanks Paul and Clive.

Would it be possible to set the Navigation Interval to say 2 Hz and shorten the potential delay to half a second?

[I do have it wired to the 3.3 pin on the UNO.]

Thanks

Johnny

Hi Johnny,

myGNSS.setNavigationFrequency(2);

will produce two solutions per second.

Faster navigation frequencies can overload the UART serial port - even though you’re not using it. Please see this post for more information: viewtopic.php?p=246423#p246423

Again, please be careful will your voltages. You are powering the u-blox board from 3.3V, that’s good. But the R4 Minima IO pins are still 5V. The SDA and SCL pins are expecting 5V and - according to the schematic - have no pull-ups. The module is outputting (pulling to) 3.3V. This could introduce errors on the bus. I recommend using a level-shifter. Or switch to the R4 WiFi - which has a level-shifted Qwiic connector.

Best wishes,

Paul

The latency from measurement to output shouldn’t be more than 100-200 ms in any case

Simple/Fast NMEA Sentence decode

https://github.com/cturvey/RandomNinjaC … C_NMEA.ino

The SAM-M8Q should be able to hit 10 Hz, but you’ll want to cull the message output, especially the GSV sentences, and stop all output to the slow UART (9600 baud) which isn’t going to be able to sustain the volume of data.

Paul

Thanks for the heads up on the data and clock voltages - I had no idea the data and clock voltages were incompatible with my UNOs.

Pardon my ignorance but are SPI communication voltages standardized? I have several modules on prototyping shields using SPI and am concerned I cannot move this shield from Redboard to UNO etc.

I will take your advice and use a UNO R4 Wifi in order to get the QWIIC connector [with associated level shifters built in].

Thanks

Johnny

Hi Johnny,

We have a RedBoard with switchable IO voltage: https://www.sparkfun.com/products/18158 . But it only has 2K of RAM and 32K of Flash which can cause other issues. Our u-blox library just about fills it…

On the I2C bus, the high voltage / logic level is provided by the pull-up resistors. The SCL and SDA lines are pulled low by the microcontroller and peripheral. SPI is worse as the clock and data out/in lines are actually driven to the IO voltage. If you connect a 3.3V peripheral to a microcontroller which has 5V IO, you are overdriving the peripheral clock-in and data-in pins.

The best solution is to use a level-shifter. This one is ideal for SPI: https://www.sparkfun.com/products/12009 . For I2C, this one is ideal: https://www.sparkfun.com/products/15439 .

I hope this helps,

Best wishes,

Paul

It’s not a “standard” in the sense you want to make it.

Something like RS232 has a standard/expectation for signaling voltages, but needs a transceiver for both 5V and 3.3V based systems.

Normally for MCU to MCU systems running at the same voltages you can just wire up with a common ground. If you need to go between systems with different voltages schemes you’ll need level shifters.

Most of the industry diverged away from 5V in the 1990’s, and now 3.3V, 3.0V, 1.8V and 1.2V are quite common. Inside the core of your part is most likely at 1.25V with just the pin IO cells being 5V.

Ok I can get the GPS example 1 NMEA sentences to work on the Redboard Turbo with the GPS module connected by QWIIC.

I cannot get that example to work on the UNO R4 Wifi [also connected by QWIIC]. When I compile it I get screens of red errors. I suspect that particular GPS library doesn’t support the UNO R4 Wifi.

*Can someone post or direct me to a simple GPS example that works on the UNO R4 wifi?

Thankyou

Johnny

Hi Johnny,

The library works just fine on the UNO R4 WiFi…

Please check you have the correct board package installed: Arduino UNO R4 Boards (1.0.5). Select Arduino Renesas UNO R4 Boards \ Arduino UNO R4 WiFi.

The Qwiic Wire (I2C) port on the R4 Wifi is Wire1, not Wire. In your code, please call:

Wire1.begin();

myGNSS.begin(Wire1);

There are much simpler GPS / GNSS libraries out there. TinyGPSPlus is excellent. But they (generally) parse NMEA on Serial (UART), not I2C, and have no direct support for the u-blox modules and the UBX protocol / configuration interface.

Best wishes,

Paul

Thanks Paul

I did have that board loaded correctly but did not understand the difference in Wire and Wire1 - it is working now.

I have determined checking if the RTC is connected takes 2-3 seconds and this is what is causing the delay. The RTC is “hot swapable” and I was hoping to utilize one code to either update the RTC from the GPS [when the RTC is connected], or write GPS data to the SD card [when the RTC is not found].

Is there a better way to check if the RTC is connected than if (rtc.begin()) in the loop?

Again thank you for your help with this.

PS I also discovered the GPS example doesn’t work with Serial.begin(9600). Must be 115200.

Johnny

Hi Johnny,

Great - glad the GNSS example is working for you.

For the DS3231 RTC:

Check that you are passing the Wire1 port into the code correctly. In some libraries you have to pass it through the constructor - where you instantiate the class. In others, you pass it through .begin.

Check you are calling the .now method - if available. This reads all the time/date registers in one go. Much faster than reading them individually.

All the best,

Paul

Paul

I don’t understand the part about “Check that you are passing the Wire1 port…”. I’ve posted my code [less the huge comments] below, hopefully that is ok to do.

I am not reading the RTC, only checking if its there and writing the GPS time to it.

Johnny

16 #include <Wire.h> //Needed for I2C to GNSS

17

18 #include <SparkFun_u-blox_GNSS_Arduino_Library.h> //http://librarymanager/All#SparkFun_u-blox_GNSS

19 SFE_UBLOX_GNSS myGNSS;

20

21 long lastTime = 0; //Simple local timer. Limits amount if I2C traffic to u-blox module.

29

30 int count;

31

32 #include <SPI.h> //Needed for the Adafruit SPI SD module

33 #include <SD.h>

34

35 File myFile;

41 #include “RTClib.h”

42

43 RTC_DS3231 rtc;

44

48 int mySecond;

49 int myMinute;

50 int myHour;

51 int myDay;

52 int myMonth;

53 int myYear;

60

61 void setup () {

62

63 Serial.begin(115200); // Needs to be 115200 for the GPS to work properly

64

65 while (!Serial);

66 delay(3000);

67

76

77 Wire1.begin();

78

79 if (myGNSS.begin(Wire1) == false)

80 {

81 Serial.println(F(“u-blox GNSS not detected at default I2C address. Please check wiring. Freezing.”));

82 while (1);

83 }

84

85 Serial.println(“GPS initialized.”);

86

87 myGNSS.setI2COutput(COM_TYPE_UBX | COM_TYPE_NMEA); //Set the I2C port to output both NMEA and UBX messages

88 myGNSS.saveConfigSelective(VAL_CFG_SUBSEC_IOPORT); //Save (only) the communications port settings to flash and BBR

89

102 delay(3000);

103 if (!SD.begin(10)) {

104 Serial.println(“SD card initialization failed!”);

105 while (1);

106 }

107 Serial.println(“SD card initialized.”);

108

109 myFile = SD.open(“GPS_data.txt”,FILE_WRITE);

115 }

119

120 void loop () {

121

127 if (millis() - lastTime > 1000)

128 {

129 lastTime = millis(); //Update the timer

130

131 // Seems GPS seconds need to be adjusted slightly by the time it gets to the RTC adjustment.

132 mySecond = myGNSS.getSecond() - 1;

133 myMinute = myGNSS.getMinute();

134 myMonth = myGNSS.getMonth();

135 myYear = myGNSS.getYear();

136

137 // Timezone adjustment, -8 hours

138 if (myGNSS.getHour() < 8) {

139 myHour = myGNSS.getHour() + 12 -8;

140 myDay = myGNSS.getDay() -1;

141 } else {

142 myHour = myGNSS.getHour() -8;

143 myDay = myGNSS.getDay();

144 }

149

151 if (rtc.begin()) {

152 // RTC connected.

153 // Under this condition the code should just be writing GPS date & time to the RTC.

154 Serial.println(“RTC connected, updating with GPS date & time…”);

155 rtc.adjust(DateTime(myYear, myMonth, myDay, myHour, myMinute, mySecond));

156 Serial.flush();

157 } else {

158 // RTC not connected.

159 // Under this condition the code should just be writing GPS data to the SD card.

160 Serial.println(“RTC not connected, writing GPS data to SD card…”);

161 }

196

197 myFile.print(“GPS: ,”);

198 myFile.print(myYear);

199 myFile.print(‘/’);

200 myFile.print(myMonth);

201 myFile.print(‘/’);

202 myFile.print(myDay);

203 myFile.print(“,”);

204 myFile.print(myHour);

205 myFile.print(‘:’);

206 myFile.print(myMinute);

207 myFile.print(‘:’);

208 myFile.print(myGNSS.getSecond());

209 myFile.print(“,”);

210

211 myFile.print(myGNSS.getLatitude() / 10000000., 7);

212 myFile.print(“,”);

213 myFile.print(myGNSS.getLongitude() / 10000000., 7);

214 myFile.print(“,”);

215 myFile.print(myGNSS.getGroundSpeed());

216 myFile.print(“,”);

217 myFile.print(myGNSS.getHeading());

218 myFile.print(“,”);

219 myFile.print(myGNSS.getFixType());

220 myFile.println();

222

225 myFile.flush();

229

230 }

231 }

Hi Johnny,

I don’t know which DS3231 / RTClib library you are using. There are many. I’m guessing you are using the Adafruit one? https://github.com/adafruit/RTClib

Likewise, I don’t know which DS3231 board you are using. Again maybe the Adafruit one? https://www.adafruit.com/product/5188

The DS3231 will work both at 3.3V and 5V. You are OK to connect it directly to the 5V IO pins on the UNO R4 WiFi. But it will also work at 3.3V too. (The u-blox GNSS modules are 3.3V-only)

From your code, I guess you have the DS3231 connected to the SCL / SDA I2C IO pins / breakout pads. That is the Wire bus. That’s fine. But you could also connect it to the Qwiic bus (Wire1) and run it at 3.3V. If you are using the Adafruit library, you would call rtc.begin(Wire1); instead.

I think I understand why your code is ‘slow’. When you call myGNSS.getSecond() the library checks if the second value has been read before. If it hasn’t, it requests it from the module and waits for it to arrive. That could take up to a full second depending on when you do it and if the module is running at 1Hz navigation rate. The minute, hour, day, month, year all come from the same Position Velocity Time (UBX-NAV-PVT) message. The library knows those are valid, haven’t been read before, and so returns them immediately. But - here’s the important bit - when you call myGNSS.getSecond() a second time, the library knows the seconds have been read before and so it requests a new PVT message from the module and waits for it to arrive. This is why your code takes 2 or 3 seconds to run. The myGNSS.getHour() at line 138 will return immediately because the getSecond at line 132 got fresh data. But you read getHour again in line 139 or 142, delaying the code by another second. The getSecond at line 208 should return immediately because the second getHour read fresh data…

It would be better to use myGNSS.getPVT(). You actually don’t need lastTime as getPVT will wait until it receives new data - or times out. lastTime could slow your code down even more.

void loop() {
  if (myGNSS.getPVT()) {
    mySecond = myGNSS.getSecond();
    myMinute = myGNSS.getMinute();
    myHour = myGNSS.getHour();
    myDay = myGNSS.getDay();
    myMonth = myGNSS.getMonth();
    myYear = myGNSS.getYear();

    // Do your time zone manipulation using myHour etc.. Don't call myGNSS.getHour a second time
  }

You might find myGNSS.getUnixEpoch(); useful for time zone stuff. That returns the number of seconds since the Unix epoch (midnight Jan 1st 1970). It’s then easy to add / subtract your time zone in seconds. The fun comes when you want to convert it back to HMSDMY format. But there are many code examples and time libraries out there to help with that.

Best,

Paul

Paul

I really like the idea of just relying on the 1Hz data update and eliminating the if milli statement. The part about just calling the data once makes sense too.

It now runs and reports on the serial monitor every second. Great!

However when I add in this one line for latitude it degrades to every 2 seconds. Is there some other way to obtain latitude, longitude and fix type? [Groundspeed reports ridiculous values with fix type 3, not sure what is going on with that].

if (myGNSS.getPVT()) {

mySecond = myGNSS.getSecond();

myMinute = myGNSS.getMinute();

myHour = myGNSS.getHour();

myDay = myGNSS.getDay();

myMonth = myGNSS.getMonth();

myYear = myGNSS.getYear();

// myLatitude = myGNSS.getLatitude();

// Do your time zone manipulation using myHour etc… Don’t call myGNSS.getHour a second time

Thank you

Johnny

Hi Johnny,

getLatitude reads the latitude from the same UBX-NAV-PVT message and uses the same data-stale flag to check if it has been read before. Please check you are only calling myGNSS.getLatitude() once. Please check you changed the print at line 211 so it uses myLatitude, instead of calling getLatitude.

The ground speed is mm/sec. Expect large values!

Best wishes,

Paul

Paul

I have commented out all the extra getLatitude()'s however the below code returns 0.00 for latitude:

if (myGNSS.getPVT()) {

mySecond = myGNSS.getSecond();

myMinute = myGNSS.getMinute();

myHour = myGNSS.getHour();

myDay = myGNSS.getDay();

myMonth = myGNSS.getMonth();

myYear = myGNSS.getYear();

myLatitude = myGNSS.getLatitude();

Serial.println(myLatitude);

Shouldn’t the PVT provide a latitude value?

Thanks

Johnny

You only need yo update your RTC one time, not once a second.