2013-05-29 20:51:34 -03:00
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
2012-10-26 20:59:07 -03:00
# ifndef __AP_MOTORS_CLASS_H__
# define __AP_MOTORS_CLASS_H__
# include <AP_Common.h>
2012-12-16 20:55:33 -04:00
# include <AP_Progmem.h>
2012-10-26 20:59:07 -03:00
# include <AP_Math.h> // ArduPilot Mega Vector/Matrix math Library
2013-08-08 10:15:04 -03:00
# include <AP_Notify.h> // Notify library
2012-10-26 20:59:07 -03:00
# include <RC_Channel.h> // RC Channel Library
2015-02-23 00:51:54 -04:00
# include <Filter.h> // filter library
2012-10-26 20:59:07 -03:00
// offsets for motors in motor_out, _motor_filtered and _motor_to_channel_map arrays
# define AP_MOTORS_MOT_1 0
# define AP_MOTORS_MOT_2 1
# define AP_MOTORS_MOT_3 2
# define AP_MOTORS_MOT_4 3
# define AP_MOTORS_MOT_5 4
# define AP_MOTORS_MOT_6 5
# define AP_MOTORS_MOT_7 6
# define AP_MOTORS_MOT_8 7
# define APM1_MOTOR_TO_CHANNEL_MAP CH_1,CH_2,CH_3,CH_4,CH_7,CH_8,CH_10,CH_11
# define APM2_MOTOR_TO_CHANNEL_MAP CH_1,CH_2,CH_3,CH_4,CH_5,CH_6,CH_7,CH_8
# define AP_MOTORS_MAX_NUM_MOTORS 8
# define AP_MOTORS_DEFAULT_MIN_THROTTLE 130
2013-05-26 23:21:31 -03:00
# define AP_MOTORS_DEFAULT_MID_THROTTLE 500
2013-05-14 04:05:22 -03:00
# define AP_MOTORS_DEFAULT_MAX_THROTTLE 1000
2012-10-26 20:59:07 -03:00
// APM board definitions
# define AP_MOTORS_APM1 1
# define AP_MOTORS_APM2 2
// frame definitions
2013-03-31 00:41:02 -03:00
# define AP_MOTORS_PLUS_FRAME 0
# define AP_MOTORS_X_FRAME 1
# define AP_MOTORS_V_FRAME 2
2013-05-03 11:13:40 -03:00
# define AP_MOTORS_H_FRAME 3 // same as X frame but motors spin in opposite direction
2014-04-01 23:51:41 -03:00
# define AP_MOTORS_VTAIL_FRAME 4 // Lynxmotion Hunter VTail 400/500
2014-08-18 02:25:24 -03:00
# define AP_MOTORS_ATAIL_FRAME 5 // A-Shaped VTail Quads
2013-12-03 10:22:14 -04:00
# define AP_MOTORS_NEW_PLUS_FRAME 10 // NEW frames are same as original 4 but with motor orders changed to be clockwise from the front
# define AP_MOTORS_NEW_X_FRAME 11
# define AP_MOTORS_NEW_V_FRAME 12
# define AP_MOTORS_NEW_H_FRAME 13 // same as X frame but motors spin in opposite direction
2012-10-26 20:59:07 -03:00
// motor update rate
2013-05-14 04:56:55 -03:00
# define AP_MOTORS_SPEED_DEFAULT 490 // default output rate to the motors
2012-10-26 20:59:07 -03:00
# define THROTTLE_CURVE_ENABLED 1 // throttle curve disabled by default
# define THROTTLE_CURVE_MID_THRUST 52 // throttle which produces 1/2 the maximum thrust. expressed as a percentage of the full throttle range (i.e 0 ~ 100)
# define THROTTLE_CURVE_MAX_THRUST 93 // throttle which produces the maximum thrust. expressed as a percentage of the full throttle range (i.e 0 ~ 100)
2013-11-18 04:00:17 -04:00
# define AP_MOTORS_SPIN_WHEN_ARMED 70 // spin motors at this PWM value when armed
2013-06-22 22:07:35 -03:00
2015-02-11 08:03:39 -04:00
# define AP_MOTORS_YAW_HEADROOM_DEFAULT 200
2015-02-21 02:06:56 -04:00
# define AP_MOTORS_THR_LOW_CMP_DEFAULT 0.5f // ratio controlling the max throttle output during competing requests of low throttle from the pilot (or autopilot) and higher throttle for attitude control. Higher favours Attitude over pilot input
2015-02-23 02:35:10 -04:00
# define AP_MOTORS_THST_EXPO_DEFAULT 0.5f // set to 0 for linear and 1 for second order approximation
# define AP_MOTORS_THST_MAX_DEFAULT 0.95f // throttle which produces the maximum thrust. (i.e. 0 ~ 1 ) of the full throttle range
# define AP_MOTORS_THST_BAT_MAX_DEFAULT 0.0f
# define AP_MOTORS_THST_BAT_MIN_DEFAULT 0.0f
2015-02-23 02:36:23 -04:00
# define AP_MOTORS_CURR_MAX_DEFAULT 0.0f // current limiting max default
2015-02-23 00:51:54 -04:00
# define AP_MOTORS_BATT_VOLT_FILT_HZ 0.5f // battery voltage filtered at 0.5hz
2015-05-03 11:21:05 -03:00
# define AP_MOTORS_THR_MIX_MIN_DEFAULT 0.1f // minimum throttle mix
# define AP_MOTORS_THR_MIX_MID_DEFAULT 0.5f // manual throttle mix
# define AP_MOTORS_THR_MIX_MAX_DEFAULT 0.9f // maximum throttle mix
2015-02-23 02:35:10 -04:00
2012-10-26 20:59:07 -03:00
// bit mask for recording which limits we have reached when outputting to motors
# define AP_MOTOR_NO_LIMITS_REACHED 0x00
# define AP_MOTOR_ROLLPITCH_LIMIT 0x01
# define AP_MOTOR_YAW_LIMIT 0x02
# define AP_MOTOR_THROTTLE_LIMIT 0x04
# define AP_MOTOR_ANY_LIMIT 0xFF
2014-10-18 08:54:08 -03:00
// To-Do: replace this hard coded counter with a timer
2015-05-04 03:16:48 -03:00
# if HAL_CPU_CLASS < HAL_CPU_CLASS_75 || CONFIG_HAL_BOARD == HAL_BOARD_SITL || CONFIG_HAL_BOARD == HAL_BOARD_LINUX
2014-10-18 08:54:08 -03:00
// slow start increments - throttle increase per (100hz) iteration. i.e. 5 = full speed in 2 seconds
# define AP_MOTOR_SLOW_START_INCREMENT 10 // max throttle ramp speed (i.e. motors can reach full throttle in 1 second)
# define AP_MOTOR_SLOW_START_LOW_END_INCREMENT 2 // min throttle ramp speed (i.e. motors will speed up from zero to _spin_when_armed speed in about 1 second)
# else
// slow start increments - throttle increase per (400hz) iteration. i.e. 1 = full speed in 2.5 seconds
# define AP_MOTOR_SLOW_START_INCREMENT 3 // max throttle ramp speed (i.e. motors can reach full throttle in 0.8 seconds)
# define AP_MOTOR_SLOW_START_LOW_END_INCREMENT 1 // min throttle ramp speed (i.e. motors will speed up from zero to _spin_when_armed speed in about 0.3 second)
# endif
2012-10-26 20:59:07 -03:00
/// @class AP_Motors
class AP_Motors {
public :
// Constructor
2015-02-21 02:45:56 -04:00
AP_Motors ( RC_Channel & rc_roll , RC_Channel & rc_pitch , RC_Channel & rc_throttle , RC_Channel & rc_yaw , uint16_t loop_rate , uint16_t speed_hz = AP_MOTORS_SPEED_DEFAULT ) ;
2012-10-26 20:59:07 -03:00
// init
2015-02-21 04:33:13 -04:00
virtual void Init ( ) { }
2012-10-26 20:59:07 -03:00
// set update rate to motors - a value in hertz
2013-05-14 06:23:36 -03:00
virtual void set_update_rate ( uint16_t speed_hz ) { _speed_hz = speed_hz ; } ;
2012-10-26 20:59:07 -03:00
// set frame orientation (normally + or X)
2013-09-12 10:27:44 -03:00
virtual void set_frame_orientation ( uint8_t new_orientation ) { _flags . frame_orientation = new_orientation ; } ;
2012-10-26 20:59:07 -03:00
// enable - starts allowing signals to be sent to motors
2013-05-31 02:17:08 -03:00
virtual void enable ( ) = 0 ;
2012-10-26 20:59:07 -03:00
// arm, disarm or check status status of motors
2014-01-30 03:29:53 -04:00
bool armed ( ) const { return _flags . armed ; } ;
2013-08-08 10:15:04 -03:00
void armed ( bool arm ) ;
2012-10-26 20:59:07 -03:00
2015-04-17 12:58:26 -03:00
// set motor interlock status
void set_interlock ( bool set ) { _flags . interlock = set ; }
2015-05-01 00:16:32 -03:00
// get motor interlock status. true means motors run, false motors don't run
2015-04-17 12:58:26 -03:00
bool get_interlock ( ) const { return _flags . interlock ; } ;
2012-10-26 20:59:07 -03:00
// set_min_throttle - sets the minimum throttle that will be sent to the engines when they're not off (i.e. to prevents issues with some motors spinning and some not at very low throttle)
2013-05-26 23:21:31 -03:00
void set_min_throttle ( uint16_t min_throttle ) ;
2015-02-22 22:27:51 -04:00
// set_hover_throttle - sets the mid throttle which is close to the hover throttle of the copter
2013-05-26 23:21:31 -03:00
// this is used to limit the amount that the stability patch will increase the throttle to give more room for roll, pitch and yaw control
2015-02-22 22:27:51 -04:00
void set_hover_throttle ( uint16_t hov_thr ) { _hover_out = hov_thr ; }
// get_hover_throttle_as_pwm - converts hover throttle to pwm (i.e. range 1000 ~ 2000)
int16_t get_hover_throttle_as_pwm ( ) const ;
2012-10-26 20:59:07 -03:00
2014-01-05 23:28:57 -04:00
int16_t throttle_min ( ) const { return _min_throttle ; }
int16_t throttle_max ( ) const { return _max_throttle ; }
2014-02-10 00:20:26 -04:00
// set_roll, set_pitch, set_yaw, set_throttle
void set_roll ( int16_t roll_in ) { _rc_roll . servo_out = roll_in ; } ; // range -4500 ~ 4500
void set_pitch ( int16_t pitch_in ) { _rc_pitch . servo_out = pitch_in ; } ; // range -4500 ~ 4500
void set_yaw ( int16_t yaw_in ) { _rc_yaw . servo_out = yaw_in ; } ; // range -4500 ~ 4500
2015-04-24 01:35:04 -03:00
void set_throttle ( float throttle_in ) { _throttle_in = throttle_in ; } ; // range 0 ~ 1000
2015-04-02 17:54:15 -03:00
void set_stabilizing ( bool stabilizing ) { _flags . stabilizing = stabilizing ; }
2014-02-10 00:20:26 -04:00
2015-01-24 00:57:32 -04:00
// accessors for roll, pitch, yaw and throttle inputs to motors
2015-03-31 22:30:06 -03:00
int16_t get_roll ( ) const { return _rc_roll . servo_out ; }
int16_t get_pitch ( ) const { return _rc_pitch . servo_out ; }
int16_t get_yaw ( ) const { return _rc_yaw . servo_out ; }
2014-08-04 05:50:07 -03:00
int16_t get_throttle_out ( ) const { return _rc_throttle . servo_out ; }
2014-07-17 06:15:07 -03:00
2015-04-16 16:36:02 -03:00
void set_throttle_filter_cutoff ( float filt_hz ) { _throttle_filter . set_cutoff_frequency ( filt_hz ) ; }
2015-04-16 01:55:12 -03:00
2012-10-26 20:59:07 -03:00
// output - sends commands to the motors
2015-05-21 15:33:18 -03:00
virtual void output ( ) ;
2012-10-26 20:59:07 -03:00
// output_min - sends minimum values out to the motors
2013-05-31 02:17:08 -03:00
virtual void output_min ( ) = 0 ;
2012-10-26 20:59:07 -03:00
2014-04-28 04:28:40 -03:00
// output_test - spin a motor at the pwm value specified
// motor_seq is the motor's sequence number from 1 to the number of motors on the frame
// pwm value is an actual pwm value that will be output, normally in the range of 1000 ~ 2000
virtual void output_test ( uint8_t motor_seq , int16_t pwm ) = 0 ;
2012-10-26 20:59:07 -03:00
2014-09-19 10:06:29 -03:00
// throttle_pass_through - passes provided pwm directly to all motors - dangerous but used for initialising ESCs
// pwm value is an actual pwm value that will be output, normally in the range of 1000 ~ 2000
virtual void throttle_pass_through ( int16_t pwm ) ;
2012-10-26 20:59:07 -03:00
2015-02-11 08:03:39 -04:00
// set_yaw_headroom - set yaw headroom (yaw is given at least this amount of pwm)
virtual void set_yaw_headroom ( int16_t pwm ) { _yaw_headroom = pwm ; }
2015-02-23 02:35:10 -04:00
// set_voltage - set voltage to be used for output scaling
virtual void set_voltage ( float volts ) { _batt_voltage = volts ; }
// set_current - set current to be used for output scaling
virtual void set_current ( float current ) { _batt_current = current ; }
2015-04-28 15:19:50 -03:00
// set_density_ratio - sets air density as a proportion of sea level density
void set_air_density_ratio ( float ratio ) { _air_density_ratio = ratio ; }
2015-05-03 11:21:05 -03:00
// set_throttle_thr_mix - set desired throttle_thr_mix (actual throttle_thr_mix is slewed towards this value over 1~2 seconds)
2015-03-08 07:17:28 -03:00
// low values favour pilot/autopilot throttle over attitude control, high values favour attitude control over throttle
// has no effect when throttle is above hover throttle
2015-05-03 11:21:05 -03:00
void set_throttle_mix_min ( ) { _throttle_thr_mix_desired = _thr_mix_min ; }
void set_throttle_mix_mid ( ) { _throttle_thr_mix_desired = AP_MOTORS_THR_MIX_MID_DEFAULT ; }
void set_throttle_mix_max ( ) { _throttle_thr_mix_desired = AP_MOTORS_THR_MIX_MAX_DEFAULT ; }
2015-03-08 07:17:28 -03:00
2015-05-03 11:21:05 -03:00
// get_throttle_thr_mix - get low throttle compensation value
2015-05-07 22:26:47 -03:00
bool is_throttle_mix_min ( ) { return ( _throttle_thr_mix < 1.25f * _thr_mix_min ) ; }
2015-04-14 09:56:01 -03:00
2015-01-24 00:57:32 -04:00
// get_lift_max - get maximum lift ratio
float get_lift_max ( ) { return _lift_max ; }
// get_batt_voltage_filt - get battery voltage ratio
float get_batt_voltage_filt ( ) { return _batt_voltage_filt . get ( ) ; }
// get_batt_resistance - get battery resistance approximation
float get_batt_resistance ( ) { return _batt_resistance ; }
// get_throttle_limit - throttle limit ratio
float get_throttle_limit ( ) { return _throttle_limit ; }
2015-04-28 15:56:10 -03:00
// returns warning throttle
float get_throttle_warn ( ) { return rel_pwm_to_thr_range ( _spin_when_armed ) ; }
2012-10-26 20:59:07 -03:00
// 1 if motor is enabled, 0 otherwise
2013-05-14 04:05:22 -03:00
bool motor_enabled [ AP_MOTORS_MAX_NUM_MOTORS ] ;
2012-10-26 20:59:07 -03:00
2013-09-12 10:27:44 -03:00
// slow_start - set to true to slew motors from current speed to maximum
// Note: this must be set immediately before a step up in throttle
void slow_start ( bool true_false ) ;
2014-07-26 04:27:16 -03:00
// get_motor_mask - returns a bitmask of which outputs are being used for motors (1 means being used)
// this can be used to ensure other pwm outputs (i.e. for servos) do not conflict
virtual uint16_t get_motor_mask ( ) = 0 ;
2013-07-21 01:58:24 -03:00
// structure for holding motor limit flags
struct AP_Motors_limit {
uint8_t roll_pitch : 1 ; // we have reached roll or pitch limit
uint8_t yaw : 1 ; // we have reached yaw limit
2013-07-25 12:45:59 -03:00
uint8_t throttle_lower : 1 ; // we have reached throttle's lower limit
uint8_t throttle_upper : 1 ; // we have reached throttle's upper limit
2013-07-21 01:58:24 -03:00
} limit ;
2012-10-26 20:59:07 -03:00
// var_info for holding Parameter information
static const struct AP_Param : : GroupInfo var_info [ ] ;
protected :
// output functions that should be overloaded by child classes
2015-04-02 17:54:15 -03:00
virtual void output_armed_stabilizing ( ) = 0 ;
virtual void output_armed_not_stabilizing ( ) = 0 ;
2015-04-17 12:58:26 -03:00
virtual void output_armed_zero_throttle ( ) { output_min ( ) ; }
2015-03-31 21:56:27 -03:00
virtual void output_disarmed ( ) = 0 ;
2012-10-26 20:59:07 -03:00
2015-04-01 00:16:43 -03:00
// update the throttle input filter
2015-05-21 15:19:25 -03:00
virtual void update_throttle_filter ( ) ;
2015-04-01 00:16:43 -03:00
2015-02-23 05:09:08 -04:00
// update_max_throttle - updates the limits on _max_throttle for slow_start and current limiting flag
2013-09-12 10:27:44 -03:00
void update_max_throttle ( ) ;
2015-02-23 05:09:08 -04:00
// current_limit_max_throttle - current limit maximum throttle (called from update_max_throttle)
void current_limit_max_throttle ( ) ;
2013-05-26 23:21:31 -03:00
2015-02-21 03:55:13 -04:00
// apply_thrust_curve_and_volt_scaling - thrust curve and voltage adjusted pwm value (i.e. 1000 ~ 2000)
int16_t apply_thrust_curve_and_volt_scaling ( int16_t pwm_out , int16_t pwm_min , int16_t pwm_max ) const ;
2015-02-21 05:15:11 -04:00
// update_lift_max_from_batt_voltage - used for voltage compensation
void update_lift_max_from_batt_voltage ( ) ;
2015-02-23 00:52:39 -04:00
// update_battery_resistance - calculate battery resistance when throttle is above hover_out
void update_battery_resistance ( ) ;
2015-05-03 11:21:05 -03:00
// update_throttle_thr_mix - updates thr_low_comp value towards the target
void update_throttle_thr_mix ( ) ;
2015-03-08 07:17:28 -03:00
2015-04-28 15:19:50 -03:00
// return gain scheduling gain based on voltage and air density
float get_compensation_gain ( ) const ;
2015-02-23 08:48:54 -04:00
2015-04-02 17:54:15 -03:00
float rel_pwm_to_thr_range ( float pwm ) const ;
float thr_range_to_rel_pwm ( float thr ) const ;
2013-09-12 10:27:44 -03:00
// flag bitmask
struct AP_Motors_flags {
2015-04-02 17:54:15 -03:00
uint8_t armed : 1 ; // 0 if disarmed, 1 if armed
uint8_t stabilizing : 1 ; // 0 if not controlling attitude, 1 if controlling attitude
uint8_t frame_orientation : 4 ; // PLUS_FRAME 0, X_FRAME 1, V_FRAME 2, H_FRAME 3, NEW_PLUS_FRAME 10, NEW_X_FRAME, NEW_V_FRAME, NEW_H_FRAME
uint8_t slow_start : 1 ; // 1 if slow start is active
uint8_t slow_start_low_end : 1 ; // 1 just after arming so we can ramp up the spin_when_armed value
2015-05-01 00:16:32 -03:00
uint8_t interlock : 1 ; // 1 if the motor interlock is enabled (i.e. motors run), 0 if disabled (motors don't run)
2013-09-12 10:27:44 -03:00
} _flags ;
2013-07-16 00:45:40 -03:00
2014-02-10 02:58:24 -04:00
// mapping of motor number (as received from upper APM code) to RC channel output - used to account for differences between APM1 and APM2
static const uint8_t _motor_to_channel_map [ AP_MOTORS_MAX_NUM_MOTORS ] PROGMEM ;
2013-09-12 10:27:44 -03:00
// parameters
2014-08-18 02:25:24 -03:00
AP_Int16 _spin_when_armed ; // used to control whether the motors always spin when armed. pwm value above radio_min
2013-09-12 10:27:44 -03:00
2015-02-11 08:03:39 -04:00
AP_Int16 _yaw_headroom ; // yaw control is given at least this pwm range
2015-02-23 02:35:10 -04:00
AP_Float _thrust_curve_expo ; // curve used to linearize pwm to thrust conversion. set to 0 for linear and 1 for second order approximation
AP_Float _thrust_curve_max ; // throttle which produces the maximum thrust. (i.e. 0 ~ 1 ) of the full throttle range
AP_Float _batt_voltage_max ; // maximum voltage used to scale lift
AP_Float _batt_voltage_min ; // minimum voltage used to scale lift
2015-02-23 02:36:23 -04:00
AP_Float _batt_current_max ; // current over which maximum throttle is limited
2015-05-03 11:21:05 -03:00
AP_Float _thr_mix_min ; // current over which maximum throttle is limited
2015-02-11 08:03:39 -04:00
2013-09-12 10:27:44 -03:00
// internal variables
2014-02-10 00:20:26 -04:00
RC_Channel & _rc_roll ; // roll input in from users is held in servo_out
RC_Channel & _rc_pitch ; // pitch input in from users is held in servo_out
RC_Channel & _rc_throttle ; // throttle input in from users is held in servo_out
RC_Channel & _rc_yaw ; // yaw input in from users is held in servo_out
2015-02-21 02:45:56 -04:00
uint16_t _loop_rate ; // rate at which output() function is called (normally 400hz)
2013-09-12 10:27:44 -03:00
uint16_t _speed_hz ; // speed in hz to send updates to motors
int16_t _min_throttle ; // the minimum throttle to be sent to the motors when they're on (prevents motors stalling while flying)
int16_t _max_throttle ; // the maximum throttle to be sent to the motors (sometimes limited by slow start)
2015-02-22 22:27:51 -04:00
int16_t _hover_out ; // the estimated hover throttle as pct * 10 (i.e. 0 ~ 1000)
2013-10-20 02:51:35 -03:00
int16_t _spin_when_armed_ramped ; // equal to _spin_when_armed parameter but slowly ramped up from zero
2015-05-03 11:21:05 -03:00
float _throttle_thr_mix ; // mix between throttle and hover throttle for 0 to 1 and ratio above hover throttle for >1
float _throttle_thr_mix_desired ; // desired throttle_low_comp value, actual throttle_low_comp is slewed towards this value over 1~2 seconds
2015-02-23 02:35:10 -04:00
// battery voltage compensation variables
float _batt_voltage ; // latest battery voltage reading
float _batt_voltage_resting ; // battery voltage reading at minimum throttle
2015-02-23 00:51:54 -04:00
LowPassFilterFloat _batt_voltage_filt ; // filtered battery voltage expressed as a percentage (0 ~ 1.0) of batt_voltage_max
2015-02-23 02:35:10 -04:00
float _batt_current ; // latest battery current reading
float _batt_current_resting ; // battery's current when motors at minimum
float _batt_resistance ; // battery's resistance calculated by comparing resting voltage vs in flight voltage
int16_t _batt_timer ; // timer used in battery resistance calcs
2015-04-28 15:19:50 -03:00
float _air_density_ratio ; // air density / sea level density - decreases in altitude
2015-02-23 02:35:10 -04:00
float _lift_max ; // maximum lift ratio from battery voltage
2015-02-23 02:36:23 -04:00
float _throttle_limit ; // ratio of throttle limit between hover and maximum
2015-04-01 00:16:43 -03:00
float _throttle_in ; // last throttle input from set_throttle caller
LowPassFilterFloat _throttle_filter ; // throttle input filter
2012-10-26 20:59:07 -03:00
} ;
# endif // __AP_MOTORS_CLASS_H__