2012-11-05 00:31:02 -04:00
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
2012-11-07 09:24:00 -04:00
# ifndef __AP_INERTIALNAV_H__
# define __AP_INERTIALNAV_H__
2012-11-05 00:31:02 -04:00
# include <AP_AHRS.h>
# include <AP_InertialSensor.h> // ArduPilot Mega IMU Library
# include <AP_Baro.h> // ArduPilot Mega Barometer Library
2012-12-09 11:43:11 -04:00
# include <AP_Buffer.h> // FIFO buffer library
2013-09-19 03:56:23 -03:00
# include <AP_GPS_Glitch.h> // GPS Glitch detection library
2012-11-05 00:31:02 -04:00
2013-04-30 00:52:28 -03:00
# define AP_INTERTIALNAV_TC_XY 2.5f // default time constant for complementary filter's X & Y axis
2013-03-31 23:51:12 -03:00
# define AP_INTERTIALNAV_TC_Z 5.0f // default time constant for complementary filter's Z axis
2013-01-10 03:56:21 -04:00
2012-12-09 11:43:11 -04:00
// #defines to control how often historical accel based positions are saved
// so they can later be compared to laggy gps readings
# define AP_INTERTIALNAV_SAVE_POS_AFTER_ITERATIONS 10
# define AP_INTERTIALNAV_GPS_LAG_IN_10HZ_INCREMENTS 4 // must not be larger than size of _hist_position_estimate_x and _hist_position_estimate_y
2013-01-22 05:16:18 -04:00
# define AP_INTERTIALNAV_GPS_TIMEOUT_MS 300 // timeout after which position error from GPS will fall to zero
2012-12-09 11:43:11 -04:00
2012-11-05 00:31:02 -04:00
/*
2013-10-25 10:55:31 -03:00
* AP_InertialNav blends accelerometer data with gps and barometer data to improve altitude and position hold .
*
* Most of the functions have to be called at 100 Hz . ( see defines above )
*
* The accelerometer values are integrated over time to approximate velocity and position .
* The inaccurcy of these estimates grows over time due to noisy sensor data .
* To improve the accuracy , baro and gps readings are used :
* An error value is calculated as the difference between the sensor ' s measurement and the last position estimation .
* This value is weighted with a gain factor and incorporated into the new estimation
2012-11-05 00:31:02 -04:00
*/
2012-11-07 09:24:00 -04:00
class AP_InertialNav
2012-11-05 00:31:02 -04:00
{
public :
// Constructor
2013-10-25 10:55:31 -03:00
AP_InertialNav ( const AP_AHRS * ahrs , AP_Baro * baro , GPS * & gps , GPS_Glitch & gps_glitch ) :
2012-11-05 00:31:02 -04:00
_ahrs ( ahrs ) ,
_baro ( baro ) ,
2013-09-19 03:56:23 -03:00
_gps ( gps ) ,
2012-11-05 00:31:02 -04:00
_xy_enabled ( false ) ,
2012-12-09 11:43:11 -04:00
_gps_last_update ( 0 ) ,
2013-04-17 10:00:52 -03:00
_gps_last_time ( 0 ) ,
2013-08-07 11:00:09 -03:00
_historic_xy_counter ( 0 ) ,
2013-09-19 03:56:23 -03:00
_baro_last_update ( 0 ) ,
_glitch_detector ( gps_glitch )
2012-12-12 17:46:06 -04:00
{
AP_Param : : setup_object_defaults ( this , var_info ) ;
}
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* initializes the object .
*
* AP_InertialNav : : set_home_position ( int32_t , int32_t ) should be called later ,
* to enable all " horizontal related " getter - methods .
*/
2013-03-19 05:51:16 -03:00
void init ( ) ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* update - updates velocity and position estimates using latest info from accelerometers
* augmented with gps and baro readings
*
* @ param dt : time since last update in seconds
*/
2013-03-19 05:51:16 -03:00
void update ( float dt ) ;
2012-11-05 00:31:02 -04:00
2012-12-09 11:43:11 -04:00
//
// XY Axis specific methods
//
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* set_time_constant_xy - sets time constant used by complementary filter for horizontal position estimate
*
* smaller values means higher influence of gps on position estimation
* bigger values favor the integrated accelerometer data for position estimation
*
* @ param time_constant_in_seconds : constant in seconds ; 0 < constant < 30 must hold
*/
2013-03-19 05:51:16 -03:00
void set_time_constant_xy ( float time_constant_in_seconds ) ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* position_ok - true if inertial based altitude and position can be trusted
* @ return
*/
2013-04-21 08:04:47 -03:00
bool position_ok ( ) const ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* check_gps - checks if new gps readings have arrived and calls correct_with_gps to
* calculate the horizontal position error
* @ see correct_with_gps ( int32_t lon , int32_t lat , float dt ) ;
*/
2013-03-19 05:51:16 -03:00
void check_gps ( ) ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* correct_with_gps - calculates horizontal position error using gps
*
* @ param now : current time since boot in milliseconds
* @ param lon : longitude in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
* @ param lat : latitude in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
*/
2013-09-19 03:56:23 -03:00
void correct_with_gps ( uint32_t now , int32_t lon , int32_t lat ) ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* get_position - returns the current position relative to the home location in cm .
*
* the home location was set with AP_InertialNav : : set_home_position ( int32_t , int32_t )
*
* @ return
*/
const Vector3f & get_position ( ) const { return _position ; }
/**
* get_latitude - returns the latitude of the current position estimation in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
* @ return
*/
2013-04-21 08:04:47 -03:00
int32_t get_latitude ( ) const ;
2013-10-25 10:55:31 -03:00
/**
* get_longitude - returns the longitude of the current position estimation in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
* @ return
*/
2013-04-21 08:04:47 -03:00
int32_t get_longitude ( ) const ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* set_home_position - sets home position
*
* all internal calculations are recorded as the distances from this point .
*
* @ param lon : longitude in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
* @ param lat : latitude in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
*/
2013-09-19 03:57:22 -03:00
void set_home_position ( int32_t lon , int32_t lat ) ;
2012-12-09 11:43:11 -04:00
2013-10-25 10:55:31 -03:00
/**
* get_latitude_diff - returns the current latitude difference from the home location .
*
* @ return difference in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
*/
2013-04-21 08:04:47 -03:00
float get_latitude_diff ( ) const ;
2013-10-25 10:55:31 -03:00
/**
* get_longitude_diff - returns the current longitude difference from the home location .
*
* @ return difference in 100 nano degrees ( i . e . degree value multiplied by 10 , 000 , 000 )
*/
2013-04-21 08:04:47 -03:00
float get_longitude_diff ( ) const ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* get_latitude_velocity - returns the current velocity in latitude direction .
*
* @ see get_velocity ( ) . x
*
* @ return : latitude velocity in cm / s
*/
2013-04-21 08:04:47 -03:00
float get_latitude_velocity ( ) const ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* get_longitude_velocity - returns the current velocity in longitude direction .
*
* @ see get_velocity ( ) . y
*
* @ return : longitude velocity in cm / s
*/
float get_longitude_velocity ( ) const ;
2013-03-27 02:09:04 -03:00
2013-10-25 10:55:31 -03:00
/**
* get_velocity - returns the current velocity in cm / s
*
* @ return velocity vector :
* . x : latitude velocity in cm / s
* . y : longitude velocity in cm / s
* . z : vertical velocity in cm / s
*/
const Vector3f & get_velocity ( ) const { return _velocity ; }
/**
* set_velocity_xy - overwrites the current horizontal velocity in cm / s
*
* @ param x : latitude velocity in cm / s
* @ param y : longitude velocity in cm / s
*/
2013-03-19 05:51:16 -03:00
void set_velocity_xy ( float x , float y ) ;
2012-11-05 00:31:02 -04:00
2012-12-09 11:43:11 -04:00
//
// Z Axis methods
//
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* set_time_constant_z - sets timeconstant used by complementary filter for vertical position estimation
*
* smaller values means higher influence of barometer in altitude estimation
* bigger values favor the integrated accelerometer data for altitude estimation
*
* @ param time_constant_in_seconds : constant in s ; 0 < constant < 30 must hold
*/
2013-03-19 05:51:16 -03:00
void set_time_constant_z ( float time_constant_in_seconds ) ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* altitude_ok - returns true if inertial based altitude and position can be trusted
* @ return
*/
2013-04-21 08:04:47 -03:00
bool altitude_ok ( ) const { return true ; }
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* check_baro - checks if new baro readings have arrived and calls correct_with_baro to
* calculate the vertical position error
*
* @ see correct_with_baro ( float baro_alt , float dt ) ;
*/
2013-03-19 05:51:16 -03:00
void check_baro ( ) ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* correct_with_baro - calculates vertical position error using barometer .
*
* @ param baro_alt : altitude in cm
* @ param dt : time since last baro reading in s
*/
2013-03-19 05:51:16 -03:00
void correct_with_baro ( float baro_alt , float dt ) ;
2012-11-05 00:31:02 -04:00
2013-10-25 10:55:31 -03:00
/**
* get_altitude - get latest altitude estimate in cm
* @ return
*/
float get_altitude ( ) const { return _position . z ; }
/**
* set_altitude - overwrites the current altitude value .
*
* @ param new_altitude : altitude in cm
*/
2013-03-19 05:51:16 -03:00
void set_altitude ( float new_altitude ) ;
2012-12-09 11:43:11 -04:00
2013-10-25 10:55:31 -03:00
/**
* get_velocity_z - returns the current climbrate .
*
* @ see get_velocity ( ) . z
*
* @ return climbrate in cm / s
*/
2013-04-21 08:04:47 -03:00
float get_velocity_z ( ) const { return _velocity . z ; }
2013-10-25 10:55:31 -03:00
/**
* set_velocity_z - overwrites the current climbrate .
*
* @ param new_velocity : climbrate in cm / s
*/
2013-03-19 05:51:16 -03:00
void set_velocity_z ( float new_velocity ) ;
2012-12-09 11:43:11 -04:00
// class level parameters
static const struct AP_Param : : GroupInfo var_info [ ] ;
// public variables
Vector3f accel_correction_ef ; // earth frame accelerometer corrections. here for logging purposes only
protected :
2013-10-25 10:55:31 -03:00
/**
* update gains from time constant .
*
* The time constants ( in s ) can be set with the following methods :
*
* @ see : AP_InertialNav : : set_time_constant_xy ( float )
* @ see : AP_InertialNav : : set_time_constant_z ( float )
*/
void update_gains ( ) ;
/**
* set_position_xy - overwrites the current position relative to the home location in cm
*
* the home location was set with AP_InertialNav : : set_home_location ( int32_t , int32_t )
*
* @ param x : relative latitude position in cm
* @ param y : relative longitude position in cm
*/
void set_position_xy ( float x , float y ) ;
2012-12-09 11:43:11 -04:00
2013-09-19 03:56:23 -03:00
// structure for holding flags
struct InertialNav_flags {
uint8_t gps_glitching : 1 ; // 1 if glitch detector was previously indicating a gps glitch
} _flags ;
2013-10-25 10:55:31 -03:00
const AP_AHRS * const _ahrs ; // pointer to ahrs object
2012-12-09 11:43:11 -04:00
AP_Baro * _baro ; // pointer to barometer
2013-09-19 03:56:23 -03:00
GPS * & _gps ; // pointer to gps
2012-12-09 11:43:11 -04:00
// XY Axis specific variables
bool _xy_enabled ; // xy position estimates enabled
2013-10-25 10:55:31 -03:00
AP_Float _time_constant_xy ; // time constant for horizontal corrections in s
2012-12-09 11:43:11 -04:00
float _k1_xy ; // gain for horizontal position correction
float _k2_xy ; // gain for horizontal velocity correction
float _k3_xy ; // gain for horizontal accelerometer offset correction
2013-10-25 10:55:31 -03:00
uint32_t _gps_last_update ; // system time of last gps update in ms
uint32_t _gps_last_time ; // time of last gps update according to the gps itself in ms
2012-12-09 11:43:11 -04:00
uint8_t _historic_xy_counter ; // counter used to slow saving of position estimates for later comparison to gps
2013-10-25 10:55:31 -03:00
AP_BufferFloat_Size5 _hist_position_estimate_x ; // buffer of historic accel based position to account for gpslag
AP_BufferFloat_Size5 _hist_position_estimate_y ; // buffer of historic accel based position to account for gps lag
int32_t _base_lat ; // base latitude (home location) in 100 nano degrees (i.e. degree value multiplied by 10,000,000)
int32_t _base_lon ; // base longitude (home location) in 100 nano degrees (i.e. degree value multiplied by 10,000,000)
2013-08-07 11:00:09 -03:00
float _lon_to_cm_scaling ; // conversion of longitude to centimeters
2012-12-09 11:43:11 -04:00
// Z Axis specific variables
2013-10-25 10:55:31 -03:00
AP_Float _time_constant_z ; // time constant for vertical corrections in s
2012-12-09 11:43:11 -04:00
float _k1_z ; // gain for vertical position correction
float _k2_z ; // gain for vertical velocity correction
float _k3_z ; // gain for vertical accelerometer offset correction
2013-10-25 10:55:31 -03:00
uint32_t _baro_last_update ; // time of last barometer update in ms
AP_BufferFloat_Size15 _hist_position_estimate_z ; // buffer of historic accel based altitudes to account for barometer lag
2012-12-09 11:43:11 -04:00
// general variables
2013-10-25 10:55:31 -03:00
Vector3f _position_base ; // (uncorrected) position estimate in cm - relative to the home location (_base_lat, _base_lon, 0)
Vector3f _position_correction ; // sum of corrections to _position_base from delayed 1st order samples in cm
Vector3f _velocity ; // latest velocity estimate (integrated from accelerometer values) in cm/s
Vector3f _position_error ; // current position error in cm - is set by the check_* methods and used by update method to calculate the correction terms
Vector3f _position ; // sum(_position_base, _position_correction) - corrected position estimate in cm - relative to the home location (_base_lat, _base_lon, 0)
2013-09-19 03:56:23 -03:00
// GPS Glitch detector
GPS_Glitch & _glitch_detector ;
2012-11-05 00:31:02 -04:00
} ;
2012-11-07 09:24:00 -04:00
# endif // __AP_INERTIALNAV_H__