239 lines
7.6 KiB
C++
239 lines
7.6 KiB
C++
/*
|
|
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, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
Author: Francisco Ferreira (some code is copied from sitl_gps.cpp)
|
|
|
|
*/
|
|
#define ALLOW_DOUBLE_MATH_FUNCTIONS
|
|
|
|
#include "AP_NMEA_Output.h"
|
|
|
|
#if HAL_NMEA_OUTPUT_ENABLED
|
|
|
|
#include <AP_Math/definitions.h>
|
|
#include <AP_RTC/AP_RTC.h>
|
|
#include <AP_AHRS/AP_AHRS.h>
|
|
#include <AP_GPS/AP_GPS.h>
|
|
#include <AP_SerialManager/AP_SerialManager.h>
|
|
|
|
#include <stdio.h>
|
|
#include <time.h>
|
|
|
|
AP_NMEA_Output::AP_NMEA_Output()
|
|
{
|
|
AP_SerialManager& sm = AP::serialmanager();
|
|
|
|
for (uint8_t i = 0; i < ARRAY_SIZE(_uart); i++) {
|
|
_uart[i] = sm.find_serial(AP_SerialManager::SerialProtocol_NMEAOutput, i);
|
|
|
|
if (_uart[i] == nullptr) {
|
|
break;
|
|
}
|
|
_num_outputs++;
|
|
}
|
|
}
|
|
|
|
AP_NMEA_Output* AP_NMEA_Output::probe()
|
|
{
|
|
AP_NMEA_Output *ret = new AP_NMEA_Output();
|
|
if (ret == nullptr || ret->_num_outputs == 0) {
|
|
delete ret;
|
|
return nullptr;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
uint8_t AP_NMEA_Output::_nmea_checksum(const char *str)
|
|
{
|
|
uint8_t checksum = 0;
|
|
const uint8_t* bytes = (const uint8_t*) str;
|
|
|
|
for (uint16_t i = 1; str[i]; i++) {
|
|
checksum ^= bytes[i];
|
|
}
|
|
|
|
return checksum;
|
|
}
|
|
|
|
void AP_NMEA_Output::update()
|
|
{
|
|
const uint32_t now_ms = AP_HAL::millis();
|
|
|
|
// only send at 10Hz
|
|
if ((now_ms - _last_run_ms) < 100) {
|
|
return;
|
|
}
|
|
_last_run_ms = now_ms;
|
|
|
|
// get time and date
|
|
uint64_t time_usec;
|
|
if (!AP::rtc().get_utc_usec(time_usec)) {
|
|
return;
|
|
}
|
|
|
|
// not completely accurate, our time includes leap seconds and time_t should be without
|
|
const time_t time_sec = time_usec / 1000000;
|
|
struct tm* tm = gmtime(&time_sec);
|
|
|
|
// format time string
|
|
char tstring[11];
|
|
snprintf(tstring, sizeof(tstring), "%02u%02u%06.3f", tm->tm_hour, tm->tm_min, tm->tm_sec + (time_usec % 1000000) * 1.0e-6);
|
|
|
|
// format date string
|
|
char dstring[7];
|
|
snprintf(dstring, sizeof(dstring), "%02u%02u%02u", tm->tm_mday, tm->tm_mon+1, tm->tm_year % 100);
|
|
|
|
auto &ahrs = AP::ahrs();
|
|
|
|
// get location (note: get_location from AHRS always returns true after having GPS position once)
|
|
Location loc;
|
|
bool pos_valid = ahrs.get_location(loc);
|
|
|
|
// format latitude
|
|
char lat_string[13];
|
|
double deg = fabs(loc.lat * 1.0e-7f);
|
|
double min_dec = ((fabs(loc.lat) - (unsigned)deg * 1.0e7)) * 60 * 1.e-7f;
|
|
snprintf(lat_string,
|
|
sizeof(lat_string),
|
|
"%02u%08.5f,%c",
|
|
(unsigned) deg,
|
|
min_dec,
|
|
loc.lat < 0 ? 'S' : 'N');
|
|
|
|
|
|
// format longitude
|
|
char lng_string[14];
|
|
deg = fabs(loc.lng * 1.0e-7f);
|
|
min_dec = ((fabs(loc.lng) - (unsigned)deg * 1.0e7)) * 60 * 1.e-7f;
|
|
snprintf(lng_string,
|
|
sizeof(lng_string),
|
|
"%03u%08.5f,%c",
|
|
(unsigned) deg,
|
|
min_dec,
|
|
loc.lng < 0 ? 'W' : 'E');
|
|
|
|
// format GGA message
|
|
char* gga = nullptr;
|
|
int16_t gga_res = asprintf(&gga,
|
|
"$GPGGA,%s,%s,%s,%01d,%02d,%04.1f,%07.2f,M,0.0,M,,",
|
|
tstring,
|
|
lat_string,
|
|
lng_string,
|
|
pos_valid ? 1 : 0,
|
|
pos_valid ? 6 : 3,
|
|
2.0,
|
|
loc.alt * 0.01f);
|
|
if (gga_res == -1) {
|
|
return;
|
|
}
|
|
char gga_end[6];
|
|
snprintf(gga_end, sizeof(gga_end), "*%02X\r\n", (unsigned) _nmea_checksum(gga));
|
|
|
|
// get speed
|
|
Vector2f speed = ahrs.groundspeed_vector();
|
|
float speed_knots = speed.length() * M_PER_SEC_TO_KNOTS;
|
|
float heading = wrap_360(degrees(atan2f(speed.x, speed.y)));
|
|
|
|
// format RMC message
|
|
char* rmc = nullptr;
|
|
int16_t rmc_res = asprintf(&rmc,
|
|
"$GPRMC,%s,%c,%s,%s,%.2f,%.2f,%s,,",
|
|
tstring,
|
|
pos_valid ? 'A' : 'V',
|
|
lat_string,
|
|
lng_string,
|
|
speed_knots,
|
|
heading,
|
|
dstring);
|
|
if (rmc_res == -1) {
|
|
free(gga);
|
|
return;
|
|
}
|
|
char rmc_end[6];
|
|
snprintf(rmc_end, sizeof(rmc_end), "*%02X\r\n", (unsigned) _nmea_checksum(rmc));
|
|
|
|
|
|
// get roll, pitch, yaw
|
|
const float roll_deg = wrap_180(degrees(ahrs.get_roll()));
|
|
const float pitch_deg = wrap_180(degrees(ahrs.get_pitch()));
|
|
const float yaw_deg = wrap_360(degrees(ahrs.get_yaw()));
|
|
const float heave_m = 0; // instantaneous heave in meters
|
|
const float roll_deg_accuracy = 0; // stddev of roll_deg;
|
|
const float pitch_deg_accuracy = 0; // stddev of pitch_deg;
|
|
const float heading_deg_accuracy = 0; // stddev of yaw_deg;
|
|
|
|
// GPS Update Quality Flag:
|
|
// 0 = no position
|
|
// 1 = All non-RTK fixed integer positions
|
|
// 2 = RTK fixed integer positions
|
|
const AP_GPS::GPS_Status gps_status = AP::gps().status();
|
|
const uint8_t gps_status_flag = (gps_status >= AP_GPS::GPS_OK_FIX_3D_RTK_FIXED) ? 2 :
|
|
(gps_status >= AP_GPS::GPS_OK_FIX_2D ? 1 : 0);
|
|
|
|
// INS Status Flag:
|
|
// 0 = All SPAN Pre-Alignment INS Status
|
|
// 1 = All SPAN Post-Alignment INS Status
|
|
const bool ins_status_flag = ahrs.initialised() &&
|
|
ahrs.healthy() &&
|
|
(!ahrs.have_inertial_nav() || AP::ins().accel_calibrated_ok_all());
|
|
|
|
// format PASHR message
|
|
char* pashr = nullptr;
|
|
int16_t pashr_res = asprintf(&pashr,
|
|
"$PASHR,%s,%.2f,T,%c%.2f,%c%.2f,%c%.2f,%.3f,%.3f,%.3f,%u,%u",
|
|
tstring,
|
|
yaw_deg, // This is a TRUE NORTH value
|
|
roll_deg<0? '-':'+', fabs(roll_deg), // always show + or - symbol
|
|
pitch_deg<0?'-':'+', fabs(pitch_deg), // always show + or - symbol
|
|
heave_m<0? '-':'+', fabs(heave_m), // always show + or - symbol
|
|
roll_deg_accuracy,
|
|
pitch_deg_accuracy,
|
|
heading_deg_accuracy,
|
|
(unsigned)gps_status_flag,
|
|
(unsigned)ins_status_flag);
|
|
if (pashr_res == -1) {
|
|
free(gga);
|
|
free(rmc);
|
|
return;
|
|
}
|
|
char pashr_end[6];
|
|
snprintf(pashr_end, sizeof(pashr_end), "*%02X\r\n", (unsigned) _nmea_checksum(pashr));
|
|
|
|
|
|
const uint32_t space_required = strlen(gga) + strlen(gga_end) + strlen(rmc) + strlen(rmc_end) + strlen(pashr) + strlen(pashr_end);
|
|
|
|
// send to all NMEA output ports
|
|
for (uint8_t i = 0; i < _num_outputs; i++) {
|
|
if (_uart[i]->txspace() < space_required) {
|
|
continue;
|
|
}
|
|
|
|
_uart[i]->write(gga);
|
|
_uart[i]->write(gga_end);
|
|
|
|
_uart[i]->write(rmc);
|
|
_uart[i]->write(rmc_end);
|
|
|
|
_uart[i]->write(pashr);
|
|
_uart[i]->write(pashr_end);
|
|
}
|
|
|
|
free(gga);
|
|
free(rmc);
|
|
free(pashr);
|
|
}
|
|
|
|
#endif // HAL_NMEA_OUTPUT_ENABLED
|