2014-03-28 16:52:27 -03:00
|
|
|
/*
|
|
|
|
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.
|
2010-09-08 02:06:56 -03:00
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
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.
|
2010-09-08 02:06:56 -03:00
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2016-02-17 21:25:23 -04:00
|
|
|
#pragma once
|
2012-12-18 20:40:18 -04:00
|
|
|
|
2015-08-11 03:28:43 -03:00
|
|
|
#include <AP_HAL/AP_HAL.h>
|
2014-03-28 16:52:27 -03:00
|
|
|
#include <inttypes.h>
|
2015-08-11 03:28:43 -03:00
|
|
|
#include <AP_Common/AP_Common.h>
|
|
|
|
#include <AP_Param/AP_Param.h>
|
|
|
|
#include <AP_Math/AP_Math.h>
|
|
|
|
#include <GCS_MAVLink/GCS_MAVLink.h>
|
|
|
|
#include <AP_Vehicle/AP_Vehicle.h>
|
2014-03-28 16:52:27 -03:00
|
|
|
#include "GPS_detect_state.h"
|
2015-08-15 19:50:18 -03:00
|
|
|
#include <AP_SerialManager/AP_SerialManager.h>
|
2014-03-28 16:52:27 -03:00
|
|
|
|
|
|
|
/**
|
|
|
|
maximum number of GPS instances available on this platform. If more
|
2017-01-05 14:07:14 -04:00
|
|
|
than 1 then redundant sensors may be available
|
2014-03-28 16:52:27 -03:00
|
|
|
*/
|
2017-01-29 19:02:57 -04:00
|
|
|
#define GPS_MAX_RECEIVERS 2 // maximum number of physical GPS sensors allowed - does not include virtual GPS created by blending receiver data
|
|
|
|
#define GPS_MAX_INSTANCES (GPS_MAX_RECEIVERS + 1) // maximumum number of GPs instances including the 'virtual' GPS created by blending receiver data
|
2015-06-27 04:26:25 -03:00
|
|
|
#define GPS_RTK_INJECT_TO_ALL 127
|
2017-03-07 23:33:31 -04:00
|
|
|
#define GPS_MAX_RATE_MS 200 // maximum value of rate_ms (i.e. slowest update rate) is 5hz or 200ms
|
2015-06-27 04:26:25 -03:00
|
|
|
|
2017-01-10 05:58:42 -04:00
|
|
|
// the number of GPS leap seconds
|
|
|
|
#define GPS_LEAPSECONDS_MILLIS 18000ULL
|
|
|
|
|
2017-02-11 05:26:18 -04:00
|
|
|
#define UNIX_OFFSET_MSEC (17000ULL * 86400ULL + 52 * 10 * MSEC_PER_WEEK - GPS_LEAPSECONDS_MILLIS)
|
2017-02-03 18:51:15 -04:00
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
class DataFlash_Class;
|
|
|
|
class AP_GPS_Backend;
|
|
|
|
|
|
|
|
/// @class AP_GPS
|
|
|
|
/// GPS driver main class
|
|
|
|
class AP_GPS
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2017-03-08 06:11:38 -04:00
|
|
|
friend class AP_GPS_ERB;
|
|
|
|
friend class AP_GPS_GSOF;
|
|
|
|
friend class AP_GPS_MAV;
|
|
|
|
friend class AP_GPS_MTK;
|
|
|
|
friend class AP_GPS_MTK19;
|
|
|
|
friend class AP_GPS_NMEA;
|
|
|
|
friend class AP_GPS_NOVA;
|
|
|
|
friend class AP_GPS_PX4;
|
|
|
|
friend class AP_GPS_QURT;
|
|
|
|
friend class AP_GPS_SBF;
|
|
|
|
friend class AP_GPS_SBP;
|
|
|
|
friend class AP_GPS_SIRF;
|
|
|
|
friend class AP_GPS_UBLOX;
|
|
|
|
friend class AP_GPS_Backend;
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2017-03-08 05:56:52 -04:00
|
|
|
// constructor
|
|
|
|
AP_GPS();
|
2014-03-28 16:52:27 -03:00
|
|
|
|
|
|
|
// GPS driver types
|
|
|
|
enum GPS_Type {
|
|
|
|
GPS_TYPE_NONE = 0,
|
|
|
|
GPS_TYPE_AUTO = 1,
|
|
|
|
GPS_TYPE_UBLOX = 2,
|
|
|
|
GPS_TYPE_MTK = 3,
|
|
|
|
GPS_TYPE_MTK19 = 4,
|
|
|
|
GPS_TYPE_NMEA = 5,
|
2014-04-01 03:25:15 -03:00
|
|
|
GPS_TYPE_SIRF = 6,
|
2014-04-04 20:05:54 -03:00
|
|
|
GPS_TYPE_HIL = 7,
|
2014-11-03 08:37:06 -04:00
|
|
|
GPS_TYPE_SBP = 8,
|
2015-09-02 06:13:38 -03:00
|
|
|
GPS_TYPE_PX4 = 9,
|
|
|
|
GPS_TYPE_SBF = 10,
|
2016-10-16 21:56:34 -03:00
|
|
|
GPS_TYPE_GSOF = 11,
|
|
|
|
GPS_TYPE_QURT = 12,
|
2016-01-18 17:54:40 -04:00
|
|
|
GPS_TYPE_ERB = 13,
|
2016-05-19 20:24:08 -03:00
|
|
|
GPS_TYPE_MAV = 14,
|
2016-10-16 21:56:34 -03:00
|
|
|
GPS_TYPE_NOVA = 15,
|
2014-03-28 16:52:27 -03:00
|
|
|
};
|
|
|
|
|
|
|
|
/// GPS status codes
|
|
|
|
enum GPS_Status {
|
2016-10-16 21:56:34 -03:00
|
|
|
NO_GPS = GPS_FIX_TYPE_NO_GPS, ///< No GPS connected/detected
|
|
|
|
NO_FIX = GPS_FIX_TYPE_NO_FIX, ///< Receiving valid GPS messages but no lock
|
|
|
|
GPS_OK_FIX_2D = GPS_FIX_TYPE_2D_FIX, ///< Receiving valid messages and 2D lock
|
|
|
|
GPS_OK_FIX_3D = GPS_FIX_TYPE_3D_FIX, ///< Receiving valid messages and 3D lock
|
|
|
|
GPS_OK_FIX_3D_DGPS = GPS_FIX_TYPE_DGPS, ///< Receiving valid messages and 3D lock with differential improvements
|
|
|
|
GPS_OK_FIX_3D_RTK_FLOAT = GPS_FIX_TYPE_RTK_FLOAT, ///< Receiving valid messages and 3D RTK Float
|
|
|
|
GPS_OK_FIX_3D_RTK_FIXED = GPS_FIX_TYPE_RTK_FIXED, ///< Receiving valid messages and 3D RTK Fixed
|
2014-03-28 16:52:27 -03:00
|
|
|
};
|
|
|
|
|
|
|
|
// GPS navigation engine settings. Not all GPS receivers support
|
|
|
|
// this
|
|
|
|
enum GPS_Engine_Setting {
|
|
|
|
GPS_ENGINE_NONE = -1,
|
|
|
|
GPS_ENGINE_PORTABLE = 0,
|
|
|
|
GPS_ENGINE_STATIONARY = 2,
|
|
|
|
GPS_ENGINE_PEDESTRIAN = 3,
|
|
|
|
GPS_ENGINE_AUTOMOTIVE = 4,
|
|
|
|
GPS_ENGINE_SEA = 5,
|
|
|
|
GPS_ENGINE_AIRBORNE_1G = 6,
|
|
|
|
GPS_ENGINE_AIRBORNE_2G = 7,
|
|
|
|
GPS_ENGINE_AIRBORNE_4G = 8
|
|
|
|
};
|
|
|
|
|
2016-02-02 03:58:33 -04:00
|
|
|
enum GPS_Config {
|
|
|
|
GPS_ALL_CONFIGURED = 255
|
|
|
|
};
|
|
|
|
|
2014-03-31 06:48:22 -03:00
|
|
|
/*
|
|
|
|
The GPS_State structure is filled in by the backend driver as it
|
|
|
|
parses each message from the GPS.
|
|
|
|
*/
|
2014-03-28 16:52:27 -03:00
|
|
|
struct GPS_State {
|
|
|
|
uint8_t instance; // the instance number of this GPS
|
|
|
|
|
|
|
|
// all the following fields must all be filled by the backend driver
|
|
|
|
GPS_Status status; ///< driver fix status
|
|
|
|
uint32_t time_week_ms; ///< GPS time (milliseconds from start of GPS week)
|
|
|
|
uint16_t time_week; ///< GPS week number
|
|
|
|
Location location; ///< last fix location
|
|
|
|
float ground_speed; ///< ground speed in m/sec
|
2016-05-04 22:28:35 -03:00
|
|
|
float ground_course; ///< ground course in degrees
|
2014-04-01 17:49:29 -03:00
|
|
|
uint16_t hdop; ///< horizontal dilution of precision in cm
|
2015-09-07 19:22:12 -03:00
|
|
|
uint16_t vdop; ///< vertical dilution of precision in cm
|
2016-05-12 14:02:29 -03:00
|
|
|
uint8_t num_sats; ///< Number of visible satellites
|
2014-03-28 16:52:27 -03:00
|
|
|
Vector3f velocity; ///< 3D velocitiy in m/s, in NED format
|
2014-10-28 16:44:07 -03:00
|
|
|
float speed_accuracy;
|
|
|
|
float horizontal_accuracy;
|
|
|
|
float vertical_accuracy;
|
2014-03-28 16:52:27 -03:00
|
|
|
bool have_vertical_velocity:1; ///< does this GPS give vertical velocity?
|
2014-10-28 16:44:07 -03:00
|
|
|
bool have_speed_accuracy:1;
|
|
|
|
bool have_horizontal_accuracy:1;
|
|
|
|
bool have_vertical_accuracy:1;
|
2014-03-28 16:52:27 -03:00
|
|
|
uint32_t last_gps_time_ms; ///< the system time we got the last GPS timestamp, milliseconds
|
|
|
|
};
|
|
|
|
|
2017-03-08 06:11:38 -04:00
|
|
|
/// Startup initialisation.
|
|
|
|
void init(DataFlash_Class *dataflash, const AP_SerialManager& serial_manager);
|
|
|
|
|
|
|
|
/// Update GPS state based on possible bytes received from the module.
|
|
|
|
/// This routine must be called periodically (typically at 10Hz or
|
|
|
|
/// more) to process incoming data.
|
|
|
|
void update(void);
|
|
|
|
|
2016-05-19 20:24:08 -03:00
|
|
|
// Pass mavlink data to message handlers (for MAV type)
|
2016-10-04 04:37:01 -03:00
|
|
|
void handle_msg(const mavlink_message_t *msg);
|
2016-05-19 20:24:08 -03:00
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
// Accessor functions
|
|
|
|
|
2014-04-03 18:32:34 -03:00
|
|
|
// return number of active GPS sensors. Note that if the first GPS
|
2017-01-29 19:02:57 -04:00
|
|
|
// is not present but the 2nd is then we return 2. Note that a blended
|
2017-03-07 23:33:31 -04:00
|
|
|
// GPS solution is treated as an additional sensor.
|
2017-03-08 05:56:52 -04:00
|
|
|
uint8_t num_sensors(void) const;
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2017-01-29 19:02:57 -04:00
|
|
|
// Return the index of the primary sensor which is the index of the sensor contributing to
|
2017-03-07 23:33:31 -04:00
|
|
|
// the output. A blended solution is available as an additional instance
|
2014-06-06 18:58:11 -03:00
|
|
|
uint8_t primary_sensor(void) const {
|
|
|
|
return primary_instance;
|
|
|
|
}
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
/// Query GPS status
|
|
|
|
GPS_Status status(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].status;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
GPS_Status status(void) const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return status(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
2017-03-08 05:49:58 -04:00
|
|
|
// Query the highest status this GPS supports (always reports GPS_OK_FIX_3D for the blended GPS)
|
2014-06-06 18:58:11 -03:00
|
|
|
GPS_Status highest_supported_status(uint8_t instance) const;
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
// location of last fix
|
|
|
|
const Location &location(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].location;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
const Location &location() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return location(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
2017-03-08 05:56:52 -04:00
|
|
|
// report speed accuracy
|
|
|
|
bool speed_accuracy(uint8_t instance, float &sacc) const;
|
2014-10-28 16:44:07 -03:00
|
|
|
bool speed_accuracy(float &sacc) const {
|
|
|
|
return speed_accuracy(primary_instance, sacc);
|
|
|
|
}
|
|
|
|
|
2017-03-08 05:56:52 -04:00
|
|
|
bool horizontal_accuracy(uint8_t instance, float &hacc) const;
|
2014-10-28 16:44:07 -03:00
|
|
|
bool horizontal_accuracy(float &hacc) const {
|
|
|
|
return horizontal_accuracy(primary_instance, hacc);
|
|
|
|
}
|
|
|
|
|
2017-03-08 05:56:52 -04:00
|
|
|
bool vertical_accuracy(uint8_t instance, float &vacc) const;
|
2014-10-28 16:44:07 -03:00
|
|
|
bool vertical_accuracy(float &vacc) const {
|
|
|
|
return vertical_accuracy(primary_instance, vacc);
|
|
|
|
}
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
// 3D velocity in NED format
|
|
|
|
const Vector3f &velocity(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].velocity;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
const Vector3f &velocity() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return velocity(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// ground speed in m/s
|
|
|
|
float ground_speed(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].ground_speed;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
float ground_speed() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return ground_speed(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// ground speed in cm/s
|
|
|
|
uint32_t ground_speed_cm(void) {
|
|
|
|
return ground_speed() * 100;
|
|
|
|
}
|
|
|
|
|
2017-03-08 06:01:29 -04:00
|
|
|
// ground course in degrees
|
2016-05-04 22:28:35 -03:00
|
|
|
float ground_course(uint8_t instance) const {
|
|
|
|
return state[instance].ground_course;
|
|
|
|
}
|
|
|
|
float ground_course() const {
|
|
|
|
return ground_course(primary_instance);
|
|
|
|
}
|
2017-03-08 06:01:29 -04:00
|
|
|
// ground course in centi-degrees
|
2014-03-28 16:52:27 -03:00
|
|
|
int32_t ground_course_cd(uint8_t instance) const {
|
2016-05-04 22:28:35 -03:00
|
|
|
return ground_course(instance) * 100;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
int32_t ground_course_cd() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return ground_course_cd(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// number of locked satellites
|
|
|
|
uint8_t num_sats(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].num_sats;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
uint8_t num_sats() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return num_sats(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// GPS time of week in milliseconds
|
|
|
|
uint32_t time_week_ms(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].time_week_ms;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
uint32_t time_week_ms() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return time_week_ms(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// GPS week
|
|
|
|
uint16_t time_week(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].time_week;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
uint16_t time_week() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return time_week(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// horizontal dilution of precision
|
2014-04-01 17:49:29 -03:00
|
|
|
uint16_t get_hdop(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].hdop;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
2014-04-01 17:49:29 -03:00
|
|
|
uint16_t get_hdop() const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return get_hdop(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
2015-09-07 19:37:13 -03:00
|
|
|
// vertical dilution of precision
|
|
|
|
uint16_t get_vdop(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].vdop;
|
2015-09-07 19:37:13 -03:00
|
|
|
}
|
|
|
|
uint16_t get_vdop() const {
|
|
|
|
return get_vdop(primary_instance);
|
|
|
|
}
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
// the time we got our last fix in system milliseconds. This is
|
|
|
|
// used when calculating how far we might have moved since that fix
|
|
|
|
uint32_t last_fix_time_ms(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return timing[instance].last_fix_time_ms;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
uint32_t last_fix_time_ms(void) const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return last_fix_time_ms(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
2015-11-03 09:17:25 -04:00
|
|
|
// the time we last processed a message in milliseconds. This is
|
|
|
|
// used to indicate that we have new GPS data to process
|
|
|
|
uint32_t last_message_time_ms(uint8_t instance) const {
|
|
|
|
return timing[instance].last_message_time_ms;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
uint32_t last_message_time_ms(void) const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return last_message_time_ms(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// return true if the GPS supports vertical velocity values
|
|
|
|
bool have_vertical_velocity(uint8_t instance) const {
|
2015-11-03 09:17:25 -04:00
|
|
|
return state[instance].have_vertical_velocity;
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
bool have_vertical_velocity(void) const {
|
2014-04-03 18:32:34 -03:00
|
|
|
return have_vertical_velocity(primary_instance);
|
2014-03-28 16:52:27 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// the expected lag (in seconds) in the position and velocity readings from the gps
|
2016-12-18 19:31:28 -04:00
|
|
|
float get_lag(uint8_t instance) const;
|
|
|
|
float get_lag(void) const { return get_lag(primary_instance); }
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2017-01-05 14:07:14 -04:00
|
|
|
// return a 3D vector defining the offset of the GPS antenna in meters relative to the body frame origin
|
2017-03-08 05:56:52 -04:00
|
|
|
const Vector3f &get_antenna_offset(uint8_t instance) const;
|
2016-10-07 19:04:14 -03:00
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
// set position for HIL
|
2014-04-01 17:49:29 -03:00
|
|
|
void setHIL(uint8_t instance, GPS_Status status, uint64_t time_epoch_ms,
|
2014-08-09 09:47:54 -03:00
|
|
|
const Location &location, const Vector3f &velocity, uint8_t num_sats,
|
2016-05-04 22:28:35 -03:00
|
|
|
uint16_t hdop);
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2016-05-04 05:16:05 -03:00
|
|
|
// set accuracy for HIL
|
2016-05-05 03:04:18 -03:00
|
|
|
void setHIL_Accuracy(uint8_t instance, float vdop, float hacc, float vacc, float sacc, bool _have_vertical_velocity, uint32_t sample_ms);
|
2017-03-08 06:11:38 -04:00
|
|
|
|
|
|
|
// lock out a GPS port, allowing another application to use the port
|
|
|
|
void lock_port(uint8_t instance, bool locked);
|
|
|
|
|
|
|
|
//Inject a packet of raw binary to a GPS
|
|
|
|
void inject_data(uint8_t *data, uint8_t len);
|
|
|
|
void inject_data(uint8_t instance, uint8_t *data, uint8_t len);
|
|
|
|
|
|
|
|
//MAVLink Status Sending
|
|
|
|
void send_mavlink_gps_raw(mavlink_channel_t chan);
|
|
|
|
void send_mavlink_gps2_raw(mavlink_channel_t chan);
|
|
|
|
|
|
|
|
void send_mavlink_gps_rtk(mavlink_channel_t chan);
|
|
|
|
void send_mavlink_gps2_rtk(mavlink_channel_t chan);
|
|
|
|
|
|
|
|
// Returns the index of the first unconfigured GPS (returns GPS_ALL_CONFIGURED if all instances report as being configured)
|
|
|
|
uint8_t first_unconfigured_gps(void) const;
|
|
|
|
void broadcast_first_configuration_failure_reason(void) const;
|
|
|
|
|
|
|
|
// return true if all GPS instances have finished configuration
|
|
|
|
bool all_configured(void) const {
|
|
|
|
return first_unconfigured_gps() == GPS_ALL_CONFIGURED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle sending of initialisation strings to the GPS - only used by backends
|
|
|
|
void send_blob_start(uint8_t instance, const char *_blob, uint16_t size);
|
|
|
|
void send_blob_update(uint8_t instance);
|
|
|
|
|
|
|
|
// return last fix time since the 1/1/1970 in microseconds
|
|
|
|
uint64_t time_epoch_usec(uint8_t instance);
|
|
|
|
uint64_t time_epoch_usec(void) {
|
|
|
|
return time_epoch_usec(primary_instance);
|
|
|
|
}
|
|
|
|
|
|
|
|
// convert GPS week and millis to unix epoch in ms
|
|
|
|
static uint64_t time_epoch_convert(uint16_t gps_week, uint32_t gps_ms);
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
static const struct AP_Param::GroupInfo var_info[];
|
|
|
|
|
2017-03-08 06:11:38 -04:00
|
|
|
protected:
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
// dataflash for logging, if available
|
|
|
|
DataFlash_Class *_DataFlash;
|
|
|
|
|
|
|
|
// configuration parameters
|
2017-01-29 19:02:57 -04:00
|
|
|
AP_Int8 _type[GPS_MAX_RECEIVERS];
|
2014-03-28 16:52:27 -03:00
|
|
|
AP_Int8 _navfilter;
|
2014-06-06 18:58:11 -03:00
|
|
|
AP_Int8 _auto_switch;
|
|
|
|
AP_Int8 _min_dgps;
|
2015-04-22 20:10:46 -03:00
|
|
|
AP_Int16 _sbp_logmask;
|
|
|
|
AP_Int8 _inject_to;
|
2015-05-21 18:07:59 -03:00
|
|
|
uint32_t _last_instance_swap_ms;
|
2014-09-04 01:27:05 -03:00
|
|
|
AP_Int8 _sbas_mode;
|
2014-09-04 01:43:29 -03:00
|
|
|
AP_Int8 _min_elevation;
|
2015-05-04 05:18:34 -03:00
|
|
|
AP_Int8 _raw_data;
|
2017-01-29 19:02:57 -04:00
|
|
|
AP_Int8 _gnss_mode[GPS_MAX_RECEIVERS];
|
|
|
|
AP_Int16 _rate_ms[GPS_MAX_RECEIVERS];
|
2015-10-28 21:40:50 -03:00
|
|
|
AP_Int8 _save_config;
|
2016-02-02 03:58:33 -04:00
|
|
|
AP_Int8 _auto_config;
|
2017-01-29 19:02:57 -04:00
|
|
|
AP_Vector3f _antenna_offset[GPS_MAX_RECEIVERS];
|
|
|
|
AP_Int16 _delay_ms[GPS_MAX_RECEIVERS];
|
|
|
|
AP_Int8 _blend_mask;
|
|
|
|
AP_Float _blend_tc;
|
2016-10-07 19:04:14 -03:00
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
private:
|
2017-03-08 06:11:38 -04:00
|
|
|
// return gps update rate in milliseconds
|
|
|
|
uint16_t get_rate_ms(uint8_t instance) const;
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
struct GPS_timing {
|
|
|
|
// the time we got our last fix in system milliseconds
|
|
|
|
uint32_t last_fix_time_ms;
|
|
|
|
|
|
|
|
// the time we got our last fix in system milliseconds
|
|
|
|
uint32_t last_message_time_ms;
|
|
|
|
};
|
2017-01-29 19:02:57 -04:00
|
|
|
// Note allowance for an additional instance to contain blended data
|
|
|
|
GPS_timing timing[GPS_MAX_RECEIVERS+1];
|
|
|
|
GPS_State state[GPS_MAX_RECEIVERS+1];
|
|
|
|
AP_GPS_Backend *drivers[GPS_MAX_RECEIVERS];
|
|
|
|
AP_HAL::UARTDriver *_port[GPS_MAX_RECEIVERS];
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2014-04-03 18:32:34 -03:00
|
|
|
/// primary GPS instance
|
|
|
|
uint8_t primary_instance:2;
|
|
|
|
|
|
|
|
/// number of GPS instances present
|
|
|
|
uint8_t num_instances:2;
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2014-04-04 17:33:34 -03:00
|
|
|
// which ports are locked
|
|
|
|
uint8_t locked_ports:2;
|
|
|
|
|
2014-03-28 16:52:27 -03:00
|
|
|
// state of auto-detection process, per instance
|
|
|
|
struct detect_state {
|
|
|
|
uint32_t detect_started_ms;
|
|
|
|
uint32_t last_baud_change_ms;
|
2016-09-24 03:16:40 -03:00
|
|
|
uint8_t current_baud;
|
2014-03-28 16:52:27 -03:00
|
|
|
struct UBLOX_detect_state ublox_detect_state;
|
|
|
|
struct MTK_detect_state mtk_detect_state;
|
|
|
|
struct MTK19_detect_state mtk19_detect_state;
|
|
|
|
struct SIRF_detect_state sirf_detect_state;
|
|
|
|
struct NMEA_detect_state nmea_detect_state;
|
2014-04-04 20:05:54 -03:00
|
|
|
struct SBP_detect_state sbp_detect_state;
|
2016-01-18 17:54:40 -04:00
|
|
|
struct ERB_detect_state erb_detect_state;
|
2017-01-29 19:02:57 -04:00
|
|
|
} detect_state[GPS_MAX_RECEIVERS];
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2014-03-31 06:48:22 -03:00
|
|
|
struct {
|
2015-10-26 08:25:44 -03:00
|
|
|
const char *blob;
|
2014-03-31 06:48:22 -03:00
|
|
|
uint16_t remaining;
|
2017-01-29 19:02:57 -04:00
|
|
|
} initblob_state[GPS_MAX_RECEIVERS];
|
2014-03-31 06:48:22 -03:00
|
|
|
|
2014-04-04 20:05:54 -03:00
|
|
|
static const uint32_t _baudrates[];
|
2015-10-26 08:25:44 -03:00
|
|
|
static const char _initialisation_blob[];
|
|
|
|
static const char _initialisation_raw_blob[];
|
2014-03-28 16:52:27 -03:00
|
|
|
|
|
|
|
void detect_instance(uint8_t instance);
|
|
|
|
void update_instance(uint8_t instance);
|
2016-02-09 16:59:12 -04:00
|
|
|
void _broadcast_gps_type(const char *type, uint8_t instance, int8_t baud_index);
|
2016-10-04 04:37:01 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
buffer for re-assembling RTCM data for GPS injection.
|
|
|
|
The 8 bit flags field in GPS_RTCM_DATA is interpreted as:
|
|
|
|
1 bit for "is fragmented"
|
|
|
|
2 bits for fragment number
|
|
|
|
5 bits for sequence number
|
|
|
|
|
|
|
|
The rtcm_buffer is allocated on first use. Once a block of data
|
|
|
|
is successfully reassembled it is injected into all active GPS
|
|
|
|
backends. This assumes we don't want more than 4*180=720 bytes
|
|
|
|
in a RTCM data block
|
|
|
|
*/
|
|
|
|
struct rtcm_buffer {
|
|
|
|
uint8_t fragments_received:4;
|
|
|
|
uint8_t sequence:5;
|
|
|
|
uint8_t fragment_count;
|
|
|
|
uint16_t total_length;
|
|
|
|
uint8_t buffer[MAVLINK_MSG_GPS_RTCM_DATA_FIELD_DATA_LEN*4];
|
|
|
|
} *rtcm_buffer;
|
|
|
|
|
|
|
|
// re-assemble GPS_RTCM_DATA message
|
|
|
|
void handle_gps_rtcm_data(const mavlink_message_t *msg);
|
|
|
|
|
2017-01-05 14:07:14 -04:00
|
|
|
// inject data into all backends
|
2016-10-04 04:37:01 -03:00
|
|
|
void inject_data_all(const uint8_t *data, uint16_t len);
|
2014-03-28 16:52:27 -03:00
|
|
|
|
2017-01-29 19:02:57 -04:00
|
|
|
// GPS blending and switching
|
|
|
|
Vector2f _NE_pos_offset_m[GPS_MAX_RECEIVERS]; // Filtered North,East position offset from GPS instance to blended solution in _output_state.location (m)
|
|
|
|
float _hgt_offset_cm[GPS_MAX_RECEIVERS]; // Filtered height offset from GPS instance relative to blended solution in _output_state.location (cm)
|
|
|
|
Vector3f _blended_antenna_offset; // blended antenna offset
|
|
|
|
float _blended_lag_sec = 0.001f * GPS_MAX_RATE_MS; // blended receiver lag in seconds
|
|
|
|
float _blend_weights[GPS_MAX_RECEIVERS]; // blend weight for each GPS. The blend weights must sum to 1.0 across all instances.
|
|
|
|
uint32_t _last_time_updated[GPS_MAX_RECEIVERS]; // the last value of state.last_gps_time_ms read for that GPS instance - used to detect new data.
|
|
|
|
float _omega_lpf; // cutoff frequency in rad/sec of LPF applied to position offsets
|
|
|
|
bool _output_is_blended; // true when a blended GPS solution being output
|
|
|
|
|
|
|
|
// calculate the blend weight. Returns true if blend could be calculated, false if not
|
|
|
|
bool calc_blend_weights(void);
|
|
|
|
|
|
|
|
// calculate the blended state
|
|
|
|
void calc_blended_state(void);
|
|
|
|
|
|
|
|
};
|