2015-11-24 04:24:04 -04:00
# include "Plane.h"
const AP_Param : : GroupInfo QuadPlane : : var_info [ ] = {
2015-12-26 06:40:40 -04:00
// @Param: ENABLE
// @DisplayName: Enable QuadPlane
2016-04-22 07:20:45 -03:00
// @Description: This enables QuadPlane functionality, assuming multicopter motors start on output 5. If this is set to 2 then when starting AUTO mode it will initially be in VTOL AUTO mode.
// @Values: 0:Disable,1:Enable,2:Enable VTOL AUTO
2015-12-26 06:40:40 -04:00
// @User: Standard
2015-12-29 07:43:43 -04:00
AP_GROUPINFO_FLAGS ( " ENABLE " , 1 , QuadPlane , enable , 0 , AP_PARAM_FLAG_ENABLE ) ,
2015-12-26 06:40:40 -04:00
// @Group: M_
2015-11-24 04:24:04 -04:00
// @Path: ../libraries/AP_Motors/AP_MotorsMulticopter.cpp
2017-01-09 04:19:42 -04:00
AP_SUBGROUPPTR ( motors , " M_ " , 2 , QuadPlane , AP_MotorsMulticopter ) ,
2015-11-24 04:24:04 -04:00
2016-02-15 04:45:56 -04:00
// 3 ~ 8 were used by quadplane attitude control PIDs
2015-11-24 04:24:04 -04:00
// @Param: ANGLE_MAX
// @DisplayName: Angle Max
2016-04-28 21:23:26 -03:00
// @Description: Maximum lean angle in all VTOL flight modes
2015-11-24 04:24:04 -04:00
// @Units: Centi-degrees
// @Range: 1000 8000
// @User: Advanced
2016-04-28 21:23:26 -03:00
AP_GROUPINFO ( " ANGLE_MAX " , 10 , QuadPlane , aparm . angle_max , 3000 ) ,
2015-11-24 04:24:04 -04:00
// @Param: TRANSITION_MS
// @DisplayName: Transition time
2015-12-26 04:51:05 -04:00
// @Description: Transition time in milliseconds after minimum airspeed is reached
2016-12-30 01:15:56 -04:00
// @Units: milliseconds
2015-11-24 04:24:04 -04:00
// @Range: 0 30000
// @User: Advanced
2015-12-26 06:40:40 -04:00
AP_GROUPINFO ( " TRANSITION_MS " , 11 , QuadPlane , transition_time_ms , 5000 ) ,
2015-11-24 04:24:04 -04:00
2015-12-26 03:45:42 -04:00
// @Param: PZ_P
2015-11-24 04:24:04 -04:00
// @DisplayName: Position (vertical) controller P gain
// @Description: Position (vertical) controller P gain. Converts the difference between the desired altitude and actual altitude into a climb or descent rate which is passed to the throttle rate controller
// @Range: 1.000 3.000
// @User: Standard
2015-12-26 06:40:40 -04:00
AP_SUBGROUPINFO ( p_alt_hold , " PZ_ " , 12 , QuadPlane , AC_P ) ,
2015-11-24 04:24:04 -04:00
2015-12-26 03:45:42 -04:00
// @Param: PXY_P
2015-11-24 04:24:04 -04:00
// @DisplayName: Position (horizonal) controller P gain
// @Description: Loiter position controller P gain. Converts the distance (in the latitude direction) to the target location into a desired speed which is then passed to the loiter latitude rate controller
// @Range: 0.500 2.000
// @User: Standard
2015-12-26 06:40:40 -04:00
AP_SUBGROUPINFO ( p_pos_xy , " PXY_ " , 13 , QuadPlane , AC_P ) ,
2015-11-24 04:24:04 -04:00
2015-12-26 03:45:42 -04:00
// @Param: VXY_P
2015-11-24 04:24:04 -04:00
// @DisplayName: Velocity (horizontal) P gain
// @Description: Velocity (horizontal) P gain. Converts the difference between desired velocity to a target acceleration
// @Range: 0.1 6.0
// @Increment: 0.1
// @User: Advanced
2015-12-26 03:45:42 -04:00
// @Param: VXY_I
2015-11-24 04:24:04 -04:00
// @DisplayName: Velocity (horizontal) I gain
// @Description: Velocity (horizontal) I gain. Corrects long-term difference in desired velocity to a target acceleration
// @Range: 0.02 1.00
// @Increment: 0.01
// @User: Advanced
2015-12-26 03:45:42 -04:00
// @Param: VXY_IMAX
2015-11-24 04:24:04 -04:00
// @DisplayName: Velocity (horizontal) integrator maximum
// @Description: Velocity (horizontal) integrator maximum. Constrains the target acceleration that the I gain will output
// @Range: 0 4500
// @Increment: 10
// @Units: cm/s/s
// @User: Advanced
2015-12-26 06:40:40 -04:00
AP_SUBGROUPINFO ( pi_vel_xy , " VXY_ " , 14 , QuadPlane , AC_PI_2D ) ,
2015-11-24 04:24:04 -04:00
2015-12-26 03:45:42 -04:00
// @Param: VZ_P
2015-11-24 04:24:04 -04:00
// @DisplayName: Velocity (vertical) P gain
// @Description: Velocity (vertical) P gain. Converts the difference between desired vertical speed and actual speed into a desired acceleration that is passed to the throttle acceleration controller
// @Range: 1.000 8.000
// @User: Standard
2015-12-26 06:40:40 -04:00
AP_SUBGROUPINFO ( p_vel_z , " VZ_ " , 15 , QuadPlane , AC_P ) ,
2015-11-24 04:24:04 -04:00
2015-12-26 03:45:42 -04:00
// @Param: AZ_P
2015-11-24 04:24:04 -04:00
// @DisplayName: Throttle acceleration controller P gain
// @Description: Throttle acceleration controller P gain. Converts the difference between desired vertical acceleration and actual acceleration into a motor output
// @Range: 0.500 1.500
// @User: Standard
2015-12-26 03:45:42 -04:00
// @Param: AZ_I
2015-11-24 04:24:04 -04:00
// @DisplayName: Throttle acceleration controller I gain
// @Description: Throttle acceleration controller I gain. Corrects long-term difference in desired vertical acceleration and actual acceleration
// @Range: 0.000 3.000
// @User: Standard
2015-12-26 03:45:42 -04:00
// @Param: AZ_IMAX
2015-11-24 04:24:04 -04:00
// @DisplayName: Throttle acceleration controller I gain maximum
// @Description: Throttle acceleration controller I gain maximum. Constrains the maximum pwm that the I term will generate
// @Range: 0 1000
// @Units: Percent*10
// @User: Standard
2015-12-26 03:45:42 -04:00
// @Param: AZ_D
2015-11-24 04:24:04 -04:00
// @DisplayName: Throttle acceleration controller D gain
// @Description: Throttle acceleration controller D gain. Compensates for short-term change in desired vertical acceleration vs actual acceleration
// @Range: 0.000 0.400
// @User: Standard
2015-12-26 03:45:42 -04:00
// @Param: AZ_FILT_HZ
2015-11-24 04:24:04 -04:00
// @DisplayName: Throttle acceleration filter
// @Description: Filter applied to acceleration to reduce noise. Lower values reduce noise but add delay.
// @Range: 1.000 100.000
// @Units: Hz
// @User: Standard
2015-12-26 06:40:40 -04:00
AP_SUBGROUPINFO ( pid_accel_z , " AZ_ " , 16 , QuadPlane , AC_PID ) ,
2015-11-24 04:24:04 -04:00
2015-12-26 03:45:42 -04:00
// @Group: P_
2015-11-24 04:24:04 -04:00
// @Path: ../libraries/AC_AttitudeControl/AC_PosControl.cpp
2015-12-26 06:40:40 -04:00
AP_SUBGROUPPTR ( pos_control , " P " , 17 , QuadPlane , AC_PosControl ) ,
2015-12-26 03:45:42 -04:00
// @Param: VELZ_MAX
// @DisplayName: Pilot maximum vertical speed
// @Description: The maximum vertical velocity the pilot may request in cm/s
// @Units: Centimeters/Second
// @Range: 50 500
// @Increment: 10
// @User: Standard
2015-12-26 06:40:40 -04:00
AP_GROUPINFO ( " VELZ_MAX " , 18 , QuadPlane , pilot_velocity_z_max , 250 ) ,
2015-11-24 04:24:04 -04:00
2015-12-26 03:45:42 -04:00
// @Param: ACCEL_Z
// @DisplayName: Pilot vertical acceleration
// @Description: The vertical acceleration used when pilot is controlling the altitude
// @Units: cm/s/s
// @Range: 50 500
// @Increment: 10
// @User: Standard
2015-12-26 06:40:40 -04:00
AP_GROUPINFO ( " ACCEL_Z " , 19 , QuadPlane , pilot_accel_z , 250 ) ,
2015-12-26 03:45:42 -04:00
2015-12-26 04:51:05 -04:00
// @Group: WP_
// @Path: ../libraries/AC_WPNav/AC_WPNav.cpp
2015-12-26 06:40:40 -04:00
AP_SUBGROUPPTR ( wp_nav , " WP_ " , 20 , QuadPlane , AC_WPNav ) ,
// @Param: RC_SPEED
// @DisplayName: RC output speed in Hz
// @Description: This is the PWM refresh rate in Hz for QuadPlane quad motors
// @Units: Hz
// @Range: 50 500
// @Increment: 10
// @User: Standard
AP_GROUPINFO ( " RC_SPEED " , 21 , QuadPlane , rc_speed , 490 ) ,
// @Param: THR_MIN_PWM
// @DisplayName: Minimum PWM output
// @Description: This is the minimum PWM output for the quad motors
// @Units: Hz
// @Range: 800 2200
// @Increment: 1
// @User: Standard
AP_GROUPINFO ( " THR_MIN_PWM " , 22 , QuadPlane , thr_min_pwm , 1000 ) ,
// @Param: THR_MAX_PWM
// @DisplayName: Maximum PWM output
// @Description: This is the maximum PWM output for the quad motors
// @Units: Hz
// @Range: 800 2200
// @Increment: 1
// @User: Standard
2016-01-01 20:56:31 -04:00
AP_GROUPINFO ( " THR_MAX_PWM " , 23 , QuadPlane , thr_max_pwm , 2000 ) ,
2015-12-26 06:40:40 -04:00
// @Param: ASSIST_SPEED
// @DisplayName: Quadplane assistance speed
// @Description: This is the speed below which the quad motors will provide stability and lift assistance in fixed wing modes. Zero means no assistance except during transition
// @Units: m/s
// @Range: 0 100
// @Increment: 0.1
// @User: Standard
AP_GROUPINFO ( " ASSIST_SPEED " , 24 , QuadPlane , assist_speed , 0 ) ,
2016-01-02 00:25:49 -04:00
// @Param: YAW_RATE_MAX
// @DisplayName: Maximum yaw rate
// @Description: This is the maximum yaw rate in degrees/second
// @Units: degrees/second
// @Range: 50 500
// @Increment: 1
// @User: Standard
AP_GROUPINFO ( " YAW_RATE_MAX " , 25 , QuadPlane , yaw_rate_max , 100 ) ,
2016-01-02 02:55:56 -04:00
// @Param: LAND_SPEED
// @DisplayName: Land speed
// @Description: The descent speed for the final stage of landing in cm/s
// @Units: cm/s
// @Range: 30 200
// @Increment: 10
// @User: Standard
AP_GROUPINFO ( " LAND_SPEED " , 26 , QuadPlane , land_speed_cms , 50 ) ,
// @Param: LAND_FINAL_ALT
// @DisplayName: Land final altitude
// @Description: The altitude at which we should switch to Q_LAND_SPEED descent rate
// @Units: m
// @Range: 0.5 50
// @Increment: 0.1
// @User: Standard
AP_GROUPINFO ( " LAND_FINAL_ALT " , 27 , QuadPlane , land_final_alt , 6 ) ,
2016-01-06 00:19:57 -04:00
2016-06-16 00:02:41 -03:00
// 28 was used by THR_MID
2016-01-15 01:49:49 -04:00
// @Param: TRAN_PIT_MAX
// @DisplayName: Transition max pitch
// @Description: Maximum pitch during transition to auto fixed wing flight
// @User: Standard
// @Range: 0 30
// @Units: Degrees
// @Increment: 1
AP_GROUPINFO ( " TRAN_PIT_MAX " , 29 , QuadPlane , transition_pitch_max , 3 ) ,
2016-02-07 20:00:19 -04:00
2016-12-30 19:50:01 -04:00
// frame class was moved from 30 when consolidating AP_Motors classes
# define FRAME_CLASS_OLD_IDX 30
2016-02-07 20:00:19 -04:00
// @Param: FRAME_CLASS
// @DisplayName: Frame Class
// @Description: Controls major frame class for multicopter component
2016-12-12 09:08:16 -04:00
// @Values: 0:Undefined, 1:Quad, 2:Hexa, 3:Octa, 4:OctaQuad, 5:Y6, 7:Tri
2016-02-07 20:00:19 -04:00
// @User: Standard
2016-12-30 19:50:01 -04:00
AP_GROUPINFO ( " FRAME_CLASS " , 46 , QuadPlane , frame_class , 1 ) ,
2016-02-07 20:00:19 -04:00
// @Param: FRAME_TYPE
// @DisplayName: Frame Type (+, X or V)
// @Description: Controls motor mixing for multicopter component
// @Values: 0:Plus, 1:X, 2:V, 3:H, 4:V-Tail, 5:A-Tail, 10:Y6B
// @User: Standard
AP_GROUPINFO ( " FRAME_TYPE " , 31 , QuadPlane , frame_type , 1 ) ,
2016-04-20 02:13:20 -03:00
// @Param: VFWD_GAIN
// @DisplayName: Forward velocity hold gain
2016-06-03 22:04:03 -03:00
// @Description: Controls use of forward motor in vtol modes. If this is zero then the forward motor will not be used for position control in VTOL modes. A value of 0.05 is a good place to start if you want to use the forward motor for position control. No forward motor will be used in QSTABILIZE or QHOVER modes. Use QLOITER for position hold with the forward motor.
2016-04-20 02:13:20 -03:00
// @Range: 0 0.5
// @Increment: 0.01
// @User: Standard
AP_GROUPINFO ( " VFWD_GAIN " , 32 , QuadPlane , vel_forward . gain , 0 ) ,
2016-04-20 03:23:17 -03:00
// @Param: WVANE_GAIN
// @DisplayName: Weathervaning gain
2016-06-03 22:04:03 -03:00
// @Description: This controls the tendency to yaw to face into the wind. A value of 0.1 is to start with and will give a slow turn into the wind. Use a value of 0.4 for more rapid response. The weathervaning works by turning into the direction of roll.
2016-04-20 03:23:17 -03:00
// @Range: 0 1
// @Increment: 0.01
// @User: Standard
AP_GROUPINFO ( " WVANE_GAIN " , 33 , QuadPlane , weathervane . gain , 0 ) ,
2016-04-21 08:52:25 -03:00
// @Param: WVANE_MINROLL
// @DisplayName: Weathervaning min roll
// @Description: This set the minimum roll in degrees before active weathervaning will start. This may need to be larger if your aircraft has bad roll trim.
// @Range: 0 10
// @Increment: 0.1
// @User: Standard
AP_GROUPINFO ( " WVANE_MINROLL " , 34 , QuadPlane , weathervane . min_roll , 1 ) ,
2016-04-29 02:31:08 -03:00
// @Param: RTL_ALT
// @DisplayName: QRTL return altitude
// @Description: The altitude which QRTL mode heads to initially
// @Units: m
// @Range: 1 200
// @Increment: 1
// @User: Standard
AP_GROUPINFO ( " RTL_ALT " , 35 , QuadPlane , qrtl_alt , 15 ) ,
2016-04-29 03:31:22 -03:00
// @Param: RTL_MODE
// @DisplayName: VTOL RTL mode
// @Description: If this is set to 1 then an RTL will change to QRTL when the loiter target is reached
// @Values: 0:Disabled,1:Enabled
// @User: Standard
AP_GROUPINFO ( " RTL_MODE " , 36 , QuadPlane , rtl_mode , 0 ) ,
2016-05-01 05:07:52 -03:00
// @Param: TILT_MASK
// @DisplayName: Tiltrotor mask
// @Description: This is a bitmask of motors that are tiltable in a tiltrotor (or tiltwing). The mask is in terms of the standard motor order for the frame type.
// @User: Standard
AP_GROUPINFO ( " TILT_MASK " , 37 , QuadPlane , tilt . tilt_mask , 0 ) ,
// @Param: TILT_RATE
// @DisplayName: Tiltrotor tilt rate
// @Description: This is the maximum speed at which the motor angle will change for a tiltrotor
// @Units: degrees/second
// @Increment: 1
// @Range: 10 300
// @User: Standard
AP_GROUPINFO ( " TILT_RATE " , 38 , QuadPlane , tilt . max_rate_dps , 40 ) ,
// @Param: TILT_MAX
// @DisplayName: Tiltrotor maximum VTOL angle
// @Description: This is the maximum angle of the tiltable motors at which multicopter control will be enabled. Beyond this angle the plane will fly solely as a fixed wing aircraft and the motors will tilt to their maximum angle at the TILT_RATE
// @Units: degrees
// @Increment: 1
// @Range: 20 80
// @User: Standard
AP_GROUPINFO ( " TILT_MAX " , 39 , QuadPlane , tilt . max_angle_deg , 45 ) ,
2016-05-05 19:27:47 -03:00
// @Param: GUIDED_MODE
// @DisplayName: Enable VTOL in GUIDED mode
// @Description: This enables use of VTOL in guided mode. When enabled the aircraft will switch to VTOL flight when the guided destination is reached and hover at the destination.
// @Values: 0:Disabled,1:Enabled
// @User: Standard
AP_GROUPINFO ( " GUIDED_MODE " , 40 , QuadPlane , guided_mode , 0 ) ,
2016-05-07 19:11:00 -03:00
2016-06-16 00:02:41 -03:00
// 41 was used by THR_MIN
2016-06-02 00:10:39 -03:00
// @Param: ESC_CAL
// @DisplayName: ESC Calibration
// @Description: This is used to calibrate the throttle range of the VTOL motors. Please read http://ardupilot.org/plane/docs/quadplane-esc-calibration.html before using. This parameter is automatically set back to 0 on every boot. This parameter only takes effect in QSTABILIZE mode. When set to 1 the output of all motors will come directly from the throttle stick when armed, and will be zero when disarmed. When set to 2 the output of all motors will be maximum when armed and zero when disarmed. Make sure you remove all properllers before using.
// @Values: 0:Disabled,1:ThrottleInput,2:FullInput
// @User: Standard
AP_GROUPINFO ( " ESC_CAL " , 42 , QuadPlane , esc_calibration , 0 ) ,
2016-06-03 19:35:09 -03:00
// @Param: VFWD_ALT
// @DisplayName: Forward velocity alt cutoff
// @Description: Controls altitude to disable forward velocity assist when below this relative altitude. This is useful to keep the forward velocity propeller from hitting the ground. Rangefinder height data is incorporated when available.
// @Range: 0 10
// @Increment: 0.25
// @User: Standard
AP_GROUPINFO ( " VFWD_ALT " , 43 , QuadPlane , vel_forward_alt_cutoff , 0 ) ,
2016-07-25 02:46:17 -03:00
// @Param: LAND_ICE_CUT
// @DisplayName: Cut IC engine on landing
// @Description: This controls stopping an internal combustion engine in the final landing stage of a VTOL. This is important for aircraft where the forward thrust engine may experience prop-strike if left running during landing. This requires the engine controls are enabled using the ICE_* parameters.
// @Values: 0:Disabled,1:Enabled
// @User: Standard
AP_GROUPINFO ( " LAND_ICE_CUT " , 44 , QuadPlane , land_icengine_cut , 1 ) ,
2016-04-20 03:23:17 -03:00
2016-08-29 03:44:54 -03:00
// @Param: ASSIST_ANGLE
// @DisplayName: Quadplane assistance angle
// @Description: This is the angular error in attitude beyond which the quadplane VTOL motors will provide stability assistance. This will only be used if Q_ASSIST_SPEED is also non-zero. Assistance will be given if the attitude is outside the normal attitude limits by at least 5 degrees and the angular error in roll or pitch is greater than this angle for at least 1 second. Set to zero to disable angle assistance.
// @Units: degrees
// @Range: 0 90
// @Increment: 1
// @User: Standard
AP_GROUPINFO ( " ASSIST_ANGLE " , 45 , QuadPlane , assist_angle , 30 ) ,
2016-07-03 09:02:38 -03:00
// @Param: TILT_TYPE
// @DisplayName: Tiltrotor type
// @Description: This is the type of tiltrotor when TILT_MASK is non-zero. A continuous tiltrotor can tilt the rotors to any angle on demand. A binary tiltrotor assumes a retract style servo where the servo is either fully forward or fully up. In both cases the servo can't move faster than Q_TILT_RATE
// @Values: 0:Continuous,1:Binary
2017-01-22 18:29:51 -04:00
AP_GROUPINFO ( " TILT_TYPE " , 47 , QuadPlane , tilt . tilt_type , TILT_TYPE_CONTINUOUS ) ,
2016-07-03 09:02:38 -03:00
2015-11-24 04:24:04 -04:00
AP_GROUPEND
} ;
2016-04-01 02:40:06 -03:00
static const struct {
const char * name ;
float value ;
} defaults_table [ ] = {
2016-04-01 02:44:49 -03:00
{ " Q_A_RAT_RLL_P " , 0.25 } ,
{ " Q_A_RAT_RLL_I " , 0.25 } ,
{ " Q_A_RAT_RLL_FILT " , 10.0 } ,
{ " Q_A_RAT_PIT_P " , 0.25 } ,
{ " Q_A_RAT_PIT_I " , 0.25 } ,
{ " Q_A_RAT_PIT_FILT " , 10.0 } ,
2017-02-13 06:35:41 -04:00
{ " Q_M_SPOOL_TIME " , 0.25 } ,
2016-04-01 02:40:06 -03:00
} ;
2015-11-24 04:24:04 -04:00
QuadPlane : : QuadPlane ( AP_AHRS_NavEKF & _ahrs ) :
ahrs ( _ahrs )
{
AP_Param : : setup_object_defaults ( this , var_info ) ;
}
2016-02-07 20:00:19 -04:00
// setup default motors for the frame class
void QuadPlane : : setup_default_channels ( uint8_t num_motors )
{
for ( uint8_t i = 0 ; i < num_motors ; i + + ) {
2016-10-22 07:27:57 -03:00
SRV_Channels : : set_aux_channel_default ( ( SRV_Channel : : Aux_servo_function_t ) ( SRV_Channel : : k_motor1 + i ) , CH_5 + i ) ;
2016-02-07 20:00:19 -04:00
}
}
2016-01-06 03:17:08 -04:00
bool QuadPlane : : setup ( void )
2015-11-24 04:24:04 -04:00
{
2016-01-06 03:17:08 -04:00
if ( initialised ) {
return true ;
}
if ( ! enable | | hal . util - > get_soft_armed ( ) ) {
return false ;
2015-12-26 06:40:40 -04:00
}
2016-04-02 17:49:38 -03:00
float loop_delta_t = 1.0 / plane . scheduler . get_loop_rate_hz ( ) ;
2016-12-30 19:50:01 -04:00
2017-01-09 04:19:42 -04:00
enum AP_Motors : : motor_frame_class motor_class ;
2016-12-30 19:50:01 -04:00
/*
cope with upgrade from old AP_Motors values for frame_class
*/
AP_Int8 old_class ;
const AP_Param : : ConversionInfo cinfo { Parameters : : k_param_quadplane , FRAME_CLASS_OLD_IDX , AP_PARAM_INT8 , nullptr } ;
if ( AP_Param : : find_old_parameter ( & cinfo , & old_class ) & & ! frame_class . load ( ) ) {
uint8_t new_value = 0 ;
// map from old values to new values
switch ( old_class . get ( ) ) {
case 0 :
new_value = AP_Motors : : MOTOR_FRAME_QUAD ;
break ;
case 1 :
new_value = AP_Motors : : MOTOR_FRAME_HEXA ;
break ;
case 2 :
new_value = AP_Motors : : MOTOR_FRAME_OCTA ;
break ;
case 3 :
new_value = AP_Motors : : MOTOR_FRAME_OCTAQUAD ;
break ;
case 4 :
new_value = AP_Motors : : MOTOR_FRAME_Y6 ;
break ;
}
frame_class . set_and_save ( new_value ) ;
}
2016-04-02 17:49:38 -03:00
2016-01-06 03:17:08 -04:00
if ( hal . util - > available_memory ( ) <
4096 + sizeof ( * motors ) + sizeof ( * attitude_control ) + sizeof ( * pos_control ) + sizeof ( * wp_nav ) ) {
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " Not enough memory for quadplane " ) ;
goto failed ;
}
2016-02-07 20:00:19 -04:00
2015-12-26 06:40:40 -04:00
/*
dynamically allocate the key objects for quadplane . This ensures
that the objects don ' t affect the vehicle unless enabled and
also saves memory when not in use
*/
2017-01-09 04:19:42 -04:00
motor_class = ( enum AP_Motors : : motor_frame_class ) frame_class . get ( ) ;
switch ( motor_class ) {
2016-12-12 09:08:16 -04:00
case AP_Motors : : MOTOR_FRAME_QUAD :
2016-02-07 20:00:19 -04:00
setup_default_channels ( 4 ) ;
break ;
2016-12-12 09:08:16 -04:00
case AP_Motors : : MOTOR_FRAME_HEXA :
2016-02-07 20:00:19 -04:00
setup_default_channels ( 6 ) ;
break ;
2016-12-12 09:08:16 -04:00
case AP_Motors : : MOTOR_FRAME_OCTA :
case AP_Motors : : MOTOR_FRAME_OCTAQUAD :
2016-02-07 20:00:19 -04:00
setup_default_channels ( 8 ) ;
break ;
2016-12-12 09:08:16 -04:00
case AP_Motors : : MOTOR_FRAME_Y6 :
2016-04-21 21:28:06 -03:00
setup_default_channels ( 7 ) ;
2016-03-12 19:04:25 -04:00
break ;
2017-01-09 04:19:42 -04:00
case AP_Motors : : MOTOR_FRAME_TRI :
SRV_Channels : : set_default_function ( CH_5 , SRV_Channel : : k_motor1 ) ;
SRV_Channels : : set_default_function ( CH_6 , SRV_Channel : : k_motor2 ) ;
SRV_Channels : : set_default_function ( CH_8 , SRV_Channel : : k_motor4 ) ;
SRV_Channels : : set_default_function ( CH_11 , SRV_Channel : : k_motor7 ) ;
2017-02-07 19:47:17 -04:00
AP_Param : : set_frame_type_flags ( AP_PARAM_FRAME_TRICOPTER ) ;
2017-01-09 04:19:42 -04:00
break ;
2016-02-07 20:00:19 -04:00
default :
hal . console - > printf ( " Unknown frame class %u \n " , ( unsigned ) frame_class . get ( ) ) ;
goto failed ;
}
2017-01-09 04:19:42 -04:00
if ( motor_class = = AP_Motors : : MOTOR_FRAME_TRI ) {
motors = new AP_MotorsTri ( plane . scheduler . get_loop_rate_hz ( ) ) ;
} else {
motors = new AP_MotorsMatrix ( plane . scheduler . get_loop_rate_hz ( ) ) ;
}
2016-10-06 09:47:33 -03:00
const static char * strUnableToAllocate = " Unable to allocate " ;
2015-12-26 06:40:40 -04:00
if ( ! motors ) {
2016-10-06 09:47:33 -03:00
hal . console - > printf ( " %s motors \n " , strUnableToAllocate ) ;
2016-01-06 03:17:08 -04:00
goto failed ;
2015-12-26 06:40:40 -04:00
}
2016-02-07 20:00:19 -04:00
2015-12-26 06:40:40 -04:00
AP_Param : : load_object_from_eeprom ( motors , motors - > var_info ) ;
2016-04-02 17:49:38 -03:00
attitude_control = new AC_AttitudeControl_Multi ( ahrs , aparm , * motors , loop_delta_t ) ;
2015-12-26 06:40:40 -04:00
if ( ! attitude_control ) {
2016-10-06 09:47:33 -03:00
hal . console - > printf ( " %s attitude_control \n " , strUnableToAllocate ) ;
2016-01-06 03:17:08 -04:00
goto failed ;
2015-12-26 06:40:40 -04:00
}
AP_Param : : load_object_from_eeprom ( attitude_control , attitude_control - > var_info ) ;
pos_control = new AC_PosControl ( ahrs , inertial_nav , * motors , * attitude_control ,
p_alt_hold , p_vel_z , pid_accel_z ,
p_pos_xy , pi_vel_xy ) ;
if ( ! pos_control ) {
2016-10-06 09:47:33 -03:00
hal . console - > printf ( " %s pos_control \n " , strUnableToAllocate ) ;
2016-01-06 03:17:08 -04:00
goto failed ;
2015-12-26 06:40:40 -04:00
}
AP_Param : : load_object_from_eeprom ( pos_control , pos_control - > var_info ) ;
wp_nav = new AC_WPNav ( inertial_nav , ahrs , * pos_control , * attitude_control ) ;
2016-10-26 23:54:15 -03:00
if ( ! wp_nav ) {
2016-10-06 09:47:33 -03:00
hal . console - > printf ( " %s wp_nav \n " , strUnableToAllocate ) ;
2016-01-06 03:17:08 -04:00
goto failed ;
2015-12-26 06:40:40 -04:00
}
AP_Param : : load_object_from_eeprom ( wp_nav , wp_nav - > var_info ) ;
2016-12-12 09:08:16 -04:00
motors - > init ( ( AP_Motors : : motor_frame_class ) frame_class . get ( ) , ( AP_Motors : : motor_frame_type ) frame_type . get ( ) ) ;
2016-06-09 10:19:18 -03:00
motors - > set_throttle_range ( thr_min_pwm , thr_max_pwm ) ;
2015-12-26 06:40:40 -04:00
motors - > set_update_rate ( rc_speed ) ;
motors - > set_interlock ( true ) ;
2016-04-02 17:49:38 -03:00
pid_accel_z . set_dt ( loop_delta_t ) ;
pos_control - > set_dt ( loop_delta_t ) ;
2016-09-01 08:32:24 -03:00
attitude_control - > parameter_sanity_check ( ) ;
2015-12-26 06:40:40 -04:00
2016-01-01 22:29:05 -04:00
// setup the trim of any motors used by AP_Motors so px4io
// failsafe will disable motors
2016-05-06 05:09:07 -03:00
for ( uint8_t i = 0 ; i < 8 ; i + + ) {
2016-10-22 07:27:57 -03:00
SRV_Channel : : Aux_servo_function_t func = ( SRV_Channel : : Aux_servo_function_t ) ( SRV_Channel : : k_motor1 + i ) ;
SRV_Channels : : set_failsafe_pwm ( func , thr_min_pwm ) ;
2016-01-01 22:29:05 -04:00
}
2016-09-10 19:19:22 -03:00
# if HAVE_PX4_MIXER
2016-01-01 22:29:05 -04:00
// redo failsafe mixing on px4
plane . setup_failsafe_mixing ( ) ;
# endif
2015-12-26 06:40:40 -04:00
transition_state = TRANSITION_DONE ;
2016-04-01 02:40:06 -03:00
setup_defaults ( ) ;
2015-12-26 06:40:40 -04:00
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " QuadPlane initialised " ) ;
2016-01-06 03:17:08 -04:00
initialised = true ;
return true ;
failed :
initialised = false ;
enable . set ( 0 ) ;
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " QuadPlane setup failed " ) ;
return false ;
2015-11-24 04:24:04 -04:00
}
2016-04-01 02:40:06 -03:00
/*
setup default parameters from defaults_table
*/
void QuadPlane : : setup_defaults ( void )
{
for ( uint8_t i = 0 ; i < ARRAY_SIZE ( defaults_table ) ; i + + ) {
if ( ! AP_Param : : set_default_by_name ( defaults_table [ i ] . name , defaults_table [ i ] . value ) ) {
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " QuadPlane setup failure for %s " ,
defaults_table [ i ] . name ) ;
AP_HAL : : panic ( " quadplane bad default %s " , defaults_table [ i ] . name ) ;
}
}
2016-06-02 00:10:39 -03:00
// reset ESC calibration
if ( esc_calibration ! = 0 ) {
esc_calibration . set_and_save ( 0 ) ;
}
2016-04-01 02:40:06 -03:00
}
2016-06-02 00:10:39 -03:00
// run ESC calibration
void QuadPlane : : run_esc_calibration ( void )
{
if ( ! motors - > armed ( ) ) {
motors - > set_throttle_passthrough_for_esc_calibration ( 0 ) ;
AP_Notify : : flags . esc_calibration = false ;
return ;
}
if ( ! AP_Notify : : flags . esc_calibration ) {
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " Starting ESC calibration " ) ;
}
AP_Notify : : flags . esc_calibration = true ;
switch ( esc_calibration ) {
case 1 :
// throttle based calibration
motors - > set_throttle_passthrough_for_esc_calibration ( plane . channel_throttle - > get_control_in ( ) * 0.01f ) ;
break ;
case 2 :
// full range calibration
motors - > set_throttle_passthrough_for_esc_calibration ( 1 ) ;
break ;
}
}
2015-12-26 03:45:42 -04:00
// init quadplane stabilize mode
void QuadPlane : : init_stabilize ( void )
{
2015-12-26 05:13:20 -04:00
throttle_wait = false ;
2015-12-26 03:45:42 -04:00
}
2015-12-26 04:27:13 -04:00
// hold in stabilize with given throttle
void QuadPlane : : hold_stabilize ( float throttle_in )
2015-12-26 04:51:05 -04:00
{
2015-11-24 04:24:04 -04:00
// call attitude controller
2016-06-16 23:48:25 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( plane . nav_roll_cd ,
2015-12-26 06:40:40 -04:00
plane . nav_pitch_cd ,
2016-01-09 18:42:46 -04:00
get_desired_yaw_rate_cds ( ) ,
2015-12-26 06:40:40 -04:00
smoothing_gain ) ;
2015-11-24 04:24:04 -04:00
2016-01-05 19:32:25 -04:00
if ( throttle_in < = 0 ) {
2016-02-04 00:03:39 -04:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SPIN_WHEN_ARMED ) ;
2016-01-05 19:32:25 -04:00
attitude_control - > set_throttle_out_unstabilized ( 0 , true , 0 ) ;
} else {
2016-02-04 00:03:39 -04:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
2016-01-05 19:32:25 -04:00
attitude_control - > set_throttle_out ( throttle_in , true , 0 ) ;
}
2015-12-26 04:27:13 -04:00
}
// quadplane stabilize mode
void QuadPlane : : control_stabilize ( void )
{
2016-06-02 00:10:39 -03:00
// special check for ESC calibration in QSTABILIZE
if ( esc_calibration ! = 0 ) {
run_esc_calibration ( ) ;
return ;
}
// normal QSTABILIZE mode
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
float pilot_throttle_scaled = plane . channel_throttle - > get_control_in ( ) / 100.0f ;
2015-12-26 04:27:13 -04:00
hold_stabilize ( pilot_throttle_scaled ) ;
2015-11-24 04:24:04 -04:00
}
2015-12-26 03:45:42 -04:00
// init quadplane hover mode
void QuadPlane : : init_hover ( void )
{
// initialize vertical speeds and leash lengths
2015-12-26 06:40:40 -04:00
pos_control - > set_speed_z ( - pilot_velocity_z_max , pilot_velocity_z_max ) ;
pos_control - > set_accel_z ( pilot_accel_z ) ;
2015-12-26 03:45:42 -04:00
// initialise position and desired velocity
2015-12-26 06:40:40 -04:00
pos_control - > set_alt_target ( inertial_nav . get_altitude ( ) ) ;
pos_control - > set_desired_velocity_z ( inertial_nav . get_velocity_z ( ) ) ;
2015-12-26 05:13:20 -04:00
init_throttle_wait ( ) ;
2015-12-26 03:45:42 -04:00
}
2015-12-26 04:27:13 -04:00
/*
hold hover with target climb rate
*/
void QuadPlane : : hold_hover ( float target_climb_rate )
2015-12-26 03:45:42 -04:00
{
2016-02-04 00:03:39 -04:00
// motors use full range
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
2015-12-26 03:45:42 -04:00
// initialize vertical speeds and acceleration
2015-12-26 06:40:40 -04:00
pos_control - > set_speed_z ( - pilot_velocity_z_max , pilot_velocity_z_max ) ;
pos_control - > set_accel_z ( pilot_accel_z ) ;
2015-12-26 03:45:42 -04:00
// call attitude controller
2016-06-16 23:48:25 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( plane . nav_roll_cd ,
2015-12-26 06:40:40 -04:00
plane . nav_pitch_cd ,
2016-01-09 18:42:46 -04:00
get_desired_yaw_rate_cds ( ) ,
2015-12-26 06:40:40 -04:00
smoothing_gain ) ;
2015-12-26 03:45:42 -04:00
// call position controller
2015-12-26 06:40:40 -04:00
pos_control - > set_alt_target_from_climb_rate_ff ( target_climb_rate , plane . G_Dt , false ) ;
pos_control - > update_z_controller ( ) ;
2015-12-26 03:45:42 -04:00
2015-12-26 04:27:13 -04:00
}
/*
control QHOVER mode
*/
void QuadPlane : : control_hover ( void )
{
2015-12-26 05:13:20 -04:00
if ( throttle_wait ) {
2016-02-04 00:03:39 -04:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SPIN_WHEN_ARMED ) ;
2015-12-26 06:40:40 -04:00
attitude_control - > set_throttle_out_unstabilized ( 0 , true , 0 ) ;
pos_control - > relax_alt_hold_controllers ( 0 ) ;
2015-12-26 05:13:20 -04:00
} else {
hold_hover ( get_pilot_desired_climb_rate_cms ( ) ) ;
}
2015-12-26 03:45:42 -04:00
}
2015-12-26 04:51:05 -04:00
void QuadPlane : : init_loiter ( void )
{
// set target to current position
2015-12-26 06:40:40 -04:00
wp_nav - > init_loiter_target ( ) ;
2015-12-26 04:51:05 -04:00
// initialize vertical speed and acceleration
2015-12-26 06:40:40 -04:00
pos_control - > set_speed_z ( - pilot_velocity_z_max , pilot_velocity_z_max ) ;
pos_control - > set_accel_z ( pilot_accel_z ) ;
2015-12-26 04:51:05 -04:00
// initialise position and desired velocity
2015-12-26 06:40:40 -04:00
pos_control - > set_alt_target ( inertial_nav . get_altitude ( ) ) ;
pos_control - > set_desired_velocity_z ( inertial_nav . get_velocity_z ( ) ) ;
2015-12-26 05:13:20 -04:00
init_throttle_wait ( ) ;
2015-12-26 04:51:05 -04:00
}
2016-03-09 03:20:41 -04:00
void QuadPlane : : init_land ( void )
{
init_loiter ( ) ;
throttle_wait = false ;
2016-05-05 19:27:47 -03:00
poscontrol . state = QPOS_LAND_DESCEND ;
2016-06-16 05:17:46 -03:00
landing_detect . lower_limit_start_ms = 0 ;
2016-03-09 03:20:41 -04:00
}
2016-01-01 06:39:36 -04:00
// helper for is_flying()
bool QuadPlane : : is_flying ( void )
{
if ( ! available ( ) ) {
return false ;
}
2016-07-24 17:08:36 -03:00
if ( motors - > get_throttle ( ) > 0.01f & & ! motors - > limit . throttle_lower ) {
2016-01-01 06:39:36 -04:00
return true ;
}
return false ;
}
2015-12-26 06:40:40 -04:00
// crude landing detector to prevent tipover
bool QuadPlane : : should_relax ( void )
{
2016-06-09 10:20:50 -03:00
bool motor_at_lower_limit = motors - > limit . throttle_lower & & attitude_control - > is_throttle_mix_min ( ) ;
2016-07-24 17:08:36 -03:00
if ( motors - > get_throttle ( ) < 0.01f ) {
2016-01-01 07:09:11 -04:00
motor_at_lower_limit = true ;
}
2015-12-26 06:40:40 -04:00
if ( ! motor_at_lower_limit ) {
2016-06-16 05:17:46 -03:00
landing_detect . lower_limit_start_ms = 0 ;
2015-12-26 06:40:40 -04:00
}
2016-06-16 05:17:46 -03:00
if ( motor_at_lower_limit & & landing_detect . lower_limit_start_ms = = 0 ) {
landing_detect . lower_limit_start_ms = millis ( ) ;
2015-12-26 06:40:40 -04:00
}
2016-06-16 05:17:46 -03:00
bool relax_loiter = landing_detect . lower_limit_start_ms ! = 0 & &
( millis ( ) - landing_detect . lower_limit_start_ms ) > 1000 ;
2015-12-26 06:40:40 -04:00
return relax_loiter ;
}
2016-04-22 07:20:06 -03:00
// see if we are flying in vtol
bool QuadPlane : : is_flying_vtol ( void )
{
2016-07-26 18:43:16 -03:00
if ( ! available ( ) ) {
return false ;
}
2016-07-24 17:08:36 -03:00
if ( motors - > get_throttle ( ) > 0.01f ) {
// if we are demanding more than 1% throttle then don't consider aircraft landed
return true ;
}
if ( plane . control_mode = = QSTABILIZE | | plane . control_mode = = QHOVER | | plane . control_mode = = QLOITER ) {
// in manual flight modes only consider aircraft landed when pilot demanded throttle is zero
return plane . channel_throttle - > get_control_in ( ) > 0 ;
}
2016-06-16 05:17:46 -03:00
if ( in_vtol_mode ( ) & & millis ( ) - landing_detect . lower_limit_start_ms > 5000 ) {
2016-07-24 17:08:36 -03:00
// use landing detector
2016-04-22 07:20:06 -03:00
return true ;
}
return false ;
}
2016-04-02 08:45:51 -03:00
/*
smooth out descent rate for landing to prevent a jerk as we get to
land_final_alt .
*/
float QuadPlane : : landing_descent_rate_cms ( float height_above_ground )
{
float ret = linear_interpolate ( land_speed_cms , wp_nav - > get_speed_down ( ) ,
height_above_ground ,
land_final_alt , land_final_alt + 3 ) ;
return ret ;
}
2015-12-26 04:51:05 -04:00
// run quadplane loiter controller
void QuadPlane : : control_loiter ( )
{
2015-12-26 05:13:20 -04:00
if ( throttle_wait ) {
2016-02-04 00:03:39 -04:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SPIN_WHEN_ARMED ) ;
2015-12-26 06:40:40 -04:00
attitude_control - > set_throttle_out_unstabilized ( 0 , true , 0 ) ;
pos_control - > relax_alt_hold_controllers ( 0 ) ;
wp_nav - > init_loiter_target ( ) ;
2015-12-26 05:13:20 -04:00
return ;
}
2015-12-26 06:40:40 -04:00
if ( should_relax ( ) ) {
wp_nav - > loiter_soften_for_landing ( ) ;
}
2016-01-01 03:18:53 -04:00
if ( millis ( ) - last_loiter_ms > 500 ) {
wp_nav - > init_loiter_target ( ) ;
}
last_loiter_ms = millis ( ) ;
2016-02-04 00:03:39 -04:00
// motors use full range
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
2015-12-26 04:51:05 -04:00
// initialize vertical speed and acceleration
2015-12-26 06:40:40 -04:00
pos_control - > set_speed_z ( - pilot_velocity_z_max , pilot_velocity_z_max ) ;
pos_control - > set_accel_z ( pilot_accel_z ) ;
2015-12-26 04:51:05 -04:00
// process pilot's roll and pitch input
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
wp_nav - > set_pilot_desired_acceleration ( plane . channel_roll - > get_control_in ( ) ,
plane . channel_pitch - > get_control_in ( ) ) ;
2015-12-26 04:51:05 -04:00
// Update EKF speed limit - used to limit speed when we are using optical flow
float ekfGndSpdLimit , ekfNavVelGainScaler ;
ahrs . getEkfControlLimits ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
// run loiter controller
2015-12-26 06:40:40 -04:00
wp_nav - > update_loiter ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
2015-12-26 04:51:05 -04:00
2016-06-16 23:48:25 -03:00
// call attitude controller with conservative smoothing gain of 4.0f
2015-12-26 06:40:40 -04:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( wp_nav - > get_roll ( ) ,
wp_nav - > get_pitch ( ) ,
2016-06-16 23:48:25 -03:00
get_desired_yaw_rate_cds ( ) ,
4.0f ) ;
2015-12-26 04:51:05 -04:00
// nav roll and pitch are controller by loiter controller
2015-12-26 06:40:40 -04:00
plane . nav_roll_cd = wp_nav - > get_roll ( ) ;
plane . nav_pitch_cd = wp_nav - > get_pitch ( ) ;
2015-12-26 04:51:05 -04:00
2016-03-09 03:20:41 -04:00
if ( plane . control_mode = = QLAND ) {
2016-06-03 18:12:33 -03:00
float height_above_ground = plane . relative_ground_altitude ( plane . g . rangefinder_landing ) ;
2016-05-05 19:27:47 -03:00
if ( height_above_ground < land_final_alt & & poscontrol . state < QPOS_LAND_FINAL ) {
poscontrol . state = QPOS_LAND_FINAL ;
2016-09-28 16:18:38 -03:00
// cut IC engine if enabled
if ( land_icengine_cut ! = 0 ) {
plane . g2 . ice_control . engine_control ( 0 , 0 , 0 ) ;
}
2016-03-09 03:20:41 -04:00
}
2016-05-05 19:27:47 -03:00
float descent_rate = ( poscontrol . state = = QPOS_LAND_FINAL ) ? land_speed_cms : landing_descent_rate_cms ( height_above_ground ) ;
2016-03-09 03:20:41 -04:00
pos_control - > set_alt_target_from_climb_rate ( - descent_rate , plane . G_Dt , true ) ;
check_land_complete ( ) ;
} else {
// update altitude target and call position controller
pos_control - > set_alt_target_from_climb_rate_ff ( get_pilot_desired_climb_rate_cms ( ) , plane . G_Dt , false ) ;
}
2015-12-26 06:40:40 -04:00
pos_control - > update_z_controller ( ) ;
2015-12-26 04:51:05 -04:00
}
/*
2016-01-09 18:42:46 -04:00
get pilot input yaw rate in cd / s
2015-12-26 04:51:05 -04:00
*/
2016-01-09 18:42:46 -04:00
float QuadPlane : : get_pilot_input_yaw_rate_cds ( void )
{
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
if ( plane . channel_throttle - > get_control_in ( ) < = 0 & & ! plane . auto_throttle_mode ) {
2016-01-09 18:42:46 -04:00
// the user may be trying to disarm
return 0 ;
}
// add in rudder input
return plane . channel_rudder - > norm_input ( ) * 100 * yaw_rate_max ;
}
/*
get overall desired yaw rate in cd / s
*/
float QuadPlane : : get_desired_yaw_rate_cds ( void )
2015-12-26 04:51:05 -04:00
{
2016-01-02 00:25:49 -04:00
float yaw_cds = 0 ;
2015-12-26 06:40:40 -04:00
if ( assisted_flight ) {
// use bank angle to get desired yaw rate
2016-01-09 18:42:46 -04:00
yaw_cds + = desired_auto_yaw_rate_cds ( ) ;
2015-12-26 06:40:40 -04:00
}
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
if ( plane . channel_throttle - > get_control_in ( ) < = 0 & & ! plane . auto_throttle_mode ) {
2016-01-02 00:16:31 -04:00
// the user may be trying to disarm
return 0 ;
}
2016-01-09 18:42:46 -04:00
// add in pilot input
yaw_cds + = get_pilot_input_yaw_rate_cds ( ) ;
2016-04-20 03:23:17 -03:00
// add in weathervaning
yaw_cds + = get_weathervane_yaw_rate_cds ( ) ;
2016-01-02 00:25:49 -04:00
return yaw_cds ;
2015-12-26 04:51:05 -04:00
}
// get pilot desired climb rate in cm/s
float QuadPlane : : get_pilot_desired_climb_rate_cms ( void )
{
2015-12-26 06:40:40 -04:00
if ( plane . failsafe . ch3_failsafe | | plane . failsafe . ch3_counter > 0 ) {
// descend at 0.5m/s for now
return - 50 ;
}
uint16_t dead_zone = plane . channel_throttle - > get_dead_zone ( ) ;
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
uint16_t trim = ( plane . channel_throttle - > get_radio_max ( ) + plane . channel_throttle - > get_radio_min ( ) ) / 2 ;
2015-12-26 06:40:40 -04:00
return pilot_velocity_z_max * plane . channel_throttle - > pwm_to_angle_dz_trim ( dead_zone , trim ) / 100.0f ;
2015-12-26 04:51:05 -04:00
}
2015-12-26 05:13:20 -04:00
/*
initialise throttle_wait based on throttle and is_flying ( )
*/
void QuadPlane : : init_throttle_wait ( void )
{
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
if ( plane . channel_throttle - > get_control_in ( ) > = 10 | |
2015-12-26 05:13:20 -04:00
plane . is_flying ( ) ) {
throttle_wait = false ;
} else {
throttle_wait = true ;
}
}
2015-11-24 04:24:04 -04:00
// set motor arming
void QuadPlane : : set_armed ( bool armed )
{
2015-12-26 06:40:40 -04:00
if ( ! initialised ) {
return ;
}
motors - > armed ( armed ) ;
2015-11-24 04:24:04 -04:00
if ( armed ) {
2015-12-26 06:40:40 -04:00
motors - > enable ( ) ;
}
}
/*
estimate desired climb rate for assistance ( in cm / s )
*/
float QuadPlane : : assist_climb_rate_cms ( void )
{
2016-01-20 02:21:54 -04:00
float climb_rate ;
2015-12-26 06:40:40 -04:00
if ( plane . auto_throttle_mode ) {
2016-01-20 02:21:54 -04:00
// use altitude_error_cm, spread over 10s interval
2016-12-04 11:27:33 -04:00
climb_rate = plane . altitude_error_cm / 10.0f ;
2016-01-20 02:21:54 -04:00
} else {
// otherwise estimate from pilot input
climb_rate = plane . g . flybywire_climb_rate * ( plane . nav_pitch_cd / ( float ) plane . aparm . pitch_limit_max_cd ) ;
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
climb_rate * = plane . channel_throttle - > get_control_in ( ) ;
2015-11-24 04:24:04 -04:00
}
2016-01-20 02:21:54 -04:00
climb_rate = constrain_float ( climb_rate , - wp_nav - > get_speed_down ( ) , wp_nav - > get_speed_up ( ) ) ;
2015-12-26 06:40:40 -04:00
return climb_rate ;
2015-11-24 04:24:04 -04:00
}
2015-12-26 06:40:40 -04:00
/*
calculate desired yaw rate for assistance
*/
2016-01-09 18:42:46 -04:00
float QuadPlane : : desired_auto_yaw_rate_cds ( void )
2015-12-26 06:40:40 -04:00
{
float aspeed ;
2016-08-07 21:48:36 -03:00
if ( ! ahrs . airspeed_estimate ( & aspeed ) | | aspeed < plane . aparm . airspeed_min ) {
aspeed = plane . aparm . airspeed_min ;
2015-12-26 06:40:40 -04:00
}
if ( aspeed < 1 ) {
aspeed = 1 ;
}
float yaw_rate = degrees ( GRAVITY_MSS * tanf ( radians ( plane . nav_roll_cd * 0.01f ) ) / aspeed ) * 100 ;
return yaw_rate ;
}
2015-11-24 04:24:04 -04:00
2016-08-29 03:44:54 -03:00
/*
return true if the quadplane should provide stability assistance
*/
bool QuadPlane : : assistance_needed ( float aspeed )
{
if ( assist_speed < = 0 ) {
// assistance disabled
in_angle_assist = false ;
angle_error_start_ms = 0 ;
return false ;
}
if ( aspeed < assist_speed ) {
// assistance due to Q_ASSIST_SPEED
in_angle_assist = false ;
angle_error_start_ms = 0 ;
return true ;
}
if ( assist_angle < = 0 ) {
in_angle_assist = false ;
angle_error_start_ms = 0 ;
return false ;
}
/*
now check if we should provide assistance due to attitude error
*/
const uint16_t allowed_envelope_error_cd = 500U ;
if ( labs ( ahrs . roll_sensor ) < = plane . aparm . roll_limit_cd + allowed_envelope_error_cd & &
ahrs . pitch_sensor < plane . aparm . pitch_limit_max_cd + allowed_envelope_error_cd & &
ahrs . pitch_sensor > - ( plane . aparm . pitch_limit_min_cd + allowed_envelope_error_cd ) ) {
// we are inside allowed attitude envelope
in_angle_assist = false ;
angle_error_start_ms = 0 ;
return false ;
}
uint32_t max_angle_cd = 100U * assist_angle ;
if ( ( labs ( ahrs . roll_sensor - plane . nav_roll_cd ) < max_angle_cd & &
labs ( ahrs . pitch_sensor - plane . nav_pitch_cd ) < max_angle_cd ) ) {
// not beyond angle error
angle_error_start_ms = 0 ;
in_angle_assist = false ;
return false ;
}
if ( angle_error_start_ms = = 0 ) {
angle_error_start_ms = AP_HAL : : millis ( ) ;
}
bool ret = ( AP_HAL : : millis ( ) - angle_error_start_ms ) > = 1000U ;
if ( ret & & ! in_angle_assist ) {
in_angle_assist = true ;
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " Angle assist r=%d p=%d " ,
( int ) ( ahrs . roll_sensor / 100 ) ,
( int ) ( ahrs . pitch_sensor / 100 ) ) ;
}
return ret ;
}
2015-11-24 04:24:04 -04:00
/*
2015-12-26 04:27:13 -04:00
update for transition from quadplane to fixed wing mode
2015-11-24 04:24:04 -04:00
*/
void QuadPlane : : update_transition ( void )
{
2016-01-03 01:46:34 -04:00
if ( plane . control_mode = = MANUAL | |
plane . control_mode = = ACRO | |
plane . control_mode = = TRAINING ) {
// in manual modes quad motors are always off
2016-05-01 20:33:54 -03:00
if ( ! tilt . motors_active ) {
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SHUT_DOWN ) ;
motors - > output ( ) ;
}
2015-12-26 04:27:13 -04:00
transition_state = TRANSITION_DONE ;
2016-10-11 03:04:22 -03:00
assisted_flight = false ;
2015-12-26 04:27:13 -04:00
return ;
}
2015-12-26 06:40:40 -04:00
float aspeed ;
bool have_airspeed = ahrs . airspeed_estimate ( & aspeed ) ;
/*
see if we should provide some assistance
*/
2016-08-29 03:44:54 -03:00
if ( have_airspeed & &
assistance_needed ( aspeed ) & &
2016-01-01 03:18:53 -04:00
( plane . auto_throttle_mode | |
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
plane . channel_throttle - > get_control_in ( ) > 0 | |
2016-01-01 03:18:53 -04:00
plane . is_flying ( ) ) ) {
2015-12-26 06:40:40 -04:00
// the quad should provide some assistance to the plane
2016-06-10 04:37:14 -03:00
if ( transition_state ! = TRANSITION_AIRSPEED_WAIT ) {
2016-06-13 17:34:40 -03:00
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " Transition started airspeed %.1f " , ( double ) aspeed ) ;
2016-06-10 04:37:14 -03:00
}
2015-12-26 06:40:40 -04:00
transition_state = TRANSITION_AIRSPEED_WAIT ;
transition_start_ms = millis ( ) ;
assisted_flight = true ;
} else {
assisted_flight = false ;
}
2016-01-09 06:50:59 -04:00
2017-01-22 22:06:31 -04:00
// if rotors are fully forward then we are not transitioning
if ( tiltrotor_fully_fwd ( ) ) {
transition_state = TRANSITION_DONE ;
}
2016-01-09 06:50:59 -04:00
if ( transition_state < TRANSITION_TIMER ) {
// set a single loop pitch limit in TECS
2016-05-27 02:48:14 -03:00
if ( plane . ahrs . groundspeed ( ) < 3 ) {
// until we have some ground speed limit to zero pitch
plane . TECS_controller . set_pitch_max_limit ( 0 ) ;
} else {
plane . TECS_controller . set_pitch_max_limit ( transition_pitch_max ) ;
}
2016-01-20 03:30:48 -04:00
} else if ( transition_state < TRANSITION_DONE ) {
plane . TECS_controller . set_pitch_max_limit ( ( transition_pitch_max + 1 ) * 2 ) ;
2016-01-09 06:50:59 -04:00
}
2016-10-04 21:52:52 -03:00
if ( transition_state < TRANSITION_DONE ) {
// during transition we ask TECS to use a synthetic
// airspeed. Otherwise the pitch limits will throw off the
// throttle calculation which is driven by pitch
plane . TECS_controller . use_synthetic_airspeed ( ) ;
}
2015-12-26 06:40:40 -04:00
2015-12-26 04:27:13 -04:00
switch ( transition_state ) {
case TRANSITION_AIRSPEED_WAIT : {
2016-04-01 03:29:51 -03:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
2015-12-26 04:27:13 -04:00
// we hold in hover until the required airspeed is reached
2015-12-26 05:13:20 -04:00
if ( transition_start_ms = = 0 ) {
2015-12-26 06:40:40 -04:00
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " Transition airspeed wait " ) ;
2015-12-26 05:13:20 -04:00
transition_start_ms = millis ( ) ;
}
2016-08-07 21:48:36 -03:00
if ( have_airspeed & & aspeed > plane . aparm . airspeed_min & & ! assisted_flight ) {
2015-12-26 04:51:05 -04:00
transition_start_ms = millis ( ) ;
2015-12-26 04:27:13 -04:00
transition_state = TRANSITION_TIMER ;
2016-02-17 16:52:29 -04:00
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " Transition airspeed reached %.1f " , ( double ) aspeed ) ;
2015-12-26 04:27:13 -04:00
}
2016-01-01 05:42:10 -04:00
assisted_flight = true ;
2015-12-26 06:40:40 -04:00
hold_hover ( assist_climb_rate_cms ( ) ) ;
2016-09-20 22:46:15 -03:00
run_rate_controller ( ) ;
2016-03-24 22:11:56 -03:00
motors_output ( ) ;
2015-12-26 06:40:40 -04:00
last_throttle = motors - > get_throttle ( ) ;
2015-12-26 04:27:13 -04:00
break ;
}
case TRANSITION_TIMER : {
2016-04-01 03:29:51 -03:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
2015-12-26 04:27:13 -04:00
// after airspeed is reached we degrade throttle over the
// transition time, but continue to stabilize
2015-12-26 04:51:05 -04:00
if ( millis ( ) - transition_start_ms > ( unsigned ) transition_time_ms ) {
2015-12-26 04:27:13 -04:00
transition_state = TRANSITION_DONE ;
2015-12-26 06:40:40 -04:00
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_INFO , " Transition done " ) ;
2015-12-26 04:27:13 -04:00
}
2016-06-10 04:35:43 -03:00
float trans_time_ms = ( float ) transition_time_ms . get ( ) ;
float throttle_scaled = last_throttle * ( trans_time_ms - ( millis ( ) - transition_start_ms ) ) / trans_time_ms ;
2017-01-24 02:59:28 -04:00
if ( throttle_scaled < 0.01 ) {
// ensure we don't drop all the way to zero or the motors
// will stop stabilizing
throttle_scaled = 0.01 ;
2015-12-26 04:27:13 -04:00
}
2016-01-01 00:36:03 -04:00
assisted_flight = true ;
2015-12-26 04:27:13 -04:00
hold_stabilize ( throttle_scaled ) ;
2016-09-20 22:46:15 -03:00
run_rate_controller ( ) ;
2016-03-24 22:11:56 -03:00
motors_output ( ) ;
2015-12-26 04:27:13 -04:00
break ;
}
case TRANSITION_DONE :
2016-05-01 20:33:54 -03:00
if ( ! tilt . motors_active ) {
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SHUT_DOWN ) ;
motors - > output ( ) ;
}
2015-12-26 04:27:13 -04:00
break ;
2015-11-24 04:24:04 -04:00
}
}
2016-09-20 22:46:15 -03:00
/*
run multicopter rate controller
*/
void QuadPlane : : run_rate_controller ( void )
{
attitude_control - > set_throttle_mix_max ( ) ;
attitude_control - > rate_controller_run ( ) ;
}
2015-11-24 04:24:04 -04:00
/*
update motor output for quadplane
*/
void QuadPlane : : update ( void )
{
2016-01-06 03:17:08 -04:00
if ( ! setup ( ) ) {
2015-12-26 06:40:40 -04:00
return ;
}
2016-07-22 05:36:28 -03:00
if ( plane . afs . should_crash_vehicle ( ) ) {
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SHUT_DOWN ) ;
motors - > output ( ) ;
return ;
}
2016-03-12 19:05:10 -04:00
if ( motor_test . running ) {
motor_test_output ( ) ;
return ;
}
2016-03-09 03:20:41 -04:00
if ( ! in_vtol_mode ( ) ) {
2015-11-24 04:24:04 -04:00
update_transition ( ) ;
} else {
2015-12-26 06:40:40 -04:00
assisted_flight = false ;
2015-12-26 05:13:20 -04:00
// run low level rate controllers
2016-09-20 22:46:15 -03:00
run_rate_controller ( ) ;
2015-12-26 05:13:20 -04:00
// output to motors
2016-03-24 22:11:56 -03:00
motors_output ( ) ;
2015-12-26 05:13:20 -04:00
transition_start_ms = 0 ;
2016-01-01 03:18:53 -04:00
if ( throttle_wait & & ! plane . is_flying ( ) ) {
transition_state = TRANSITION_DONE ;
} else {
transition_state = TRANSITION_AIRSPEED_WAIT ;
}
2015-12-26 06:40:40 -04:00
last_throttle = motors - > get_throttle ( ) ;
2015-12-26 05:13:20 -04:00
}
// disable throttle_wait when throttle rises above 10%
2015-12-26 06:40:40 -04:00
if ( throttle_wait & &
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
( plane . channel_throttle - > get_control_in ( ) > 10 | |
2015-12-26 06:40:40 -04:00
plane . failsafe . ch3_failsafe | |
plane . failsafe . ch3_counter > 0 ) ) {
2015-12-26 05:13:20 -04:00
throttle_wait = false ;
2015-11-24 04:24:04 -04:00
}
2016-05-01 05:07:52 -03:00
tiltrotor_update ( ) ;
2015-11-24 04:24:04 -04:00
}
2015-12-26 06:40:40 -04:00
2016-08-01 19:21:44 -03:00
/*
see if motors should be shutdown . If they should be then change AP_Motors state to
AP_Motors : : DESIRED_SHUT_DOWN
This is a safety check to prevent accidental motor runs on the
ground , such as if RC fails and QRTL is started
*/
void QuadPlane : : check_throttle_suppression ( void )
{
// if the motors have been running in the last 2 seconds then
// allow them to run now
if ( AP_HAL : : millis ( ) - last_motors_active_ms < 2000 ) {
return ;
}
// see if motors are already disabled
if ( motors - > get_desired_spool_state ( ) < AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) {
return ;
}
// if the users throttle is above zero then allow motors to run
if ( plane . channel_throttle - > get_control_in ( ) ! = 0 ) {
return ;
}
// if we are in a fixed wing auto throttle mode and we have
// unsuppressed the throttle then allow motors to run
if ( plane . auto_throttle_mode & & ! plane . throttle_suppressed ) {
return ;
}
// if our vertical velocity is greater than 1m/s then allow motors to run
if ( fabsf ( inertial_nav . get_velocity_z ( ) ) > 100 ) {
return ;
}
// if we are more than 5m from home altitude then allow motors to run
if ( plane . relative_ground_altitude ( plane . g . rangefinder_landing ) > 5 ) {
return ;
}
// allow for takeoff
if ( plane . control_mode = = AUTO & & plane . mission . get_current_nav_cmd ( ) . id = = MAV_CMD_NAV_VTOL_TAKEOFF ) {
return ;
}
// motors should be in the spin when armed state to warn user they could become active
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SPIN_WHEN_ARMED ) ;
motors - > set_throttle ( 0 ) ;
last_motors_active_ms = 0 ;
}
2016-03-24 22:11:56 -03:00
/*
output motors and do any copter needed
*/
void QuadPlane : : motors_output ( void )
{
2016-08-29 04:55:27 -03:00
if ( ! hal . util - > get_soft_armed ( ) | | plane . afs . should_crash_vehicle ( ) ) {
2016-07-22 05:36:28 -03:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SHUT_DOWN ) ;
motors - > output ( ) ;
return ;
}
2016-06-02 00:10:39 -03:00
if ( esc_calibration & & AP_Notify : : flags . esc_calibration & & plane . control_mode = = QSTABILIZE ) {
// output is direct from run_esc_calibration()
return ;
}
2016-08-01 19:21:44 -03:00
// see if motors should be shut down
check_throttle_suppression ( ) ;
2016-03-24 22:11:56 -03:00
motors - > output ( ) ;
2016-04-03 20:50:15 -03:00
if ( motors - > armed ( ) ) {
plane . DataFlash . Log_Write_Rate ( plane . ahrs , * motors , * attitude_control , * pos_control ) ;
Log_Write_QControl_Tuning ( ) ;
2016-05-27 01:16:44 -03:00
uint32_t now = AP_HAL : : millis ( ) ;
if ( now - last_ctrl_log_ms > 100 ) {
attitude_control - > control_monitor_log ( ) ;
}
2016-04-03 20:50:15 -03:00
}
2016-08-01 19:21:44 -03:00
// remember when motors were last active for throttle suppression
if ( motors - > get_throttle ( ) > 0.01f | | tilt . motors_active ) {
last_motors_active_ms = AP_HAL : : millis ( ) ;
}
2016-03-24 22:11:56 -03:00
}
2015-12-26 06:40:40 -04:00
/*
update control mode for quadplane modes
*/
void QuadPlane : : control_run ( void )
{
if ( ! initialised ) {
return ;
}
switch ( plane . control_mode ) {
case QSTABILIZE :
control_stabilize ( ) ;
break ;
case QHOVER :
control_hover ( ) ;
break ;
case QLOITER :
2016-03-09 03:20:41 -04:00
case QLAND :
2015-12-26 06:40:40 -04:00
control_loiter ( ) ;
2016-04-29 02:31:08 -03:00
break ;
case QRTL :
control_qrtl ( ) ;
break ;
2015-12-26 06:40:40 -04:00
default :
break ;
}
// we also stabilize using fixed wing surfaces
float speed_scaler = plane . get_speed_scaler ( ) ;
plane . stabilize_roll ( speed_scaler ) ;
plane . stabilize_pitch ( speed_scaler ) ;
}
/*
enter a quadplane mode
*/
bool QuadPlane : : init_mode ( void )
{
2016-01-06 03:17:08 -04:00
if ( ! setup ( ) ) {
return false ;
}
2015-12-26 06:40:40 -04:00
if ( ! initialised ) {
GCS_MAVLINK : : send_statustext_all ( MAV_SEVERITY_CRITICAL , " QuadPlane mode refused " ) ;
return false ;
}
2016-06-02 00:10:39 -03:00
AP_Notify : : flags . esc_calibration = false ;
2015-12-26 06:40:40 -04:00
switch ( plane . control_mode ) {
case QSTABILIZE :
init_stabilize ( ) ;
break ;
case QHOVER :
init_hover ( ) ;
break ;
case QLOITER :
init_loiter ( ) ;
break ;
2016-03-09 03:20:41 -04:00
case QLAND :
init_land ( ) ;
break ;
2016-04-29 02:31:08 -03:00
case QRTL :
init_qrtl ( ) ;
break ;
2015-12-26 06:40:40 -04:00
default :
break ;
}
return true ;
}
/*
handle a MAVLink DO_VTOL_TRANSITION
*/
2016-04-22 05:22:31 -03:00
bool QuadPlane : : handle_do_vtol_transition ( enum MAV_VTOL_STATE state )
2015-12-26 06:40:40 -04:00
{
if ( ! available ( ) ) {
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " VTOL not available " ) ;
2016-04-22 05:22:31 -03:00
return false ;
2015-12-26 06:40:40 -04:00
}
if ( plane . control_mode ! = AUTO ) {
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " VTOL transition only in AUTO " ) ;
2016-04-22 05:22:31 -03:00
return false ;
2015-12-26 06:40:40 -04:00
}
2016-04-22 05:22:31 -03:00
switch ( state ) {
2015-12-26 06:40:40 -04:00
case MAV_VTOL_STATE_MC :
if ( ! plane . auto_state . vtol_mode ) {
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " Entered VTOL mode " ) ;
}
plane . auto_state . vtol_mode = true ;
2016-04-22 05:22:31 -03:00
return true ;
2015-12-26 06:40:40 -04:00
case MAV_VTOL_STATE_FW :
if ( plane . auto_state . vtol_mode ) {
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " Exited VTOL mode " ) ;
}
plane . auto_state . vtol_mode = false ;
2016-04-22 05:22:31 -03:00
return true ;
default :
break ;
2015-12-26 06:40:40 -04:00
}
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " Invalid VTOL mode " ) ;
2016-04-22 05:22:31 -03:00
return false ;
2015-12-26 06:40:40 -04:00
}
2016-01-01 06:39:36 -04:00
/*
are we in a VTOL auto state ?
*/
bool QuadPlane : : in_vtol_auto ( void )
{
2016-05-05 22:28:26 -03:00
if ( ! enable ) {
return false ;
}
2016-05-05 22:50:26 -03:00
if ( plane . control_mode ! = AUTO ) {
2016-01-01 06:39:36 -04:00
return false ;
}
if ( plane . auto_state . vtol_mode ) {
return true ;
}
switch ( plane . mission . get_current_nav_cmd ( ) . id ) {
case MAV_CMD_NAV_VTOL_LAND :
case MAV_CMD_NAV_VTOL_TAKEOFF :
return true ;
2016-05-11 02:57:41 -03:00
case MAV_CMD_NAV_LOITER_UNLIM :
case MAV_CMD_NAV_LOITER_TIME :
return plane . auto_state . vtol_loiter ;
2016-01-01 06:39:36 -04:00
default :
return false ;
}
}
2016-02-20 05:20:27 -04:00
/*
are we in a VTOL mode ?
*/
bool QuadPlane : : in_vtol_mode ( void )
{
2016-04-09 05:48:22 -03:00
if ( ! enable ) {
return false ;
}
2016-02-20 05:20:27 -04:00
return ( plane . control_mode = = QSTABILIZE | |
plane . control_mode = = QHOVER | |
plane . control_mode = = QLOITER | |
2016-03-09 03:20:41 -04:00
plane . control_mode = = QLAND | |
2016-04-29 02:31:08 -03:00
plane . control_mode = = QRTL | |
2016-08-12 17:27:48 -03:00
( ( plane . control_mode = = GUIDED | | plane . control_mode = = AVOID_ADSB ) & & plane . auto_state . vtol_loiter ) | |
2016-02-20 05:20:27 -04:00
in_vtol_auto ( ) ) ;
}
2016-04-28 23:53:20 -03:00
2015-12-26 06:40:40 -04:00
/*
2016-04-28 23:53:20 -03:00
main landing controller . Used for landing and RTL .
2015-12-26 06:40:40 -04:00
*/
2016-05-05 19:27:47 -03:00
void QuadPlane : : vtol_position_controller ( void )
2015-12-26 06:40:40 -04:00
{
2016-01-09 01:26:13 -04:00
if ( ! setup ( ) ) {
return ;
}
2016-04-01 03:29:51 -03:00
2016-04-28 23:53:20 -03:00
setup_target_position ( ) ;
2015-12-26 06:40:40 -04:00
2016-04-28 23:53:20 -03:00
const Location & loc = plane . next_WP_loc ;
float ekfGndSpdLimit , ekfNavVelGainScaler ;
ahrs . getEkfControlLimits ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
2016-01-02 02:55:56 -04:00
2016-05-05 19:27:47 -03:00
switch ( poscontrol . state ) {
case QPOS_LAND_FINAL :
2016-01-09 18:42:46 -04:00
/*
2016-01-15 01:49:49 -04:00
for land - final we use the loiter controller
2016-01-09 18:42:46 -04:00
*/
2016-01-15 01:49:49 -04:00
// run loiter controller
wp_nav - > update_loiter ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
2016-01-09 18:42:46 -04:00
2016-06-16 23:48:25 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( plane . nav_roll_cd ,
2016-01-15 01:49:49 -04:00
plane . nav_pitch_cd ,
2016-05-05 22:50:26 -03:00
get_pilot_input_yaw_rate_cds ( ) + get_weathervane_yaw_rate_cds ( ) ,
2016-01-15 01:49:49 -04:00
smoothing_gain ) ;
// nav roll and pitch are controller by position controller
plane . nav_roll_cd = pos_control - > get_roll ( ) ;
plane . nav_pitch_cd = pos_control - > get_pitch ( ) ;
2016-04-28 23:53:20 -03:00
break ;
2016-05-05 19:27:47 -03:00
case QPOS_POSITION1 : {
2016-04-10 07:36:20 -03:00
Vector2f diff_wp = location_diff ( plane . current_loc , loc ) ;
2016-05-07 18:34:30 -03:00
float distance = diff_wp . length ( ) ;
2016-04-02 06:54:01 -03:00
2016-05-05 19:27:47 -03:00
if ( poscontrol . speed_scale < = 0 ) {
2016-04-02 06:54:01 -03:00
// initialise scaling so we start off targeting our
2016-04-03 20:50:15 -03:00
// current linear speed towards the target. If this is
// less than the wpnav speed then the wpnav speed is used
// land_speed_scale is then used to linearly change
// velocity as we approach the waypoint, aiming for zero
// speed at the waypoint
Vector2f groundspeed = ahrs . groundspeed_vector ( ) ;
2016-05-07 18:34:30 -03:00
float speed_towards_target = distance > 1 ? ( diff_wp . normalized ( ) * groundspeed ) : 0 ;
// setup land_speed_scale so at current distance we
// maintain speed towards target, and slow down as we
// approach
2016-04-20 05:07:04 -03:00
// max_speed will control how fast we will fly. It will always decrease
2016-05-05 19:27:47 -03:00
poscontrol . max_speed = MAX ( speed_towards_target , wp_nav - > get_speed_xy ( ) * 0.01 ) ;
poscontrol . speed_scale = poscontrol . max_speed / MAX ( distance , 1 ) ;
2016-06-24 17:27:56 -03:00
// start with low integrator. The alt_hold controller will
// add hover throttle to initial integrator. By starting
// without it we end up with a smoother startup when
// transitioning from fixed wing flight
float aspeed ;
if ( ahrs . airspeed_estimate ( & aspeed ) & & aspeed > 6 ) {
pid_accel_z . set_integrator ( ( - motors - > get_throttle_hover ( ) ) * 1000.0f ) ;
}
2016-04-02 06:54:01 -03:00
}
2016-04-10 07:36:20 -03:00
// run fixed wing navigation
2016-04-28 23:53:20 -03:00
plane . nav_controller - > update_waypoint ( plane . prev_WP_loc , loc ) ;
2016-04-10 07:36:20 -03:00
/*
calculate target velocity , not dropping it below 2 m / s
*/
const float final_speed = 2.0f ;
2016-05-05 19:27:47 -03:00
Vector2f target_speed_xy = diff_wp * poscontrol . speed_scale ;
2016-04-20 05:07:04 -03:00
float target_speed = target_speed_xy . length ( ) ;
2016-05-07 18:34:30 -03:00
if ( distance < 1 ) {
// prevent numerical error before switching to POSITION2
target_speed_xy ( 0.1 , 0.1 ) ;
}
2016-04-20 05:07:04 -03:00
if ( target_speed < final_speed ) {
// until we enter the loiter we always aim for at least 2m/s
target_speed_xy = target_speed_xy . normalized ( ) * final_speed ;
2016-05-05 19:27:47 -03:00
poscontrol . max_speed = final_speed ;
} else if ( target_speed > poscontrol . max_speed ) {
2016-04-20 05:07:04 -03:00
// we never speed up during landing approaches
2016-05-05 19:27:47 -03:00
target_speed_xy = target_speed_xy . normalized ( ) * poscontrol . max_speed ;
2016-04-20 05:07:04 -03:00
} else {
2016-05-05 19:27:47 -03:00
poscontrol . max_speed = target_speed ;
2016-04-10 07:36:20 -03:00
}
2016-04-20 05:07:04 -03:00
pos_control - > set_desired_velocity_xy ( target_speed_xy . x * 100 ,
target_speed_xy . y * 100 ) ;
2016-04-10 07:36:20 -03:00
2016-04-02 06:54:01 -03:00
pos_control - > update_vel_controller_xyz ( ekfNavVelGainScaler ) ;
2016-04-10 07:36:20 -03:00
const Vector3f & curr_pos = inertial_nav . get_position ( ) ;
pos_control - > set_xy_target ( curr_pos . x , curr_pos . y ) ;
pos_control - > freeze_ff_xy ( ) ;
2016-04-02 06:54:01 -03:00
// nav roll and pitch are controller by position controller
plane . nav_roll_cd = pos_control - > get_roll ( ) ;
plane . nav_pitch_cd = pos_control - > get_pitch ( ) ;
2016-04-10 07:36:20 -03:00
/*
limit the pitch down with an expanding envelope . This
prevents the velocity controller demanding nose down during
the initial slowdown if the target velocity curve is higher
than the actual velocity curve ( for a high drag
aircraft ) . Nose down will cause a lot of downforce on the
wings which will draw a lot of current and also cause the
aircraft to lose altitude rapidly .
*/
float pitch_limit_cd = linear_interpolate ( - 300 , plane . aparm . pitch_limit_min_cd ,
plane . auto_state . wp_proportion , 0 , 1 ) ;
if ( plane . nav_pitch_cd < pitch_limit_cd ) {
plane . nav_pitch_cd = pitch_limit_cd ;
// tell the pos controller we have limited the pitch to
// stop integrator buildup
pos_control - > set_limit_accel_xy ( ) ;
}
2016-04-02 07:19:06 -03:00
2016-04-02 06:54:01 -03:00
// call attitude controller
2016-06-16 23:48:25 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( plane . nav_roll_cd ,
2016-04-02 06:54:01 -03:00
plane . nav_pitch_cd ,
2016-04-20 03:40:47 -03:00
desired_auto_yaw_rate_cds ( ) + get_weathervane_yaw_rate_cds ( ) ,
2016-04-02 06:54:01 -03:00
smoothing_gain ) ;
2016-04-10 07:36:20 -03:00
if ( plane . auto_state . wp_proportion > = 1 | |
plane . auto_state . wp_distance < 5 ) {
2016-05-05 19:27:47 -03:00
poscontrol . state = QPOS_POSITION2 ;
2016-04-10 07:36:20 -03:00
wp_nav - > init_loiter_target ( ) ;
2016-05-05 19:27:47 -03:00
plane . gcs_send_text_fmt ( MAV_SEVERITY_INFO , " VTOL position2 started v=%.1f d=%.1f " ,
2016-04-14 20:22:41 -03:00
( double ) ahrs . groundspeed ( ) , ( double ) plane . auto_state . wp_distance ) ;
2016-04-02 06:54:01 -03:00
}
2016-04-28 23:53:20 -03:00
break ;
}
2016-05-05 19:27:47 -03:00
case QPOS_POSITION2 :
case QPOS_LAND_DESCEND :
2016-01-15 01:49:49 -04:00
/*
2016-04-02 06:54:01 -03:00
for final land repositioning and descent we run the loiter controller
2016-01-15 01:49:49 -04:00
*/
2016-01-09 18:42:46 -04:00
// also run fixed wing navigation
2016-04-28 23:53:20 -03:00
plane . nav_controller - > update_waypoint ( plane . prev_WP_loc , loc ) ;
2016-01-15 01:49:49 -04:00
2016-05-05 19:27:47 -03:00
pos_control - > set_xy_target ( poscontrol . target . x , poscontrol . target . y ) ;
2016-01-08 20:28:10 -04:00
2016-01-15 01:49:49 -04:00
// run loiter controller
wp_nav - > update_loiter ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
// nav roll and pitch are controller by position controller
2016-01-09 18:42:46 -04:00
plane . nav_roll_cd = wp_nav - > get_roll ( ) ;
plane . nav_pitch_cd = wp_nav - > get_pitch ( ) ;
2016-01-15 01:49:49 -04:00
// call attitude controller
2016-06-16 23:48:25 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( plane . nav_roll_cd ,
2016-01-09 18:42:46 -04:00
plane . nav_pitch_cd ,
2016-05-05 22:50:26 -03:00
get_pilot_input_yaw_rate_cds ( ) + get_weathervane_yaw_rate_cds ( ) ,
2016-01-15 01:49:49 -04:00
smoothing_gain ) ;
2016-04-28 23:53:20 -03:00
break ;
2016-01-09 18:42:46 -04:00
2016-05-05 19:27:47 -03:00
case QPOS_LAND_COMPLETE :
2016-04-28 23:53:20 -03:00
// nothing to do
break ;
2016-01-02 02:55:56 -04:00
}
2015-12-26 06:40:40 -04:00
2016-04-28 23:53:20 -03:00
// now height control
2016-05-05 19:27:47 -03:00
switch ( poscontrol . state ) {
case QPOS_POSITION1 :
case QPOS_POSITION2 :
2016-04-29 02:31:08 -03:00
if ( plane . control_mode = = QRTL ) {
plane . ahrs . get_position ( plane . current_loc ) ;
2016-04-29 02:50:45 -03:00
float target_altitude = plane . next_WP_loc . alt ;
2016-05-05 19:27:47 -03:00
if ( poscontrol . slow_descent ) {
2016-04-29 02:50:45 -03:00
// gradually descend as we approach target
plane . auto_state . wp_proportion = location_path_proportion ( plane . current_loc ,
plane . prev_WP_loc , plane . next_WP_loc ) ;
target_altitude = linear_interpolate ( plane . prev_WP_loc . alt ,
plane . next_WP_loc . alt ,
plane . auto_state . wp_proportion ,
0 , 1 ) ;
}
2016-04-29 02:31:08 -03:00
pos_control - > set_alt_target ( target_altitude - plane . home . alt ) ;
} else {
pos_control - > set_alt_target_from_climb_rate ( 0 , plane . G_Dt , false ) ;
}
2016-04-28 23:53:20 -03:00
break ;
2015-12-26 06:40:40 -04:00
2016-05-05 19:27:47 -03:00
case QPOS_LAND_DESCEND : {
2016-06-11 01:37:44 -03:00
float height_above_ground = plane . relative_ground_altitude ( plane . g . rangefinder_landing ) ;
2016-04-28 23:53:20 -03:00
pos_control - > set_alt_target_from_climb_rate ( - landing_descent_rate_cms ( height_above_ground ) ,
plane . G_Dt , true ) ;
2016-01-01 05:42:10 -04:00
break ;
2016-04-28 23:53:20 -03:00
}
2016-05-05 19:27:47 -03:00
case QPOS_LAND_FINAL :
2016-04-28 23:53:20 -03:00
pos_control - > set_alt_target_from_climb_rate ( - land_speed_cms , plane . G_Dt , true ) ;
2016-01-01 05:42:10 -04:00
break ;
2016-04-28 23:53:20 -03:00
2016-05-05 19:27:47 -03:00
case QPOS_LAND_COMPLETE :
2016-01-01 05:42:10 -04:00
break ;
}
2015-12-26 06:40:40 -04:00
pos_control - > update_z_controller ( ) ;
}
2016-01-01 05:42:10 -04:00
2016-04-28 23:53:20 -03:00
/*
setup the target position based on plane . next_WP_loc
*/
void QuadPlane : : setup_target_position ( void )
{
const Location & loc = plane . next_WP_loc ;
Location origin = inertial_nav . get_origin ( ) ;
Vector2f diff2d ;
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
diff2d = location_diff ( origin , loc ) ;
2016-05-05 19:27:47 -03:00
poscontrol . target . x = diff2d . x * 100 ;
poscontrol . target . y = diff2d . y * 100 ;
poscontrol . target . z = plane . next_WP_loc . alt - origin . alt ;
2016-04-28 23:53:20 -03:00
if ( ! locations_are_same ( loc , last_auto_target ) | |
plane . next_WP_loc . alt ! = last_auto_target . alt | |
millis ( ) - last_loiter_ms > 500 ) {
2016-05-05 19:27:47 -03:00
wp_nav - > set_wp_destination ( poscontrol . target ) ;
2016-04-28 23:53:20 -03:00
last_auto_target = loc ;
}
last_loiter_ms = millis ( ) ;
// setup vertical speed and acceleration
pos_control - > set_speed_z ( - pilot_velocity_z_max , pilot_velocity_z_max ) ;
pos_control - > set_accel_z ( pilot_accel_z ) ;
}
/*
run takeoff controller to climb vertically
*/
void QuadPlane : : takeoff_controller ( void )
{
/*
for takeoff we need to use the loiter controller wpnav controller takes over the descent rate
control
*/
float ekfGndSpdLimit , ekfNavVelGainScaler ;
ahrs . getEkfControlLimits ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
setup_target_position ( ) ;
// run loiter controller
wp_nav - > update_loiter ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
2016-06-16 23:48:25 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( plane . nav_roll_cd ,
2016-04-28 23:53:20 -03:00
plane . nav_pitch_cd ,
2016-05-05 22:50:26 -03:00
get_pilot_input_yaw_rate_cds ( ) + get_weathervane_yaw_rate_cds ( ) ,
2016-04-28 23:53:20 -03:00
smoothing_gain ) ;
// nav roll and pitch are controller by position controller
plane . nav_roll_cd = pos_control - > get_roll ( ) ;
plane . nav_pitch_cd = pos_control - > get_pitch ( ) ;
pos_control - > set_alt_target_from_climb_rate ( wp_nav - > get_speed_up ( ) , plane . G_Dt , true ) ;
pos_control - > update_z_controller ( ) ;
}
/*
run waypoint controller between prev_WP_loc and next_WP_loc
*/
void QuadPlane : : waypoint_controller ( void )
{
setup_target_position ( ) ;
/*
this is full copter control of auto flight
*/
// run wpnav controller
wp_nav - > update_wpnav ( ) ;
// call attitude controller
attitude_control - > input_euler_angle_roll_pitch_yaw ( wp_nav - > get_roll ( ) ,
wp_nav - > get_pitch ( ) ,
wp_nav - > get_yaw ( ) ,
2016-06-16 23:48:25 -03:00
true , 4.0f ) ;
2016-04-28 23:53:20 -03:00
// nav roll and pitch are controller by loiter controller
plane . nav_roll_cd = wp_nav - > get_roll ( ) ;
plane . nav_pitch_cd = wp_nav - > get_pitch ( ) ;
// climb based on altitude error
pos_control - > set_alt_target_from_climb_rate_ff ( assist_climb_rate_cms ( ) , plane . G_Dt , false ) ;
pos_control - > update_z_controller ( ) ;
}
/*
handle auto - mode when auto_state . vtol_mode is true
*/
void QuadPlane : : control_auto ( const Location & loc )
{
if ( ! setup ( ) ) {
return ;
}
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
2016-05-11 02:57:41 -03:00
switch ( plane . mission . get_current_nav_cmd ( ) . id ) {
case MAV_CMD_NAV_VTOL_TAKEOFF :
2016-04-28 23:53:20 -03:00
takeoff_controller ( ) ;
2016-05-11 02:57:41 -03:00
break ;
case MAV_CMD_NAV_VTOL_LAND :
case MAV_CMD_NAV_LOITER_UNLIM :
case MAV_CMD_NAV_LOITER_TIME :
2016-05-05 19:27:47 -03:00
vtol_position_controller ( ) ;
2016-05-11 02:57:41 -03:00
break ;
default :
2016-04-28 23:53:20 -03:00
waypoint_controller ( ) ;
2016-05-11 02:57:41 -03:00
break ;
2016-04-28 23:53:20 -03:00
}
}
2016-04-29 02:31:08 -03:00
/*
handle QRTL mode
*/
void QuadPlane : : control_qrtl ( void )
{
2016-05-05 19:27:47 -03:00
vtol_position_controller ( ) ;
if ( poscontrol . state > = QPOS_POSITION2 ) {
2016-04-29 02:31:08 -03:00
// change target altitude to home alt
plane . next_WP_loc . alt = plane . home . alt ;
verify_vtol_land ( ) ;
} else {
pos_control - > set_alt_target ( qrtl_alt * 100UL ) ;
}
}
/*
handle QRTL mode
*/
void QuadPlane : : init_qrtl ( void )
{
// use do_RTL() to setup next_WP_loc
plane . do_RTL ( plane . home . alt + qrtl_alt * 100UL ) ;
plane . prev_WP_loc = plane . current_loc ;
2016-05-05 19:27:47 -03:00
poscontrol . slow_descent = ( plane . current_loc . alt > plane . next_WP_loc . alt ) ;
poscontrol . state = QPOS_POSITION1 ;
poscontrol . speed_scale = 0 ;
2016-04-29 02:31:08 -03:00
}
2016-01-01 05:42:10 -04:00
/*
start a VTOL takeoff
*/
bool QuadPlane : : do_vtol_takeoff ( const AP_Mission : : Mission_Command & cmd )
{
2016-01-09 01:26:13 -04:00
if ( ! setup ( ) ) {
2016-01-01 05:42:10 -04:00
return false ;
}
2016-04-01 03:29:51 -03:00
2016-01-01 06:57:31 -04:00
plane . set_next_WP ( cmd . content . location ) ;
2016-01-01 05:42:10 -04:00
plane . next_WP_loc . alt = plane . current_loc . alt + cmd . content . location . alt ;
throttle_wait = false ;
2016-01-08 19:47:41 -04:00
// set target to current position
wp_nav - > init_loiter_target ( ) ;
// initialize vertical speed and acceleration
pos_control - > set_speed_z ( - pilot_velocity_z_max , pilot_velocity_z_max ) ;
pos_control - > set_accel_z ( pilot_accel_z ) ;
// initialise position and desired velocity
pos_control - > set_alt_target ( inertial_nav . get_altitude ( ) ) ;
pos_control - > set_desired_velocity_z ( inertial_nav . get_velocity_z ( ) ) ;
2016-01-01 05:42:10 -04:00
// also update nav_controller for status output
plane . nav_controller - > update_waypoint ( plane . prev_WP_loc , plane . next_WP_loc ) ;
return true ;
}
/*
start a VTOL landing
*/
bool QuadPlane : : do_vtol_land ( const AP_Mission : : Mission_Command & cmd )
{
2016-01-09 01:26:13 -04:00
if ( ! setup ( ) ) {
2016-01-01 05:42:10 -04:00
return false ;
}
2016-02-15 04:45:56 -04:00
attitude_control - > get_rate_roll_pid ( ) . reset_I ( ) ;
attitude_control - > get_rate_pitch_pid ( ) . reset_I ( ) ;
2016-04-01 02:45:01 -03:00
attitude_control - > get_rate_yaw_pid ( ) . reset_I ( ) ;
2016-01-15 01:49:49 -04:00
pid_accel_z . reset_I ( ) ;
pi_vel_xy . reset_I ( ) ;
2016-01-01 06:57:31 -04:00
plane . set_next_WP ( cmd . content . location ) ;
2016-01-02 02:55:56 -04:00
// initially aim for current altitude
plane . next_WP_loc . alt = plane . current_loc . alt ;
2016-05-05 19:27:47 -03:00
poscontrol . state = QPOS_POSITION1 ;
poscontrol . speed_scale = 0 ;
2016-04-10 07:36:20 -03:00
wp_nav - > init_loiter_target ( ) ;
2016-01-15 01:49:49 -04:00
throttle_wait = false ;
2016-06-16 05:17:46 -03:00
landing_detect . lower_limit_start_ms = 0 ;
2016-01-15 01:49:49 -04:00
Location origin = inertial_nav . get_origin ( ) ;
Vector2f diff2d ;
Vector3f target ;
diff2d = location_diff ( origin , plane . next_WP_loc ) ;
target . x = diff2d . x * 100 ;
target . y = diff2d . y * 100 ;
target . z = plane . next_WP_loc . alt - origin . alt ;
2016-01-20 02:21:54 -04:00
pos_control - > set_alt_target ( inertial_nav . get_altitude ( ) ) ;
2016-01-01 07:09:11 -04:00
2016-01-01 05:42:10 -04:00
// also update nav_controller for status output
plane . nav_controller - > update_waypoint ( plane . prev_WP_loc , plane . next_WP_loc ) ;
return true ;
}
/*
check if a VTOL takeoff has completed
*/
bool QuadPlane : : verify_vtol_takeoff ( const AP_Mission : : Mission_Command & cmd )
{
2016-01-01 07:09:11 -04:00
if ( ! available ( ) ) {
return true ;
}
2016-01-08 19:47:41 -04:00
if ( plane . current_loc . alt < plane . next_WP_loc . alt ) {
2016-01-01 05:42:10 -04:00
return false ;
}
transition_state = TRANSITION_AIRSPEED_WAIT ;
2016-04-02 05:53:48 -03:00
plane . TECS_controller . set_pitch_max_limit ( transition_pitch_max ) ;
2016-09-01 22:09:42 -03:00
pos_control - > set_alt_target ( inertial_nav . get_altitude ( ) ) ;
2016-08-29 05:04:42 -03:00
plane . complete_auto_takeoff ( ) ;
2016-01-01 05:42:10 -04:00
return true ;
}
2016-06-16 05:17:46 -03:00
/*
check if a landing is complete
*/
2016-03-09 03:20:41 -04:00
void QuadPlane : : check_land_complete ( void )
{
2016-06-16 05:17:46 -03:00
if ( poscontrol . state ! = QPOS_LAND_FINAL ) {
// only apply to final landing phase
return ;
}
uint32_t now = AP_HAL : : millis ( ) ;
bool might_be_landed = ( landing_detect . lower_limit_start_ms ! = 0 & &
now - landing_detect . lower_limit_start_ms > 1000 ) ;
if ( ! might_be_landed ) {
landing_detect . land_start_ms = 0 ;
return ;
}
float height = inertial_nav . get_altitude ( ) * 0.01f ;
if ( landing_detect . land_start_ms = = 0 ) {
landing_detect . land_start_ms = now ;
landing_detect . vpos_start_m = height ;
}
// we only consider the vehicle landed when the motors have been
// at minimum for 5s and the vertical position estimate has not
// changed by more than 20cm for 4s
2016-07-08 05:38:43 -03:00
if ( fabsf ( height - landing_detect . vpos_start_m ) > 0.2 ) {
2016-06-16 05:17:46 -03:00
// height has changed, call off landing detection
landing_detect . land_start_ms = 0 ;
return ;
}
if ( ( now - landing_detect . land_start_ms ) < 4000 | |
( now - landing_detect . lower_limit_start_ms ) < 5000 ) {
// not landed yet
return ;
2016-03-09 03:20:41 -04:00
}
2016-06-16 05:17:46 -03:00
landing_detect . land_start_ms = 0 ;
// motors have been at zero for 5s, and we have had less than 0.3m
// change in altitude for last 4s. We are landed.
plane . disarm_motors ( ) ;
poscontrol . state = QPOS_LAND_COMPLETE ;
plane . gcs_send_text ( MAV_SEVERITY_INFO , " Land complete " ) ;
// reload target airspeed which could have been modified by the mission
2016-11-17 21:07:10 -04:00
plane . aparm . airspeed_cruise_cm . load ( ) ;
2016-03-09 03:20:41 -04:00
}
2016-01-01 05:42:10 -04:00
/*
check if a VTOL landing has completed
*/
2016-04-29 02:31:08 -03:00
bool QuadPlane : : verify_vtol_land ( void )
2016-01-01 05:42:10 -04:00
{
2016-01-01 07:09:11 -04:00
if ( ! available ( ) ) {
return true ;
}
2016-05-05 19:27:47 -03:00
if ( poscontrol . state = = QPOS_POSITION2 & &
2016-01-02 02:55:56 -04:00
plane . auto_state . wp_distance < 2 ) {
2016-05-05 19:27:47 -03:00
poscontrol . state = QPOS_LAND_DESCEND ;
2016-01-02 02:55:56 -04:00
plane . gcs_send_text ( MAV_SEVERITY_INFO , " Land descend started " ) ;
2016-04-29 02:31:08 -03:00
plane . set_next_WP ( plane . next_WP_loc ) ;
2016-01-02 02:55:56 -04:00
}
2016-01-01 07:09:11 -04:00
if ( should_relax ( ) ) {
wp_nav - > loiter_soften_for_landing ( ) ;
2016-01-01 05:42:10 -04:00
}
2016-01-02 02:55:56 -04:00
// at land_final_alt begin final landing
2016-06-10 04:36:46 -03:00
float height_above_ground = plane . relative_ground_altitude ( plane . g . rangefinder_landing ) ;
if ( poscontrol . state = = QPOS_LAND_DESCEND & & height_above_ground < land_final_alt ) {
2016-05-05 19:27:47 -03:00
poscontrol . state = QPOS_LAND_FINAL ;
2016-01-02 02:55:56 -04:00
pos_control - > set_alt_target ( inertial_nav . get_altitude ( ) ) ;
2016-07-25 02:46:17 -03:00
// cut IC engine if enabled
if ( land_icengine_cut ! = 0 ) {
plane . g2 . ice_control . engine_control ( 0 , 0 , 0 ) ;
}
2016-01-02 02:55:56 -04:00
plane . gcs_send_text ( MAV_SEVERITY_INFO , " Land final started " ) ;
}
2016-03-09 03:20:41 -04:00
check_land_complete ( ) ;
2016-01-01 05:42:10 -04:00
return false ;
}
2016-03-24 22:33:19 -03:00
// Write a control tuning packet
void QuadPlane : : Log_Write_QControl_Tuning ( )
{
2016-04-03 20:50:15 -03:00
const Vector3f & desired_velocity = pos_control - > get_desired_velocity ( ) ;
2016-04-10 07:36:20 -03:00
const Vector3f & accel_target = pos_control - > get_accel_target ( ) ;
2016-03-24 22:33:19 -03:00
struct log_QControl_Tuning pkt = {
LOG_PACKET_HEADER_INIT ( LOG_QTUN_MSG ) ,
time_us : AP_HAL : : micros64 ( ) ,
angle_boost : attitude_control - > angle_boost ( ) ,
throttle_out : motors - > get_throttle ( ) ,
desired_alt : pos_control - > get_alt_target ( ) / 100.0f ,
inav_alt : inertial_nav . get_altitude ( ) / 100.0f ,
baro_alt : ( int32_t ) plane . barometer . get_altitude ( ) * 100 ,
desired_climb_rate : ( int16_t ) pos_control - > get_vel_target_z ( ) ,
2016-04-03 20:50:15 -03:00
climb_rate : ( int16_t ) inertial_nav . get_velocity_z ( ) ,
dvx : desired_velocity . x * 0.01f ,
dvy : desired_velocity . y * 0.01f ,
2016-04-10 07:36:20 -03:00
dax : accel_target . x * 0.01f ,
day : accel_target . y * 0.01f ,
2016-03-24 22:33:19 -03:00
} ;
plane . DataFlash . WriteBlock ( & pkt , sizeof ( pkt ) ) ;
}
2016-04-20 02:13:20 -03:00
/*
calculate the forward throttle percentage . The forward throttle can
be used to assist with position hold and with landing approach . It
reduces the need for down pitch which reduces load on the vertical
lift motors .
*/
int8_t QuadPlane : : forward_throttle_pct ( void )
{
/*
in non - VTOL modes or modes without a velocity controller . We
don ' t use it in QHOVER or QSTABILIZE as they are the primary
recovery modes for a quadplane and need to be as simple as
possible . They will drift with the wind
*/
if ( ! in_vtol_mode ( ) | |
! motors - > armed ( ) | |
vel_forward . gain < = 0 | |
plane . control_mode = = QSTABILIZE | |
plane . control_mode = = QHOVER ) {
return 0 ;
}
2016-06-02 18:57:10 -03:00
float deltat = ( AP_HAL : : millis ( ) - vel_forward . last_ms ) * 0.001f ;
2016-04-20 02:13:20 -03:00
if ( deltat > 1 | | deltat < 0 ) {
vel_forward . integrator = 0 ;
deltat = 0.1 ;
}
if ( deltat < 0.1 ) {
// run at 10Hz
return vel_forward . last_pct ;
}
2016-06-02 18:57:10 -03:00
vel_forward . last_ms = AP_HAL : : millis ( ) ;
2016-04-20 02:13:20 -03:00
// work out the desired speed in forward direction
const Vector3f & desired_velocity_cms = pos_control - > get_desired_velocity ( ) ;
Vector3f vel_ned ;
if ( ! plane . ahrs . get_velocity_NED ( vel_ned ) ) {
// we don't know our velocity? EKF must be pretty sick
vel_forward . last_pct = 0 ;
2016-06-03 19:35:09 -03:00
vel_forward . integrator = 0 ;
2016-04-20 02:13:20 -03:00
return 0 ;
}
Vector3f vel_error_body = ahrs . get_rotation_body_to_ned ( ) . transposed ( ) * ( ( desired_velocity_cms * 0.01f ) - vel_ned ) ;
// find component of velocity error in fwd body frame direction
float fwd_vel_error = vel_error_body * Vector3f ( 1 , 0 , 0 ) ;
// scale forward velocity error by maximum airspeed
2016-08-07 21:48:36 -03:00
fwd_vel_error / = MAX ( plane . aparm . airspeed_max , 5 ) ;
2016-04-20 02:13:20 -03:00
// add in a component from our current pitch demand. This tends to
// move us to zero pitch. Assume that LIM_PITCH would give us the
// WP nav speed.
fwd_vel_error - = ( wp_nav - > get_speed_xy ( ) * 0.01f ) * plane . nav_pitch_cd / ( float ) plane . aparm . pitch_limit_max_cd ;
if ( should_relax ( ) & & vel_ned . length ( ) < 1 ) {
// we may be landed
fwd_vel_error = 0 ;
vel_forward . integrator * = 0.95f ;
}
// integrator as throttle percentage (-100 to 100)
vel_forward . integrator + = fwd_vel_error * deltat * vel_forward . gain * 100 ;
2016-06-13 18:31:50 -03:00
// inhibit reverse throttle and allow petrol engines with min > 0
int8_t fwd_throttle_min = ( plane . aparm . throttle_min < = 0 ) ? 0 : plane . aparm . throttle_min ;
vel_forward . integrator = constrain_float ( vel_forward . integrator , fwd_throttle_min , plane . aparm . throttle_max ) ;
2016-04-20 02:13:20 -03:00
2016-06-03 19:35:09 -03:00
// If we are below alt_cutoff then scale down the effect until it turns off at alt_cutoff and decay the integrator
float alt_cutoff = MAX ( 0 , vel_forward_alt_cutoff ) ;
float height_above_ground = plane . relative_ground_altitude ( plane . g . rangefinder_landing ) ;
vel_forward . last_pct = linear_interpolate ( 0 , vel_forward . integrator ,
height_above_ground , alt_cutoff , alt_cutoff + 2 ) ;
if ( vel_forward . last_pct = = 0 ) {
// if the percent is 0 then decay the integrator
vel_forward . integrator * = 0.95f ;
}
2016-04-20 02:13:20 -03:00
return vel_forward . last_pct ;
}
2016-04-20 03:23:17 -03:00
/*
get weathervaning yaw rate in cd / s
*/
float QuadPlane : : get_weathervane_yaw_rate_cds ( void )
{
/*
we only do weathervaning in modes where we are doing VTOL
position control . We also don ' t do it if the pilot has given any
yaw input in the last 3 seconds .
*/
if ( ! in_vtol_mode ( ) | |
! motors - > armed ( ) | |
weathervane . gain < = 0 | |
plane . control_mode = = QSTABILIZE | |
plane . control_mode = = QHOVER ) {
weathervane . last_output = 0 ;
return 0 ;
}
ArduPlane: Fix up after refactoring RC_Channel class
Further to refactor of RC_Channel class which included
adding get_xx set_xx methods, change reads and writes to the public members
to calls to get and set functionsss
old public member(int16_t) get function -> int16_t set function (int16_t)
(expression where c is an object of type RC_Channel)
c.radio_in c.get_radio_in() c.set_radio_in(v)
c.control_in c.get_control_in() c.set_control_in(v)
c.servo_out c.get_servo_out() c.set_servo_out(v)
c.pwm_out c.get_pwm_out() // use existing
c.radio_out c.get_radio_out() c.set_radio_out(v)
c.radio_max c.get_radio_max() c.set_radio_max(v)
c.radio_min c.get_radio_min() c.set_radio_min(v)
c.radio_trim c.get_radio_trim() c.set_radio_trim(v);
c.min_max_configured() // return true if min and max are configured
Because data members of RC_Channels are now private and so cannot be written directly
some overloads are provided in the Plane classes to provide the old functionality
new overload Plane::stick_mix_channel(RC_Channel *channel)
which forwards to the previously existing
void stick_mix_channel(RC_Channel *channel, int16_t &servo_out);
new overload Plane::channel_output_mixer(Rc_Channel* , RC_Channel*)const
which forwards to
(uint8_t mixing_type, int16_t & chan1, int16_t & chan2)const;
Rename functions
RC_Channel_aux::set_radio_trim(Aux_servo_function_t function)
to RC_Channel_aux::set_trim_to_radio_in_for(Aux_servo_function_t function)
RC_Channel_aux::set_servo_out(Aux_servo_function_t function, int16_t value)
to RC_Channel_aux::set_servo_out_for(Aux_servo_function_t function, int16_t value)
Rationale:
RC_Channel is a complicated class, which combines
several functionalities dealing with stick inputs
in pwm and logical units, logical and actual actuator
outputs, unit conversion etc, etc
The intent of this PR is to clarify existing use of
the class. At the basic level it should now be possible
to grep all places where private variable is set by
searching for the set_xx function.
(The wider purpose is to provide a more generic and
logically simpler method of output mixing. This is a small step)
2016-05-08 05:33:02 -03:00
if ( plane . channel_rudder - > get_control_in ( ) ! = 0 ) {
2016-04-20 03:23:17 -03:00
weathervane . last_pilot_input_ms = AP_HAL : : millis ( ) ;
weathervane . last_output = 0 ;
return 0 ;
}
if ( AP_HAL : : millis ( ) - weathervane . last_pilot_input_ms < 3000 ) {
weathervane . last_output = 0 ;
return 0 ;
}
2016-04-21 08:52:25 -03:00
float roll = wp_nav - > get_roll ( ) / 100.0f ;
if ( fabsf ( roll ) < weathervane . min_roll ) {
weathervane . last_output = 0 ;
return 0 ;
}
if ( roll > 0 ) {
roll - = weathervane . min_roll ;
} else {
roll + = weathervane . min_roll ;
}
float output = constrain_float ( ( roll / 45.0f ) * weathervane . gain , - 1 , 1 ) ;
2016-04-20 03:23:17 -03:00
if ( should_relax ( ) ) {
output = 0 ;
}
weathervane . last_output = 0.98f * weathervane . last_output + 0.02f * output ;
// scale over half of yaw_rate_max. This gives the pilot twice the
// authority of the weathervane controller
return weathervane . last_output * ( yaw_rate_max / 2 ) * 100 ;
}
2016-05-05 22:28:26 -03:00
/*
start guided mode control
*/
void QuadPlane : : guided_start ( void )
{
poscontrol . state = QPOS_POSITION1 ;
poscontrol . speed_scale = 0 ;
setup_target_position ( ) ;
poscontrol . slow_descent = ( plane . current_loc . alt > plane . next_WP_loc . alt ) ;
}
/*
update guided mode control
*/
void QuadPlane : : guided_update ( void )
{
// run VTOL position controller
vtol_position_controller ( ) ;
}
2016-09-26 22:46:51 -03:00
void QuadPlane : : afs_terminate ( void )
{
if ( available ( ) ) {
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SHUT_DOWN ) ;
motors - > output ( ) ;
}
}
2016-09-30 19:35:58 -03:00
/*
return true if we should do guided mode loitering using VTOL motors
*/
bool QuadPlane : : guided_mode_enabled ( void )
{
if ( ! available ( ) ) {
return false ;
}
// only use quadplane guided when in AUTO or GUIDED mode
if ( plane . control_mode ! = GUIDED & & plane . control_mode ! = AUTO ) {
return false ;
}
return guided_mode ! = 0 ;
}