mirror of
https://gitee.com/mirrors_PX4/PX4-Autopilot.git
synced 2026-04-14 10:07:39 +08:00
- saves a small amount of work for the ekf2 selector in multi-EKF mode (visual_odometry_aligned now ignored) - helps to distinguish the origin/purpose from vehicle_odometry and vehicle_visual_odometry
617 lines
22 KiB
C++
617 lines
22 KiB
C++
/****************************************************************************
|
|
*
|
|
* Copyright (c) 2020 PX4 Development Team. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* 3. Neither the name PX4 nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#include "EKF2Selector.hpp"
|
|
|
|
using namespace time_literals;
|
|
using matrix::Quatf;
|
|
using matrix::Vector2f;
|
|
using math::constrain;
|
|
using math::max;
|
|
using math::radians;
|
|
|
|
EKF2Selector::EKF2Selector() :
|
|
ModuleParams(nullptr),
|
|
ScheduledWorkItem("ekf2_selector", px4::wq_configurations::nav_and_controllers)
|
|
{
|
|
}
|
|
|
|
EKF2Selector::~EKF2Selector()
|
|
{
|
|
Stop();
|
|
|
|
px4_lockstep_unregister_component(_lockstep_component);
|
|
}
|
|
|
|
bool EKF2Selector::Start()
|
|
{
|
|
// default to first instance
|
|
_selected_instance = 0;
|
|
_instance[0].estimator_status_sub.registerCallback();
|
|
_instance[0].estimator_attitude_sub.registerCallback();
|
|
|
|
return true;
|
|
}
|
|
|
|
void EKF2Selector::Stop()
|
|
{
|
|
for (int i = 0; i < EKF2_MAX_INSTANCES; i++) {
|
|
_instance[i].estimator_attitude_sub.unregisterCallback();
|
|
_instance[i].estimator_status_sub.unregisterCallback();
|
|
}
|
|
|
|
ScheduleClear();
|
|
}
|
|
|
|
void EKF2Selector::SelectInstance(uint8_t ekf_instance)
|
|
{
|
|
if (ekf_instance != _selected_instance) {
|
|
|
|
// update sensor_selection immediately
|
|
sensor_selection_s sensor_selection{};
|
|
sensor_selection.accel_device_id = _instance[ekf_instance].estimator_status.accel_device_id;
|
|
sensor_selection.gyro_device_id = _instance[ekf_instance].estimator_status.gyro_device_id;
|
|
sensor_selection.timestamp = hrt_absolute_time();
|
|
_sensor_selection_pub.publish(sensor_selection);
|
|
|
|
if (_selected_instance != INVALID_INSTANCE) {
|
|
// switch callback registration
|
|
_instance[_selected_instance].estimator_attitude_sub.unregisterCallback();
|
|
_instance[_selected_instance].estimator_status_sub.unregisterCallback();
|
|
|
|
if (!_instance[_selected_instance].healthy) {
|
|
PX4_WARN("primary EKF changed %d (unhealthy) -> %d", _selected_instance, ekf_instance);
|
|
}
|
|
}
|
|
|
|
_instance[ekf_instance].estimator_attitude_sub.registerCallback();
|
|
_instance[ekf_instance].estimator_status_sub.registerCallback();
|
|
|
|
_selected_instance = ekf_instance;
|
|
_instance_changed_count++;
|
|
_last_instance_change = sensor_selection.timestamp;
|
|
_instance[ekf_instance].time_last_selected = _last_instance_change;
|
|
|
|
// reset all relative test ratios
|
|
for (auto &inst : _instance) {
|
|
inst.relative_test_ratio = 0;
|
|
}
|
|
|
|
// publish new data immediately with resets
|
|
PublishVehicleAttitude(true);
|
|
PublishVehicleLocalPosition(true);
|
|
PublishVehicleGlobalPosition(true);
|
|
}
|
|
}
|
|
|
|
bool EKF2Selector::UpdateErrorScores()
|
|
{
|
|
bool updated = false;
|
|
|
|
// first check imu inconsistencies
|
|
_gyro_fault_detected = false;
|
|
uint32_t faulty_gyro_id = 0;
|
|
_accel_fault_detected = false;
|
|
uint32_t faulty_accel_id = 0;
|
|
|
|
if (_sensors_status_imu.updated()) {
|
|
sensors_status_imu_s sensors_status_imu;
|
|
|
|
if (_sensors_status_imu.copy(&sensors_status_imu)) {
|
|
const float angle_rate_threshold = radians(_param_ekf2_sel_imu_angle_rate.get());
|
|
const float angle_threshold = radians(_param_ekf2_sel_imu_angle.get());
|
|
const float accel_threshold = _param_ekf2_sel_imu_accel.get();
|
|
const float velocity_threshold = _param_ekf2_sel_imu_velocity.get();
|
|
const float time_step_s = constrain((sensors_status_imu.timestamp - _last_update_us) * 1e-6f, 0.f, 0.02f);
|
|
_last_update_us = sensors_status_imu.timestamp;
|
|
|
|
uint8_t n_gyros = 0;
|
|
uint8_t n_accels = 0;
|
|
uint8_t n_gyro_exceedances = 0;
|
|
uint8_t n_accel_exceedances = 0;
|
|
float largest_accumulated_gyro_error = 0.0f;
|
|
float largest_accumulated_accel_error = 0.0f;
|
|
uint8_t largest_gyro_error_index = 0;
|
|
uint8_t largest_accel_error_index = 0;
|
|
|
|
for (unsigned i = 0; i < IMU_STATUS_SIZE; i++) {
|
|
// check for gyros with excessive difference to mean using accumulated error
|
|
if (sensors_status_imu.gyro_device_ids[i] != 0) {
|
|
n_gyros++;
|
|
_accumulated_gyro_error[i] += (sensors_status_imu.gyro_inconsistency_rad_s[i] - angle_rate_threshold) * time_step_s;
|
|
_accumulated_gyro_error[i] = fmaxf(_accumulated_gyro_error[i], 0.f);
|
|
|
|
if (_accumulated_gyro_error[i] > angle_threshold) {
|
|
n_gyro_exceedances++;
|
|
}
|
|
|
|
if (_accumulated_gyro_error[i] > largest_accumulated_gyro_error) {
|
|
largest_accumulated_gyro_error = _accumulated_gyro_error[i];
|
|
largest_gyro_error_index = i;
|
|
}
|
|
|
|
} else {
|
|
// no sensor
|
|
_accumulated_gyro_error[i] = 0.f;
|
|
}
|
|
|
|
// check for accelerometers with excessive difference to mean using accumulated error
|
|
if (sensors_status_imu.accel_device_ids[i] != 0) {
|
|
n_accels++;
|
|
_accumulated_accel_error[i] += (sensors_status_imu.accel_inconsistency_m_s_s[i] - accel_threshold) * time_step_s;
|
|
_accumulated_accel_error[i] = fmaxf(_accumulated_accel_error[i], 0.f);
|
|
|
|
if (_accumulated_accel_error[i] > velocity_threshold) {
|
|
n_accel_exceedances++;
|
|
}
|
|
|
|
if (_accumulated_accel_error[i] > largest_accumulated_accel_error) {
|
|
largest_accumulated_accel_error = _accumulated_accel_error[i];
|
|
largest_accel_error_index = i;
|
|
}
|
|
|
|
} else {
|
|
// no sensor
|
|
_accumulated_accel_error[i] = 0.f;
|
|
}
|
|
}
|
|
|
|
if (n_gyro_exceedances > 0) {
|
|
if (n_gyros >= 3) {
|
|
// If there are 3 or more sensors, the one with the largest accumulated error is faulty
|
|
_gyro_fault_detected = true;
|
|
faulty_gyro_id = _instance[largest_gyro_error_index].estimator_status.gyro_device_id;
|
|
|
|
} else if (n_gyros == 2) {
|
|
// A fault is present, but the faulty sensor identity cannot be determined
|
|
_gyro_fault_detected = true;
|
|
}
|
|
}
|
|
|
|
if (n_accel_exceedances > 0) {
|
|
if (n_accels >= 3) {
|
|
// If there are 3 or more sensors, the one with the largest accumulated error is faulty
|
|
_accel_fault_detected = true;
|
|
faulty_accel_id = _instance[largest_accel_error_index].estimator_status.accel_device_id;
|
|
|
|
} else if (n_accels == 2) {
|
|
// A fault is present, but the faulty sensor identity cannot be determined
|
|
_accel_fault_detected = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
bool primary_updated = false;
|
|
|
|
// calculate individual error scores
|
|
for (uint8_t i = 0; i < EKF2_MAX_INSTANCES; i++) {
|
|
const bool prev_healthy = _instance[i].healthy;
|
|
|
|
if (_instance[i].estimator_status_sub.update(&_instance[i].estimator_status)) {
|
|
|
|
if ((i + 1) > _available_instances) {
|
|
_available_instances = i + 1;
|
|
updated = true;
|
|
}
|
|
|
|
if (i == _selected_instance) {
|
|
primary_updated = true;
|
|
}
|
|
|
|
const estimator_status_s &status = _instance[i].estimator_status;
|
|
|
|
const bool tilt_align = status.control_mode_flags & (1 << estimator_status_s::CS_TILT_ALIGN);
|
|
const bool yaw_align = status.control_mode_flags & (1 << estimator_status_s::CS_YAW_ALIGN);
|
|
|
|
_instance[i].combined_test_ratio = 0.0f;
|
|
|
|
if (tilt_align && yaw_align) {
|
|
_instance[i].combined_test_ratio = fmaxf(_instance[i].combined_test_ratio,
|
|
0.5f * (status.vel_test_ratio + status.pos_test_ratio));
|
|
_instance[i].combined_test_ratio = fmaxf(_instance[i].combined_test_ratio, status.hgt_test_ratio);
|
|
}
|
|
|
|
_instance[i].healthy = tilt_align && yaw_align && (status.filter_fault_flags == 0);
|
|
|
|
} else if (hrt_elapsed_time(&_instance[i].estimator_status.timestamp) > 20_ms) {
|
|
_instance[i].healthy = false;
|
|
}
|
|
|
|
// if the gyro used by the EKF is faulty, declare the EKF unhealthy without delay
|
|
if (_gyro_fault_detected && _instance[i].estimator_status.gyro_device_id == faulty_gyro_id) {
|
|
_instance[i].healthy = false;
|
|
}
|
|
|
|
// if the accelerometer used by the EKF is faulty, declare the EKF unhealthy without delay
|
|
if (_accel_fault_detected && _instance[i].estimator_status.accel_device_id == faulty_accel_id) {
|
|
_instance[i].healthy = false;
|
|
}
|
|
|
|
if (prev_healthy != _instance[i].healthy) {
|
|
updated = true;
|
|
}
|
|
}
|
|
|
|
// update relative test ratios if primary has updated
|
|
if (primary_updated) {
|
|
for (uint8_t i = 0; i < _available_instances; i++) {
|
|
if (i != _selected_instance) {
|
|
|
|
const float error_delta = _instance[i].combined_test_ratio - _instance[_selected_instance].combined_test_ratio;
|
|
|
|
// reduce error only if its better than the primary instance by at least EKF2_SEL_ERR_RED to prevent unnecessary selection changes
|
|
const float threshold = _gyro_fault_detected ? fmaxf(_param_ekf2_sel_err_red.get(), 0.05f) : 0.0f;
|
|
|
|
if (error_delta > 0 || error_delta < -threshold) {
|
|
_instance[i].relative_test_ratio += error_delta;
|
|
_instance[i].relative_test_ratio = constrain(_instance[i].relative_test_ratio, -_rel_err_score_lim, _rel_err_score_lim);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return (primary_updated || updated);
|
|
}
|
|
|
|
void EKF2Selector::PublishVehicleAttitude(bool reset)
|
|
{
|
|
vehicle_attitude_s attitude;
|
|
|
|
if (_instance[_selected_instance].estimator_attitude_sub.copy(&attitude)) {
|
|
if (reset) {
|
|
// on reset compute deltas from last published data
|
|
++_quat_reset_counter;
|
|
|
|
_delta_q_reset = (Quatf(attitude.q) * Quatf(_attitude_last.q).inversed()).normalized();
|
|
|
|
// ensure monotonically increasing timestamp_sample through reset
|
|
attitude.timestamp_sample = max(attitude.timestamp_sample, _attitude_last.timestamp_sample);
|
|
|
|
} else {
|
|
// otherwise propogate deltas from estimator data while maintaining the overall reset counts
|
|
if (attitude.quat_reset_counter > _attitude_last.quat_reset_counter) {
|
|
++_quat_reset_counter;
|
|
_delta_q_reset = Quatf{attitude.delta_q_reset};
|
|
}
|
|
}
|
|
|
|
// save last primary estimator_attitude
|
|
_attitude_last = attitude;
|
|
|
|
// republish with total reset count and current timestamp
|
|
attitude.quat_reset_counter = _quat_reset_counter;
|
|
_delta_q_reset.copyTo(attitude.delta_q_reset);
|
|
|
|
attitude.timestamp = hrt_absolute_time();
|
|
_vehicle_attitude_pub.publish(attitude);
|
|
}
|
|
}
|
|
|
|
void EKF2Selector::PublishVehicleLocalPosition(bool reset)
|
|
{
|
|
// vehicle_local_position
|
|
vehicle_local_position_s local_position;
|
|
|
|
if (_instance[_selected_instance].estimator_local_position_sub.copy(&local_position)) {
|
|
if (reset) {
|
|
// on reset compute deltas from last published data
|
|
++_xy_reset_counter;
|
|
++_z_reset_counter;
|
|
++_vxy_reset_counter;
|
|
++_vz_reset_counter;
|
|
++_heading_reset_counter;
|
|
|
|
_delta_xy_reset = Vector2f{local_position.x, local_position.y} - Vector2f{_local_position_last.x, _local_position_last.y};
|
|
_delta_z_reset = local_position.z - _local_position_last.z;
|
|
_delta_vxy_reset = Vector2f{local_position.vx, local_position.vy} - Vector2f{_local_position_last.vx, _local_position_last.vy};
|
|
_delta_vz_reset = local_position.vz - _local_position_last.vz;
|
|
_delta_heading_reset = matrix::wrap_2pi(local_position.heading - _local_position_last.heading);
|
|
|
|
// ensure monotonically increasing timestamp_sample through reset
|
|
local_position.timestamp_sample = max(local_position.timestamp_sample, _local_position_last.timestamp_sample);
|
|
|
|
} else {
|
|
// otherwise propogate deltas from estimator data while maintaining the overall reset counts
|
|
|
|
// XY reset
|
|
if (local_position.xy_reset_counter > _local_position_last.xy_reset_counter) {
|
|
++_xy_reset_counter;
|
|
_delta_xy_reset = Vector2f{local_position.delta_xy};
|
|
}
|
|
|
|
// Z reset
|
|
if (local_position.z_reset_counter > _local_position_last.z_reset_counter) {
|
|
++_z_reset_counter;
|
|
_delta_z_reset = local_position.delta_z;
|
|
}
|
|
|
|
// VXY reset
|
|
if (local_position.vxy_reset_counter > _local_position_last.vxy_reset_counter) {
|
|
++_vxy_reset_counter;
|
|
_delta_vxy_reset = Vector2f{local_position.delta_vxy};
|
|
}
|
|
|
|
// VZ reset
|
|
if (local_position.vz_reset_counter > _local_position_last.vz_reset_counter) {
|
|
++_vz_reset_counter;
|
|
_delta_z_reset = local_position.delta_vz;
|
|
}
|
|
|
|
// heading reset
|
|
if (local_position.heading_reset_counter > _local_position_last.heading_reset_counter) {
|
|
++_heading_reset_counter;
|
|
_delta_heading_reset = local_position.delta_heading;
|
|
}
|
|
}
|
|
|
|
// save last primary estimator_local_position
|
|
_local_position_last = local_position;
|
|
|
|
// publish estimator's local position for system (vehicle_local_position) unless it's stale
|
|
if (local_position.timestamp >= _instance[_selected_instance].estimator_status.timestamp_sample) {
|
|
// republish with total reset count and current timestamp
|
|
local_position.xy_reset_counter = _xy_reset_counter;
|
|
local_position.z_reset_counter = _z_reset_counter;
|
|
local_position.vxy_reset_counter = _vxy_reset_counter;
|
|
local_position.vz_reset_counter = _vz_reset_counter;
|
|
local_position.heading_reset_counter = _heading_reset_counter;
|
|
|
|
_delta_xy_reset.copyTo(local_position.delta_xy);
|
|
local_position.delta_z = _delta_z_reset;
|
|
_delta_vxy_reset.copyTo(local_position.delta_vxy);
|
|
local_position.delta_vz = _delta_vz_reset;
|
|
local_position.delta_heading = _delta_heading_reset;
|
|
|
|
local_position.timestamp = hrt_absolute_time();
|
|
_vehicle_local_position_pub.publish(local_position);
|
|
}
|
|
}
|
|
}
|
|
|
|
void EKF2Selector::PublishVehicleGlobalPosition(bool reset)
|
|
{
|
|
vehicle_global_position_s global_position;
|
|
|
|
if (_instance[_selected_instance].estimator_global_position_sub.copy(&global_position)) {
|
|
if (reset) {
|
|
// on reset compute deltas from last published data
|
|
++_lat_lon_reset_counter;
|
|
++_alt_reset_counter;
|
|
|
|
_delta_lat_reset = global_position.lat - _global_position_last.lat;
|
|
_delta_lon_reset = global_position.lon - _global_position_last.lon;
|
|
_delta_alt_reset = global_position.delta_alt - _global_position_last.delta_alt;
|
|
|
|
// ensure monotonically increasing timestamp_sample through reset
|
|
global_position.timestamp_sample = max(global_position.timestamp_sample, _global_position_last.timestamp_sample);
|
|
|
|
} else {
|
|
// otherwise propogate deltas from estimator data while maintaining the overall reset counts
|
|
|
|
// lat/lon reset
|
|
if (global_position.lat_lon_reset_counter > _global_position_last.lat_lon_reset_counter) {
|
|
++_lat_lon_reset_counter;
|
|
|
|
// TODO: delta latitude/longitude
|
|
//_delta_lat_reset = global_position.delta_lat;
|
|
//_delta_lon_reset = global_position.delta_lon;
|
|
}
|
|
|
|
// alt reset
|
|
if (global_position.alt_reset_counter > _global_position_last.alt_reset_counter) {
|
|
++_alt_reset_counter;
|
|
_delta_alt_reset = global_position.delta_alt;
|
|
}
|
|
}
|
|
|
|
// save last primary estimator_global_position
|
|
_global_position_last = global_position;
|
|
|
|
// publish estimator's global position for system (vehicle_global_position) unless it's stale
|
|
if (global_position.timestamp >= _instance[_selected_instance].estimator_status.timestamp_sample) {
|
|
// republish with total reset count and current timestamp
|
|
global_position.lat_lon_reset_counter = _lat_lon_reset_counter;
|
|
global_position.alt_reset_counter = _alt_reset_counter;
|
|
global_position.delta_alt = _delta_alt_reset;
|
|
|
|
global_position.timestamp = hrt_absolute_time();
|
|
_vehicle_global_position_pub.publish(global_position);
|
|
}
|
|
}
|
|
}
|
|
|
|
void EKF2Selector::Run()
|
|
{
|
|
// re-schedule as backup timeout
|
|
ScheduleDelayed(10_ms);
|
|
|
|
// check for parameter updates
|
|
if (_parameter_update_sub.updated()) {
|
|
// clear update
|
|
parameter_update_s pupdate;
|
|
_parameter_update_sub.copy(&pupdate);
|
|
|
|
// update parameters from storage
|
|
updateParams();
|
|
}
|
|
|
|
// update combined test ratio for all estimators
|
|
const bool updated = UpdateErrorScores();
|
|
|
|
if (updated) {
|
|
bool lower_error_available = false;
|
|
float alternative_error = 0.f; // looking for instances that have error lower than the current primary
|
|
uint8_t best_ekf_instance = _selected_instance;
|
|
|
|
// loop through all available instances to find if an alternative is available
|
|
for (int i = 0; i < _available_instances; i++) {
|
|
// Use an alternative instance if -
|
|
// (healthy and has updated recently)
|
|
// AND
|
|
// (has relative error less than selected instance and has not been the selected instance for at least 10 seconds
|
|
// OR
|
|
// selected instance has stopped updating
|
|
if (_instance[i].healthy && (i != _selected_instance)) {
|
|
const float relative_error = _instance[i].relative_test_ratio;
|
|
|
|
if (relative_error < alternative_error) {
|
|
|
|
alternative_error = relative_error;
|
|
best_ekf_instance = i;
|
|
|
|
// relative error less than selected instance and has not been the selected instance for at least 10 seconds
|
|
if ((relative_error <= -_rel_err_thresh) && hrt_elapsed_time(&_instance[i].time_last_selected) > 10_s) {
|
|
lower_error_available = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((_selected_instance == INVALID_INSTANCE)
|
|
|| (!_instance[_selected_instance].healthy && (best_ekf_instance == _selected_instance))) {
|
|
|
|
// force initial selection
|
|
uint8_t best_instance = _selected_instance;
|
|
float best_test_ratio = FLT_MAX;
|
|
|
|
for (int i = 0; i < _available_instances; i++) {
|
|
if (_instance[i].healthy) {
|
|
const float test_ratio = _instance[i].combined_test_ratio;
|
|
|
|
if ((test_ratio > 0) && (test_ratio < best_test_ratio)) {
|
|
best_instance = i;
|
|
best_test_ratio = test_ratio;
|
|
}
|
|
}
|
|
}
|
|
|
|
SelectInstance(best_instance);
|
|
|
|
} else if (best_ekf_instance != _selected_instance) {
|
|
// if this instance has a significantly lower relative error to the active primary, we consider it as a
|
|
// better instance and would like to switch to it even if the current primary is healthy
|
|
// switch immediately if the current selected is no longer healthy
|
|
if ((lower_error_available && hrt_elapsed_time(&_last_instance_change) > 10_s)
|
|
|| !_instance[_selected_instance].healthy) {
|
|
|
|
SelectInstance(best_ekf_instance);
|
|
}
|
|
}
|
|
|
|
// publish selector status
|
|
estimator_selector_status_s selector_status{};
|
|
selector_status.primary_instance = _selected_instance;
|
|
selector_status.instances_available = _available_instances;
|
|
selector_status.instance_changed_count = _instance_changed_count;
|
|
selector_status.last_instance_change = _last_instance_change;
|
|
selector_status.accel_device_id = _instance[_selected_instance].estimator_status.accel_device_id;
|
|
selector_status.baro_device_id = _instance[_selected_instance].estimator_status.baro_device_id;
|
|
selector_status.gyro_device_id = _instance[_selected_instance].estimator_status.gyro_device_id;
|
|
selector_status.mag_device_id = _instance[_selected_instance].estimator_status.mag_device_id;
|
|
selector_status.gyro_fault_detected = _gyro_fault_detected;
|
|
selector_status.accel_fault_detected = _accel_fault_detected;
|
|
|
|
for (int i = 0; i < EKF2_MAX_INSTANCES; i++) {
|
|
selector_status.combined_test_ratio[i] = _instance[i].combined_test_ratio;
|
|
selector_status.relative_test_ratio[i] = _instance[i].relative_test_ratio;
|
|
selector_status.healthy[i] = _instance[i].healthy;
|
|
}
|
|
|
|
for (int i = 0; i < IMU_STATUS_SIZE; i++) {
|
|
selector_status.accumulated_gyro_error[i] = _accumulated_gyro_error[i];
|
|
selector_status.accumulated_accel_error[i] = _accumulated_accel_error[i];
|
|
}
|
|
|
|
selector_status.timestamp = hrt_absolute_time();
|
|
_estimator_selector_status_pub.publish(selector_status);
|
|
}
|
|
|
|
// republish selected estimator data for system
|
|
if (_selected_instance != INVALID_INSTANCE) {
|
|
// selected estimator_attitude -> vehicle_attitude
|
|
if (_instance[_selected_instance].estimator_attitude_sub.updated()) {
|
|
PublishVehicleAttitude();
|
|
}
|
|
|
|
// selected estimator_local_position -> vehicle_local_position
|
|
if (_instance[_selected_instance].estimator_local_position_sub.updated()) {
|
|
PublishVehicleLocalPosition();
|
|
}
|
|
|
|
// selected estimator_global_position -> vehicle_global_position
|
|
if (_instance[_selected_instance].estimator_global_position_sub.updated()) {
|
|
PublishVehicleGlobalPosition();
|
|
}
|
|
|
|
// selected estimator_odometry -> vehicle_odometry
|
|
if (_instance[_selected_instance].estimator_odometry_sub.updated()) {
|
|
vehicle_odometry_s vehicle_odometry;
|
|
|
|
if (_instance[_selected_instance].estimator_odometry_sub.update(&vehicle_odometry)) {
|
|
if (vehicle_odometry.timestamp >= _instance[_selected_instance].estimator_status.timestamp_sample) {
|
|
vehicle_odometry.timestamp = hrt_absolute_time();
|
|
_vehicle_odometry_pub.publish(vehicle_odometry);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (_lockstep_component == -1) {
|
|
_lockstep_component = px4_lockstep_register_component();
|
|
}
|
|
|
|
px4_lockstep_progress(_lockstep_component);
|
|
}
|
|
|
|
void EKF2Selector::PrintStatus()
|
|
{
|
|
PX4_INFO("available instances: %d", _available_instances);
|
|
|
|
if (_selected_instance == INVALID_INSTANCE) {
|
|
PX4_WARN("selected instance: None");
|
|
}
|
|
|
|
for (int i = 0; i < _available_instances; i++) {
|
|
const EstimatorInstance &inst = _instance[i];
|
|
|
|
PX4_INFO("%d: ACC: %d, GYRO: %d, MAG: %d, %s, test ratio: %.7f (%.5f) %s",
|
|
inst.instance, inst.estimator_status.accel_device_id, inst.estimator_status.gyro_device_id,
|
|
inst.estimator_status.mag_device_id,
|
|
inst.healthy ? "healthy" : "unhealthy",
|
|
(double)inst.combined_test_ratio, (double)inst.relative_test_ratio,
|
|
(_selected_instance == i) ? "*" : "");
|
|
}
|
|
}
|