mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-18 06:38:29 -04:00
AP_GPS: SwiftNav RTK Driver and GPS AutoSwitch param
This commit is contained in:
parent
fee79c5bac
commit
2b1169b0ab
@ -27,15 +27,17 @@ const AP_Param::GroupInfo AP_GPS::var_info[] PROGMEM = {
|
||||
// @Param: TYPE
|
||||
// @DisplayName: GPS type
|
||||
// @Description: GPS type
|
||||
// @Values: 0:None,1:AUTO,2:uBlox,3:MTK,4:MTK19,5:NMEA,6:SiRF,7:HIL,8:SwiftBinaryProtocol
|
||||
// @Values: 0:None,1:AUTO,2:uBlox,3:MTK,4:MTK19,5:NMEA,6:SiRF,7:HIL,8:SwiftNav
|
||||
AP_GROUPINFO("TYPE", 0, AP_GPS, _type[0], 1),
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
|
||||
// @Param: TYPE2
|
||||
// @DisplayName: 2nd GPS type
|
||||
// @Description: GPS type of 2nd GPS
|
||||
// @Values: 0:None,1:AUTO,2:uBlox,3:MTK,4:MTK19,5:NMEA,6:SiRF,7:HIL,8:SwiftBinaryProtocol
|
||||
// @Values: 0:None,1:AUTO,2:uBlox,3:MTK,4:MTK19,5:NMEA,6:SiRF,7:HIL,8:SwiftNav
|
||||
AP_GROUPINFO("TYPE2", 1, AP_GPS, _type[1], 0),
|
||||
|
||||
#endif
|
||||
|
||||
// @Param: NAVFILTER
|
||||
@ -44,6 +46,24 @@ const AP_Param::GroupInfo AP_GPS::var_info[] PROGMEM = {
|
||||
// @Values: 0:Portable,2:Stationary,3:Pedestrian,4:Automotive,5:Sea,6:Airborne1G,7:Airborne2G,8:Airborne4G
|
||||
AP_GROUPINFO("NAVFILTER", 2, AP_GPS, _navfilter, GPS_ENGINE_AIRBORNE_4G),
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
// @Param: AUTO_SWITCH
|
||||
// @DisplayName: Automatic Switchover Setting
|
||||
// @Description: Automatic switchover to GPS reporting best lock
|
||||
// @Values: 0:Disabled,1:Enabled
|
||||
// @User: Advanced
|
||||
AP_GROUPINFO("AUTO_SWITCH", 3, AP_GPS, _auto_switch, 1),
|
||||
#endif
|
||||
|
||||
#if GPS_RTK_AVAILABLE
|
||||
// @Param: DGPS_MIN_LOCK
|
||||
// @DisplayName: Minimum Lock Type Accepted for DGPS
|
||||
// @Description: Sets the minimum type of differential GPS corrections required before allowing to switch into DGPS mode.
|
||||
// @Values: 0:Any,50:FloatRTK,100:IntegerRTK
|
||||
// @User: Advanced
|
||||
AP_GROUPINFO("MIN_DGPS", 4, AP_GPS, _min_dgps, 100),
|
||||
#endif
|
||||
|
||||
AP_GROUPEND
|
||||
};
|
||||
|
||||
@ -53,6 +73,7 @@ void AP_GPS::init(DataFlash_Class *dataflash)
|
||||
_DataFlash = dataflash;
|
||||
hal.uartB->begin(38400UL, 256, 16);
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
primary_instance = 0;
|
||||
if (hal.uartE != NULL) {
|
||||
hal.uartE->begin(38400UL, 256, 16);
|
||||
}
|
||||
@ -165,7 +186,7 @@ AP_GPS::detect_instance(uint8_t instance)
|
||||
hal.console->print_P(PSTR(" MTK "));
|
||||
new_gps = new AP_GPS_MTK(*this, state[instance], port);
|
||||
}
|
||||
#if HAL_CPU_CLASS >= HAL_CPU_CLASS_75
|
||||
#if GPS_RTK_AVAILABLE
|
||||
else if ((_type[instance] == GPS_TYPE_AUTO || _type[instance] == GPS_TYPE_SBP) &&
|
||||
AP_GPS_SBP::_detect(dstate->sbp_detect_state, data)) {
|
||||
hal.console->print_P(PSTR(" SBP "));
|
||||
@ -198,6 +219,55 @@ AP_GPS::detect_instance(uint8_t instance)
|
||||
}
|
||||
}
|
||||
|
||||
bool
|
||||
AP_GPS::can_calculate_base_pos(void)
|
||||
{
|
||||
#if GPS_RTK_AVAILABLE
|
||||
for (uint8_t i=0; i<GPS_MAX_INSTANCES; i++) {
|
||||
if (drivers[i] != NULL && drivers[i]->can_calculate_base_pos()) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
return false;
|
||||
}
|
||||
|
||||
/*
|
||||
Tells the underlying GPS drivers to capture its current position as home.
|
||||
*/
|
||||
void
|
||||
AP_GPS::calculate_base_pos(void)
|
||||
{
|
||||
#if GPS_RTK_AVAILABLE
|
||||
for (uint8_t i = 0; i<GPS_MAX_INSTANCES; i++) {
|
||||
if (drivers[i] != NULL && drivers[i]->can_calculate_base_pos()) {
|
||||
drivers[i]->calculate_base_pos();
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
AP_GPS::GPS_Status
|
||||
AP_GPS::highest_supported_status(uint8_t instance) const
|
||||
{
|
||||
#if GPS_RTK_AVAILABLE
|
||||
if (drivers[instance] != NULL)
|
||||
return drivers[instance]->highest_supported_status();
|
||||
#endif
|
||||
return AP_GPS::GPS_OK_FIX_3D;
|
||||
}
|
||||
|
||||
AP_GPS::GPS_Status
|
||||
AP_GPS::highest_supported_status(void) const
|
||||
{
|
||||
#if GPS_RTK_AVAILABLE
|
||||
|
||||
if (drivers[primary_instance] != NULL)
|
||||
return drivers[primary_instance]->highest_supported_status();
|
||||
#endif
|
||||
return AP_GPS::GPS_OK_FIX_3D;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
update one GPS instance. This should be called at 10Hz or greater
|
||||
@ -260,6 +330,7 @@ AP_GPS::update_instance(uint8_t instance)
|
||||
void
|
||||
AP_GPS::update(void)
|
||||
{
|
||||
|
||||
for (uint8_t i=0; i<GPS_MAX_INSTANCES; i++) {
|
||||
update_instance(i);
|
||||
}
|
||||
@ -273,22 +344,26 @@ AP_GPS::update(void)
|
||||
if (state[i].status != NO_GPS) {
|
||||
num_instances = i+1;
|
||||
}
|
||||
if (i == primary_instance) {
|
||||
continue;
|
||||
}
|
||||
if (state[i].status > state[primary_instance].status) {
|
||||
// we have a higher status lock, change GPS
|
||||
primary_instance = i;
|
||||
continue;
|
||||
}
|
||||
if (state[i].status == state[primary_instance].status &&
|
||||
state[i].num_sats >= state[primary_instance].num_sats + 2) {
|
||||
// this GPS has at least 2 more satellites than the
|
||||
// current primary, switch primary. Once we switch we will
|
||||
// then tend to stick to the new GPS as primary. We don't
|
||||
// want to switch too often as it will look like a
|
||||
// position shift to the controllers.
|
||||
primary_instance = i;
|
||||
if (_auto_switch) {
|
||||
if (i == primary_instance) {
|
||||
continue;
|
||||
}
|
||||
if (state[i].status > state[primary_instance].status) {
|
||||
// we have a higher status lock, change GPS
|
||||
primary_instance = i;
|
||||
continue;
|
||||
}
|
||||
if (state[i].status == state[primary_instance].status &&
|
||||
state[i].num_sats >= state[primary_instance].num_sats + 2) {
|
||||
// this GPS has at least 2 more satellites than the
|
||||
// current primary, switch primary. Once we switch we will
|
||||
// then tend to stick to the new GPS as primary. We don't
|
||||
// want to switch too often as it will look like a
|
||||
// position shift to the controllers.
|
||||
primary_instance = i;
|
||||
}
|
||||
} else {
|
||||
primary_instance = 0;
|
||||
}
|
||||
}
|
||||
#else
|
||||
@ -343,3 +418,73 @@ AP_GPS::lock_port(uint8_t instance, bool lock)
|
||||
locked_ports &= ~(1U<<instance);
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS::send_mavlink_gps_raw(mavlink_channel_t chan)
|
||||
{
|
||||
static uint32_t last_send_time_ms;
|
||||
if (last_send_time_ms == 0 || last_send_time_ms != last_message_time_ms(0)) {
|
||||
last_send_time_ms = last_message_time_ms(0);
|
||||
const Location &loc = location(0);
|
||||
mavlink_msg_gps_raw_int_send(
|
||||
chan,
|
||||
last_fix_time_ms(0)*(uint64_t)1000,
|
||||
status(0),
|
||||
loc.lat, // in 1E7 degrees
|
||||
loc.lng, // in 1E7 degrees
|
||||
loc.alt * 10UL, // in mm
|
||||
get_hdop(0),
|
||||
65535,
|
||||
ground_speed(0)*100, // cm/s
|
||||
ground_course_cd(0), // 1/100 degrees,
|
||||
num_sats(0));
|
||||
}
|
||||
}
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
void
|
||||
AP_GPS::send_mavlink_gps2_raw(mavlink_channel_t chan)
|
||||
{
|
||||
static uint32_t last_send_time_ms2;
|
||||
if (num_sensors() > 1 &&
|
||||
status(1) > AP_GPS::NO_GPS &&
|
||||
(last_send_time_ms2 == 0 || last_send_time_ms2 != last_message_time_ms(1))) {
|
||||
const Location &loc = location(1);
|
||||
last_send_time_ms2 = last_message_time_ms(1);
|
||||
mavlink_msg_gps2_raw_send(
|
||||
chan,
|
||||
last_fix_time_ms(1)*(uint64_t)1000,
|
||||
status(1),
|
||||
loc.lat,
|
||||
loc.lng,
|
||||
loc.alt * 10UL,
|
||||
get_hdop(1),
|
||||
65535,
|
||||
ground_speed(1)*100, // cm/s
|
||||
ground_course_cd(1), // 1/100 degrees,
|
||||
num_sats(1),
|
||||
0,
|
||||
0);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#if GPS_RTK_AVAILABLE
|
||||
void
|
||||
AP_GPS::send_mavlink_gps_rtk(mavlink_channel_t chan)
|
||||
{
|
||||
if (drivers[0] != NULL && drivers[0]->highest_supported_status() > AP_GPS::GPS_OK_FIX_3D) {
|
||||
drivers[0]->send_mavlink_gps_rtk(chan);
|
||||
}
|
||||
}
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
void
|
||||
AP_GPS::send_mavlink_gps2_rtk(mavlink_channel_t chan)
|
||||
{
|
||||
if (drivers[1] != NULL && drivers[1]->highest_supported_status() > AP_GPS::GPS_OK_FIX_3D) {
|
||||
drivers[1]->send_mavlink_gps_rtk(chan);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
#endif
|
||||
|
@ -35,6 +35,12 @@
|
||||
#define GPS_MAX_INSTANCES 1
|
||||
#endif
|
||||
|
||||
#if HAL_CPU_CLASS > HAL_CPU_CLASS_75
|
||||
#define GPS_RTK_AVAILABLE 1
|
||||
#else
|
||||
#define GPS_RTK_AVAILABLE 0
|
||||
#endif
|
||||
|
||||
class DataFlash_Class;
|
||||
class AP_GPS_Backend;
|
||||
|
||||
@ -56,6 +62,14 @@ public:
|
||||
/// more) to process incoming data.
|
||||
void update(void);
|
||||
|
||||
//True if any of the underlying GPS Drivers are ready to enter
|
||||
//a dgps-based fix beyond 3D lock, such as RTK mode.
|
||||
bool can_calculate_base_pos(void);
|
||||
|
||||
//Allows the underlying GPS Drivers to enter a differential lock
|
||||
//Might cause a position jump, thus only do this on the ground.
|
||||
void calculate_base_pos(void);
|
||||
|
||||
// GPS driver types
|
||||
enum GPS_Type {
|
||||
GPS_TYPE_NONE = 0,
|
||||
@ -122,6 +136,10 @@ public:
|
||||
return num_instances;
|
||||
}
|
||||
|
||||
uint8_t primary_sensor(void) const {
|
||||
return primary_instance;
|
||||
}
|
||||
|
||||
// using these macros saves some code space on APM2
|
||||
#if GPS_MAX_INSTANCES == 1
|
||||
# define _GPS_STATE(instance) state[0]
|
||||
@ -139,6 +157,10 @@ public:
|
||||
return status(primary_instance);
|
||||
}
|
||||
|
||||
// Query the highest status this GPS supports
|
||||
GPS_Status highest_supported_status(uint8_t instance) const;
|
||||
GPS_Status highest_supported_status(void) const;
|
||||
|
||||
// location of last fix
|
||||
const Location &location(uint8_t instance) const {
|
||||
return _GPS_STATE(instance).location;
|
||||
@ -256,7 +278,11 @@ public:
|
||||
// configuration parameters
|
||||
AP_Int8 _type[GPS_MAX_INSTANCES];
|
||||
AP_Int8 _navfilter;
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
AP_Int8 _auto_switch;
|
||||
AP_Int8 _min_dgps;
|
||||
#endif
|
||||
|
||||
// handle sending of initialisation strings to the GPS
|
||||
void send_blob_start(uint8_t instance, const prog_char *_blob, uint16_t size);
|
||||
void send_blob_update(uint8_t instance);
|
||||
@ -264,6 +290,19 @@ public:
|
||||
// lock out a GPS port, allowing another application to use the port
|
||||
void lock_port(uint8_t instance, bool locked);
|
||||
|
||||
//MAVLink Status Sending
|
||||
void send_mavlink_gps_raw(mavlink_channel_t chan);
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
void send_mavlink_gps2_raw(mavlink_channel_t chan);
|
||||
#endif
|
||||
|
||||
#if GPS_RTK_AVAILABLE
|
||||
void send_mavlink_gps_rtk(mavlink_channel_t chan);
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
void send_mavlink_gps2_rtk(mavlink_channel_t chan);
|
||||
#endif
|
||||
#endif
|
||||
|
||||
private:
|
||||
struct GPS_timing {
|
||||
// the time we got our last fix in system milliseconds
|
||||
@ -295,7 +334,7 @@ private:
|
||||
struct MTK19_detect_state mtk19_detect_state;
|
||||
struct SIRF_detect_state sirf_detect_state;
|
||||
struct NMEA_detect_state nmea_detect_state;
|
||||
#if HAL_CPU_CLASS >= HAL_CPU_CLASS_75
|
||||
#if GPS_RTK_AVAILABLE
|
||||
struct SBP_detect_state sbp_detect_state;
|
||||
#endif
|
||||
} detect_state[GPS_MAX_INSTANCES];
|
||||
|
@ -18,17 +18,25 @@
|
||||
// Swift Navigation GPS driver for ArduPilot
|
||||
// Origin code by Niels Joubert njoubert.com
|
||||
//
|
||||
|
||||
|
||||
#include <AP_GPS.h>
|
||||
#include "AP_GPS_SBP.h"
|
||||
#include <DataFlash.h>
|
||||
|
||||
#if GPS_RTK_AVAILABLE
|
||||
|
||||
#define SBP_DEBUGGING 0
|
||||
#define SBP_FAKE_3DLOCK 0
|
||||
|
||||
extern const AP_HAL::HAL& hal;
|
||||
|
||||
#define SBP_MILLIS_BETWEEN_HEALTHCHECKS 1500
|
||||
#define SBP_MILLIS_BETWEEN_HEALTHCHECKS 2000U
|
||||
#define SBP_BASELINE_TIMEOUT_MS 1000U
|
||||
#define SBP_FIX_TIMEOUT_MS 1000U
|
||||
#define SBP_HEARTBEAT_TIMEOUT_MS 5000U
|
||||
|
||||
#define SBP_MILLIS_BETWEEN_TRACKING_LOG 1800U
|
||||
|
||||
#define SBP_DEBUGGING 0
|
||||
#if SBP_DEBUGGING
|
||||
@ -41,7 +49,7 @@ extern const AP_HAL::HAL& hal;
|
||||
only do detailed hardware logging on boards likely to have more log
|
||||
storage space
|
||||
*/
|
||||
#if HAL_CPU_CLASS >= HAL_CPU_CLASS_75
|
||||
#if GPS_RTK_AVAILABLE
|
||||
#define SBP_HW_LOGGING 1
|
||||
#else
|
||||
#define SBP_HW_LOGGING 0
|
||||
@ -51,91 +59,311 @@ bool AP_GPS_SBP::logging_started = false;
|
||||
|
||||
AP_GPS_SBP::AP_GPS_SBP(AP_GPS &_gps, AP_GPS::GPS_State &_state, AP_HAL::UARTDriver *_port) :
|
||||
AP_GPS_Backend(_gps, _state, _port),
|
||||
has_updated_pos(false),
|
||||
has_updated_vel(false),
|
||||
|
||||
last_baseline_received_ms(0),
|
||||
last_heatbeat_received_ms(0),
|
||||
last_tracking_state_ms(0),
|
||||
iar_num_hypotheses(-1),
|
||||
|
||||
dgps_corrections_incoming(false),
|
||||
rtk_corrections_incoming(false),
|
||||
|
||||
has_new_pos_llh(false),
|
||||
has_new_vel_ned(false),
|
||||
has_new_baseline_ecef(false),
|
||||
|
||||
has_rtk_base_pos(false),
|
||||
|
||||
pos_msg_counter(0),
|
||||
vel_msg_counter(0),
|
||||
dops_msg_counter(0),
|
||||
baseline_msg_counter(0),
|
||||
full_update_counter(0),
|
||||
crc_error_counter(0),
|
||||
last_healthcheck_millis(0)
|
||||
{
|
||||
|
||||
Debug("Initializing SBP Driver");
|
||||
|
||||
port->begin(115200, 256, 16);
|
||||
port->flush();
|
||||
|
||||
parser_state.state = sbp_parser_state_t::WAITING;
|
||||
|
||||
state.status = AP_GPS::NO_FIX;
|
||||
state.have_vertical_velocity = true;
|
||||
|
||||
state.last_gps_time_ms = last_heatbeat_received_ms = last_healthcheck_millis = hal.scheduler->millis();
|
||||
|
||||
}
|
||||
|
||||
bool
|
||||
AP_GPS_SBP::can_calculate_base_pos(void)
|
||||
{
|
||||
return (rtk_corrections_incoming && !has_rtk_base_pos);
|
||||
};
|
||||
|
||||
|
||||
void
|
||||
AP_GPS_SBP::calculate_base_pos(void)
|
||||
{
|
||||
//INVARIANT:
|
||||
// Only ever capture home with motors not armed!
|
||||
// External driver checks whether can_raise_fix_level becomes true
|
||||
// and only if it can, AND motors are not armed, will be capture home!
|
||||
if (state.status < AP_GPS::GPS_OK_FIX_3D) {
|
||||
Debug("Attempting to capture home without GPS Fix available. Can't do RTK without home lat-lon.");
|
||||
return;
|
||||
}
|
||||
|
||||
if (!rtk_corrections_incoming) {
|
||||
Debug("Attempting to capture home baseline without rtk corrections being received.");
|
||||
return;
|
||||
}
|
||||
|
||||
Vector3d current_llh;
|
||||
Vector3d current_ecef;
|
||||
Vector3d current_baseline_ecef;
|
||||
|
||||
current_llh[0] = last_sbp_pos_llh_msg.lat * DEG_TO_RAD_DOUBLE;
|
||||
current_llh[1] = last_sbp_pos_llh_msg.lon * DEG_TO_RAD_DOUBLE;
|
||||
current_llh[2] = last_sbp_pos_llh_msg.height;
|
||||
|
||||
wgsllh2ecef(current_llh, current_ecef);
|
||||
|
||||
current_baseline_ecef[0] = ((double)last_sbp_baseline_ecef_msg.x) / 1000.0;
|
||||
current_baseline_ecef[1] = ((double)last_sbp_baseline_ecef_msg.y) / 1000.0;
|
||||
current_baseline_ecef[2] = ((double)last_sbp_baseline_ecef_msg.z) / 1000.0;
|
||||
|
||||
base_pos_ecef = current_ecef - current_baseline_ecef;
|
||||
has_rtk_base_pos = true;
|
||||
|
||||
Debug("SBP Got Base Position! has_rtk_base_pos=%d, (%.2f, %.2f, %.2f)", has_rtk_base_pos,
|
||||
base_pos_ecef[0],
|
||||
base_pos_ecef[1],
|
||||
base_pos_ecef[2]);
|
||||
}
|
||||
void
|
||||
AP_GPS_SBP::invalidate_base_pos()
|
||||
{
|
||||
has_rtk_base_pos = false;
|
||||
}
|
||||
|
||||
bool
|
||||
AP_GPS_SBP::read(void)
|
||||
{
|
||||
//Invariant: Calling this function processes *all* data current in the UART buffer.
|
||||
//
|
||||
//IMPORTANT NOTICE: This function is NOT CALLED for several seconds
|
||||
// during arming. That should not cause the driver to die. Process *all* waiting messages
|
||||
|
||||
//First we process all data waiting for the queue.
|
||||
sbp_process();
|
||||
bool full_update = false;
|
||||
do {
|
||||
//Attempt to process one message at a time
|
||||
bool new_message = sbp_process();
|
||||
|
||||
//Attempt to update our internal state with this new message.
|
||||
if (update_state(new_message)) {
|
||||
full_update = true;
|
||||
full_update_counter += 1;
|
||||
}
|
||||
|
||||
} while (port->available() > 0);
|
||||
|
||||
uint32_t now = hal.scheduler->millis();
|
||||
uint32_t elapsed = now - last_healthcheck_millis;
|
||||
if (elapsed > SBP_MILLIS_BETWEEN_HEALTHCHECKS) {
|
||||
last_healthcheck_millis = now;
|
||||
|
||||
#if SBP_DEBUGGING || SBP_HW_LOGGING
|
||||
float pos_msg_hz = pos_msg_counter / (float) elapsed * 1000.0;
|
||||
float vel_msg_hz = vel_msg_counter / (float) elapsed * 1000.0;
|
||||
float dops_msg_hz = dops_msg_counter / (float) elapsed * 1000.0;
|
||||
float baseline_msg_hz = baseline_msg_counter / (float) elapsed * 1000.0;
|
||||
float crc_error_hz = crc_error_counter / (float) elapsed * 1000.0;
|
||||
float pos_msg_hz = pos_msg_counter / (float) elapsed * 1000;
|
||||
float vel_msg_hz = vel_msg_counter / (float) elapsed * 1000;
|
||||
float baseline_msg_hz = baseline_msg_counter / (float) elapsed * 1000;
|
||||
float full_update_hz = full_update_counter / (float) elapsed * 1000;
|
||||
|
||||
pos_msg_counter = 0;
|
||||
vel_msg_counter = 0;
|
||||
dops_msg_counter = 0;
|
||||
baseline_msg_counter = 0;
|
||||
crc_error_counter = 0;
|
||||
full_update_counter = 0;
|
||||
|
||||
Debug("SBP GPS perf: CRC=(%.2fHz) Pos=(%.2fHz) Vel=(%.2fHz) Dops=(%.2fHz) Baseline=(%.2fHz)\n",
|
||||
crc_error_hz,
|
||||
Debug("SBP GPS perf: Fix=(%d) CRC=(%d) Pos=(%.2fHz) Vel=(%.2fHz) Baseline=(%.2fHz) Update=(%.2fHz) DGPS=(%d) RTK=(%d) RTK_HOME=(%d) IAR=(%d)",
|
||||
state.status,
|
||||
crc_error_counter,
|
||||
pos_msg_hz,
|
||||
vel_msg_hz,
|
||||
dops_msg_hz,
|
||||
baseline_msg_hz);
|
||||
baseline_msg_hz,
|
||||
full_update_hz,
|
||||
dgps_corrections_incoming,
|
||||
rtk_corrections_incoming,
|
||||
has_rtk_base_pos,
|
||||
iar_num_hypotheses);
|
||||
|
||||
#if SBP_HW_LOGGING
|
||||
logging_log_health(pos_msg_hz,
|
||||
vel_msg_hz,
|
||||
dops_msg_hz,
|
||||
baseline_msg_hz,
|
||||
crc_error_hz);
|
||||
full_update_hz,
|
||||
crc_error_counter,
|
||||
dgps_corrections_incoming,
|
||||
rtk_corrections_incoming,
|
||||
has_rtk_base_pos,
|
||||
iar_num_hypotheses);
|
||||
|
||||
#endif
|
||||
#endif
|
||||
|
||||
}
|
||||
|
||||
//Now we check whether we've done a full update - is all the sticky bits set?
|
||||
if (has_updated_pos && has_updated_vel) {
|
||||
state.status = AP_GPS::GPS_OK_FIX_3D;
|
||||
has_updated_pos = false;
|
||||
has_updated_vel = false;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
return full_update;
|
||||
}
|
||||
|
||||
//This attempts to read all the SBP messages from the incoming port.
|
||||
void
|
||||
AP_GPS_SBP::sbp_process()
|
||||
//This consolidates all the latest messages,
|
||||
//and the current mode the driver is in
|
||||
//
|
||||
// INVARIANT:
|
||||
// If in a fix mode >= 3,
|
||||
// returns true only if a full position and velocity update happened.
|
||||
// If in fix mode 0 or 1,
|
||||
// returns true if messages are being received or we haven't timed out
|
||||
bool
|
||||
AP_GPS_SBP::update_state(bool has_new_message)
|
||||
{
|
||||
|
||||
while (port->available() > 0) {
|
||||
uint32_t now = hal.scheduler->millis();
|
||||
|
||||
//Determine the current mode the GPS is in: DGPS or plain
|
||||
//Notice that this is sticky.
|
||||
if (has_new_baseline_ecef && (now - last_baseline_received_ms < SBP_BASELINE_TIMEOUT_MS)) {
|
||||
dgps_corrections_incoming = true;
|
||||
if (gps._min_dgps >= 100) {
|
||||
//Allow only IntegerRTK baselines
|
||||
rtk_corrections_incoming = dgps_corrections_incoming && (last_sbp_baseline_ecef_msg.flags & 0x1);
|
||||
} else {
|
||||
//Allow floatRTK baselines
|
||||
rtk_corrections_incoming = dgps_corrections_incoming;
|
||||
}
|
||||
}
|
||||
|
||||
//Currently we only use relative positioning if we have RTK-level fixes,
|
||||
//we ignore float-level fixes
|
||||
bool using_relative_positioning = rtk_corrections_incoming && has_rtk_base_pos;
|
||||
|
||||
//Drop out of RTK mode if we haven't seen a baseline for a while...
|
||||
if (using_relative_positioning && (now - last_baseline_received_ms > SBP_BASELINE_TIMEOUT_MS)) {
|
||||
dgps_corrections_incoming = false;
|
||||
rtk_corrections_incoming = false;
|
||||
using_relative_positioning = false;
|
||||
}
|
||||
|
||||
//UPDATE POSITION AND VELOCITY
|
||||
if (!using_relative_positioning &&
|
||||
(has_new_pos_llh && has_new_vel_ned) &&
|
||||
(last_sbp_pos_llh_msg.tow == last_sbp_vel_ned_msg.tow)) {
|
||||
|
||||
state.last_gps_time_ms = hal.scheduler->millis();
|
||||
|
||||
state.time_week_ms = last_sbp_pos_llh_msg.tow;
|
||||
state.location.lat = (int32_t) (last_sbp_pos_llh_msg.lat*1e7);
|
||||
state.location.lng = (int32_t) (last_sbp_pos_llh_msg.lon*1e7);
|
||||
state.location.alt = (int32_t) (last_sbp_pos_llh_msg.height*1e2);
|
||||
state.num_sats = last_sbp_pos_llh_msg.n_sats;
|
||||
|
||||
update_state_velocity();
|
||||
|
||||
has_new_pos_llh = false;
|
||||
|
||||
state.status = AP_GPS::GPS_OK_FIX_3D;
|
||||
|
||||
return true;
|
||||
|
||||
} else if (using_relative_positioning &&
|
||||
(has_new_baseline_ecef && has_new_vel_ned) &&
|
||||
(last_sbp_baseline_ecef_msg.tow == last_sbp_vel_ned_msg.tow)) {
|
||||
|
||||
state.last_gps_time_ms = hal.scheduler->millis();
|
||||
|
||||
//Generate a new lat-lon from baseline
|
||||
|
||||
//Grab the current baseline
|
||||
Vector3d current_baseline_ecef; //units are currently in mm
|
||||
current_baseline_ecef[0] = ((double)last_sbp_baseline_ecef_msg.x) / 1000.0;
|
||||
current_baseline_ecef[1] = ((double)last_sbp_baseline_ecef_msg.y) / 1000.0;
|
||||
current_baseline_ecef[2] = ((double)last_sbp_baseline_ecef_msg.z) / 1000.0;
|
||||
|
||||
//Offset the reference point from that
|
||||
Vector3d current_pos_ecef;
|
||||
current_pos_ecef = base_pos_ecef + current_baseline_ecef;
|
||||
|
||||
Vector3d current_pos_llh;
|
||||
wgsecef2llh(current_pos_ecef, current_pos_llh);
|
||||
|
||||
current_pos_llh[0] *= RAD_TO_DEG_DOUBLE;
|
||||
current_pos_llh[1] *= RAD_TO_DEG_DOUBLE;
|
||||
|
||||
state.time_week_ms = last_sbp_baseline_ecef_msg.tow;
|
||||
state.location.lat = (int32_t) (current_pos_llh[0] * 1e7);
|
||||
state.location.lng = (int32_t) (current_pos_llh[1] * 1e7);
|
||||
state.location.alt = (int32_t) (current_pos_llh[2] * 1e3);
|
||||
state.num_sats = last_sbp_baseline_ecef_msg.n_sats;
|
||||
|
||||
update_state_velocity();
|
||||
|
||||
has_new_baseline_ecef = false;
|
||||
|
||||
state.status = AP_GPS::GPS_OK_FIX_3D_RTK;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
//If we get here,
|
||||
//We have not been able to update the GPS state yet for this process call.
|
||||
//Check whether the GPS is still alive and processing messages!
|
||||
|
||||
if (!using_relative_positioning && (now - state.last_gps_time_ms > SBP_FIX_TIMEOUT_MS)) {
|
||||
state.status = AP_GPS::NO_FIX;
|
||||
return (now - last_heatbeat_received_ms < SBP_HEARTBEAT_TIMEOUT_MS);
|
||||
}
|
||||
|
||||
if (now - last_heatbeat_received_ms > SBP_HEARTBEAT_TIMEOUT_MS) {
|
||||
state.status = AP_GPS::NO_GPS;
|
||||
return false;
|
||||
}
|
||||
|
||||
if (state.status < AP_GPS::GPS_OK_FIX_3D) {
|
||||
//If we are receiving messages, but dont have a fix yet, thats okay.
|
||||
return has_new_message;
|
||||
} else {
|
||||
//If we have a fix and we got here, then we're in between message synchronizations
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::update_state_velocity(void)
|
||||
{
|
||||
|
||||
state.time_week_ms = last_sbp_vel_ned_msg.tow;
|
||||
state.velocity[0] = (float)(last_sbp_vel_ned_msg.n / 1000.0);
|
||||
state.velocity[1] = (float)(last_sbp_vel_ned_msg.e / 1000.0);
|
||||
state.velocity[2] = (float)(last_sbp_vel_ned_msg.d / 1000.0);
|
||||
|
||||
float ground_vector_sq = state.velocity[0]*state.velocity[0] + state.velocity[1]*state.velocity[1];
|
||||
state.ground_speed = safe_sqrt(ground_vector_sq);
|
||||
|
||||
state.ground_course_cd = (int32_t) 100*ToDeg(atan2f(state.velocity[1], state.velocity[0]));
|
||||
if (state.ground_course_cd < 0) {
|
||||
state.ground_course_cd += 36000;
|
||||
}
|
||||
|
||||
has_new_vel_ned = false;
|
||||
|
||||
}
|
||||
|
||||
//This attempts to read a SINGLE SBP messages from the incoming port.
|
||||
//Returns true if a new message was read, false if we failed to read a message.
|
||||
bool
|
||||
AP_GPS_SBP::sbp_process()
|
||||
{
|
||||
while (port->available() > 0) {
|
||||
uint8_t temp = port->read();
|
||||
uint16_t crc;
|
||||
|
||||
|
||||
//This switch reads one character at a time,
|
||||
//parsing it into buffers until a full message is dispatched
|
||||
switch(parser_state.state) {
|
||||
@ -190,15 +418,8 @@ AP_GPS_SBP::sbp_process()
|
||||
crc = crc16_ccitt(&(parser_state.msg_len), 1, crc);
|
||||
crc = crc16_ccitt(parser_state.msg_buff, parser_state.msg_len, crc);
|
||||
if (parser_state.crc == crc) {
|
||||
|
||||
//OK, we have a valid message. Dispatch the appropriate function:
|
||||
switch(parser_state.msg_type) {
|
||||
case SBP_GPS_TIME_MSGTYPE:
|
||||
sbp_process_gpstime(parser_state.msg_buff);
|
||||
break;
|
||||
case SBP_DOPS_MSGTYPE:
|
||||
sbp_process_dops(parser_state.msg_buff);
|
||||
break;
|
||||
case SBP_POS_ECEF_MSGTYPE:
|
||||
sbp_process_pos_ecef(parser_state.msg_buff);
|
||||
break;
|
||||
@ -217,12 +438,29 @@ AP_GPS_SBP::sbp_process()
|
||||
case SBP_VEL_NED_MSGTYPE:
|
||||
sbp_process_vel_ned(parser_state.msg_buff);
|
||||
break;
|
||||
default:
|
||||
Debug("Unknown message received: msg_type=0x%x", parser_state.msg_type);
|
||||
case SBP_GPS_TIME_MSGTYPE:
|
||||
sbp_process_gpstime(parser_state.msg_buff);
|
||||
break;
|
||||
case SBP_DOPS_MSGTYPE:
|
||||
sbp_process_dops(parser_state.msg_buff);
|
||||
break;
|
||||
case SBP_TRACKING_STATE_MSGTYPE:
|
||||
sbp_process_tracking_state(parser_state.msg_buff, parser_state.msg_len);
|
||||
break;
|
||||
case SBP_IAR_STATE_MSGTYPE:
|
||||
sbp_process_iar_state(parser_state.msg_buff);
|
||||
break;
|
||||
case SBP_HEARTBEAT_MSGTYPE:
|
||||
sbp_process_heartbeat(parser_state.msg_buff);
|
||||
break;
|
||||
case SBP_STARTUP_MSGTYPE:
|
||||
sbp_process_startup(parser_state.msg_buff);
|
||||
break;
|
||||
}
|
||||
return true;
|
||||
|
||||
} else {
|
||||
Debug("CRC Error Occurred!\n");
|
||||
Debug("CRC Error Occurred!");
|
||||
crc_error_counter += 1;
|
||||
}
|
||||
|
||||
@ -235,7 +473,13 @@ AP_GPS_SBP::sbp_process()
|
||||
}
|
||||
}
|
||||
//We have parsed all the waiting messages
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_heartbeat(uint8_t* msg)
|
||||
{
|
||||
last_heatbeat_received_ms = hal.scheduler->millis();
|
||||
}
|
||||
|
||||
void
|
||||
@ -244,7 +488,6 @@ AP_GPS_SBP::sbp_process_gpstime(uint8_t* msg)
|
||||
struct sbp_gps_time_t* t = (struct sbp_gps_time_t*)msg;
|
||||
state.time_week = t->wn;
|
||||
state.time_week_ms = t->tow;
|
||||
state.last_gps_time_ms = hal.scheduler->millis();
|
||||
}
|
||||
|
||||
void
|
||||
@ -252,76 +495,113 @@ AP_GPS_SBP::sbp_process_dops(uint8_t* msg)
|
||||
{
|
||||
struct sbp_dops_t* d = (struct sbp_dops_t*) msg;
|
||||
state.time_week_ms = d->tow;
|
||||
state.last_gps_time_ms = hal.scheduler->millis();
|
||||
state.hdop = d->hdop;
|
||||
dops_msg_counter += 1;
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_pos_ecef(uint8_t* msg)
|
||||
{
|
||||
//Ideally we'd like this data in LLH format, not ECEF
|
||||
//Using LLH, not ECEF
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_pos_llh(uint8_t* msg)
|
||||
{
|
||||
struct sbp_pos_llh_t* pos = (struct sbp_pos_llh_t*)msg;
|
||||
state.time_week_ms = pos->tow;
|
||||
state.last_gps_time_ms = hal.scheduler->millis();
|
||||
state.location.lat = (int32_t) (pos->lat*1e7);
|
||||
state.location.lng = (int32_t) (pos->lon*1e7);
|
||||
state.location.alt = (int32_t) (pos->height*1e2);
|
||||
state.num_sats = pos->n_sats;
|
||||
last_sbp_pos_llh_msg = *pos;
|
||||
|
||||
has_new_pos_llh = true;
|
||||
|
||||
#if SBP_DEBUGGING || SBP_HW_LOGGING
|
||||
pos_msg_counter += 1;
|
||||
has_updated_pos = true;
|
||||
#endif
|
||||
|
||||
#if SBP_HW_LOGGING
|
||||
logging_log_llh(pos);
|
||||
#endif
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_baseline_ecef(uint8_t* msg)
|
||||
{
|
||||
struct sbp_baseline_ecef_t* b = (struct sbp_baseline_ecef_t*)msg;
|
||||
last_sbp_baseline_ecef_msg = *b;
|
||||
|
||||
last_baseline_received_ms = hal.scheduler->millis();
|
||||
has_new_baseline_ecef = true;
|
||||
|
||||
#if SBP_DEBUGGING || SBP_HW_LOGGING
|
||||
baseline_msg_counter += 1;
|
||||
#endif
|
||||
|
||||
#if SBP_HW_LOGGING
|
||||
logging_log_baseline(b);
|
||||
logging_log_baseline_ecef(b);
|
||||
#endif
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_baseline_ned(uint8_t* msg)
|
||||
{
|
||||
//Ideally we'd like this data in ECEF format, not NED
|
||||
//Currently we use ECEF baselines.
|
||||
//This is just for logging purposes.
|
||||
struct sbp_baseline_ned_t* b = (struct sbp_baseline_ned_t*)msg;
|
||||
last_sbp_baseline_ned_msg = *b;
|
||||
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_vel_ecef(uint8_t* msg)
|
||||
{
|
||||
//Ideally we'd like this data in NED format, not ECEF
|
||||
//Currently we use NED velocity.
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_vel_ned(uint8_t* msg)
|
||||
{
|
||||
struct sbp_vel_ned_t* vel = (struct sbp_vel_ned_t*)msg;
|
||||
state.time_week_ms = vel->tow;
|
||||
state.last_gps_time_ms = hal.scheduler->millis();
|
||||
state.velocity[0] = (float)vel->n / 1000.0;
|
||||
state.velocity[1] = (float)vel->e / 1000.0;
|
||||
state.velocity[2] = (float)vel->d / 1000.0;
|
||||
state.num_sats = vel->n_sats;
|
||||
last_sbp_vel_ned_msg = *vel;
|
||||
|
||||
float ground_vector_sq = state.velocity[0]*state.velocity[0] + state.velocity[1]*state.velocity[1];
|
||||
state.ground_speed = safe_sqrt(ground_vector_sq);
|
||||
has_new_vel_ned = true;
|
||||
|
||||
#if SBP_DEBUGGING || SBP_HW_LOGGING
|
||||
vel_msg_counter += 1;
|
||||
#endif
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_tracking_state(uint8_t* msg, uint8_t len)
|
||||
{
|
||||
uint32_t now = hal.scheduler->millis();
|
||||
|
||||
struct sbp_tracking_state_t* tracking_state = (struct sbp_tracking_state_t*)msg;
|
||||
last_sbp_tracking_state_msg = *tracking_state;
|
||||
|
||||
uint8_t num = len / sizeof(sbp_tracking_state_t);
|
||||
last_sbp_tracking_state_msg_num = num;
|
||||
|
||||
//Rate-limit the tracking state messages to no more than 1.8 seconds
|
||||
if (now - last_tracking_state_ms > SBP_MILLIS_BETWEEN_TRACKING_LOG) {
|
||||
last_tracking_state_ms = now;
|
||||
|
||||
#ifdef SBP_HW_LOGGING
|
||||
logging_log_tracking_state(tracking_state, num);
|
||||
#endif
|
||||
|
||||
state.ground_course_cd = (int32_t) 100*ToDeg(atan2f(state.velocity[1], state.velocity[0]));
|
||||
if (state.ground_course_cd < 0) {
|
||||
state.ground_course_cd += 36000;
|
||||
}
|
||||
|
||||
vel_msg_counter += 1;
|
||||
has_updated_vel = true;
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_iar_state(uint8_t* msg)
|
||||
{
|
||||
struct sbp_iar_state_t* iar_state = (struct sbp_iar_state_t*)msg;
|
||||
iar_num_hypotheses = (int32_t) iar_state->num_hypotheses;
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::sbp_process_startup(uint8_t* msg)
|
||||
{
|
||||
invalidate_base_pos();
|
||||
}
|
||||
|
||||
bool
|
||||
@ -390,42 +670,160 @@ AP_GPS_SBP::_detect(struct SBP_detect_state &state, uint8_t data)
|
||||
return false;
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SBP::send_mavlink_gps_rtk(mavlink_channel_t chan)
|
||||
{
|
||||
|
||||
uint8_t health = dgps_corrections_incoming |
|
||||
(rtk_corrections_incoming << 1) |
|
||||
(has_rtk_base_pos << 2);
|
||||
|
||||
mavlink_msg_gps_rtk_send(
|
||||
chan,
|
||||
last_baseline_received_ms, // Time since boot of last baseline message received in ms.
|
||||
AP_GPS::GPS_TYPE_SBP, // Identification of connected RTK receiver.
|
||||
state.time_week, // GPS Week Number of last baseline
|
||||
last_sbp_baseline_ned_msg.tow, // GPS Time of Week of last baseline
|
||||
health, // GPS-specific health report for RTK data.
|
||||
baseline_recv_rate, // Rate of baseline messages being received by GPS, in HZ*10
|
||||
last_sbp_baseline_ned_msg.n_sats, // Current number of sats used for RTK calculation.
|
||||
1, // Coordinate system of baseline. 0 == ECEF, 1 == NED
|
||||
last_sbp_baseline_ned_msg.n, // Current baseline in ECEF x or NED north component in mm
|
||||
last_sbp_baseline_ned_msg.e, // Current baseline in ECEF y or NED east component in mm
|
||||
last_sbp_baseline_ned_msg.d, // Current baseline in ECEF z or NED down component in mm
|
||||
last_sbp_baseline_ned_msg.h_accuracy, // Current estimate of baseline accuracy.
|
||||
iar_num_hypotheses // Current number of integer ambiguity hypotheses.
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
void
|
||||
AP_GPS_SBP::send_mavlink_gps2_rtk(mavlink_channel_t chan)
|
||||
{
|
||||
|
||||
uint8_t health = dgps_corrections_incoming |
|
||||
(rtk_corrections_incoming << 1) |
|
||||
(has_rtk_base_pos << 2);
|
||||
|
||||
mavlink_msg_gps2_rtk_send(
|
||||
chan,
|
||||
last_baseline_received_ms, // Time since boot of last baseline message received in ms.
|
||||
AP_GPS::GPS_TYPE_SBP, // Identification of connected RTK receiver.
|
||||
state.time_week, // GPS Week Number of last baseline
|
||||
last_sbp_baseline_ned_msg.tow, // GPS Time of Week of last baseline
|
||||
health, // GPS-specific health report for RTK data.
|
||||
baseline_recv_rate, // Rate of baseline messages being received by GPS, in HZ*10
|
||||
last_sbp_baseline_ned_msg.n_sats, // Current number of sats used for RTK calculation.
|
||||
1, // Coordinate system of baseline. 0 == ECEF, 1 == NED
|
||||
last_sbp_baseline_ned_msg.n, // Current baseline in ECEF x or NED north component in mm
|
||||
last_sbp_baseline_ned_msg.e, // Current baseline in ECEF y or NED east component in mm
|
||||
last_sbp_baseline_ned_msg.d, // Current baseline in ECEF z or NED down component in mm
|
||||
last_sbp_baseline_ned_msg.h_accuracy, // Current estimate of baseline accuracy.
|
||||
iar_num_hypotheses // Current number of integer ambiguity hypotheses.
|
||||
);
|
||||
}
|
||||
#endif
|
||||
|
||||
#if SBP_HW_LOGGING
|
||||
|
||||
#define LOG_MSG_SBPHEALTH 202
|
||||
#define LOG_MSG_SBPBASELINE 203
|
||||
#define LOG_MSG_SBPLLH 203
|
||||
#define LOG_MSG_SBPBASELINE 204
|
||||
#define LOG_MSG_SBPTRACKING1 205
|
||||
#define LOG_MSG_SBPTRACKING2 206
|
||||
|
||||
struct PACKED log_SbpHealth {
|
||||
LOG_PACKET_HEADER;
|
||||
uint32_t timestamp;
|
||||
float pos_msg_hz;
|
||||
float vel_msg_hz;
|
||||
float dops_msg_hz;
|
||||
float baseline_msg_hz;
|
||||
float crc_error_hz;
|
||||
float full_update_hz;
|
||||
uint32_t crc_error_counter;
|
||||
uint8_t dgps_corrections_incoming;
|
||||
uint8_t rtk_corrections_incoming;
|
||||
uint8_t has_rtk_base_pos;
|
||||
int32_t iar_num_hypotheses;
|
||||
};
|
||||
|
||||
struct PACKED log_SbpLLH {
|
||||
LOG_PACKET_HEADER;
|
||||
uint32_t timestamp;
|
||||
uint32_t tow;
|
||||
int32_t lat;
|
||||
int32_t lon;
|
||||
int32_t alt;
|
||||
uint8_t n_sats;
|
||||
};
|
||||
|
||||
struct PACKED log_SbpBaseline {
|
||||
LOG_PACKET_HEADER;
|
||||
uint32_t timestamp;
|
||||
uint32_t tow;
|
||||
int32_t baseline_x;
|
||||
int32_t baseline_y;
|
||||
int32_t baseline_z;
|
||||
uint16_t baseline_accuracy;
|
||||
uint8_t num_sats;
|
||||
uint8_t flags;
|
||||
uint32_t tow; //< GPS Time of Week of ECEF Baseline (unit: ms)
|
||||
int32_t x; //< Baseline ECEF X coordinate
|
||||
int32_t y; //< Baseline ECEF Y coordinate
|
||||
int32_t z; //< Baseline ECEF Z coordinate
|
||||
int32_t length; //< Baseline length
|
||||
uint16_t accuracy; //< Horizontal position accuracy estimate (unit: mm)
|
||||
uint8_t n_sats; //< Number of satellites used in solution
|
||||
uint8_t flags; //< Status flags (reserved)
|
||||
};
|
||||
|
||||
struct PACKED log_SbpTracking1 {
|
||||
LOG_PACKET_HEADER;
|
||||
uint32_t timestamp;
|
||||
uint8_t ch1_prn;
|
||||
float ch1_cn0;
|
||||
uint8_t ch2_prn;
|
||||
float ch2_cn0;
|
||||
uint8_t ch3_prn;
|
||||
float ch3_cn0;
|
||||
uint8_t ch4_prn;
|
||||
float ch4_cn0;
|
||||
uint8_t ch5_prn;
|
||||
float ch5_cn0;
|
||||
uint8_t ch6_prn;
|
||||
float ch6_cn0;
|
||||
uint8_t ch7_prn;
|
||||
float ch7_cn0;
|
||||
};
|
||||
|
||||
struct PACKED log_SbpTracking2 {
|
||||
LOG_PACKET_HEADER;
|
||||
uint32_t timestamp;
|
||||
uint8_t ch8_prn;
|
||||
float ch8_cn0;
|
||||
uint8_t ch9_prn;
|
||||
float ch9_cn0;
|
||||
uint8_t ch10_prn;
|
||||
float ch10_cn0;
|
||||
uint8_t ch11_prn;
|
||||
float ch11_cn0;
|
||||
uint8_t ch12_prn;
|
||||
float ch12_cn0;
|
||||
uint8_t ch13_prn;
|
||||
float ch13_cn0;
|
||||
uint8_t ch14_prn;
|
||||
float ch14_cn0;
|
||||
};
|
||||
|
||||
static const struct LogStructure sbp_log_structures[] PROGMEM = {
|
||||
{ LOG_MSG_SBPHEALTH, sizeof(log_SbpHealth),
|
||||
"SBPH", "Ifffff", "TimeMS,PosHz,VelHz,DopsHz,BaseHz,CrcHz" },
|
||||
"SBPH", "IffffIBBBi", "TimeMS,PHz,VHz,BHz,UpHz,CrcError,dgpsOn,rtkOn,hasRtkBase,IAR" },
|
||||
{ LOG_MSG_SBPLLH, sizeof(log_SbpLLH),
|
||||
"SBPL", "IIiiiB", "TimeMS,tow,lat,lon,alt,num_sats" },
|
||||
{ LOG_MSG_SBPBASELINE, sizeof(log_SbpBaseline),
|
||||
"SBPB", "IIiiiHBB", "TimeMS,tow,bx,by,bz,bacc,num_sats,flags" }
|
||||
"SBPB", "IIiiiiHBB", "TimeMS,tow,x,y,z,len,acc,num_sats,flags" },
|
||||
{ LOG_MSG_SBPTRACKING1, sizeof(log_SbpTracking1),
|
||||
"SBT1", "IBfBfBfBfBfBfBf", "TimeMS,s1,c1,s2,c2,s3,c3,s4,c4,s5,c5,s6,c6,s7,c7" },
|
||||
{ LOG_MSG_SBPTRACKING2, sizeof(log_SbpTracking2),
|
||||
"SBT2", "IBfBfBfBfBfBfBF", "TimeMS,s8,c8,s9,c9,s10,c10,s11,c11,s12,c12,s13,c13,s14,c14" }
|
||||
|
||||
};
|
||||
|
||||
void AP_GPS_SBP::logging_write_headers(void)
|
||||
void
|
||||
AP_GPS_SBP::logging_write_headers(void)
|
||||
{
|
||||
if (!logging_started) {
|
||||
logging_started = true;
|
||||
@ -433,7 +831,10 @@ void AP_GPS_SBP::logging_write_headers(void)
|
||||
}
|
||||
}
|
||||
|
||||
void AP_GPS_SBP::logging_log_health(float pos_msg_hz, float vel_msg_hz, float dops_msg_hz, float baseline_msg_hz, float crc_error_hz)
|
||||
void
|
||||
AP_GPS_SBP::logging_log_health(float pos_msg_hz, float vel_msg_hz, float baseline_msg_hz, float full_update_hz,
|
||||
uint32_t crc_error_counter, bool dgps_corrections_incoming, bool rtk_corrections_incoming,
|
||||
bool has_rtk_base_pos, int32_t iar_num_hypotheses)
|
||||
{
|
||||
|
||||
if (gps._DataFlash == NULL || !gps._DataFlash->logging_started()) {
|
||||
@ -444,18 +845,23 @@ void AP_GPS_SBP::logging_log_health(float pos_msg_hz, float vel_msg_hz, float do
|
||||
|
||||
struct log_SbpHealth pkt = {
|
||||
LOG_PACKET_HEADER_INIT(LOG_MSG_SBPHEALTH),
|
||||
timestamp : hal.scheduler->millis(),
|
||||
pos_msg_hz : pos_msg_hz,
|
||||
vel_msg_hz : vel_msg_hz,
|
||||
dops_msg_hz : dops_msg_hz,
|
||||
baseline_msg_hz : baseline_msg_hz,
|
||||
crc_error_hz : crc_error_hz
|
||||
timestamp : hal.scheduler->millis(),
|
||||
pos_msg_hz : pos_msg_hz,
|
||||
vel_msg_hz : vel_msg_hz,
|
||||
baseline_msg_hz : baseline_msg_hz,
|
||||
full_update_hz : full_update_hz,
|
||||
crc_error_counter : crc_error_counter,
|
||||
dgps_corrections_incoming : dgps_corrections_incoming,
|
||||
rtk_corrections_incoming : rtk_corrections_incoming,
|
||||
has_rtk_base_pos : has_rtk_base_pos,
|
||||
iar_num_hypotheses : iar_num_hypotheses
|
||||
};
|
||||
gps._DataFlash->WriteBlock(&pkt, sizeof(pkt));
|
||||
|
||||
}
|
||||
};
|
||||
|
||||
void AP_GPS_SBP::logging_log_baseline(struct sbp_baseline_ecef_t* b)
|
||||
void
|
||||
AP_GPS_SBP::logging_log_llh(struct sbp_pos_llh_t* p)
|
||||
{
|
||||
|
||||
if (gps._DataFlash == NULL || !gps._DataFlash->logging_started()) {
|
||||
@ -464,19 +870,108 @@ void AP_GPS_SBP::logging_log_baseline(struct sbp_baseline_ecef_t* b)
|
||||
|
||||
logging_write_headers();
|
||||
|
||||
struct log_SbpBaseline pkt = {
|
||||
LOG_PACKET_HEADER_INIT(LOG_MSG_SBPBASELINE),
|
||||
timestamp : hal.scheduler->millis(),
|
||||
tow : b->tow,
|
||||
baseline_x : b->x,
|
||||
baseline_y : b->y,
|
||||
baseline_z : b->z,
|
||||
baseline_accuracy : b->accuracy,
|
||||
num_sats : b->n_sats,
|
||||
flags : b->flags
|
||||
struct log_SbpLLH pkt = {
|
||||
LOG_PACKET_HEADER_INIT(LOG_MSG_SBPLLH),
|
||||
timestamp : hal.scheduler->millis(),
|
||||
tow : p->tow,
|
||||
lat : (int32_t) (p->lat*1e7),
|
||||
lon : (int32_t) (p->lon*1e7),
|
||||
alt : (int32_t) (p->height*1e2),
|
||||
n_sats : p->n_sats,
|
||||
};
|
||||
gps._DataFlash->WriteBlock(&pkt, sizeof(pkt));
|
||||
|
||||
}
|
||||
};
|
||||
|
||||
void
|
||||
AP_GPS_SBP::logging_log_baseline_ecef(struct sbp_baseline_ecef_t* b)
|
||||
{
|
||||
|
||||
if (gps._DataFlash == NULL || !gps._DataFlash->logging_started()) {
|
||||
return;
|
||||
}
|
||||
|
||||
logging_write_headers();
|
||||
|
||||
float x = b->x / 1000.0;
|
||||
float y = b->y / 1000.0;
|
||||
float z = b->z / 1000.0;
|
||||
int32_t len = (int32_t) safe_sqrt(x*x+y*y+z*z) * 1000.0;
|
||||
|
||||
struct log_SbpBaseline pkt = {
|
||||
LOG_PACKET_HEADER_INIT(LOG_MSG_SBPBASELINE),
|
||||
timestamp : hal.scheduler->millis(),
|
||||
tow : b->tow,
|
||||
x : b->x,
|
||||
y : b->y,
|
||||
z : b->z,
|
||||
length : len,
|
||||
accuracy : b->accuracy,
|
||||
n_sats : b->n_sats,
|
||||
flags : b->flags
|
||||
};
|
||||
|
||||
gps._DataFlash->WriteBlock(&pkt, sizeof(pkt));
|
||||
|
||||
};
|
||||
|
||||
|
||||
void
|
||||
AP_GPS_SBP::logging_log_tracking_state(struct sbp_tracking_state_t* state, uint8_t num)
|
||||
{
|
||||
if (gps._DataFlash == NULL || !gps._DataFlash->logging_started()) {
|
||||
return;
|
||||
}
|
||||
|
||||
logging_write_headers();
|
||||
|
||||
struct log_SbpTracking1 pkt = {
|
||||
LOG_PACKET_HEADER_INIT(LOG_MSG_SBPTRACKING1),
|
||||
timestamp : hal.scheduler->millis(),
|
||||
ch1_prn : state[0].prn,
|
||||
ch1_cn0 : state[0].cn0,
|
||||
ch2_prn : num < 1 ? 0 : state[1].prn,
|
||||
ch2_cn0 : num < 1 ? 0 : state[1].cn0,
|
||||
ch3_prn : num < 2 ? 0 : state[2].prn,
|
||||
ch3_cn0 : num < 2 ? 0 : state[2].cn0,
|
||||
ch4_prn : num < 3 ? 0 : state[3].prn,
|
||||
ch4_cn0 : num < 3 ? 0 : state[3].cn0,
|
||||
ch5_prn : num < 4 ? 0 : state[4].prn,
|
||||
ch5_cn0 : num < 4 ? 0 : state[4].cn0,
|
||||
ch6_prn : num < 5 ? 0 : state[5].prn,
|
||||
ch6_cn0 : num < 5 ? 0 : state[5].cn0,
|
||||
ch7_prn : num < 6 ? 0 : state[6].prn,
|
||||
ch7_cn0 : num < 6 ? 0 : state[6].cn0,
|
||||
};
|
||||
gps._DataFlash->WriteBlock(&pkt, sizeof(pkt));
|
||||
|
||||
if (num > 6) {
|
||||
|
||||
struct log_SbpTracking2 pkt2 = {
|
||||
LOG_PACKET_HEADER_INIT(LOG_MSG_SBPTRACKING2),
|
||||
timestamp : hal.scheduler->millis(),
|
||||
ch8_prn : num < 7 ? 0 : state[7].prn,
|
||||
ch8_cn0 : num < 7 ? 0 : state[7].cn0,
|
||||
ch9_prn : num < 8 ? 0 : state[8].prn,
|
||||
ch9_cn0 : num < 8 ? 0 : state[8].cn0,
|
||||
ch10_prn : num < 9 ? 0 : state[9].prn,
|
||||
ch10_cn0 : num < 9 ? 0 : state[9].cn0,
|
||||
ch11_prn : num < 10 ? 0 : state[10].prn,
|
||||
ch11_cn0 : num < 10 ? 0 : state[10].cn0,
|
||||
ch12_prn : num < 11 ? 0 : state[11].prn,
|
||||
ch12_cn0 : num < 11 ? 0 : state[11].cn0,
|
||||
ch13_prn : num < 12 ? 0 : state[12].prn,
|
||||
ch13_cn0 : num < 12 ? 0 : state[12].cn0,
|
||||
ch14_prn : num < 13 ? 0 : state[13].prn,
|
||||
ch14_cn0 : num < 13 ? 0 : state[13].cn0,
|
||||
};
|
||||
gps._DataFlash->WriteBlock(&pkt2, sizeof(pkt));
|
||||
|
||||
};
|
||||
|
||||
|
||||
};
|
||||
|
||||
#endif // SBP_HW_LOGGING
|
||||
|
||||
#endif // GPS_RTK_AVAILABLE
|
||||
|
@ -23,6 +23,8 @@
|
||||
#ifndef __AP_GPS_SBP_H__
|
||||
#define __AP_GPS_SBP_H__
|
||||
|
||||
#if GPS_RTK_AVAILABLE
|
||||
|
||||
#include <AP_GPS.h>
|
||||
|
||||
|
||||
@ -48,11 +50,24 @@ class AP_GPS_SBP : public AP_GPS_Backend
|
||||
public:
|
||||
AP_GPS_SBP(AP_GPS &_gps, AP_GPS::GPS_State &_state, AP_HAL::UARTDriver *_port);
|
||||
|
||||
// Methods
|
||||
bool can_calculate_base_pos(void);
|
||||
|
||||
void calculate_base_pos(void);
|
||||
|
||||
void invalidate_base_pos(void);
|
||||
|
||||
AP_GPS::GPS_Status highest_supported_status(void) { return AP_GPS::GPS_OK_FIX_3D_RTK; }
|
||||
|
||||
bool read();
|
||||
|
||||
static bool _detect(struct SBP_detect_state &state, uint8_t data);
|
||||
|
||||
virtual void send_mavlink_gps_rtk(mavlink_channel_t chan);
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
virtual void send_mavlink_gps2_rtk(mavlink_channel_t chan);
|
||||
#endif
|
||||
|
||||
private:
|
||||
|
||||
// ************************************************************************
|
||||
@ -79,15 +94,19 @@ private:
|
||||
static const uint8_t SBP_PREAMBLE = 0x55;
|
||||
|
||||
//Message types supported by this driver
|
||||
static const uint16_t SBP_GPS_TIME_MSGTYPE = 0x0100;
|
||||
static const uint16_t SBP_DOPS_MSGTYPE = 0x0206;
|
||||
static const uint16_t SBP_POS_ECEF_MSGTYPE = 0x0200;
|
||||
static const uint16_t SBP_POS_LLH_MSGTYPE = 0x0201;
|
||||
static const uint16_t SBP_BASELINE_ECEF_MSGTYPE = 0x0202;
|
||||
static const uint16_t SBP_BASELINE_NED_MSGTYPE = 0x0203;
|
||||
static const uint16_t SBP_VEL_ECEF_MSGTYPE = 0x0204;
|
||||
static const uint16_t SBP_VEL_NED_MSGTYPE = 0x0205;
|
||||
|
||||
static const uint16_t SBP_STARTUP_MSGTYPE = 0xFF00;
|
||||
static const uint16_t SBP_HEARTBEAT_MSGTYPE = 0xFFFF;
|
||||
static const uint16_t SBP_GPS_TIME_MSGTYPE = 0x0100;
|
||||
static const uint16_t SBP_DOPS_MSGTYPE = 0x0206;
|
||||
static const uint16_t SBP_POS_ECEF_MSGTYPE = 0x0200;
|
||||
static const uint16_t SBP_POS_LLH_MSGTYPE = 0x0201;
|
||||
static const uint16_t SBP_BASELINE_ECEF_MSGTYPE = 0x0202;
|
||||
static const uint16_t SBP_BASELINE_NED_MSGTYPE = 0x0203;
|
||||
static const uint16_t SBP_VEL_ECEF_MSGTYPE = 0x0204;
|
||||
static const uint16_t SBP_VEL_NED_MSGTYPE = 0x0205;
|
||||
static const uint16_t SBP_TRACKING_STATE_MSGTYPE = 0x0016;
|
||||
static const uint16_t SBP_IAR_STATE_MSGTYPE = 0x0019;
|
||||
|
||||
// GPS Time
|
||||
struct PACKED sbp_gps_time_t {
|
||||
uint16_t wn; //< GPS week number (unit: weeks)
|
||||
@ -175,17 +194,36 @@ private:
|
||||
uint8_t flags; //< Status flags (reserved)
|
||||
};
|
||||
|
||||
// Activity and Signal-to-Noise data of a tracking channel on the GPS.
|
||||
struct PACKED sbp_tracking_state_t {
|
||||
uint8_t state; //< 0 if disabled, 1 if running
|
||||
uint8_t prn; //< PRN identifier of tracked satellite
|
||||
float cn0; //< carrier to noise power ratio.
|
||||
};
|
||||
|
||||
// Integer Ambiguity Resolution state - how is the RTK resolution doing?
|
||||
struct PACKED sbp_iar_state_t {
|
||||
uint32_t num_hypotheses;
|
||||
};
|
||||
|
||||
|
||||
|
||||
// ************************************************************************
|
||||
// Swift Navigation SBP protocol parsing and processing
|
||||
// ************************************************************************
|
||||
|
||||
//Pulls data from the port, dispatches messages to processing functions
|
||||
void sbp_process();
|
||||
//Returns true if a new message was successfully decoded.
|
||||
bool sbp_process();
|
||||
|
||||
bool update_state(bool has_new_message);
|
||||
|
||||
void update_state_velocity(void);
|
||||
|
||||
//Processes individual messages
|
||||
//When a message is received, it sets a sticky bit that it has updated
|
||||
//itself. This is used to track when a full update of GPS_State has occurred
|
||||
void sbp_process_heartbeat(uint8_t* msg);
|
||||
void sbp_process_gpstime(uint8_t* msg);
|
||||
void sbp_process_dops(uint8_t* msg);
|
||||
void sbp_process_pos_ecef(uint8_t* msg);
|
||||
@ -194,21 +232,46 @@ private:
|
||||
void sbp_process_baseline_ned(uint8_t* msg);
|
||||
void sbp_process_vel_ecef(uint8_t* msg);
|
||||
void sbp_process_vel_ned(uint8_t* msg);
|
||||
void sbp_process_tracking_state(uint8_t* msg, uint8_t len);
|
||||
void sbp_process_iar_state(uint8_t* msg);
|
||||
void sbp_process_startup(uint8_t* msg);
|
||||
|
||||
|
||||
//Internal last-received-messages
|
||||
sbp_pos_llh_t last_sbp_pos_llh_msg;
|
||||
sbp_vel_ned_t last_sbp_vel_ned_msg;
|
||||
sbp_baseline_ecef_t last_sbp_baseline_ecef_msg;
|
||||
sbp_baseline_ned_t last_sbp_baseline_ned_msg;
|
||||
sbp_tracking_state_t last_sbp_tracking_state_msg;
|
||||
uint8_t last_sbp_tracking_state_msg_num;
|
||||
|
||||
//Tracking GPS health and received time-of-week
|
||||
uint32_t last_baseline_received_ms;
|
||||
uint32_t last_heatbeat_received_ms;
|
||||
uint32_t last_tracking_state_ms;
|
||||
int32_t iar_num_hypotheses;
|
||||
|
||||
//Sticky bits to track updating of state
|
||||
bool has_updated_pos:1;
|
||||
bool has_updated_vel:1;
|
||||
bool dgps_corrections_incoming:1;
|
||||
bool rtk_corrections_incoming:1;
|
||||
|
||||
bool has_new_pos_llh:1;
|
||||
bool has_new_vel_ned:1;
|
||||
bool has_new_baseline_ecef:1;
|
||||
|
||||
//RTK-specific relative-to-absolute positioning
|
||||
bool has_rtk_base_pos:1;
|
||||
Vector3d base_pos_ecef;
|
||||
|
||||
// ************************************************************************
|
||||
// Monitoring and Performance Counting
|
||||
// ************************************************************************
|
||||
|
||||
uint8_t pos_msg_counter;
|
||||
uint8_t vel_msg_counter;
|
||||
uint8_t dops_msg_counter;
|
||||
uint8_t baseline_msg_counter;
|
||||
uint16_t crc_error_counter;
|
||||
uint8_t full_update_counter;
|
||||
uint32_t crc_error_counter;
|
||||
uint32_t last_healthcheck_millis;
|
||||
|
||||
// ************************************************************************
|
||||
@ -219,8 +282,16 @@ private:
|
||||
static bool logging_started;
|
||||
|
||||
void logging_write_headers();
|
||||
void logging_log_health(float pos_msg_hz, float vel_msg_hz, float dops_msg_hz, float baseline_msg_hz, float crc_error_hz);
|
||||
void logging_log_baseline(struct sbp_baseline_ecef_t*);
|
||||
|
||||
void logging_log_health(float pos_msg_hz, float vel_msg_hz, float baseline_msg_hz, float full_update_hz,
|
||||
uint32_t crc_error_counter, bool dgps_corrections_incoming, bool rtk_corrections_incoming,
|
||||
bool has_rtk_base_pos, int32_t iar_num_hypotheses);
|
||||
void logging_log_llh(struct sbp_pos_llh_t* p);
|
||||
void logging_log_baseline_ecef(struct sbp_baseline_ecef_t*);
|
||||
void logging_log_tracking_state(struct sbp_tracking_state_t*, uint8_t num);
|
||||
|
||||
};
|
||||
|
||||
#endif // __AP_GPS_SBP_H__
|
||||
#endif // GPS_RTK_AVAILABLE
|
||||
|
||||
#endif // __AP_GPS_SBP_H__
|
||||
|
@ -20,6 +20,9 @@
|
||||
#ifndef __AP_GPS_BACKEND_H__
|
||||
#define __AP_GPS_BACKEND_H__
|
||||
|
||||
#include <GCS_MAVLink.h>
|
||||
#include <AP_GPS.h>
|
||||
|
||||
class AP_GPS_Backend
|
||||
{
|
||||
public:
|
||||
@ -34,6 +37,29 @@ public:
|
||||
// valid packet from the GPS.
|
||||
virtual bool read() = 0;
|
||||
|
||||
#if GPS_RTK_AVAILABLE
|
||||
|
||||
// true once an RTK GPS Driver has a converged baseline vector and
|
||||
// absolute single point solution to enter into an RTK Fix.
|
||||
virtual bool can_calculate_base_pos(void) { return false; };
|
||||
|
||||
// tells a RTK GPS Driver to capture the current single-point solution
|
||||
// and baseline solution as the current home data.
|
||||
virtual void calculate_base_pos(void) {};
|
||||
|
||||
// Highest status supported by this GPS.
|
||||
// Allows external system to identify type of receiver connected.
|
||||
virtual AP_GPS::GPS_Status highest_supported_status(void) { return AP_GPS::GPS_OK_FIX_3D; }
|
||||
|
||||
//MAVLink methods
|
||||
virtual void send_mavlink_gps_rtk(mavlink_channel_t chan) { return ; }
|
||||
|
||||
#if GPS_MAX_INSTANCES > 1
|
||||
virtual void send_mavlink_gps2_rtk(mavlink_channel_t chan) { return ; }
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
||||
protected:
|
||||
AP_HAL::UARTDriver *port; ///< UART we are attached to
|
||||
AP_GPS &gps; ///< access to frontend (for parameters)
|
||||
|
Loading…
Reference in New Issue
Block a user