2011-02-22 10:16:50 +00:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* @addtogroup OpenPilotModules OpenPilot Modules
|
|
|
|
* @{
|
|
|
|
* @addtogroup GSPModule GPS Module
|
|
|
|
* @brief Process GPS information
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file NMEA.c
|
|
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
|
|
|
|
* @brief GPS module, handles GPS and NMEA stream
|
|
|
|
* @see The GNU Public License (GPL) Version 3
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
/*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
|
|
|
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
2010-08-27 02:15:00 +00:00
|
|
|
#include "openpilot.h"
|
|
|
|
#include "pios.h"
|
|
|
|
#include "NMEA.h"
|
|
|
|
#include "gpsposition.h"
|
2010-08-27 19:59:25 +00:00
|
|
|
#include "gpstime.h"
|
2011-02-22 11:02:17 +00:00
|
|
|
#ifdef ENABLE_GPS_NMEA
|
|
|
|
#include "gpssatellites.h"
|
|
|
|
#endif
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2011-02-22 10:16:50 +00:00
|
|
|
#if defined(ENABLE_GPS_NMEA) || defined(ENABLE_GPS_ONESENTENCE_GTOP)
|
|
|
|
|
2010-08-27 02:15:00 +00:00
|
|
|
// Debugging
|
|
|
|
//#define GPSDEBUG
|
|
|
|
#ifdef GPSDEBUG
|
2010-09-27 07:28:34 +00:00
|
|
|
#define NMEA_DEBUG_PKT ///< define to enable debug of all NMEA messages
|
|
|
|
#define NMEA_DEBUG_GGA ///< define to enable debug of GGA messages
|
|
|
|
#define NMEA_DEBUG_VTG ///< define to enable debug of VTG messages
|
|
|
|
#define NMEA_DEBUG_RMC ///< define to enable debug of RMC messages
|
|
|
|
#define NMEA_DEBUG_GSA ///< define to enable debug of GSA messages
|
|
|
|
#define NMEA_DEBUG_GSV ///< define to enable debug of GSV messages
|
|
|
|
#define NMEA_DEBUG_ZDA ///< define to enable debug of ZDA messages
|
2010-10-27 10:39:38 +00:00
|
|
|
#define NMEA_DEBUG_PGTOP ///< define to enable debug of PGTOP messages
|
2010-08-27 02:15:00 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* NMEA sentence parsers */
|
|
|
|
|
|
|
|
struct nmea_parser {
|
2010-09-27 07:28:34 +00:00
|
|
|
const char *prefix;
|
|
|
|
bool(*handler) (GPSPositionData * GpsData, char *sentence);
|
2010-08-27 02:15:00 +00:00
|
|
|
};
|
|
|
|
|
2011-02-22 10:16:50 +00:00
|
|
|
#ifdef ENABLE_GPS_NMEA
|
|
|
|
static bool nmeaProcessGPGGA(GPSPositionData * GpsData, char *sentence);
|
|
|
|
static bool nmeaProcessGPRMC(GPSPositionData * GpsData, char *sentence);
|
|
|
|
static bool nmeaProcessGPVTG(GPSPositionData * GpsData, char *sentence);
|
|
|
|
static bool nmeaProcessGPGSA(GPSPositionData * GpsData, char *sentence);
|
|
|
|
static bool nmeaProcessGPZDA(GPSPositionData * GpsData, char *sentence);
|
|
|
|
static bool nmeaProcessGPGSV(GPSPositionData * GpsData, char *sentence);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static bool nmeaProcessPGTOP(GPSPositionData * GpsData, char *sentence);
|
|
|
|
|
|
|
|
static struct nmea_parser nmea_parsers[] = {
|
|
|
|
|
|
|
|
#ifdef ENABLE_GPS_NMEA
|
|
|
|
{
|
|
|
|
.prefix = "GPGGA",
|
|
|
|
.handler = nmeaProcessGPGGA,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.prefix = "GPVTG",
|
|
|
|
.handler = nmeaProcessGPVTG,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.prefix = "GPGSA",
|
|
|
|
.handler = nmeaProcessGPGSA,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.prefix = "GPRMC",
|
|
|
|
.handler = nmeaProcessGPRMC,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.prefix = "GPZDA",
|
|
|
|
.handler = nmeaProcessGPZDA,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.prefix = "GPGSV",
|
|
|
|
.handler = nmeaProcessGPGSV,
|
|
|
|
},
|
|
|
|
#endif
|
2010-10-27 10:39:38 +00:00
|
|
|
{
|
|
|
|
.prefix = "PGTOP",
|
|
|
|
.handler = nmeaProcessPGTOP,
|
|
|
|
},
|
2010-08-27 02:15:00 +00:00
|
|
|
};
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
static struct nmea_parser *NMEA_find_parser_by_prefix(char *prefix)
|
2010-08-27 02:15:00 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
if (!prefix) {
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < NELEMENTS(nmea_parsers); i++) {
|
|
|
|
struct nmea_parser *parser = &nmea_parsers[i];
|
|
|
|
|
|
|
|
/* Use strcmp to check for exact equality over the entire prefix */
|
|
|
|
if (!strcmp(prefix, parser->prefix)) {
|
|
|
|
/* Found an appropriate parser */
|
|
|
|
return (parser);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No matching parser for this prefix */
|
|
|
|
return (NULL);
|
2010-08-27 02:15:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Computes NMEA sentence checksum
|
|
|
|
* \param[in] Buffer for parsed nmea sentence
|
|
|
|
* \return false checksum not valid
|
|
|
|
* \return true checksum valid
|
|
|
|
*/
|
2010-09-27 07:28:34 +00:00
|
|
|
bool NMEA_checksum(char *nmea_sentence)
|
2010-08-27 02:15:00 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
uint8_t checksum_computed = 0;
|
|
|
|
uint8_t checksum_received;
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
while (*nmea_sentence != '\0' && *nmea_sentence != '*') {
|
|
|
|
checksum_computed ^= *nmea_sentence;
|
|
|
|
nmea_sentence++;
|
|
|
|
}
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
/* Make sure we're now pointing at the checksum */
|
|
|
|
if (*nmea_sentence == '\0') {
|
|
|
|
/* Buffer ran out before we found a checksum marker */
|
|
|
|
return false;
|
|
|
|
}
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
/* Load the checksum from the buffer */
|
|
|
|
checksum_received = strtol(nmea_sentence + 1, NULL, 16);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
//PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART,"$%d=%d\r\n",checksum_received,checksum_computed);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
return (checksum_computed == checksum_received);
|
2010-08-27 02:15:00 +00:00
|
|
|
}
|
|
|
|
|
2011-02-22 10:16:50 +00:00
|
|
|
/*
|
|
|
|
* This function only exists to deal with a linking
|
|
|
|
* failure in the stdlib function strtof(). This
|
|
|
|
* implementation does not rely on the _sbrk() syscall
|
|
|
|
* like strtof() does.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Parse a number encoded in a string of the format:
|
|
|
|
* [-]NN.nnnnn
|
|
|
|
* into a signed whole part and an unsigned fractional part.
|
|
|
|
* The fract_units field indicates the units of the fractional part as
|
|
|
|
* 1 whole = 10^fract_units fract
|
|
|
|
*/
|
|
|
|
static bool NMEA_parse_real(int32_t * whole, uint32_t * fract, uint8_t * fract_units, char *field)
|
|
|
|
{
|
|
|
|
char *s = field;
|
|
|
|
char *field_w;
|
|
|
|
char *field_f;
|
|
|
|
|
|
|
|
PIOS_DEBUG_Assert(whole);
|
|
|
|
PIOS_DEBUG_Assert(fract);
|
|
|
|
PIOS_DEBUG_Assert(fract_units);
|
|
|
|
|
|
|
|
field_w = strsep(&s, ".");
|
|
|
|
field_f = s;
|
|
|
|
|
|
|
|
*whole = strtol(field_w, NULL, 10);
|
|
|
|
|
|
|
|
if (field_w) {
|
|
|
|
/* decimal was found so we may have a fractional part */
|
|
|
|
*fract = strtoul(field_f, NULL, 10);
|
|
|
|
*fract_units = strlen(field_f);
|
|
|
|
} else {
|
|
|
|
/* no decimal was found, fractional part is zero */
|
|
|
|
*fract = 0;
|
|
|
|
*fract_units = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static float NMEA_real_to_float(char *nmea_real)
|
|
|
|
{
|
|
|
|
int32_t whole;
|
|
|
|
uint32_t fract;
|
|
|
|
uint8_t fract_units;
|
|
|
|
|
|
|
|
/* Sanity checks */
|
|
|
|
PIOS_DEBUG_Assert(nmea_real);
|
|
|
|
|
|
|
|
if (!NMEA_parse_real(&whole, &fract, &fract_units, nmea_real)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert to float */
|
|
|
|
return (((float)whole) + fract * pow(10, -fract_units));
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef ENABLE_GPS_NMEA
|
|
|
|
/*
|
|
|
|
* Parse a field in the format:
|
|
|
|
* DD[D]MM.mmmm[mm]
|
|
|
|
* into a fixed-point representation in units of (degrees * 1e-7)
|
|
|
|
*/
|
|
|
|
static bool NMEA_latlon_to_fixed_point(int32_t * latlon, char *nmea_latlon)
|
|
|
|
{
|
|
|
|
int32_t num_DDDMM;
|
|
|
|
uint32_t num_m;
|
|
|
|
uint8_t units;
|
|
|
|
|
|
|
|
/* Sanity checks */
|
|
|
|
PIOS_DEBUG_Assert(nmea_latlon);
|
|
|
|
PIOS_DEBUG_Assert(latlon);
|
|
|
|
|
|
|
|
if (!NMEA_parse_real(&num_DDDMM, &num_m, &units, nmea_latlon)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* scale up the mmmm[mm] field apropriately depending on # of digits */
|
|
|
|
switch (units) {
|
|
|
|
case 0:
|
|
|
|
/* no digits, value is zero so no scaling */
|
|
|
|
break;
|
|
|
|
case 1: /* m */
|
|
|
|
num_m *= 1e6; /* m000000 */
|
|
|
|
break;
|
|
|
|
case 2: /* mm */
|
|
|
|
num_m *= 1e5; /* mm00000 */
|
|
|
|
break;
|
|
|
|
case 3: /* mmm */
|
|
|
|
num_m *= 1e4; /* mmm0000 */
|
|
|
|
break;
|
|
|
|
case 4: /* mmmm */
|
|
|
|
num_m *= 1e3; /* mmmm000 */
|
|
|
|
break;
|
|
|
|
case 5: /* mmmmm */
|
|
|
|
num_m *= 1e2; /* mmmmm00 */
|
|
|
|
break;
|
|
|
|
case 6: /* mmmmmm */
|
|
|
|
num_m *= 1e1; /* mmmmmm0 */
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* unhandled format */
|
|
|
|
num_m = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
*latlon = (num_DDDMM / 100) * 1e7; /* scale the whole degrees */
|
|
|
|
*latlon += (num_DDDMM % 100) * 1e7 / 60; /* add in the scaled decimal whole minutes */
|
|
|
|
*latlon += num_m / 60; /* add in the scaled decimal fractional minutes */
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif // ENABLE_GPS_NMEA
|
|
|
|
|
2010-08-27 02:15:00 +00:00
|
|
|
/**
|
|
|
|
* Parses a complete NMEA sentence and updates the GPSPosition UAVObject
|
|
|
|
* \param[in] An NMEA sentence with a valid checksum
|
|
|
|
* \return true if the sentence was successfully parsed
|
|
|
|
* \return false if any errors were encountered with the parsing
|
|
|
|
*/
|
2010-09-27 07:28:34 +00:00
|
|
|
bool NMEA_update_position(char *nmea_sentence)
|
2010-08-27 02:15:00 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
char *sentence = nmea_sentence;
|
|
|
|
struct nmea_parser *parser;
|
|
|
|
char *prefix;
|
|
|
|
|
|
|
|
/* Find out what kind of NMEA packet we're dealing with */
|
|
|
|
prefix = strsep(&sentence, ",");
|
|
|
|
|
|
|
|
/* Check if we have a parser for this packet type */
|
|
|
|
parser = NMEA_find_parser_by_prefix(prefix);
|
|
|
|
|
|
|
|
if (!parser) {
|
|
|
|
/* Valid but unhandled packet type */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Found a matching parser for this packet type */
|
|
|
|
|
|
|
|
/* Reject empty (but valid) packets without parsing */
|
|
|
|
if (sentence[0] == ',' && sentence[1] == ',' && sentence[2] == ',') {
|
|
|
|
/* Nothing to parse, */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Parse the sentence and update the GpsData object */
|
|
|
|
|
|
|
|
GPSPositionData GpsData;
|
|
|
|
GPSPositionGet(&GpsData);
|
|
|
|
if (!parser->handler(&GpsData, sentence)) {
|
|
|
|
/* Parse failed for valid checksum. Do not update the UAVObject. */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
GPSPositionSet(&GpsData);
|
|
|
|
|
|
|
|
/* Tell the caller what kind of packet we just parsed */
|
|
|
|
return true;
|
2010-08-27 02:15:00 +00:00
|
|
|
}
|
|
|
|
|
2011-02-22 10:16:50 +00:00
|
|
|
#ifdef ENABLE_GPS_NMEA
|
|
|
|
|
2010-08-27 02:15:00 +00:00
|
|
|
/**
|
|
|
|
* Parse an NMEA GPGGA sentence and update the given UAVObject
|
|
|
|
* \param[in] A pointer to a GPSPosition UAVObject to be updated.
|
|
|
|
* \param[in] An NMEA sentence with a valid checksum
|
|
|
|
*/
|
2010-09-27 07:28:34 +00:00
|
|
|
static bool nmeaProcessGPGGA(GPSPositionData * GpsData, char *sentence)
|
2010-08-27 02:15:00 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
char *next = sentence;
|
|
|
|
char *tokens;
|
|
|
|
char *delimiter = ",*";
|
2010-08-27 02:15:00 +00:00
|
|
|
|
|
|
|
#ifdef NMEA_DEBUG_GGA
|
2010-09-27 07:28:34 +00:00
|
|
|
PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART, "$%s\r\n", sentence);
|
2010-08-27 02:15:00 +00:00
|
|
|
#endif
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// get UTC time [hhmmss.sss]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
//strcpy(GpsInfo.TimeOfFix,tokens);
|
|
|
|
|
|
|
|
// next field: latitude
|
|
|
|
// get latitude [DDMM.mmmmm]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (!NMEA_latlon_to_fixed_point(&GpsData->Latitude, tokens)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// next field: N/S indicator
|
2010-10-27 10:39:38 +00:00
|
|
|
// correct latitude for N/S
|
2010-09-27 07:28:34 +00:00
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (tokens[0] == 'S')
|
|
|
|
GpsData->Latitude = -GpsData->Latitude;
|
|
|
|
|
|
|
|
// next field: longitude
|
|
|
|
// get longitude [dddmm.mmmmm]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (!NMEA_latlon_to_fixed_point(&GpsData->Longitude, tokens)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// next field: E/W indicator
|
|
|
|
// correct longitude for E/W
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (tokens[0] == 'W')
|
|
|
|
GpsData->Longitude = -GpsData->Longitude;
|
|
|
|
|
|
|
|
// next field: position fix status
|
|
|
|
// position fix status
|
|
|
|
// 0 = Invalid, 1 = Valid SPS, 2 = Valid DGPS, 3 = Valid PPS
|
|
|
|
// check for good position fix
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
//if((tokens[0] != '0') || (tokens[0] != 0))
|
|
|
|
// GpsData.Updates++;
|
|
|
|
|
|
|
|
// next field: satellites used
|
|
|
|
// get number of satellites used in GPS solution
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Satellites = atoi(tokens);
|
|
|
|
|
|
|
|
// next field: HDOP (horizontal dilution of precision)
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: altitude
|
|
|
|
// get altitude (in meters mm.m)
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Altitude = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: altitude units, always 'M'
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: geoid separation
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->GeoidSeparation = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: separation units
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: DGPS age
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: DGPS station ID
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: checksum
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
return true;
|
2010-08-27 02:15:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse an NMEA GPRMC sentence and update the given UAVObject
|
|
|
|
* \param[in] A pointer to a GPSPosition UAVObject to be updated.
|
|
|
|
* \param[in] An NMEA sentence with a valid checksum
|
|
|
|
*/
|
2010-09-27 07:28:34 +00:00
|
|
|
static bool nmeaProcessGPRMC(GPSPositionData * GpsData, char *sentence)
|
2010-08-27 02:15:00 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
char *next = sentence;
|
|
|
|
char *tokens;
|
|
|
|
char *delimiter = ",*";
|
|
|
|
GPSTimeData gpst;
|
|
|
|
GPSTimeGet(&gpst);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
|
|
|
#ifdef NMEA_DEBUG_RMC
|
2010-09-27 07:28:34 +00:00
|
|
|
PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART, "$%s\r\n", sentence);
|
2010-08-27 02:15:00 +00:00
|
|
|
#endif
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// get UTC time [hhmmss.sss]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
float hms = NMEA_real_to_float(tokens);
|
|
|
|
gpst.Second = (int)hms % 100;
|
|
|
|
gpst.Minute = (((int)hms - gpst.Second) / 100) % 100;
|
|
|
|
gpst.Hour = (int)hms / 10000;
|
|
|
|
|
|
|
|
// next field: Navigation receiver warning A = OK, V = warning
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: latitude
|
|
|
|
// get latitude [ddmm.mmmmm]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (!NMEA_latlon_to_fixed_point(&GpsData->Latitude, tokens)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// next field: N/S indicator
|
2010-10-27 10:39:38 +00:00
|
|
|
// correct latitude for N/S
|
2010-09-27 07:28:34 +00:00
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (tokens[0] == 'S')
|
|
|
|
GpsData->Latitude = -GpsData->Latitude;
|
|
|
|
|
|
|
|
// next field: longitude
|
|
|
|
// get longitude [dddmm.mmmmm]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (!NMEA_latlon_to_fixed_point(&GpsData->Longitude, tokens)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// next field: E/W indicator
|
2010-10-27 10:39:38 +00:00
|
|
|
// correct longitude for E/W
|
2010-09-27 07:28:34 +00:00
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (tokens[0] == 'W')
|
|
|
|
GpsData->Longitude = -GpsData->Longitude;
|
|
|
|
|
|
|
|
// next field: speed (knots)
|
|
|
|
// get speed in knots
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Groundspeed = NMEA_real_to_float(tokens);
|
2010-10-27 10:39:38 +00:00
|
|
|
// to m/s
|
|
|
|
GpsData->Groundspeed *= 0.51444;
|
2010-09-27 07:28:34 +00:00
|
|
|
|
|
|
|
// next field: True course
|
|
|
|
// get True course
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Heading = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: Date of fix
|
|
|
|
// get Date of fix
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
// TODO: Should really not use a float here to be safe
|
|
|
|
float date = NMEA_real_to_float(tokens);
|
|
|
|
gpst.Year = (int)date % 100;
|
|
|
|
gpst.Month = (((int)date - gpst.Year) / 100) % 100;
|
|
|
|
gpst.Day = (int)(date / 10000);
|
|
|
|
gpst.Year += 2000;
|
|
|
|
|
|
|
|
// next field: Magnetic variation
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: E or W
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: Mode: A=autonomous, D=differential, E=Estimated, N=not valid, S=Simulator
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: checksum
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
GPSTimeSet(&gpst);
|
|
|
|
return true;
|
|
|
|
}
|
2010-08-27 02:15:00 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse an NMEA GPVTG sentence and update the given UAVObject
|
|
|
|
* \param[in] A pointer to a GPSPosition UAVObject to be updated.
|
|
|
|
* \param[in] An NMEA sentence with a valid checksum
|
|
|
|
*/
|
2010-09-27 07:28:34 +00:00
|
|
|
static bool nmeaProcessGPVTG(GPSPositionData * GpsData, char *sentence)
|
2010-08-27 02:15:00 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
char *next = sentence;
|
|
|
|
char *tokens;
|
|
|
|
char *delimiter = ",*";
|
2010-08-27 02:15:00 +00:00
|
|
|
|
|
|
|
#ifdef NMEA_DEBUG_VTG
|
2010-09-27 07:28:34 +00:00
|
|
|
PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART, "$%s\r\n", sentence);
|
2010-08-27 02:15:00 +00:00
|
|
|
#endif
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// get course (true north ref) in degrees [ddd.dd]
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-27 10:39:38 +00:00
|
|
|
GpsData->Heading = NMEA_real_to_float(tokens);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: 'T'
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: course (magnetic north)
|
|
|
|
// get course (magnetic north ref) in degrees [ddd.dd]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
// next field: 'M'
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: speed (knots)
|
|
|
|
// get speed in knots
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-27 10:39:38 +00:00
|
|
|
GpsData->Groundspeed = NMEA_real_to_float(tokens);
|
|
|
|
// to m/s
|
|
|
|
GpsData->Groundspeed *= 0.51444;
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: 'N'
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: speed (km/h)
|
|
|
|
// get speed in km/h
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: 'K'
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: checksum
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-08-27 02:15:00 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
return true;
|
2010-08-27 02:15:00 +00:00
|
|
|
}
|
|
|
|
|
2010-08-27 19:59:25 +00:00
|
|
|
/**
|
|
|
|
* Parse an NMEA GPZDA sentence and update the @ref GPSTime object
|
|
|
|
* \param[in] A pointer to a GPSPosition UAVObject to be updated (unused).
|
|
|
|
* \param[in] An NMEA sentence with a valid checksum
|
|
|
|
*/
|
2010-09-27 07:28:34 +00:00
|
|
|
static bool nmeaProcessGPZDA(GPSPositionData * GpsData, char *sentence)
|
2010-08-27 19:59:25 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
char *next = sentence;
|
|
|
|
char *tokens;
|
|
|
|
char *delimiter = ",*";
|
|
|
|
|
2010-10-28 13:50:34 +00:00
|
|
|
#ifdef NMEA_DEBUG_ZDA
|
2010-09-27 07:28:34 +00:00
|
|
|
PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART, "$%s\r\n", sentence);
|
2010-08-27 19:59:25 +00:00
|
|
|
#endif
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
GPSTimeData gpst;
|
|
|
|
GPSTimeGet(&gpst);
|
|
|
|
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
float hms = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
gpst.Second = (int)hms % 100;
|
|
|
|
gpst.Minute = (((int)hms - gpst.Second) / 100) % 100;
|
|
|
|
gpst.Hour = (int)hms / 10000;
|
|
|
|
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 13:50:34 +00:00
|
|
|
gpst.Day = atoi(tokens);
|
2010-09-27 07:28:34 +00:00
|
|
|
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 13:50:34 +00:00
|
|
|
gpst.Month = atoi(tokens);
|
2010-09-27 07:28:34 +00:00
|
|
|
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 13:50:34 +00:00
|
|
|
gpst.Year = atoi(tokens);
|
2010-09-27 07:28:34 +00:00
|
|
|
|
|
|
|
GPSTimeSet(&gpst);
|
|
|
|
|
|
|
|
return true;
|
2010-08-27 19:59:25 +00:00
|
|
|
}
|
|
|
|
|
2010-08-29 01:46:14 +00:00
|
|
|
static GPSSatellitesData gsv_partial;
|
|
|
|
/* Bitmaps of which sentences we're looking for to allow us to handle out-of-order GSVs */
|
2010-09-27 07:28:34 +00:00
|
|
|
static uint8_t gsv_expected_mask;
|
|
|
|
static uint8_t gsv_processed_mask;
|
2010-08-29 01:46:14 +00:00
|
|
|
/* Error counters */
|
2010-09-27 07:28:34 +00:00
|
|
|
static uint16_t gsv_incomplete_error;
|
|
|
|
static uint16_t gsv_duplicate_error;
|
2010-08-29 01:46:14 +00:00
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
static bool nmeaProcessGPGSV(GPSPositionData * GpsData, char *sentence)
|
2010-08-29 01:46:14 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
char *next = sentence;
|
|
|
|
char *tokens;
|
|
|
|
char *delimiter = ",";
|
2010-08-29 01:46:14 +00:00
|
|
|
|
|
|
|
#ifdef NMEA_DEBUG_GSV
|
2010-09-27 07:28:34 +00:00
|
|
|
PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART, "$%s\r\n", sentence);
|
2010-08-29 01:46:14 +00:00
|
|
|
#endif
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
/* Drop the checksum */
|
|
|
|
char *tmp = sentence;
|
|
|
|
char *tmp_delim = "*";
|
|
|
|
next = strsep(&tmp, tmp_delim);
|
|
|
|
|
|
|
|
/* # of sentences in full GSV data set */
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
uint8_t total_sentences = atoi(tokens);
|
|
|
|
if ((total_sentences < 1) || (total_sentences > 8)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sentence number within the current GSV data set */
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
uint8_t current_sentence = atoi(tokens);
|
|
|
|
if (current_sentence < 1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* # of satellites currently in view */
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
gsv_partial.SatsInView = atoi(tokens);
|
|
|
|
|
|
|
|
/* Find out if this is the first sentence in the GSV set */
|
|
|
|
if (current_sentence == 1) {
|
|
|
|
if (gsv_expected_mask != gsv_processed_mask) {
|
|
|
|
/* We are starting over when we haven't yet finished our previous GSV group */
|
|
|
|
gsv_incomplete_error++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* First GSV sentence in the sequence, reset our expected_mask */
|
|
|
|
gsv_expected_mask = (1 << total_sentences) - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t current_sentence_id = (1 << (current_sentence - 1));
|
|
|
|
if (gsv_processed_mask & current_sentence_id) {
|
|
|
|
/* Duplicate sentence in this GSV set */
|
|
|
|
gsv_duplicate_error++;
|
|
|
|
} else {
|
|
|
|
/* Note that we've seen this sentence */
|
|
|
|
gsv_processed_mask |= current_sentence_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure this sentence can fit in our GPSSatellites object */
|
|
|
|
if ((current_sentence * 4) <= NELEMENTS(gsv_partial.PRN)) {
|
|
|
|
/* Process 4 blocks of satellite info */
|
|
|
|
for (uint8_t i = 0; next && i < 4; i++) {
|
|
|
|
uint8_t sat_index = ((current_sentence - 1) * 4) + i;
|
|
|
|
|
|
|
|
/* PRN number */
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
gsv_partial.PRN[sat_index] = atoi(tokens);
|
|
|
|
|
|
|
|
/* Elevation */
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
gsv_partial.Elevation[sat_index] = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
/* Azimuth */
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
gsv_partial.Azimuth[sat_index] = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
/* SNR */
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
gsv_partial.SNR[sat_index] = atoi(tokens);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find out if we're finished processing all GSV sentences in the set */
|
|
|
|
if ((gsv_expected_mask != 0) && (gsv_processed_mask == gsv_expected_mask)) {
|
|
|
|
/* GSV set has been fully processed. Update the GPSSatellites object. */
|
|
|
|
GPSSatellitesSet(&gsv_partial);
|
|
|
|
memset((void *)&gsv_partial, 0, sizeof(gsv_partial));
|
|
|
|
gsv_expected_mask = 0;
|
|
|
|
gsv_processed_mask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2010-08-29 01:46:14 +00:00
|
|
|
}
|
|
|
|
|
2010-08-27 02:15:00 +00:00
|
|
|
/**
|
|
|
|
* Parse an NMEA GPGSA sentence and update the given UAVObject
|
|
|
|
* \param[in] A pointer to a GPSPosition UAVObject to be updated.
|
|
|
|
* \param[in] An NMEA sentence with a valid checksum
|
|
|
|
*/
|
2010-09-27 07:28:34 +00:00
|
|
|
static bool nmeaProcessGPGSA(GPSPositionData * GpsData, char *sentence)
|
2010-08-27 02:15:00 +00:00
|
|
|
{
|
2010-09-27 07:28:34 +00:00
|
|
|
char *next = sentence;
|
|
|
|
char *tokens;
|
|
|
|
char *delimiter = ",*";
|
2010-08-27 02:15:00 +00:00
|
|
|
|
|
|
|
#ifdef NMEA_DEBUG_GSA
|
2010-09-27 07:28:34 +00:00
|
|
|
PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART, "$%s\r\n", sentence);
|
2010-08-27 02:15:00 +00:00
|
|
|
#endif
|
|
|
|
|
2010-09-27 07:28:34 +00:00
|
|
|
// next field: Mode
|
|
|
|
// Mode: M=Manual, forced to operate in 2D or 3D, A=Automatic, 3D/2D
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: Mode
|
|
|
|
// Mode: 1=Fix not available, 2=2D, 3=3D
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
switch (atoi(tokens)) {
|
|
|
|
case 1:
|
|
|
|
GpsData->Status = GPSPOSITION_STATUS_NOFIX;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
GpsData->Status = GPSPOSITION_STATUS_FIX2D;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
GpsData->Status = GPSPOSITION_STATUS_FIX3D;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* Unhandled */
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// next field: 3-14 IDs of SVs used in position fix (null for unused fields)
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: PDOP
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->PDOP = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: HDOP
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->HDOP = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: VDOP
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->VDOP = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: checksum
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
return true;
|
2010-08-27 02:15:00 +00:00
|
|
|
}
|
|
|
|
|
2011-02-22 10:16:50 +00:00
|
|
|
#endif // ENABLE_GPS_NMEA
|
|
|
|
|
2010-10-27 10:39:38 +00:00
|
|
|
/**
|
|
|
|
* Parse an NMEA PGTOP sentence and update the given UAVObject
|
|
|
|
* \param[in] A pointer to a GPSPosition UAVObject to be updated.
|
|
|
|
* \param[in] An NMEA sentence with a valid checksum
|
|
|
|
*/
|
|
|
|
static bool nmeaProcessPGTOP(GPSPositionData * GpsData, char *sentence)
|
|
|
|
{
|
|
|
|
char *next = sentence;
|
|
|
|
char *tokens;
|
|
|
|
char *delimiter = ",*";
|
|
|
|
|
|
|
|
#ifdef NMEA_DEBUG_PGTOP
|
|
|
|
PIOS_COM_SendFormattedStringNonBlocking(COM_DEBUG_USART, "$%s\r\n", sentence);
|
|
|
|
#endif
|
|
|
|
GPSTimeData gpst;
|
|
|
|
GPSTimeGet(&gpst);
|
|
|
|
|
|
|
|
// get UTC time [hhmmss.sss]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
float hms = NMEA_real_to_float(tokens);
|
|
|
|
gpst.Second = (int)hms % 100;
|
|
|
|
gpst.Minute = (((int)hms - gpst.Second) / 100) % 100;
|
|
|
|
gpst.Hour = (int)hms / 10000;
|
|
|
|
|
|
|
|
// next field: latitude
|
2010-10-28 14:58:39 +00:00
|
|
|
// get latitude decimal degrees
|
2010-10-27 10:39:38 +00:00
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 14:58:39 +00:00
|
|
|
GpsData->Latitude = NMEA_real_to_float(tokens)*1e7;
|
|
|
|
|
2010-10-27 10:39:38 +00:00
|
|
|
// next field: N/S indicator
|
|
|
|
// correct latitude for N/S
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (tokens[0] == 'S')
|
|
|
|
GpsData->Latitude = -GpsData->Latitude;
|
|
|
|
|
|
|
|
// next field: longitude
|
2010-10-28 14:58:39 +00:00
|
|
|
// get longitude decimal degrees
|
2010-10-27 10:39:38 +00:00
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 14:58:39 +00:00
|
|
|
GpsData->Longitude = NMEA_real_to_float(tokens)*1e7;
|
|
|
|
|
2010-10-27 10:39:38 +00:00
|
|
|
// next field: E/W indicator
|
|
|
|
// correct longitude for E/W
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
if (tokens[0] == 'W')
|
|
|
|
GpsData->Longitude = -GpsData->Longitude;
|
|
|
|
|
|
|
|
// next field: Fix Quality
|
|
|
|
// Mode: 0=Fix not available, 1=GPS fix, 2=DGPS fix
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
// next field: satellites used
|
|
|
|
// get number of satellites used in GPS solution
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Satellites = atoi(tokens);
|
|
|
|
|
|
|
|
// next field: HDOP
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->HDOP = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: altitude
|
|
|
|
// get altitude (in meters mm.m)
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Altitude = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: geoid separation
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->GeoidSeparation = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: Fix Type
|
|
|
|
// Mode: 1=Fix not available, 2=2D, 3=3D
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
switch (atoi(tokens)) {
|
|
|
|
case 1:
|
|
|
|
GpsData->Status = GPSPOSITION_STATUS_NOFIX;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
GpsData->Status = GPSPOSITION_STATUS_FIX2D;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
GpsData->Status = GPSPOSITION_STATUS_FIX3D;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* Unhandled */
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get course over ground in degrees [ddd.dd]
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Heading = NMEA_real_to_float(tokens);
|
|
|
|
|
|
|
|
// next field: speed (km/h)
|
|
|
|
// get speed in km/h
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
GpsData->Groundspeed = NMEA_real_to_float(tokens);
|
|
|
|
// to m/s
|
|
|
|
GpsData->Groundspeed /= 3.6;
|
|
|
|
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 13:50:34 +00:00
|
|
|
gpst.Day = atoi(tokens);
|
2010-10-27 10:39:38 +00:00
|
|
|
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 13:50:34 +00:00
|
|
|
gpst.Month = atoi(tokens);
|
2010-10-27 10:39:38 +00:00
|
|
|
|
|
|
|
tokens = strsep(&next, delimiter);
|
2010-10-28 13:50:34 +00:00
|
|
|
gpst.Year = atoi(tokens);
|
2010-10-27 10:39:38 +00:00
|
|
|
|
|
|
|
GPSTimeSet(&gpst);
|
|
|
|
|
|
|
|
// next field: checksum
|
|
|
|
tokens = strsep(&next, delimiter);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-02-22 10:16:50 +00:00
|
|
|
#endif // #if defined(ENABLE_GPS_NMEA) || defined(ENABLE_GPS_ONESENTENCE_GTOP)
|