ESP32-PaxCounter/src/gpsread.cpp

189 lines
5.4 KiB
C++
Raw Normal View History

2019-04-14 14:35:07 +02:00
#if (HAS_GPS)
2018-06-08 22:41:37 +02:00
#include "globals.h"
2020-03-29 18:08:52 +02:00
#include "gpsread.h"
2018-06-08 22:41:37 +02:00
// Local logging tag
2019-02-27 00:52:27 +01:00
static const char TAG[] = __FILE__;
2018-06-09 17:59:59 +02:00
2020-09-29 17:27:52 +02:00
// we use NMEA ZDA sentence field 1 for time synchronization
// ZDA gives time for preceding pps pulse
// downsight is that it does not have a constant offset
// thus precision is only +/- 1 second
TinyGPSPlus gps;
TinyGPSCustom gpstime(gps, "GPZDA", 1); // field 1 = UTC time (hhmmss.ss)
TinyGPSCustom gpsday(gps, "GPZDA", 2); // field 2 = day (01..31)
TinyGPSCustom gpsmonth(gps, "GPZDA", 3); // field 3 = month (01..12)
TinyGPSCustom gpsyear(gps, "GPZDA", 4); // field 4 = year (4-digit)
static const String ZDA_Request = "$EIGPQ,ZDA*39\r\n";
2018-10-03 16:24:45 +02:00
TaskHandle_t GpsTask;
2018-07-19 21:53:56 +02:00
2018-11-25 16:05:30 +01:00
#ifdef GPS_SERIAL
2019-01-26 12:32:58 +01:00
HardwareSerial GPS_Serial(1); // use UART #1
static uint16_t nmea_txDelay_ms =
2019-08-03 14:01:25 +02:00
(tx_Ticks(NMEA_FRAME_SIZE, GPS_SERIAL) / portTICK_PERIOD_MS);
2019-04-14 14:35:07 +02:00
#else
static uint16_t nmea_txDelay_ms = 0;
2018-11-25 16:05:30 +01:00
#endif
2018-06-09 13:18:59 +02:00
2018-11-25 16:05:30 +01:00
// initialize and configure GPS
int gps_init(void) {
2018-11-25 16:05:30 +01:00
int ret = 1;
2019-02-09 13:02:38 +01:00
if (!gps_config()) {
ESP_LOGE(TAG, "GPS chip initializiation error");
return 0;
}
#ifdef GPS_SERIAL
GPS_Serial.begin(GPS_SERIAL);
2018-11-25 16:05:30 +01:00
ESP_LOGI(TAG, "Using serial GPS");
#elif defined GPS_I2C
Wire.begin(GPS_I2C, 400000); // I2C connect to GPS device with 400 KHz
Wire.beginTransmission(GPS_ADDR);
Wire.write(0x00); // dummy write
ret = Wire.endTransmission(); // check if chip is seen on i2c bus
if (ret) {
ESP_LOGE(TAG,
"Quectel L76 GPS chip not found on i2c bus, bus error %d. "
"Stopping GPS-Task.",
ret);
2018-11-25 16:05:30 +01:00
ret = 0;
} else {
2018-11-25 16:05:30 +01:00
ESP_LOGI(TAG, "Quectel L76 GPS chip found");
}
#endif
2018-11-25 16:05:30 +01:00
return ret;
} // gps_init()
2019-02-09 13:02:38 +01:00
// detect gps chipset type and configure it with device specific settings
int gps_config() {
int rslt = 1; // success
#if defined GPS_SERIAL
/* insert user configuration here, if needed */
2019-02-09 13:02:38 +01:00
#elif defined GPS_I2C
/* insert user configuration here, if needed */
2019-02-09 13:02:38 +01:00
#endif
return rslt;
}
2019-04-15 12:57:55 +02:00
// store current GPS location data in struct
2019-07-28 23:51:24 +02:00
void gps_storelocation(gpsStatus_t *gps_store) {
if (gps.location.isUpdated() && gps.location.isValid() &&
(gps.location.age() < 1500)) {
gps_store->latitude = (int32_t)(gps.location.lat() * 1e6);
gps_store->longitude = (int32_t)(gps.location.lng() * 1e6);
2019-07-28 23:51:24 +02:00
gps_store->satellites = (uint8_t)gps.satellites.value();
gps_store->hdop = (uint16_t)gps.hdop.value();
gps_store->altitude = (int16_t)gps.altitude.meters();
}
2018-11-25 16:05:30 +01:00
}
2020-01-03 09:59:10 +01:00
bool gps_hasfix() {
// adapted from source:
// https://github.com/hottimuc/Lora-TTNMapper-T-Beam/blob/master/fromV08/gps.cpp
return (gps.location.isValid() && gps.location.age() < 4000 &&
gps.hdop.isValid() && gps.hdop.value() <= 600 &&
gps.hdop.age() < 4000 && gps.altitude.isValid() &&
gps.altitude.age() < 4000);
}
2020-10-04 23:17:39 +02:00
// function to poll current time from GPS data; note: this is costly
2020-03-11 23:47:33 +01:00
time_t get_gpstime(uint16_t *msec) {
2020-09-29 17:27:52 +02:00
// poll NMEA ZDA sentence
#ifdef GPS_SERIAL
GPS_Serial.print(ZDA_Request);
2019-08-18 17:44:42 +02:00
// wait for gps NMEA answer
// vTaskDelay(tx_Ticks(NMEA_FRAME_SIZE, GPS_SERIAL));
#elif defined GPS_I2C
Wire.print(ZDA_Request);
#endif
// did we get a current date & time?
2020-10-05 13:40:22 +02:00
if (gpstime.isValid() && gpsday.isValid()) {
2019-04-15 12:57:55 +02:00
2020-10-05 13:40:22 +02:00
time_t t = 0;
tmElements_t tm;
2019-08-18 17:44:42 +02:00
uint32_t delay_ms =
gpstime.age() + nmea_txDelay_ms + NMEA_COMPENSATION_FACTOR;
2020-10-05 13:40:22 +02:00
uint32_t zdatime = atof(gpstime.value());
// convert time to maketime format and make time
tm.Second = zdatime % 100; // second
tm.Minute = (zdatime / 100) % 100; // minute
tm.Hour = zdatime / 10000; // hour
tm.Day = atoi(gpsday.value()); // day
tm.Month = atoi(gpsmonth.value()); // month
tm.Year = CalendarYrToTm(atoi(gpsyear.value())); // year offset from 1970
t = makeTime(tm);
2019-04-15 12:57:55 +02:00
2020-10-05 13:40:22 +02:00
ESP_LOGD(TAG, "GPS time/date = %2d:%2d:%2d / %2d.%2d.%2d", tm.Hour,
tm.Minute, tm.Second, tm.Day, tm.Month, tm.Year + 1970);
// add protocol delay with millisecond precision
t += delay_ms / 1000 - 1; // whole seconds
*msec = delay_ms % 1000; // fractional seconds
return t;
}
2019-04-15 12:57:55 +02:00
2020-10-05 13:40:22 +02:00
ESP_LOGD(TAG, "no valid GPS time");
return 0;
2019-04-14 14:35:07 +02:00
2020-03-11 23:47:33 +01:00
} // get_gpstime()
2019-01-28 23:59:52 +01:00
2020-03-11 23:47:33 +01:00
time_t get_gpstime(void) {
uint16_t msec;
2020-03-11 23:47:33 +01:00
return get_gpstime(&msec);
}
2018-11-25 16:05:30 +01:00
// GPS serial feed FreeRTos Task
void gps_loop(void *pvParameters) {
configASSERT(((uint32_t)pvParameters) == 1); // FreeRTOS check
while (1) {
2019-09-01 00:11:20 +02:00
if (cfg.payloadmask & GPS_DATA) {
#ifdef GPS_SERIAL
// feed GPS decoder with serial NMEA data from GPS device
while (GPS_Serial.available()) {
gps.encode(GPS_Serial.read());
}
#elif defined GPS_I2C
Wire.requestFrom(GPS_ADDR, 32); // caution: this is a blocking call
while (Wire.available()) {
gps.encode(Wire.read());
2018-12-19 12:32:25 +01:00
delay(2); // 2ms delay according L76 datasheet
}
#endif
2020-10-05 13:40:22 +02:00
// (only) while device time is not set or unsynched, and we have a valid
// GPS time, we trigger a device time update to poll time from GPS
if (timeSource == _unsynced && gpstime.isUpdated())
calibrateTime();
2018-11-19 00:41:15 +01:00
} // if
// show NMEA data in verbose mode, useful only for debugging GPS, very noisy
2020-01-03 09:59:10 +01:00
// ESP_LOGV(TAG, "GPS NMEA data: passed %u / failed: %u / with fix: %u",
2019-09-23 15:39:58 +02:00
// gps.passedChecksum(), gps.failedChecksum(),
// gps.sentencesWithFix());
2019-07-28 23:51:24 +02:00
2018-12-19 12:32:25 +01:00
delay(2); // yield to CPU
} // end of infinite loop
2018-06-09 19:20:34 +02:00
} // gps_loop()
2018-06-08 22:41:37 +02:00
#endif // HAS_GPS