mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-14 12:48:31 -04:00
f31cf7debe
also implement replacement mechanism for UAVCAN compasses
501 lines
15 KiB
C++
501 lines
15 KiB
C++
#include <AP_HAL/AP_HAL.h>
|
|
#include <AP_Notify/AP_Notify.h>
|
|
#include <AP_GPS/AP_GPS.h>
|
|
#include <GCS_MAVLink/GCS.h>
|
|
#include <AP_AHRS/AP_AHRS.h>
|
|
#include <AP_GPS/AP_GPS.h>
|
|
|
|
#include "AP_Compass.h"
|
|
|
|
const extern AP_HAL::HAL& hal;
|
|
|
|
#if COMPASS_CAL_ENABLED
|
|
|
|
void Compass::cal_update()
|
|
{
|
|
if (hal.util->get_soft_armed()) {
|
|
return;
|
|
}
|
|
|
|
bool running = false;
|
|
|
|
for (Priority i(0); i<COMPASS_MAX_INSTANCES; i++) {
|
|
bool failure;
|
|
_calibrator[i].update(failure);
|
|
if (failure) {
|
|
AP_Notify::events.compass_cal_failed = 1;
|
|
}
|
|
|
|
if (_calibrator[i].check_for_timeout()) {
|
|
AP_Notify::events.compass_cal_failed = 1;
|
|
cancel_calibration_all();
|
|
}
|
|
|
|
if (_calibrator[i].running()) {
|
|
running = true;
|
|
} else if (_cal_autosave && !_cal_saved[i] && _calibrator[i].get_status() == CompassCalibrator::Status::SUCCESS) {
|
|
_accept_calibration(uint8_t(i));
|
|
}
|
|
}
|
|
|
|
AP_Notify::flags.compass_cal_running = running;
|
|
|
|
if (is_calibrating()) {
|
|
_cal_has_run = true;
|
|
return;
|
|
} else if (_cal_has_run && _auto_reboot()) {
|
|
hal.scheduler->delay(1000);
|
|
hal.scheduler->reboot(false);
|
|
}
|
|
}
|
|
|
|
bool Compass::_start_calibration(uint8_t i, bool retry, float delay)
|
|
{
|
|
if (!healthy(i)) {
|
|
return false;
|
|
}
|
|
if (!use_for_yaw(i)) {
|
|
return false;
|
|
}
|
|
Priority prio = Priority(i);
|
|
if (_priority_did_list[prio] != _priority_did_stored_list[prio]) {
|
|
gcs().send_text(MAV_SEVERITY_ERROR, "Compass cal requires reboot after priority change");
|
|
return false;
|
|
}
|
|
if (_options.get() & uint16_t(Option::CAL_REQUIRE_GPS)) {
|
|
if (AP::gps().status() < AP_GPS::GPS_OK_FIX_2D) {
|
|
gcs().send_text(MAV_SEVERITY_ERROR, "Compass cal requires GPS lock");
|
|
return false;
|
|
}
|
|
}
|
|
if (!is_calibrating()) {
|
|
AP_Notify::events.initiated_compass_cal = 1;
|
|
}
|
|
if (i == 0 && _get_state(prio).external != 0) {
|
|
_calibrator[prio].set_tolerance(_calibration_threshold);
|
|
} else {
|
|
// internal compasses or secondary compasses get twice the
|
|
// threshold. This is because internal compasses tend to be a
|
|
// lot noisier
|
|
_calibrator[prio].set_tolerance(_calibration_threshold*2);
|
|
}
|
|
if (_rotate_auto) {
|
|
enum Rotation r = _get_state(prio).external?(enum Rotation)_get_state(prio).orientation.get():ROTATION_NONE;
|
|
if (r != ROTATION_CUSTOM) {
|
|
_calibrator[prio].set_orientation(r, _get_state(prio).external, _rotate_auto>=2);
|
|
}
|
|
}
|
|
_cal_saved[prio] = false;
|
|
_calibrator[prio].start(retry, delay, get_offsets_max(), i);
|
|
|
|
// disable compass learning both for calibration and after completion
|
|
_learn.set_and_save(0);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Compass::_start_calibration_mask(uint8_t mask, bool retry, bool autosave, float delay, bool autoreboot)
|
|
{
|
|
_cal_autosave = autosave;
|
|
_compass_cal_autoreboot = autoreboot;
|
|
|
|
for (uint8_t i=0; i<COMPASS_MAX_INSTANCES; i++) {
|
|
if ((1<<i) & mask) {
|
|
if (!_start_calibration(i,retry,delay)) {
|
|
_cancel_calibration_mask(mask);
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void Compass::start_calibration_all(bool retry, bool autosave, float delay, bool autoreboot)
|
|
{
|
|
_cal_autosave = autosave;
|
|
_compass_cal_autoreboot = autoreboot;
|
|
|
|
for (uint8_t i=0; i<COMPASS_MAX_INSTANCES; i++) {
|
|
// ignore any compasses that fail to start calibrating
|
|
// start all should only calibrate compasses that are being used
|
|
_start_calibration(i,retry,delay);
|
|
}
|
|
}
|
|
|
|
void Compass::_cancel_calibration(uint8_t i)
|
|
{
|
|
AP_Notify::events.initiated_compass_cal = 0;
|
|
Priority prio = Priority(i);
|
|
|
|
if (_calibrator[prio].running() || _calibrator[prio].get_status() == CompassCalibrator::Status::WAITING_TO_START) {
|
|
AP_Notify::events.compass_cal_canceled = 1;
|
|
}
|
|
_cal_saved[prio] = false;
|
|
_calibrator[prio].stop();
|
|
}
|
|
|
|
void Compass::_cancel_calibration_mask(uint8_t mask)
|
|
{
|
|
for (uint8_t i=0; i<COMPASS_MAX_INSTANCES; i++) {
|
|
if ((1<<i) & mask) {
|
|
_cancel_calibration(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Compass::cancel_calibration_all()
|
|
{
|
|
_cancel_calibration_mask(0xFF);
|
|
}
|
|
|
|
bool Compass::_accept_calibration(uint8_t i)
|
|
{
|
|
Priority prio = Priority(i);
|
|
|
|
CompassCalibrator& cal = _calibrator[prio];
|
|
const CompassCalibrator::Status cal_status = cal.get_status();
|
|
|
|
if (_cal_saved[prio] || cal_status == CompassCalibrator::Status::NOT_STARTED) {
|
|
return true;
|
|
} else if (cal_status == CompassCalibrator::Status::SUCCESS) {
|
|
_cal_complete_requires_reboot = true;
|
|
_cal_saved[prio] = true;
|
|
|
|
Vector3f ofs, diag, offdiag;
|
|
float scale_factor;
|
|
cal.get_calibration(ofs, diag, offdiag, scale_factor);
|
|
|
|
set_and_save_offsets(i, ofs);
|
|
set_and_save_diagonals(i,diag);
|
|
set_and_save_offdiagonals(i,offdiag);
|
|
set_and_save_scale_factor(i,scale_factor);
|
|
|
|
if (_get_state(prio).external && _rotate_auto >= 2) {
|
|
set_and_save_orientation(i, cal.get_orientation());
|
|
}
|
|
|
|
if (!is_calibrating()) {
|
|
AP_Notify::events.compass_cal_saved = 1;
|
|
}
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool Compass::_accept_calibration_mask(uint8_t mask)
|
|
{
|
|
bool success = true;
|
|
for (Priority i(0); i<COMPASS_MAX_INSTANCES; i++) {
|
|
if ((1<<uint8_t(i)) & mask) {
|
|
if (!_accept_calibration(uint8_t(i))) {
|
|
success = false;
|
|
}
|
|
_calibrator[i].stop();
|
|
}
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
bool Compass::send_mag_cal_progress(const GCS_MAVLINK& link)
|
|
{
|
|
uint8_t cal_mask = _get_cal_mask();
|
|
|
|
for (Priority compass_id(0); compass_id<COMPASS_MAX_INSTANCES; compass_id++) {
|
|
// ensure we don't try to send with no space available
|
|
if (!HAVE_PAYLOAD_SPACE(link.get_chan(), MAG_CAL_PROGRESS)) {
|
|
// ideally we would only send one progress message per
|
|
// call. If we don't return true here we may end up
|
|
// hogging *all* the bandwidth
|
|
return true;
|
|
}
|
|
|
|
auto& calibrator = _calibrator[compass_id];
|
|
const CompassCalibrator::Status cal_status = calibrator.get_status();
|
|
|
|
if (cal_status == CompassCalibrator::Status::WAITING_TO_START ||
|
|
cal_status == CompassCalibrator::Status::RUNNING_STEP_ONE ||
|
|
cal_status == CompassCalibrator::Status::RUNNING_STEP_TWO) {
|
|
uint8_t completion_pct = calibrator.get_completion_percent();
|
|
const CompassCalibrator::completion_mask_t& completion_mask = calibrator.get_completion_mask();
|
|
const Vector3f direction;
|
|
uint8_t attempt = _calibrator[compass_id].get_attempt();
|
|
|
|
mavlink_msg_mag_cal_progress_send(
|
|
link.get_chan(),
|
|
uint8_t(compass_id), cal_mask,
|
|
(uint8_t)cal_status, attempt, completion_pct, completion_mask,
|
|
direction.x, direction.y, direction.z
|
|
);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Compass::send_mag_cal_report(const GCS_MAVLINK& link)
|
|
{
|
|
uint8_t cal_mask = _get_cal_mask();
|
|
|
|
for (Priority compass_id(0); compass_id<COMPASS_MAX_INSTANCES; compass_id++) {
|
|
// ensure we don't try to send with no space available
|
|
if (!HAVE_PAYLOAD_SPACE(link.get_chan(), MAG_CAL_REPORT)) {
|
|
// ideally we would only send one progress message per
|
|
// call. If we don't return true here we may end up
|
|
// hogging *all* the bandwidth
|
|
return true;
|
|
}
|
|
|
|
const CompassCalibrator::Status cal_status = _calibrator[compass_id].get_status();
|
|
if (cal_status == CompassCalibrator::Status::SUCCESS ||
|
|
cal_status == CompassCalibrator::Status::FAILED ||
|
|
cal_status == CompassCalibrator::Status::BAD_ORIENTATION) {
|
|
float fitness = _calibrator[compass_id].get_fitness();
|
|
Vector3f ofs, diag, offdiag;
|
|
float scale_factor;
|
|
_calibrator[compass_id].get_calibration(ofs, diag, offdiag, scale_factor);
|
|
uint8_t autosaved = _cal_saved[compass_id];
|
|
|
|
mavlink_msg_mag_cal_report_send(
|
|
link.get_chan(),
|
|
uint8_t(compass_id), cal_mask,
|
|
(uint8_t)cal_status, autosaved,
|
|
fitness,
|
|
ofs.x, ofs.y, ofs.z,
|
|
diag.x, diag.y, diag.z,
|
|
offdiag.x, offdiag.y, offdiag.z,
|
|
_calibrator[compass_id].get_orientation_confidence(),
|
|
_calibrator[compass_id].get_original_orientation(),
|
|
_calibrator[compass_id].get_orientation(),
|
|
scale_factor
|
|
);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Compass::is_calibrating() const
|
|
{
|
|
for (Priority i(0); i<COMPASS_MAX_INSTANCES; i++) {
|
|
switch(_calibrator[i].get_status()) {
|
|
case CompassCalibrator::Status::NOT_STARTED:
|
|
case CompassCalibrator::Status::SUCCESS:
|
|
case CompassCalibrator::Status::FAILED:
|
|
case CompassCalibrator::Status::BAD_ORIENTATION:
|
|
break;
|
|
default:
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
uint8_t Compass::_get_cal_mask() const
|
|
{
|
|
uint8_t cal_mask = 0;
|
|
for (Priority i(0); i<COMPASS_MAX_INSTANCES; i++) {
|
|
if (_calibrator[i].get_status() != CompassCalibrator::Status::NOT_STARTED) {
|
|
cal_mask |= 1 << uint8_t(i);
|
|
}
|
|
}
|
|
return cal_mask;
|
|
}
|
|
|
|
/*
|
|
handle an incoming MAG_CAL command
|
|
*/
|
|
MAV_RESULT Compass::handle_mag_cal_command(const mavlink_command_long_t &packet)
|
|
{
|
|
MAV_RESULT result = MAV_RESULT_FAILED;
|
|
|
|
switch (packet.command) {
|
|
case MAV_CMD_DO_START_MAG_CAL: {
|
|
result = MAV_RESULT_ACCEPTED;
|
|
if (hal.util->get_soft_armed()) {
|
|
gcs().send_text(MAV_SEVERITY_NOTICE, "Disarm to allow compass calibration");
|
|
result = MAV_RESULT_FAILED;
|
|
break;
|
|
}
|
|
if (packet.param1 < 0 || packet.param1 > 255) {
|
|
result = MAV_RESULT_FAILED;
|
|
break;
|
|
}
|
|
|
|
uint8_t mag_mask = packet.param1;
|
|
bool retry = !is_zero(packet.param2);
|
|
bool autosave = !is_zero(packet.param3);
|
|
float delay = packet.param4;
|
|
bool autoreboot = !is_zero(packet.param5);
|
|
|
|
if (mag_mask == 0) { // 0 means all
|
|
_reset_compass_id();
|
|
start_calibration_all(retry, autosave, delay, autoreboot);
|
|
} else {
|
|
if (!_start_calibration_mask(mag_mask, retry, autosave, delay, autoreboot)) {
|
|
result = MAV_RESULT_FAILED;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case MAV_CMD_DO_ACCEPT_MAG_CAL: {
|
|
result = MAV_RESULT_ACCEPTED;
|
|
if (packet.param1 < 0 || packet.param1 > 255) {
|
|
result = MAV_RESULT_FAILED;
|
|
break;
|
|
}
|
|
|
|
uint8_t mag_mask = packet.param1;
|
|
|
|
if (mag_mask == 0) { // 0 means all
|
|
mag_mask = 0xFF;
|
|
}
|
|
|
|
if (!_accept_calibration_mask(mag_mask)) {
|
|
result = MAV_RESULT_FAILED;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case MAV_CMD_DO_CANCEL_MAG_CAL: {
|
|
result = MAV_RESULT_ACCEPTED;
|
|
if (packet.param1 < 0 || packet.param1 > 255) {
|
|
result = MAV_RESULT_FAILED;
|
|
break;
|
|
}
|
|
|
|
uint8_t mag_mask = packet.param1;
|
|
|
|
if (mag_mask == 0) { // 0 means all
|
|
cancel_calibration_all();
|
|
break;
|
|
}
|
|
|
|
_cancel_calibration_mask(mag_mask);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
get mag field with the effects of offsets, diagonals and
|
|
off-diagonals removed
|
|
*/
|
|
bool Compass::get_uncorrected_field(uint8_t instance, Vector3f &field)
|
|
{
|
|
// form eliptical correction matrix and invert it. This is
|
|
// needed to remove the effects of the eliptical correction
|
|
// when calculating new offsets
|
|
const Vector3f &diagonals = get_diagonals(instance);
|
|
const Vector3f &offdiagonals = get_offdiagonals(instance);
|
|
Matrix3f mat {
|
|
diagonals.x, offdiagonals.x, offdiagonals.y,
|
|
offdiagonals.x, diagonals.y, offdiagonals.z,
|
|
offdiagonals.y, offdiagonals.z, diagonals.z
|
|
};
|
|
if (!mat.invert()) {
|
|
return false;
|
|
}
|
|
|
|
// get corrected field
|
|
field = get_field(instance);
|
|
|
|
// remove impact of diagonals and off-diagonals
|
|
field = mat * field;
|
|
|
|
// remove impact of offsets
|
|
field -= get_offsets(instance);
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
fast compass calibration given vehicle position and yaw. This
|
|
results in zero diagonal and off-diagonal elements, so is only
|
|
suitable for vehicles where the field is close to spherical. It is
|
|
useful for large vehicles where moving the vehicle to calibrate it
|
|
is difficult.
|
|
|
|
The offsets of the selected compasses are set to values to bring
|
|
them into consistency with the WMM tables at the given latitude and
|
|
longitude. If compass_mask is zero then all enabled compasses are
|
|
calibrated.
|
|
|
|
This assumes that the compass is correctly scaled in milliGauss
|
|
*/
|
|
MAV_RESULT Compass::mag_cal_fixed_yaw(float yaw_deg, uint8_t compass_mask,
|
|
float lat_deg, float lon_deg)
|
|
{
|
|
_reset_compass_id();
|
|
if (is_zero(lat_deg) && is_zero(lon_deg)) {
|
|
Location loc;
|
|
// get AHRS position. If unavailable then try GPS location
|
|
if (!AP::ahrs().get_position(loc)) {
|
|
if (AP::gps().status() < AP_GPS::GPS_OK_FIX_3D) {
|
|
gcs().send_text(MAV_SEVERITY_ERROR, "Mag: no position available");
|
|
return MAV_RESULT_FAILED;
|
|
}
|
|
loc = AP::gps().location();
|
|
}
|
|
lat_deg = loc.lat * 1.0e-7;
|
|
lon_deg = loc.lng * 1.0e-7;
|
|
}
|
|
|
|
// get the magnetic field intensity and orientation
|
|
float intensity;
|
|
float declination;
|
|
float inclination;
|
|
if (!AP_Declination::get_mag_field_ef(lat_deg, lon_deg, intensity, declination, inclination)) {
|
|
gcs().send_text(MAV_SEVERITY_ERROR, "Mag: WMM table error");
|
|
return MAV_RESULT_FAILED;
|
|
}
|
|
|
|
// create a field vector and rotate to the required orientation
|
|
Vector3f field(1e3f * intensity, 0.0f, 0.0f);
|
|
Matrix3f R;
|
|
R.from_euler(0.0f, -ToRad(inclination), ToRad(declination));
|
|
field = R * field;
|
|
|
|
Matrix3f dcm;
|
|
dcm.from_euler(AP::ahrs().roll, AP::ahrs().pitch, radians(yaw_deg));
|
|
|
|
// Rotate into body frame using provided yaw
|
|
field = dcm.transposed() * field;
|
|
|
|
for (uint8_t i=0; i<COMPASS_MAX_INSTANCES; i++) {
|
|
if (compass_mask != 0 && ((1U<<i) & compass_mask) == 0) {
|
|
// skip this compass
|
|
continue;
|
|
}
|
|
if (!use_for_yaw(i)) {
|
|
continue;
|
|
}
|
|
if (!healthy(i)) {
|
|
gcs().send_text(MAV_SEVERITY_ERROR, "Mag[%u]: unhealthy\n", i);
|
|
return MAV_RESULT_FAILED;
|
|
}
|
|
|
|
Vector3f measurement;
|
|
if (!get_uncorrected_field(i, measurement)) {
|
|
gcs().send_text(MAV_SEVERITY_ERROR, "Mag[%u]: bad uncorrected field", i);
|
|
return MAV_RESULT_FAILED;
|
|
}
|
|
|
|
Vector3f offsets = field - measurement;
|
|
set_and_save_offsets(i, offsets);
|
|
Vector3f one{1,1,1};
|
|
set_and_save_diagonals(i, one);
|
|
Vector3f zero{0,0,0};
|
|
set_and_save_offdiagonals(i, zero);
|
|
}
|
|
|
|
return MAV_RESULT_ACCEPTED;
|
|
}
|
|
|
|
|
|
#endif // COMPASS_CAL_ENABLED
|