mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-05 07:28:29 -04:00
AP_GPS: Factor out moving base offset helper
This commit is contained in:
parent
230798546e
commit
4161e2eb28
@ -293,7 +293,7 @@ const AP_Param::GroupInfo AP_GPS::var_info[] = {
|
|||||||
AP_GROUPINFO("BLEND_TC", 21, AP_GPS, _blend_tc, 10.0f),
|
AP_GROUPINFO("BLEND_TC", 21, AP_GPS, _blend_tc, 10.0f),
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
// @Param: DRV_OPTIONS
|
// @Param: DRV_OPTIONS
|
||||||
// @DisplayName: driver options
|
// @DisplayName: driver options
|
||||||
// @Description: Additional backend specific options
|
// @Description: Additional backend specific options
|
||||||
@ -322,6 +322,20 @@ const AP_Param::GroupInfo AP_GPS::var_info[] = {
|
|||||||
AP_GROUPINFO("COM_PORT2", 24, AP_GPS, _com_port[1], 1),
|
AP_GROUPINFO("COM_PORT2", 24, AP_GPS, _com_port[1], 1),
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if GPS_MOVING_BASELINE
|
||||||
|
|
||||||
|
// @Group: MB1_
|
||||||
|
// @Path: MovingBase.cpp
|
||||||
|
AP_SUBGROUPINFO(mb_params[0], "MB1_", 25, AP_GPS, MovingBase),
|
||||||
|
|
||||||
|
#if GPS_MAX_RECEIVERS > 1
|
||||||
|
// @Group: MB2_
|
||||||
|
// @Path: MovingBase.cpp
|
||||||
|
AP_SUBGROUPINFO(mb_params[1], "MB2_", 26, AP_GPS, MovingBase),
|
||||||
|
#endif // GPS_MAX_RECEIVERS > 1
|
||||||
|
|
||||||
|
#endif // GPS_MOVING_BASELINE
|
||||||
|
|
||||||
AP_GROUPEND
|
AP_GROUPEND
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -43,14 +43,18 @@
|
|||||||
|
|
||||||
#define UNIX_OFFSET_MSEC (17000ULL * 86400ULL + 52ULL * 10ULL * AP_MSEC_PER_WEEK - GPS_LEAPSECONDS_MILLIS)
|
#define UNIX_OFFSET_MSEC (17000ULL * 86400ULL + 52ULL * 10ULL * AP_MSEC_PER_WEEK - GPS_LEAPSECONDS_MILLIS)
|
||||||
|
|
||||||
#ifndef GPS_UBLOX_MOVING_BASELINE
|
#ifndef GPS_MOVING_BASELINE
|
||||||
#define GPS_UBLOX_MOVING_BASELINE !HAL_MINIMIZE_FEATURES && GPS_MAX_RECEIVERS>1
|
#define GPS_MOVING_BASELINE !HAL_MINIMIZE_FEATURES && GPS_MAX_RECEIVERS>1
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef HAL_MSP_GPS_ENABLED
|
#ifndef HAL_MSP_GPS_ENABLED
|
||||||
#define HAL_MSP_GPS_ENABLED HAL_MSP_SENSORS_ENABLED
|
#define HAL_MSP_GPS_ENABLED HAL_MSP_SENSORS_ENABLED
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if GPS_MOVING_BASELINE
|
||||||
|
#include "MovingBase.h"
|
||||||
|
#endif // GPS_MOVING_BASELINE
|
||||||
|
|
||||||
class AP_GPS_Backend;
|
class AP_GPS_Backend;
|
||||||
|
|
||||||
/// @class AP_GPS
|
/// @class AP_GPS
|
||||||
@ -519,6 +523,10 @@ protected:
|
|||||||
AP_Float _blend_tc;
|
AP_Float _blend_tc;
|
||||||
AP_Int16 _driver_options;
|
AP_Int16 _driver_options;
|
||||||
|
|
||||||
|
#if GPS_MOVING_BASELINE
|
||||||
|
MovingBase mb_params[GPS_MAX_RECEIVERS];
|
||||||
|
#endif // GPS_MOVING_BASELINE
|
||||||
|
|
||||||
uint32_t _log_gps_bit = -1;
|
uint32_t _log_gps_bit = -1;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -79,7 +79,7 @@ AP_GPS_UBLOX::AP_GPS_UBLOX(AP_GPS &_gps, AP_GPS::GPS_State &_state, AP_HAL::UART
|
|||||||
_unconfigured_messages |= CONFIG_TP5;
|
_unconfigured_messages |= CONFIG_TP5;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (role == AP_GPS::GPS_ROLE_MB_BASE && !mb_use_uart2()) {
|
if (role == AP_GPS::GPS_ROLE_MB_BASE && !mb_use_uart2()) {
|
||||||
rtcm3_parser = new RTCM3_Parser;
|
rtcm3_parser = new RTCM3_Parser;
|
||||||
if (rtcm3_parser == nullptr) {
|
if (rtcm3_parser == nullptr) {
|
||||||
@ -96,14 +96,14 @@ AP_GPS_UBLOX::AP_GPS_UBLOX(AP_GPS &_gps, AP_GPS::GPS_State &_state, AP_HAL::UART
|
|||||||
|
|
||||||
AP_GPS_UBLOX::~AP_GPS_UBLOX()
|
AP_GPS_UBLOX::~AP_GPS_UBLOX()
|
||||||
{
|
{
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (rtcm3_parser) {
|
if (rtcm3_parser) {
|
||||||
delete rtcm3_parser;
|
delete rtcm3_parser;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
/*
|
/*
|
||||||
config for F9 GPS in moving baseline base role
|
config for F9 GPS in moving baseline base role
|
||||||
See ZED-F9P integration manual section 3.1.5.6.1
|
See ZED-F9P integration manual section 3.1.5.6.1
|
||||||
@ -205,7 +205,7 @@ const AP_GPS_UBLOX::config_list AP_GPS_UBLOX::config_MB_Rover_uart2[] {
|
|||||||
{ ConfigKey::MSGOUT_RTCM_3X_TYPE1127_UART1, 0},
|
{ ConfigKey::MSGOUT_RTCM_3X_TYPE1127_UART1, 0},
|
||||||
{ ConfigKey::MSGOUT_RTCM_3X_TYPE1230_UART1, 0},
|
{ ConfigKey::MSGOUT_RTCM_3X_TYPE1230_UART1, 0},
|
||||||
};
|
};
|
||||||
#endif // GPS_UBLOX_MOVING_BASELINE
|
#endif // GPS_MOVING_BASELINE
|
||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -357,7 +357,7 @@ AP_GPS_UBLOX::_request_next_config(void)
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case STEP_RTK_MOVBASE:
|
case STEP_RTK_MOVBASE:
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (supports_F9_config()) {
|
if (supports_F9_config()) {
|
||||||
static_assert(sizeof(active_config.done_mask)*8 >= ARRAY_SIZE(config_MB_Base_uart1), "done_mask too small, base1");
|
static_assert(sizeof(active_config.done_mask)*8 >= ARRAY_SIZE(config_MB_Base_uart1), "done_mask too small, base1");
|
||||||
static_assert(sizeof(active_config.done_mask)*8 >= ARRAY_SIZE(config_MB_Base_uart2), "done_mask too small, base2");
|
static_assert(sizeof(active_config.done_mask)*8 >= ARRAY_SIZE(config_MB_Base_uart2), "done_mask too small, base2");
|
||||||
@ -572,7 +572,7 @@ AP_GPS_UBLOX::read(void)
|
|||||||
// read the next byte
|
// read the next byte
|
||||||
data = port->read();
|
data = port->read();
|
||||||
|
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (rtcm3_parser) {
|
if (rtcm3_parser) {
|
||||||
if (rtcm3_parser->read(data)) {
|
if (rtcm3_parser->read(data)) {
|
||||||
// we've found a RTCMv3 packet. We stop parsing at
|
// we've found a RTCMv3 packet. We stop parsing at
|
||||||
@ -682,7 +682,7 @@ AP_GPS_UBLOX::read(void)
|
|||||||
break; // bad checksum
|
break; // bad checksum
|
||||||
}
|
}
|
||||||
|
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (rtcm3_parser) {
|
if (rtcm3_parser) {
|
||||||
// this is a uBlox packet, discard any partial RTCMv3 state
|
// this is a uBlox packet, discard any partial RTCMv3 state
|
||||||
rtcm3_parser->reset();
|
rtcm3_parser->reset();
|
||||||
@ -856,7 +856,7 @@ uint8_t AP_GPS_UBLOX::config_key_size(ConfigKey key) const
|
|||||||
*/
|
*/
|
||||||
int8_t AP_GPS_UBLOX::find_active_config_index(ConfigKey key) const
|
int8_t AP_GPS_UBLOX::find_active_config_index(ConfigKey key) const
|
||||||
{
|
{
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (active_config.list == nullptr) {
|
if (active_config.list == nullptr) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -1122,7 +1122,7 @@ AP_GPS_UBLOX::_parse_gps(void)
|
|||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
// see if it is in active config list
|
// see if it is in active config list
|
||||||
int8_t cfg_idx = find_active_config_index(id);
|
int8_t cfg_idx = find_active_config_index(id);
|
||||||
if (cfg_idx >= 0) {
|
if (cfg_idx >= 0) {
|
||||||
@ -1141,7 +1141,7 @@ AP_GPS_UBLOX::_parse_gps(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif // GPS_UBLOX_MOVING_BASELINE
|
#endif // GPS_MOVING_BASELINE
|
||||||
|
|
||||||
// step over the value
|
// step over the value
|
||||||
uint8_t step_size = config_key_size(id);
|
uint8_t step_size = config_key_size(id);
|
||||||
@ -1326,10 +1326,10 @@ AP_GPS_UBLOX::_parse_gps(void)
|
|||||||
state.hdop = 130;
|
state.hdop = 130;
|
||||||
#endif
|
#endif
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
#if GPS_MOVING_BASELINE
|
||||||
case MSG_RELPOSNED:
|
case MSG_RELPOSNED:
|
||||||
{
|
{
|
||||||
const Vector3f &offset0 = gps._antenna_offset[state.instance^1].get();
|
|
||||||
const Vector3f &offset1 = gps._antenna_offset[state.instance].get();
|
|
||||||
// note that we require the yaw to come from a fixed solution, not a float solution
|
// note that we require the yaw to come from a fixed solution, not a float solution
|
||||||
// yaw from a float solution would only be acceptable with a very large separation between
|
// yaw from a float solution would only be acceptable with a very large separation between
|
||||||
// GPS modules
|
// GPS modules
|
||||||
@ -1342,21 +1342,6 @@ AP_GPS_UBLOX::_parse_gps(void)
|
|||||||
static_cast<uint32_t>(RELPOSNED::refObsMiss) |
|
static_cast<uint32_t>(RELPOSNED::refObsMiss) |
|
||||||
static_cast<uint32_t>(RELPOSNED::carrSolnFloat);
|
static_cast<uint32_t>(RELPOSNED::carrSolnFloat);
|
||||||
|
|
||||||
const Vector3f antenna_offset = offset0 - offset1;
|
|
||||||
const float offset_dist = antenna_offset.length();
|
|
||||||
const float rel_dist = _buffer.relposned.relPosLength * 0.01;
|
|
||||||
const float dist_error = offset_dist - rel_dist;
|
|
||||||
const float strict_length_error_allowed = 0.2; // allow for up to 20% error
|
|
||||||
const float min_separation = 0.05;
|
|
||||||
bool tilt_ok = true;
|
|
||||||
const float min_dist = MIN(offset_dist, rel_dist);
|
|
||||||
#ifndef HAL_BUILD_AP_PERIPH
|
|
||||||
// when ahrs is available use it to constrain vertical component
|
|
||||||
const Vector3f antenna_tilt = AP::ahrs().get_rotation_body_to_ned() * antenna_offset;
|
|
||||||
const float alt_error = _buffer.relposned.relPosD*0.01 + antenna_tilt.z;
|
|
||||||
tilt_ok = fabsf(alt_error) < strict_length_error_allowed * min_dist;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
_check_new_itow(_buffer.relposned.iTOW);
|
_check_new_itow(_buffer.relposned.iTOW);
|
||||||
if (_buffer.relposned.iTOW != _last_relposned_itow+200) {
|
if (_buffer.relposned.iTOW != _last_relposned_itow+200) {
|
||||||
// useful for looking at packet loss on links
|
// useful for looking at packet loss on links
|
||||||
@ -1365,35 +1350,20 @@ AP_GPS_UBLOX::_parse_gps(void)
|
|||||||
_last_relposned_itow = _buffer.relposned.iTOW;
|
_last_relposned_itow = _buffer.relposned.iTOW;
|
||||||
_last_relposned_ms = AP_HAL::millis();
|
_last_relposned_ms = AP_HAL::millis();
|
||||||
|
|
||||||
/*
|
|
||||||
RELPOSNED messages gives the NED distance from base to
|
|
||||||
rover. It comes from the rover
|
|
||||||
*/
|
|
||||||
MB_Debug("RELPOSNED[%u]: od:%.2f rd:%.2f ae:%.2f flags:0x%04x t=%u",
|
|
||||||
state.instance+1,
|
|
||||||
offset_dist, rel_dist, alt_error,
|
|
||||||
unsigned(_buffer.relposned.flags),
|
|
||||||
unsigned(_buffer.relposned.iTOW));
|
|
||||||
|
|
||||||
if (((_buffer.relposned.flags & valid_mask) == valid_mask) &&
|
if (((_buffer.relposned.flags & valid_mask) == valid_mask) &&
|
||||||
((_buffer.relposned.flags & invalid_mask) == 0) &&
|
((_buffer.relposned.flags & invalid_mask) == 0) &&
|
||||||
rel_dist > min_separation &&
|
calculate_moving_base_yaw(_buffer.relposned.relPosHeading * 1e-5,
|
||||||
offset_dist > min_separation &&
|
_buffer.relposned.relPosLength * 0.01,
|
||||||
fabsf(dist_error) < strict_length_error_allowed * min_dist &&
|
_buffer.relposned.relPosD*0.01)) {
|
||||||
tilt_ok) {
|
|
||||||
float rotation_offset_rad;
|
|
||||||
const Vector3f diff = offset1 - offset0;
|
|
||||||
rotation_offset_rad = Vector2f(diff.x, diff.y).angle();
|
|
||||||
state.gps_yaw = wrap_360(_buffer.relposned.relPosHeading * 1e-5 - degrees(rotation_offset_rad));
|
|
||||||
state.have_gps_yaw = true;
|
|
||||||
state.gps_yaw_accuracy = _buffer.relposned.accHeading * 1e-5;
|
state.gps_yaw_accuracy = _buffer.relposned.accHeading * 1e-5;
|
||||||
state.have_gps_yaw_accuracy = true;
|
state.have_gps_yaw_accuracy = true;
|
||||||
} else {
|
} else {
|
||||||
state.have_gps_yaw = false;
|
|
||||||
state.have_gps_yaw_accuracy = false;
|
state.have_gps_yaw_accuracy = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
#endif // GPS_MOVING_BASELINE
|
||||||
|
|
||||||
case MSG_PVT:
|
case MSG_PVT:
|
||||||
Debug("MSG_PVT");
|
Debug("MSG_PVT");
|
||||||
|
|
||||||
@ -1707,7 +1677,7 @@ AP_GPS_UBLOX::_configure_valget(ConfigKey key)
|
|||||||
bool
|
bool
|
||||||
AP_GPS_UBLOX::_configure_config_set(const config_list *list, uint8_t count, uint32_t unconfig_bit)
|
AP_GPS_UBLOX::_configure_config_set(const config_list *list, uint8_t count, uint32_t unconfig_bit)
|
||||||
{
|
{
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
active_config.list = list;
|
active_config.list = list;
|
||||||
active_config.count = count;
|
active_config.count = count;
|
||||||
active_config.done_mask = 0;
|
active_config.done_mask = 0;
|
||||||
@ -1887,7 +1857,7 @@ bool AP_GPS_UBLOX::get_lag(float &lag_sec) const
|
|||||||
// F9 lag not verified yet from flight log, but likely to be at least
|
// F9 lag not verified yet from flight log, but likely to be at least
|
||||||
// as good as M8
|
// as good as M8
|
||||||
lag_sec = 0.12f;
|
lag_sec = 0.12f;
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (role == AP_GPS::GPS_ROLE_MB_BASE ||
|
if (role == AP_GPS::GPS_ROLE_MB_BASE ||
|
||||||
role == AP_GPS::GPS_ROLE_MB_ROVER) {
|
role == AP_GPS::GPS_ROLE_MB_ROVER) {
|
||||||
// the moving baseline rover will lag about 40ms from the
|
// the moving baseline rover will lag about 40ms from the
|
||||||
@ -1924,7 +1894,7 @@ void AP_GPS_UBLOX::_check_new_itow(uint32_t itow)
|
|||||||
// support for retrieving RTCMv3 data from a moving baseline base
|
// support for retrieving RTCMv3 data from a moving baseline base
|
||||||
bool AP_GPS_UBLOX::get_RTCMV3(const uint8_t *&bytes, uint16_t &len)
|
bool AP_GPS_UBLOX::get_RTCMV3(const uint8_t *&bytes, uint16_t &len)
|
||||||
{
|
{
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (rtcm3_parser) {
|
if (rtcm3_parser) {
|
||||||
len = rtcm3_parser->get_len(bytes);
|
len = rtcm3_parser->get_len(bytes);
|
||||||
return len > 0;
|
return len > 0;
|
||||||
@ -1936,7 +1906,7 @@ bool AP_GPS_UBLOX::get_RTCMV3(const uint8_t *&bytes, uint16_t &len)
|
|||||||
// clear previous RTCM3 packet
|
// clear previous RTCM3 packet
|
||||||
void AP_GPS_UBLOX::clear_RTCMV3(void)
|
void AP_GPS_UBLOX::clear_RTCMV3(void)
|
||||||
{
|
{
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if (rtcm3_parser) {
|
if (rtcm3_parser) {
|
||||||
rtcm3_parser->clear_packet();
|
rtcm3_parser->clear_packet();
|
||||||
}
|
}
|
||||||
@ -1952,7 +1922,7 @@ bool AP_GPS_UBLOX::is_healthy(void) const
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
if ((role == AP_GPS::GPS_ROLE_MB_BASE ||
|
if ((role == AP_GPS::GPS_ROLE_MB_BASE ||
|
||||||
role == AP_GPS::GPS_ROLE_MB_ROVER) &&
|
role == AP_GPS::GPS_ROLE_MB_ROVER) &&
|
||||||
!supports_F9_config()) {
|
!supports_F9_config()) {
|
||||||
|
@ -742,7 +742,7 @@ private:
|
|||||||
return (uint8_t)(ubx_msg + (state.instance * UBX_MSG_TYPES));
|
return (uint8_t)(ubx_msg + (state.instance * UBX_MSG_TYPES));
|
||||||
}
|
}
|
||||||
|
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
// see if we should use uart2 for moving baseline config
|
// see if we should use uart2 for moving baseline config
|
||||||
bool mb_use_uart2(void) const {
|
bool mb_use_uart2(void) const {
|
||||||
return (driver_options() & unsigned(DRV_OPTIONS::MB_USE_UART2))?true:false;
|
return (driver_options() & unsigned(DRV_OPTIONS::MB_USE_UART2))?true:false;
|
||||||
@ -770,7 +770,7 @@ private:
|
|||||||
// return true if GPS is capable of F9 config
|
// return true if GPS is capable of F9 config
|
||||||
bool supports_F9_config(void) const;
|
bool supports_F9_config(void) const;
|
||||||
|
|
||||||
#if GPS_UBLOX_MOVING_BASELINE
|
#if GPS_MOVING_BASELINE
|
||||||
// config for moving baseline base
|
// config for moving baseline base
|
||||||
static const config_list config_MB_Base_uart1[];
|
static const config_list config_MB_Base_uart1[];
|
||||||
static const config_list config_MB_Base_uart2[];
|
static const config_list config_MB_Base_uart2[];
|
||||||
@ -789,5 +789,5 @@ private:
|
|||||||
|
|
||||||
// RTCM3 parser for when in moving baseline base mode
|
// RTCM3 parser for when in moving baseline base mode
|
||||||
RTCM3_Parser *rtcm3_parser;
|
RTCM3_Parser *rtcm3_parser;
|
||||||
#endif // GPS_UBLOX_MOVING_BASELINE
|
#endif // GPS_MOVING_BASELINE
|
||||||
};
|
};
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
#include <AP_Logger/AP_Logger.h>
|
#include <AP_Logger/AP_Logger.h>
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <AP_RTC/AP_RTC.h>
|
#include <AP_RTC/AP_RTC.h>
|
||||||
|
#include <AP_InternalError/AP_InternalError.h>
|
||||||
|
|
||||||
#define GPS_BACKEND_DEBUGGING 0
|
#define GPS_BACKEND_DEBUGGING 0
|
||||||
|
|
||||||
@ -301,3 +302,80 @@ void AP_GPS_Backend::check_new_itow(uint32_t itow, uint32_t msg_length)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if GPS_MOVING_BASELINE
|
||||||
|
bool AP_GPS_Backend::calculate_moving_base_yaw(const float reported_heading_deg, const float reported_distance, const float reported_D)
|
||||||
|
{
|
||||||
|
constexpr float minimum_antenna_seperation = 0.05; // meters
|
||||||
|
constexpr float permitted_error_length_pct = 0.2; // percentage
|
||||||
|
|
||||||
|
bool selectedOffset = false;
|
||||||
|
Vector3f offset;
|
||||||
|
switch (MovingBase::Type(gps.mb_params[state.instance].type.get())) {
|
||||||
|
case MovingBase::Type::RelativeToAlternateInstance:
|
||||||
|
offset = gps._antenna_offset[state.instance^1].get() - gps._antenna_offset[state.instance].get();
|
||||||
|
selectedOffset = true;
|
||||||
|
break;
|
||||||
|
case MovingBase::Type::RelativeToCustomBase:
|
||||||
|
offset = gps.mb_params[state.instance].base_offset.get();
|
||||||
|
selectedOffset = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!selectedOffset) {
|
||||||
|
// invalid type, let's throw up a flag
|
||||||
|
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
|
||||||
|
goto bad_yaw;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const float offset_dist = offset.length();
|
||||||
|
const float min_dist = MIN(offset_dist, reported_distance);
|
||||||
|
|
||||||
|
if (offset_dist < minimum_antenna_seperation) {
|
||||||
|
// offsets have to be sufficently large to get a meaningful angle off of them
|
||||||
|
Debug("Insufficent antenna offset (%f, %f, %f)", (double)offset.x, (double)offset.y, (double)offset.z);
|
||||||
|
goto bad_yaw;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (reported_distance < minimum_antenna_seperation) {
|
||||||
|
// if the reported distance is less then the minimum seperation it's not sufficently robust
|
||||||
|
Debug("Reported baseline distance (%f) was less then the minimum antenna seperation (%f)",
|
||||||
|
(double)reported_distance, (double)minimum_antenna_seperation);
|
||||||
|
goto bad_yaw;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if ((offset_dist - reported_distance) > (min_dist * permitted_error_length_pct)) {
|
||||||
|
// the magnitude of the vector is much further then we were expecting
|
||||||
|
Debug("Exceeded the permitted error margin %f > %f",
|
||||||
|
(double)(offset_dist - reported_distance), (double)(min_dist * permitted_error_length_pct));
|
||||||
|
goto bad_yaw;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef HAL_BUILD_AP_PERIPH
|
||||||
|
{
|
||||||
|
const Vector3f antenna_tilt = AP::ahrs().get_rotation_body_to_ned() * offset;
|
||||||
|
const float alt_error = reported_D + antenna_tilt.z;
|
||||||
|
if (fabsf(alt_error) > permitted_error_length_pct * min_dist) {
|
||||||
|
// the vertical component is out of range, reject it
|
||||||
|
goto bad_yaw;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // HAL_BUILD_AP_PERIPH
|
||||||
|
|
||||||
|
{
|
||||||
|
// at this point the offsets are looking okay, go ahead and actually calculate a useful heading
|
||||||
|
const float rotation_offset_rad = Vector2f(-offset.x, -offset.y).angle();
|
||||||
|
state.gps_yaw = wrap_360(reported_heading_deg - degrees(rotation_offset_rad));
|
||||||
|
state.have_gps_yaw = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
bad_yaw:
|
||||||
|
state.have_gps_yaw = false;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
#endif // GPS_MOVING_BASELINE
|
||||||
|
@ -118,6 +118,10 @@ protected:
|
|||||||
return uint16_t(gps._driver_options.get());
|
return uint16_t(gps._driver_options.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if GPS_MOVING_BASELINE
|
||||||
|
bool calculate_moving_base_yaw(const float reported_heading_deg, const float reported_distance, const float reported_D);
|
||||||
|
#endif //GPS_MOVING_BASELINE
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// itow from previous message
|
// itow from previous message
|
||||||
uint32_t _last_itow;
|
uint32_t _last_itow;
|
||||||
|
43
libraries/AP_GPS/MovingBase.cpp
Normal file
43
libraries/AP_GPS/MovingBase.cpp
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
#include "MovingBase.h"
|
||||||
|
|
||||||
|
const AP_Param::GroupInfo MovingBase::var_info[] = {
|
||||||
|
// @Param: TYPE
|
||||||
|
// @DisplayName: Moving base type
|
||||||
|
// @Description: Controls the type of moving base used if using moving base.
|
||||||
|
// @Values: 0:Relative to alternate GPS instance,1:RelativeToCustomBase
|
||||||
|
// @User: Advanced
|
||||||
|
// @RebootRequired: True
|
||||||
|
AP_GROUPINFO_FLAGS("TYPE", 1, MovingBase, type, int8_t(Type::RelativeToAlternateInstance), AP_PARAM_FLAG_ENABLE),
|
||||||
|
|
||||||
|
// @Param: OFS_X
|
||||||
|
// @DisplayName: Base antenna X position offset
|
||||||
|
// @Description: X position of the base GPS antenna in body frame. Positive X is forward of the origin. Use antenna phase centroid location if provided by the manufacturer.
|
||||||
|
// @Units: m
|
||||||
|
// @Range: -5 5
|
||||||
|
// @Increment: 0.01
|
||||||
|
// @User: Advanced
|
||||||
|
|
||||||
|
// @Param: OFS_Y
|
||||||
|
// @DisplayName: Base antenna Y position offset
|
||||||
|
// @Description: Y position of the base GPS antenna in body frame. Positive Y is to the right of the origin. Use antenna phase centroid location if provided by the manufacturer.
|
||||||
|
// @Units: m
|
||||||
|
// @Range: -5 5
|
||||||
|
// @Increment: 0.01
|
||||||
|
// @User: Advanced
|
||||||
|
|
||||||
|
// @Param: OFS_Z
|
||||||
|
// @DisplayName: Base antenna Z position offset
|
||||||
|
// @Description: Z position of the base GPS antenna in body frame. Positive Z is down from the origin. Use antenna phase centroid location if provided by the manufacturer.
|
||||||
|
// @Units: m
|
||||||
|
// @Range: -5 5
|
||||||
|
// @Increment: 0.01
|
||||||
|
// @User: Advanced
|
||||||
|
AP_GROUPINFO("OFS", 2, MovingBase, base_offset, 0.0f),
|
||||||
|
|
||||||
|
AP_GROUPEND
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
MovingBase::MovingBase(void) {
|
||||||
|
AP_Param::setup_object_defaults(this, var_info);
|
||||||
|
}
|
24
libraries/AP_GPS/MovingBase.h
Normal file
24
libraries/AP_GPS/MovingBase.h
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AP_Param/AP_Param.h>
|
||||||
|
#include <AP_Math/AP_Math.h>
|
||||||
|
|
||||||
|
class MovingBase {
|
||||||
|
public:
|
||||||
|
static const struct AP_Param::GroupInfo var_info[];
|
||||||
|
|
||||||
|
MovingBase(void);
|
||||||
|
|
||||||
|
/* Do not allow copies */
|
||||||
|
MovingBase(const MovingBase &other) = delete;
|
||||||
|
MovingBase &operator=(const MovingBase&) = delete;
|
||||||
|
|
||||||
|
enum class Type : int8_t {
|
||||||
|
RelativeToAlternateInstance = 0,
|
||||||
|
RelativeToCustomBase = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
AP_Int8 type; // an option from MovingBaseType
|
||||||
|
AP_Vector3f base_offset; // base position offset from the selected GPS reciever
|
||||||
|
|
||||||
|
};
|
Loading…
Reference in New Issue
Block a user