2013-05-29 20:51:34 -03:00
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
2013-08-29 02:34:34 -03:00
/*
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/>.
*/
2012-04-02 05:26:37 -03:00
/*
2012-08-17 03:20:26 -03:00
* AP_MotorsTri . cpp - ArduCopter motors library
* Code by RandyMackay . DIYDrones . com
*
*/
2015-08-11 03:28:44 -03:00
# include <AP_HAL/AP_HAL.h>
# include <AP_Math/AP_Math.h>
2012-04-02 05:26:37 -03:00
# include "AP_MotorsTri.h"
2012-10-26 20:59:07 -03:00
extern const AP_HAL : : HAL & hal ;
2015-10-25 14:03:46 -03:00
const AP_Param : : GroupInfo AP_MotorsTri : : var_info [ ] = {
2015-05-14 22:00:46 -03:00
// variables from parent vehicle
2015-07-15 04:58:43 -03:00
AP_NESTEDGROUPINFO ( AP_MotorsMulticopter , 0 ) ,
2015-05-14 22:00:46 -03:00
2015-07-02 18:15:09 -03:00
// parameters 1 ~ 29 were reserved for tradheli
2015-05-25 08:20:53 -03:00
// parameters 30 ~ 39 reserved for tricopter
// parameters 40 ~ 49 for single copter and coax copter (these have identical parameter files)
2015-05-14 22:00:46 -03:00
// @Param: YAW_SV_REV
// @DisplayName: Yaw Servo Reverse
2015-05-25 08:20:53 -03:00
// @Description: Yaw servo reversing. Set to 1 for normal (forward) operation. Set to -1 to reverse this channel.
// @Values: -1:Reversed,1:Normal
2015-05-14 22:00:46 -03:00
// @User: Standard
2016-01-19 23:11:50 -04:00
AP_GROUPINFO ( " YAW_SV_REV " , 31 , AP_MotorsTri , _yaw_reverse , 1 ) ,
2015-05-14 22:00:46 -03:00
// @Param: YAW_SV_TRIM
// @DisplayName: Yaw Servo Trim/Center
// @Description: Trim or center position of yaw servo
// @Range: 1250 1750
// @Units: PWM
// @Increment: 1
// @User: Standard
2015-05-25 08:20:53 -03:00
AP_GROUPINFO ( " YAW_SV_TRIM " , 32 , AP_MotorsTri , _yaw_servo_trim , 1500 ) ,
2015-05-14 22:00:46 -03:00
// @Param: YAW_SV_MIN
2016-02-08 01:37:25 -04:00
// @DisplayName: Yaw Servo Min PWM
// @Description: Yaw servo's minimum pwm value
2015-05-14 22:00:46 -03:00
// @Range: 1000 1400
// @Units: PWM
// @Increment: 1
// @User: Standard
2015-05-25 08:20:53 -03:00
AP_GROUPINFO ( " YAW_SV_MIN " , 33 , AP_MotorsTri , _yaw_servo_min , 1250 ) ,
2015-05-14 22:00:46 -03:00
// @Param: YAW_SV_MAX
2016-02-08 01:37:25 -04:00
// @DisplayName: Yaw Servo Max PWM
// @Description: Yaw servo's maximum pwm value
2015-05-14 22:00:46 -03:00
// @Range: 1600 2000
// @Units: PWM
// @Increment: 1
// @User: Standard
2015-05-25 08:20:53 -03:00
AP_GROUPINFO ( " YAW_SV_MAX " , 34 , AP_MotorsTri , _yaw_servo_max , 1750 ) ,
2015-05-14 22:00:46 -03:00
2016-02-08 01:40:43 -04:00
// @Param: YAW_SV_ANGLE
// @DisplayName: Yaw Servo Max Lean Angle
// @Description: Yaw servo's maximum lean angle
2016-03-25 05:56:58 -03:00
// @Range: 5 80
2016-02-08 01:40:43 -04:00
// @Units: Degrees
// @Increment: 1
// @User: Standard
AP_GROUPINFO ( " YAW_SV_ANGLE " , 35 , AP_MotorsTri , _yaw_servo_angle_max_deg , 30 ) ,
2015-05-14 22:00:46 -03:00
AP_GROUPEND
} ;
2012-04-02 05:26:37 -03:00
// init
void AP_MotorsTri : : Init ( )
{
2016-04-28 09:36:41 -03:00
add_motor_num ( AP_MOTORS_MOT_1 ) ;
add_motor_num ( AP_MOTORS_MOT_2 ) ;
add_motor_num ( AP_MOTORS_MOT_4 ) ;
2012-08-17 03:20:26 -03:00
// set update rate for the 3 motors (but not the servo on channel 7)
set_update_rate ( _speed_hz ) ;
2013-05-19 05:28:39 -03:00
// set the motor_enabled flag so that the ESCs can be calibrated like other frame types
motor_enabled [ AP_MOTORS_MOT_1 ] = true ;
motor_enabled [ AP_MOTORS_MOT_2 ] = true ;
motor_enabled [ AP_MOTORS_MOT_4 ] = true ;
2014-02-07 09:02:31 -04:00
2016-04-21 04:32:25 -03:00
// allow mapping of motor7
add_motor_num ( AP_MOTORS_CH_TRI_YAW ) ;
2012-04-02 05:26:37 -03:00
}
2012-09-13 09:31:13 -03:00
// set update rate to motors - a value in hertz
2012-04-02 05:26:37 -03:00
void AP_MotorsTri : : set_update_rate ( uint16_t speed_hz )
{
2012-08-17 03:20:26 -03:00
// record requested speed
_speed_hz = speed_hz ;
2012-04-02 05:26:37 -03:00
2012-08-17 03:20:26 -03:00
// set update rate for the 3 motors (but not the servo on channel 7)
2013-01-10 00:52:46 -04:00
uint32_t mask =
2015-09-29 00:01:12 -03:00
1U < < AP_MOTORS_MOT_1 |
1U < < AP_MOTORS_MOT_2 |
1U < < AP_MOTORS_MOT_4 ;
2016-01-04 06:24:06 -04:00
rc_set_freq ( mask , _speed_hz ) ;
2012-04-02 05:26:37 -03:00
}
2012-08-17 03:20:26 -03:00
// enable - starts allowing signals to be sent to motors
2012-04-02 05:26:37 -03:00
void AP_MotorsTri : : enable ( )
{
2012-08-17 03:20:26 -03:00
// enable output channels
2016-01-04 06:24:06 -04:00
rc_enable_ch ( AP_MOTORS_MOT_1 ) ;
rc_enable_ch ( AP_MOTORS_MOT_2 ) ;
rc_enable_ch ( AP_MOTORS_MOT_4 ) ;
rc_enable_ch ( AP_MOTORS_CH_TRI_YAW ) ;
2012-04-02 05:26:37 -03:00
}
2016-01-19 23:15:10 -04:00
void AP_MotorsTri : : output_to_motors ( )
{
switch ( _multicopter_flags . spool_mode ) {
case SHUT_DOWN :
// sends minimum values out to the motors
hal . rcout - > cork ( ) ;
rc_write ( AP_MOTORS_MOT_1 , _throttle_radio_min ) ;
rc_write ( AP_MOTORS_MOT_2 , _throttle_radio_min ) ;
rc_write ( AP_MOTORS_MOT_4 , _throttle_radio_min ) ;
rc_write ( AP_MOTORS_CH_TRI_YAW , _yaw_servo_trim ) ;
hal . rcout - > push ( ) ;
break ;
case SPIN_WHEN_ARMED :
// sends output to motors when armed but not flying
hal . rcout - > cork ( ) ;
rc_write ( AP_MOTORS_MOT_1 , constrain_int16 ( _throttle_radio_min + _throttle_low_end_pct * _min_throttle , _throttle_radio_min , _throttle_radio_min + _min_throttle ) ) ;
rc_write ( AP_MOTORS_MOT_2 , constrain_int16 ( _throttle_radio_min + _throttle_low_end_pct * _min_throttle , _throttle_radio_min , _throttle_radio_min + _min_throttle ) ) ;
rc_write ( AP_MOTORS_MOT_4 , constrain_int16 ( _throttle_radio_min + _throttle_low_end_pct * _min_throttle , _throttle_radio_min , _throttle_radio_min + _min_throttle ) ) ;
rc_write ( AP_MOTORS_CH_TRI_YAW , _yaw_servo_trim ) ;
hal . rcout - > push ( ) ;
break ;
case SPOOL_UP :
case THROTTLE_UNLIMITED :
case SPOOL_DOWN :
// set motor output based on thrust requests
hal . rcout - > cork ( ) ;
rc_write ( AP_MOTORS_MOT_1 , calc_thrust_to_pwm ( _thrust_right ) ) ;
rc_write ( AP_MOTORS_MOT_2 , calc_thrust_to_pwm ( _thrust_left ) ) ;
rc_write ( AP_MOTORS_MOT_4 , calc_thrust_to_pwm ( _thrust_rear ) ) ;
2016-02-08 01:40:43 -04:00
rc_write ( AP_MOTORS_CH_TRI_YAW , calc_yaw_radio_output ( _pivot_angle , radians ( _yaw_servo_angle_max_deg ) ) ) ;
2016-01-19 23:15:10 -04:00
hal . rcout - > push ( ) ;
break ;
}
}
2014-07-26 04:27:39 -03:00
// get_motor_mask - returns a bitmask of which outputs are being used for motors or servos (1 means being used)
// this can be used to ensure other pwm outputs (i.e. for servos) do not conflict
uint16_t AP_MotorsTri : : get_motor_mask ( )
{
2015-05-25 10:26:48 -03:00
// tri copter uses channels 1,2,4 and 7
2016-04-21 04:32:25 -03:00
return rc_map_mask ( ( 1U < < AP_MOTORS_MOT_1 ) |
( 1U < < AP_MOTORS_MOT_2 ) |
( 1U < < AP_MOTORS_MOT_4 ) |
( 1U < < AP_MOTORS_CH_TRI_YAW ) ) ;
2014-07-26 04:27:39 -03:00
}
2016-01-19 23:14:20 -04:00
// output_armed - sends commands to the motors
// includes new scaling stability patch
2015-04-02 17:54:15 -03:00
void AP_MotorsTri : : output_armed_stabilizing ( )
{
2016-01-19 23:14:20 -04:00
float roll_thrust ; // roll thrust input value, +/- 1.0
float pitch_thrust ; // pitch thrust input value, +/- 1.0
float yaw_thrust ; // yaw thrust input value, +/- 1.0
float throttle_thrust ; // throttle thrust input value, 0.0 - 1.0
float throttle_thrust_best_rpy ; // throttle providing maximum roll, pitch and yaw range without climbing
float throttle_thrust_rpy_mix ; // partial calculation of throttle_thrust_best_rpy
float rpy_scale = 1.0f ; // this is used to scale the roll, pitch and yaw to fit within the motor limits
float rpy_low = 0.0f ; // lowest motor value
float rpy_high = 0.0f ; // highest motor value
float thr_adj ; // the difference between the pilot's desired throttle and throttle_thrust_best_rpy
float throttle_thrust_hover = get_hover_throttle_as_high_end_pct ( ) ; // throttle hover thrust value, 0.0 - 1.0
2016-03-25 05:56:58 -03:00
// sanity check YAW_SV_ANGLE parameter value to avoid divide by zero
_yaw_servo_angle_max_deg = constrain_float ( _yaw_servo_angle_max_deg , AP_MOTORS_TRI_SERVO_RANGE_DEG_MIN , AP_MOTORS_TRI_SERVO_RANGE_DEG_MAX ) ;
2016-01-19 23:14:20 -04:00
// apply voltage and air pressure compensation
2016-01-21 22:09:32 -04:00
roll_thrust = _roll_in * get_compensation_gain ( ) ;
pitch_thrust = _pitch_in * get_compensation_gain ( ) ;
2016-02-08 01:40:43 -04:00
yaw_thrust = _yaw_in * get_compensation_gain ( ) * sinf ( radians ( _yaw_servo_angle_max_deg ) ) ; // we scale this so a thrust request of 1.0f will ask for full servo deflection at full rear throttle
2016-01-19 23:14:20 -04:00
throttle_thrust = get_throttle ( ) * get_compensation_gain ( ) ;
2016-03-19 11:15:05 -03:00
// calculate angle of yaw pivot
_pivot_angle = safe_asin ( yaw_thrust ) ;
if ( fabsf ( _pivot_angle ) > radians ( _yaw_servo_angle_max_deg ) ) {
limit . yaw = true ;
_pivot_angle = constrain_float ( _pivot_angle , - radians ( _yaw_servo_angle_max_deg ) , radians ( _yaw_servo_angle_max_deg ) ) ;
}
float pivot_thrust_max = cosf ( _pivot_angle ) ;
2016-01-19 23:14:20 -04:00
float thrust_max = 1.0f ;
// sanity check throttle is above zero and below current limited throttle
if ( throttle_thrust < = 0.0f ) {
throttle_thrust = 0.0f ;
2014-10-04 11:27:11 -03:00
limit . throttle_lower = true ;
}
2016-01-19 23:14:20 -04:00
if ( throttle_thrust > = _throttle_thrust_max ) {
throttle_thrust = _throttle_thrust_max ;
2014-10-04 11:27:11 -03:00
limit . throttle_upper = true ;
}
2012-08-17 03:20:26 -03:00
2016-01-19 23:14:20 -04:00
throttle_thrust_rpy_mix = MAX ( throttle_thrust , throttle_thrust * MAX ( 0.0f , 1.0f - _throttle_rpy_mix ) + throttle_thrust_hover * _throttle_rpy_mix ) ;
2015-01-21 07:16:34 -04:00
2016-01-19 23:14:20 -04:00
// The following mix may be offer less coupling between axis but needs testing
//_thrust_right = roll_thrust * -0.5f + pitch_thrust * 1.0f;
//_thrust_left = roll_thrust * 0.5f + pitch_thrust * 1.0f;
//_thrust_rear = 0;
2012-08-17 03:20:26 -03:00
2016-01-19 23:14:20 -04:00
_thrust_right = roll_thrust * - 0.5f + pitch_thrust * 0.5f ;
_thrust_left = roll_thrust * 0.5f + pitch_thrust * 0.5f ;
_thrust_rear = pitch_thrust * - 0.5f ;
2013-09-12 14:36:58 -03:00
2016-01-19 23:14:20 -04:00
// calculate roll and pitch for each motor
// set rpy_low and rpy_high to the lowest and highest values of the motors
2013-07-16 00:46:34 -03:00
2016-01-19 23:14:20 -04:00
// record lowest roll pitch command
rpy_low = MIN ( _thrust_right , _thrust_left ) ;
rpy_high = MAX ( _thrust_right , _thrust_left ) ;
if ( rpy_low > _thrust_rear ) {
rpy_low = _thrust_rear ;
}
// check to see if the rear motor will reach maximum thrust before the front two motors
if ( ( 1.0f - rpy_high ) > ( pivot_thrust_max - _thrust_rear ) ) {
thrust_max = pivot_thrust_max ;
rpy_high = _thrust_rear ;
}
2015-04-02 17:54:15 -03:00
2016-01-19 23:14:20 -04:00
// calculate throttle that gives most possible room for yaw (range 1000 ~ 2000) which is the lower of:
// 1. 0.5f - (rpy_low+rpy_high)/2.0 - this would give the maximum possible room margin above the highest motor and below the lowest
// 2. the higher of:
// a) the pilot's throttle input
// b) the point _throttle_rpy_mix between the pilot's input throttle and hover-throttle
// Situation #2 ensure we never increase the throttle above hover throttle unless the pilot has commanded this.
// Situation #2b allows us to raise the throttle above what the pilot commanded but not so far that it would actually cause the copter to rise.
// We will choose #1 (the best throttle for yaw control) if that means reducing throttle to the motors (i.e. we favor reducing throttle *because* it provides better yaw control)
// We will choose #2 (a mix of pilot and hover throttle) only when the throttle is quite low. We favor reducing throttle instead of better yaw control because the pilot has commanded it
// check everything fits
throttle_thrust_best_rpy = MIN ( 0.5f * thrust_max - ( rpy_low + rpy_high ) / 2.0 , throttle_thrust_rpy_mix ) ;
if ( is_zero ( rpy_low ) ) {
rpy_scale = 1.0f ;
} else {
rpy_scale = constrain_float ( - throttle_thrust_best_rpy / rpy_low , 0.0f , 1.0f ) ;
}
2013-07-16 00:46:34 -03:00
2016-01-19 23:14:20 -04:00
// calculate how close the motors can come to the desired throttle
thr_adj = throttle_thrust - throttle_thrust_best_rpy ;
if ( rpy_scale < 1.0f ) {
// Full range is being used by roll, pitch, and yaw.
limit . roll_pitch = true ;
2016-02-03 00:49:06 -04:00
if ( thr_adj > 0.0f ) {
2016-01-19 23:14:20 -04:00
limit . throttle_upper = true ;
2013-07-16 00:46:34 -03:00
}
2016-01-19 23:14:20 -04:00
thr_adj = 0.0f ;
} else {
if ( thr_adj < - ( throttle_thrust_best_rpy + rpy_low ) ) {
// Throttle can't be reduced to desired value
thr_adj = - ( throttle_thrust_best_rpy + rpy_low ) ;
} else if ( thr_adj > thrust_max - ( throttle_thrust_best_rpy + rpy_high ) ) {
// Throttle can't be increased to desired value
thr_adj = thrust_max - ( throttle_thrust_best_rpy + rpy_high ) ;
limit . throttle_upper = true ;
2013-07-16 00:46:34 -03:00
}
2016-01-19 23:14:20 -04:00
}
2013-07-16 00:46:34 -03:00
2016-01-19 23:14:20 -04:00
// add scaled roll, pitch, constrained yaw and throttle for each motor
_thrust_right = throttle_thrust_best_rpy + thr_adj + rpy_scale * _thrust_right ;
_thrust_left = throttle_thrust_best_rpy + thr_adj + rpy_scale * _thrust_left ;
_thrust_rear = throttle_thrust_best_rpy + thr_adj + rpy_scale * _thrust_rear ;
// scale pivot thrust to account for pivot angle
2016-03-25 05:56:58 -03:00
// we should not need to check for divide by zero as _pivot_angle is constrained to the 5deg ~ 80 deg range
2016-01-19 23:14:20 -04:00
_thrust_rear = _thrust_rear / cosf ( _pivot_angle ) ;
// constrain all outputs to 0.0f to 1.0f
// test code should be run with these lines commented out as they should not do anything
_thrust_right = constrain_float ( _thrust_right , 0.0f , 1.0f ) ;
_thrust_left = constrain_float ( _thrust_left , 0.0f , 1.0f ) ;
_thrust_rear = constrain_float ( _thrust_rear , 0.0f , 1.0f ) ;
2012-04-02 05:26:37 -03:00
}
2014-04-28 04:30:26 -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
void AP_MotorsTri : : output_test ( uint8_t motor_seq , int16_t pwm )
2012-04-02 05:26:37 -03:00
{
2014-04-28 04:30:26 -03:00
// exit immediately if not armed
2015-04-02 17:54:15 -03:00
if ( ! armed ( ) ) {
2014-04-28 04:30:26 -03:00
return ;
}
2012-10-26 20:59:07 -03:00
2014-04-28 04:30:26 -03:00
// output to motors and servos
switch ( motor_seq ) {
case 1 :
// front right motor
2016-01-04 01:56:54 -04:00
rc_write ( AP_MOTORS_MOT_1 , pwm ) ;
2014-04-28 04:30:26 -03:00
break ;
case 2 :
// back motor
2016-01-04 01:56:54 -04:00
rc_write ( AP_MOTORS_MOT_4 , pwm ) ;
2014-04-28 04:30:26 -03:00
break ;
case 3 :
// back servo
2016-01-04 01:56:54 -04:00
rc_write ( AP_MOTORS_CH_TRI_YAW , pwm ) ;
2014-04-28 04:30:26 -03:00
break ;
case 4 :
// front left motor
2016-01-04 01:56:54 -04:00
rc_write ( AP_MOTORS_MOT_2 , pwm ) ;
2014-04-28 04:30:26 -03:00
break ;
default :
// do nothing
break ;
}
2012-10-26 20:59:07 -03:00
}
2015-05-14 22:00:46 -03:00
// calc_yaw_radio_output - calculate final radio output for yaw channel
2016-01-19 23:15:10 -04:00
int16_t AP_MotorsTri : : calc_yaw_radio_output ( float yaw_input , float yaw_input_max )
2015-05-14 22:00:46 -03:00
{
int16_t ret ;
2016-02-18 02:55:11 -04:00
if ( _yaw_reverse < 0 ) {
yaw_input = - yaw_input ;
}
2016-01-19 23:15:10 -04:00
if ( yaw_input > = 0 ) {
ret = ( _yaw_servo_trim + ( yaw_input / yaw_input_max * ( _yaw_servo_max - _yaw_servo_trim ) ) ) ;
2015-05-14 22:00:46 -03:00
} else {
2016-01-19 23:15:10 -04:00
ret = ( _yaw_servo_trim + ( yaw_input / yaw_input_max * ( _yaw_servo_trim - _yaw_servo_min ) ) ) ;
2015-05-14 22:00:46 -03:00
}
return ret ;
}
2016-05-01 19:00:45 -03:00
/*
call vehicle supplied thrust compensation if set . This allows for
vehicle specific thrust compensation for motor arrangements such as
the forward motors tilting
*/
void AP_MotorsTri : : thrust_compensation ( void )
{
if ( _thrust_compensation_callback ) {
// convert 3 thrust values into an array indexed by motor number
float thrust [ 4 ] { _thrust_right , _thrust_left , 0 , _thrust_rear } ;
// apply vehicle supplied compensation function
_thrust_compensation_callback ( thrust , 4 ) ;
// extract compensated thrust values
_thrust_right = thrust [ 0 ] ;
_thrust_left = thrust [ 1 ] ;
_thrust_rear = thrust [ 3 ] ;
}
}