Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make blended backend a real backend #26919

Merged
merged 3 commits into from
Aug 26, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
79 changes: 75 additions & 4 deletions Tools/autotest/arducopter.py
Original file line number Diff line number Diff line change
Expand Up @@ -9140,20 +9140,90 @@ def GPSWeightedBlending(self):
if m.TimeUS != current_ts:
current_ts = None
continue
measurements[m.I] = (m.Lat, m.Lng)
measurements[m.I] = (m.Lat, m.Lng, m.Alt)
if len(measurements) == 3:
# check lat:
for n in 0, 1:
for n in 0, 1, 2:
expected_blended = 0.8*measurements[0][n] + 0.2*measurements[1][n]
epsilon = 0.0000002
axis_epsilons = [0.0000002, 0.0000002, 0.2]
epsilon = axis_epsilons[n]
error = abs(measurements[2][n] - expected_blended)
if error > epsilon:
raise NotAchievedException(f"Blended diverged {measurements[0][n]=} {measurements[1][n]=}")
raise NotAchievedException(f"Blended diverged {n=} {measurements[0][n]=} {measurements[1][n]=}")
current_ts = None

self.context_pop()
self.reboot_sitl()

def GPSBlendingAffinity(self):
'''test blending when affinity in use'''
# configure:
self.set_parameters({
"WP_YAW_BEHAVIOR": 0, # do not yaw
"GPS2_TYPE": 1,
"SIM_GPS2_TYPE": 1,
"SIM_GPS2_DISABLE": 0,
"SIM_GPS_POS_X": 1.0,
"SIM_GPS_POS_Y": -1.0,
"SIM_GPS2_POS_X": -1.0,
"SIM_GPS2_POS_Y": 1.0,
"GPS_AUTO_SWITCH": 2,

"EK3_AFFINITY" : 1,
"EK3_IMU_MASK": 7,
"SIM_IMU_COUNT": 3,
"INS_ACC3OFFS_X": 0.001,
"INS_ACC3OFFS_Y": 0.001,
"INS_ACC3OFFS_Z": 0.001,
})
# force-calibration of accel:
self.run_cmd(mavutil.mavlink.MAV_CMD_PREFLIGHT_CALIBRATION, p5=76)
self.reboot_sitl()

alt = 10
self.takeoff(alt, mode='GUIDED')
self.fly_guided_move_local(30, 0, alt)
self.fly_guided_move_local(30, 30, alt)
self.fly_guided_move_local(0, 30, alt)
self.fly_guided_move_local(0, 0, alt)
self.change_mode('LAND')

current_log_file = self.dfreader_for_current_onboard_log()

self.wait_disarmed()

# ensure that the blended solution is always about half-way
# between the two GPSs:
current_ts = None
max_errors = [0, 0, 0]
while True:
m = current_log_file.recv_match(type='XKF1')
if m is None:
break
if current_ts is None:
if m.C != 0: # noqa
continue
current_ts = m.TimeUS
measurements = {}
if m.TimeUS != current_ts:
current_ts = None
continue
measurements[m.C] = (m.PN, m.PE, m.PD)
if len(measurements) == 3:
# check lat:
for n in 0, 1, 2:
expected_blended = 0.5*measurements[0][n] + 0.5*measurements[1][n]
axis_epsilons = [0.02, 0.02, 0.03]
epsilon = axis_epsilons[n]
error = abs(measurements[2][n] - expected_blended)
# self.progress(f"{n=} {error=}")
if error > max_errors[n]:
max_errors[n] = error
if error > epsilon:
raise NotAchievedException(f"Blended diverged {n=} {measurements[0][n]=} {measurements[1][n]=} {measurements[2][n]=} {error=}") # noqa:E501
current_ts = None
self.progress(f"{max_errors=}")

def Callisto(self):
'''Test Callisto'''
self.customise_SITL_commandline(
Expand Down Expand Up @@ -11789,6 +11859,7 @@ def tests2b(self): # this block currently around 9.5mins here
self.GPSBlending,
self.GPSWeightedBlending,
self.GPSBlendingLog,
self.GPSBlendingAffinity,
self.DataFlash,
Test(self.DataFlashErase, attempts=8),
self.Callisto,
Expand Down
6 changes: 0 additions & 6 deletions libraries/AP_Arming/AP_Arming.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -649,12 +649,6 @@ bool AP_Arming::gps_checks(bool report)
(double)distance_m);
return false;
}
#if AP_GPS_BLENDED_ENABLED
if (!gps.blend_health_check()) {
check_failed(ARMING_CHECK_GPS, report, "GPS blending unhealthy");
return false;
}
#endif

// check AHRS and GPS are within 10m of each other
if (gps.num_sensors() > 0) {
Expand Down
51 changes: 25 additions & 26 deletions libraries/AP_GPS/AP_GPS.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
#include <AP_SerialManager/AP_SerialManager.h>

#include "AP_GPS_NOVA.h"
#include "AP_GPS_Blended.h"
#include "AP_GPS_ERB.h"
#include "AP_GPS_GSOF.h"
#include "AP_GPS_NMEA.h"
Expand Down Expand Up @@ -59,15 +60,19 @@
#include "RTCM3_Parser.h"
#endif

#if !AP_GPS_BLENDED_ENABLED
#if defined(GPS_BLENDED_INSTANCE)
#error GPS_BLENDED_INSTANCE should not be defined when AP_GPS_BLENDED_ENABLED is false
#endif
#endif

#define GPS_RTK_INJECT_TO_ALL 127
#ifndef GPS_MAX_RATE_MS
#define GPS_MAX_RATE_MS 200 // maximum value of rate_ms (i.e. slowest update rate) is 5hz or 200ms
#endif
#define GPS_BAUD_TIME_MS 1200
#define GPS_TIMEOUT_MS 4000u

#define BLEND_COUNTER_FAILURE_INCREMENT 10

extern const AP_HAL::HAL &hal;

// baudrates to try to detect GPSes with
Expand Down Expand Up @@ -347,6 +352,11 @@ void AP_GPS::init()
rate_ms.set(GPS_MAX_RATE_MS);
}
}

// create the blended instance if appropriate:
#if AP_GPS_BLENDED_ENABLED
drivers[GPS_BLENDED_INSTANCE] = NEW_NOTHROW AP_GPS_Blended(*this, params[GPS_BLENDED_INSTANCE], state[GPS_BLENDED_INSTANCE], timing[GPS_BLENDED_INSTANCE]);
#endif
}

void AP_GPS::convert_parameters()
Expand Down Expand Up @@ -1087,25 +1097,15 @@ void AP_GPS::update_primary(void)
*/
const bool using_moving_base = is_rtk_base(0) || is_rtk_base(1);
if ((GPSAutoSwitch)_auto_switch.get() == GPSAutoSwitch::BLEND && !using_moving_base) {
_output_is_blended = calc_blend_weights();
// adjust blend health counter
if (!_output_is_blended) {
_blend_health_counter = MIN(_blend_health_counter+BLEND_COUNTER_FAILURE_INCREMENT, 100);
} else if (_blend_health_counter > 0) {
_blend_health_counter--;
}
// stop blending if unhealthy
if (_blend_health_counter >= 50) {
_output_is_blended = false;
}
_output_is_blended = ((AP_GPS_Blended*)drivers[GPS_BLENDED_INSTANCE])->calc_weights();
} else {
_output_is_blended = false;
_blend_health_counter = 0;
((AP_GPS_Blended*)drivers[GPS_BLENDED_INSTANCE])->zero_health_counter();
}

if (_output_is_blended) {
// Use the weighting to calculate blended GPS states
calc_blended_state();
((AP_GPS_Blended*)drivers[GPS_BLENDED_INSTANCE])->calc_state();
// set primary to the virtual instance
primary_instance = GPS_BLENDED_INSTANCE;
return;
Expand Down Expand Up @@ -1698,10 +1698,7 @@ bool AP_GPS::get_lag(uint8_t instance, float &lag_sec) const
#if AP_GPS_BLENDED_ENABLED
// return lag of blended GPS
if (instance == GPS_BLENDED_INSTANCE) {
lag_sec = _blended_lag_sec;
// auto switching uses all GPS receivers, so all must be configured
uint8_t inst; // we don't actually care what the number is, but must pass it
return first_unconfigured_gps(inst);
return drivers[instance]->get_lag(lag_sec);
}
#endif

Expand Down Expand Up @@ -1735,7 +1732,7 @@ const Vector3f &AP_GPS::get_antenna_offset(uint8_t instance) const
#if AP_GPS_BLENDED_ENABLED
if (instance == GPS_BLENDED_INSTANCE) {
// return an offset for the blended GPS solution
return _blended_antenna_offset;
return ((AP_GPS_Blended*)drivers[instance])->get_antenna_offset();
}
#endif

Expand Down Expand Up @@ -1791,12 +1788,6 @@ bool AP_GPS::is_healthy(uint8_t instance) const
}
#endif // HAL_BUILD_AP_PERIPH

#if AP_GPS_BLENDED_ENABLED
if (instance == GPS_BLENDED_INSTANCE) {
return blend_health_check();
}
#endif

return drivers[instance] != nullptr &&
drivers[instance]->is_healthy();
}
Expand Down Expand Up @@ -1829,6 +1820,14 @@ bool AP_GPS::pre_arm_checks(char failure_msg[], uint16_t failure_msg_len)
}
}
}

#if AP_GPS_BLENDED_ENABLED
if (!drivers[GPS_BLENDED_INSTANCE]->is_healthy()) {
hal.util->snprintf(failure_msg, failure_msg_len, "GPS blending unhealthy");
return false;
}
#endif

return true;
}

Expand Down
19 changes: 3 additions & 16 deletions libraries/AP_GPS/AP_GPS.h
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ class RTCM3_Parser;
/// GPS driver main class
class AP_GPS
{
friend class AP_GPS_Blended;
friend class AP_GPS_ERB;
friend class AP_GPS_GSOF;
friend class AP_GPS_MAV;
Expand Down Expand Up @@ -512,9 +513,6 @@ class AP_GPS
// pre-arm check that all GPSs are close to each other. farthest distance between GPSs (in meters) is returned
bool all_consistent(float &distance) const;

// pre-arm check of GPS blending. False if blending is unhealthy, True if healthy or blending is not being used
bool blend_health_check() const;

// handle sending of initialisation strings to the GPS - only used by backends
void send_blob_start(uint8_t instance);
void send_blob_start(uint8_t instance, const char *_blob, uint16_t size);
Expand Down Expand Up @@ -607,7 +605,7 @@ class AP_GPS
protected:

// configuration parameters
Params params[GPS_MAX_RECEIVERS];
Params params[GPS_MAX_INSTANCES];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we remove GPS_MAX_RECEIVERS now?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not yet. There are all sorts of interesting decisions to be made revolving around it.

Many of them will end up turning into a :-style iteration across the instances array.

AP_Int8 _navfilter;
AP_Int8 _auto_switch;
AP_Int16 _sbp_logmask;
Expand Down Expand Up @@ -669,7 +667,7 @@ class AP_GPS
// Note allowance for an additional instance to contain blended data
GPS_timing timing[GPS_MAX_INSTANCES];
GPS_State state[GPS_MAX_INSTANCES];
AP_GPS_Backend *drivers[GPS_MAX_RECEIVERS];
AP_GPS_Backend *drivers[GPS_MAX_INSTANCES];
AP_HAL::UARTDriver *_port[GPS_MAX_RECEIVERS];

/// primary GPS instance
Expand Down Expand Up @@ -757,18 +755,7 @@ class AP_GPS
void inject_data(uint8_t instance, const uint8_t *data, uint16_t len);

#if AP_GPS_BLENDED_ENABLED
// GPS blending and switching
Vector3f _blended_antenna_offset; // blended antenna offset
float _blended_lag_sec; // 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.
bool _output_is_blended; // true when a blended GPS solution being output
uint8_t _blend_health_counter; // 0 = perfectly health, 100 = very unhealthy

// 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);
#endif

bool should_log() const;
Expand Down
Loading
Loading