mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-04 15:08:28 -04:00
dcd7f9d26d
this allows us to detect when the user has the ports the wrong way around, and mark the sensor unhealthy
200 lines
6.1 KiB
C++
200 lines
6.1 KiB
C++
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
/*
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
/*
|
|
* APM_Airspeed.cpp - airspeed (pitot) driver
|
|
*/
|
|
|
|
|
|
#include <AP_HAL.h>
|
|
#include <AP_Math.h>
|
|
#include <AP_Common.h>
|
|
#include <AP_ADC.h>
|
|
#include <AP_Airspeed.h>
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_APM1
|
|
#include <AP_ADC_AnalogSource.h>
|
|
#define ARSPD_DEFAULT_PIN 64
|
|
#elif CONFIG_HAL_BOARD == HAL_BOARD_APM2
|
|
#define ARSPD_DEFAULT_PIN 0
|
|
#elif CONFIG_HAL_BOARD == HAL_BOARD_AVR_SITL
|
|
#define ARSPD_DEFAULT_PIN 1
|
|
#elif CONFIG_HAL_BOARD == HAL_BOARD_PX4
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <systemlib/airspeed.h>
|
|
#include <drivers/drv_airspeed.h>
|
|
#include <uORB/topics/differential_pressure.h>
|
|
#ifdef CONFIG_ARCH_BOARD_PX4FMU_V1
|
|
#define ARSPD_DEFAULT_PIN 11
|
|
#else
|
|
#define ARSPD_DEFAULT_PIN 15
|
|
#endif
|
|
#elif CONFIG_HAL_BOARD == HAL_BOARD_FLYMAPLE
|
|
#define ARSPD_DEFAULT_PIN 16
|
|
#elif CONFIG_HAL_BOARD == HAL_BOARD_LINUX
|
|
#define ARSPD_DEFAULT_PIN AP_AIRSPEED_I2C_PIN
|
|
#else
|
|
#define ARSPD_DEFAULT_PIN 0
|
|
#endif
|
|
|
|
// table of user settable parameters
|
|
const AP_Param::GroupInfo AP_Airspeed::var_info[] PROGMEM = {
|
|
|
|
// @Param: ENABLE
|
|
// @DisplayName: Airspeed enable
|
|
// @Description: enable airspeed sensor
|
|
// @Values: 0:Disable,1:Enable
|
|
AP_GROUPINFO("ENABLE", 0, AP_Airspeed, _enable, 1),
|
|
|
|
// @Param: USE
|
|
// @DisplayName: Airspeed use
|
|
// @Description: use airspeed for flight control
|
|
// @Values: 1:Use,0:Don't Use
|
|
AP_GROUPINFO("USE", 1, AP_Airspeed, _use, 0),
|
|
|
|
// @Param: OFFSET
|
|
// @DisplayName: Airspeed offset
|
|
// @Description: Airspeed calibration offset
|
|
// @Increment: 0.1
|
|
AP_GROUPINFO("OFFSET", 2, AP_Airspeed, _offset, 0),
|
|
|
|
// @Param: RATIO
|
|
// @DisplayName: Airspeed ratio
|
|
// @Description: Airspeed calibration ratio
|
|
// @Increment: 0.1
|
|
AP_GROUPINFO("RATIO", 3, AP_Airspeed, _ratio, 1.9936f),
|
|
|
|
// @Param: PIN
|
|
// @DisplayName: Airspeed pin
|
|
// @Description: The analog pin number that the airspeed sensor is connected to. Set this to 0..9 for the APM2 analog pins. Set to 64 on an APM1 for the dedicated airspeed port on the end of the board. Set to 11 on PX4 for the analog airspeed port. Set to 15 on the Pixhawk for the analog airspeed port. Set to 65 on the PX4 or Pixhawk for an EagleTree or MEAS I2C airspeed sensor.
|
|
// @User: Advanced
|
|
AP_GROUPINFO("PIN", 4, AP_Airspeed, _pin, ARSPD_DEFAULT_PIN),
|
|
|
|
// @Param: AUTOCAL
|
|
// @DisplayName: Automatic airspeed ratio calibration
|
|
// @Description: If this is enabled then the APM will automatically adjust the ARSPD_RATIO during flight, based upon an estimation filter using ground speed and true airspeed. The automatic calibration will save the new ratio to EEPROM every 2 minutes if it changes by more than 5%
|
|
// @User: Advanced
|
|
AP_GROUPINFO("AUTOCAL", 5, AP_Airspeed, _autocal, 0),
|
|
|
|
AP_GROUPEND
|
|
};
|
|
|
|
|
|
/*
|
|
this scaling factor converts from the old system where we used a
|
|
0 to 4095 raw ADC value for 0-5V to the new system which gets the
|
|
voltage in volts directly from the ADC driver
|
|
*/
|
|
#define SCALING_OLD_CALIBRATION 819 // 4095/5
|
|
|
|
void AP_Airspeed::init()
|
|
{
|
|
_last_pressure = 0;
|
|
_calibration.init(_ratio);
|
|
_last_saved_ratio = _ratio;
|
|
_counter = 0;
|
|
|
|
analog.init();
|
|
digital.init();
|
|
}
|
|
|
|
// read the airspeed sensor
|
|
float AP_Airspeed::get_pressure(void)
|
|
{
|
|
if (!_enable) {
|
|
return 0;
|
|
}
|
|
float pressure = 0;
|
|
if (_pin == AP_AIRSPEED_I2C_PIN) {
|
|
_healthy = digital.get_differential_pressure(pressure);
|
|
} else {
|
|
_healthy = analog.get_differential_pressure(pressure);
|
|
}
|
|
return pressure;
|
|
}
|
|
|
|
// get a temperature reading if possible
|
|
bool AP_Airspeed::get_temperature(float &temperature)
|
|
{
|
|
if (!_enable) {
|
|
return false;
|
|
}
|
|
if (_pin == AP_AIRSPEED_I2C_PIN) {
|
|
return digital.get_temperature(temperature);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// calibrate the airspeed. This must be called at least once before
|
|
// the get_airspeed() interface can be used
|
|
void AP_Airspeed::calibrate()
|
|
{
|
|
float sum = 0;
|
|
uint8_t count = 0;
|
|
if (!_enable) {
|
|
return;
|
|
}
|
|
// discard first reading
|
|
get_pressure();
|
|
for (uint8_t i = 0; i < 10; i++) {
|
|
hal.scheduler->delay(100);
|
|
float p = get_pressure();
|
|
if (_healthy) {
|
|
sum += p;
|
|
count++;
|
|
}
|
|
}
|
|
if (count == 0) {
|
|
// unhealthy sensor
|
|
hal.console->println_P(PSTR("Airspeed sensor unhealthy"));
|
|
_offset.set(0);
|
|
return;
|
|
}
|
|
float raw = sum/count;
|
|
_offset.set_and_save(raw);
|
|
_airspeed = 0;
|
|
_raw_airspeed = 0;
|
|
}
|
|
|
|
// read the airspeed sensor
|
|
void AP_Airspeed::read(void)
|
|
{
|
|
float airspeed_pressure;
|
|
if (!_enable) {
|
|
return;
|
|
}
|
|
/*
|
|
when we get negative pressures it means the tubes are probably
|
|
connected the wrong way around. We don't just use the absolute
|
|
value as otherwise we could get bad readings in some flight
|
|
attitudes (eg. a spin) due to pressure on the static port
|
|
*/
|
|
airspeed_pressure = get_pressure() - _offset;
|
|
if (airspeed_pressure < -32) {
|
|
// we're reading more than about -8m/s. The user probably has
|
|
// the ports the wrong way around
|
|
_healthy = false;
|
|
}
|
|
airspeed_pressure = max(airspeed_pressure, 0);
|
|
_last_pressure = airspeed_pressure;
|
|
_raw_airspeed = sqrtf(airspeed_pressure * _ratio);
|
|
_airspeed = 0.7f * _airspeed + 0.3f * _raw_airspeed;
|
|
}
|