#if (HAS_GPS) #include "globals.h" // Local logging tag static const char TAG[] = __FILE__; TinyGPSPlus gps; gpsStatus_t gps_status = {0}; TaskHandle_t GpsTask; #ifdef GPS_SERIAL HardwareSerial GPS_Serial(1); // use UART #1 static uint16_t nmea_txDelay_ms = (tx_Ticks(NMEA_FRAME_SIZE, GPS_SERIAL) / portTICK_PERIOD_MS); #else static uint16_t nmea_txDelay_ms = 0; #endif // initialize and configure GPS int gps_init(void) { int ret = 1; if (!gps_config()) { ESP_LOGE(TAG, "GPS chip initializiation error"); return 0; } #if defined GPS_SERIAL GPS_Serial.begin(GPS_SERIAL); 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); ret = 0; } else { ESP_LOGI(TAG, "Quectel L76 GPS chip found"); } #endif return ret; } // gps_init() // detect gps chipset type and configure it with device specific settings int gps_config() { int rslt = 1; // success #if defined GPS_SERIAL /* to come */ #elif defined GPS_I2C /* to come */ #endif return rslt; } // store current GPS location data in struct void gps_storelocation(gpsStatus_t *gps_store) { if (gps.location.isUpdated() && gps.location.isValid() && (gps.time.age() < 1500)) { gps_store->latitude = (int32_t)(gps.location.lat() * 1e6); gps_store->longitude = (int32_t)(gps.location.lng() * 1e6); gps_store->satellites = (uint8_t)gps.satellites.value(); gps_store->hdop = (uint16_t)gps.hdop.value(); gps_store->altitude = (int16_t)gps.altitude.meters(); } } // store current GPS timedate in struct void IRAM_ATTR gps_storetime(gpsStatus_t *gps_store) { if (gps.time.isUpdated() && gps.date.isValid() && (gps.time.age() < 1000)) { gps_store->time_age = gps.time.age() + nmea_txDelay_ms; gps_store->timedate.Second = gps.time.second(); gps_store->timedate.Minute = gps.time.minute(); gps_store->timedate.Hour = gps.time.hour(); gps_store->timedate.Day = gps.date.day(); gps_store->timedate.Month = gps.date.month(); gps_store->timedate.Year = CalendarYrToTm(gps.date.year()); // year offset from 1970 in microTime.h } else gps_store->timedate = {0}; } // function to fetch current time from struct; note: this is costly time_t fetch_gpsTime(gpsStatus_t value, uint16_t *msec) { *msec = 1000 - value.time_age; time_t t = timeIsValid(makeTime(value.timedate)); ESP_LOGD(TAG, "GPS time: %d | time age: %d", t, value.time_age); return t; } // fetch_gpsTime() // GPS serial feed FreeRTos Task void gps_loop(void *pvParameters) { configASSERT(((uint32_t)pvParameters) == 1); // FreeRTOS check while (1) { if (cfg.payloadmask && GPS_DATA) { #if defined 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()); delay(2); // 2ms delay according L76 datasheet } #endif } // if // show NMEA data in verbose mode, useful for debugging GPS ESP_LOGV(TAG, "GPS NMEA data: passed %d / failed: %d / with fix: %d", gps.passedChecksum(), gps.failedChecksum(), gps.sentencesWithFix()); delay(2); // yield to CPU } // end of infinite loop } // gps_loop() #endif // HAS_GPS