2015-11-24 04:24:04 -04:00
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
# include "Plane.h"
const AP_Param : : GroupInfo QuadPlane : : var_info [ ] = {
2015-12-26 06:40:40 -04:00
// @Param: ENABLE
// @DisplayName: Enable QuadPlane
// @Description: This enables QuadPlane functionality, assuming quad motors on outputs 5 to 8
// @Values: 0:Disable,1:Enable
// @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
2016-02-07 20:00:19 -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
// @Description: Maximum lean angle in all flight modes
// @Units: Centi-degrees
// @Range: 1000 8000
// @User: Advanced
2015-12-26 06:40:40 -04:00
AP_GROUPINFO ( " ANGLE_MAX " , 10 , QuadPlane , aparm . angle_max , 4500 ) ,
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
2015-11-24 04:24:04 -04:00
// @Units: milli-seconds
// @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
// @Param: THR_MID
// @DisplayName: Throttle Mid Position
// @Description: The throttle output (0 ~ 1000) when throttle stick is in mid position. Used to scale the manual throttle so that the mid throttle stick position is close to the throttle required to hover
// @User: Standard
// @Range: 300 700
// @Units: Percent*10
2016-03-07 20:23:51 -04:00
// @Increment: 10
2016-01-06 00:19:57 -04:00
AP_GROUPINFO ( " THR_MID " , 28 , QuadPlane , throttle_mid , 500 ) ,
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
// @Param: FRAME_CLASS
// @DisplayName: Frame Class
// @Description: Controls major frame class for multicopter component
2016-03-12 19:05:10 -04:00
// @Values: 0:Quad, 1:Hexa, 2:Octa, 3:OctaQuad
2016-02-07 20:00:19 -04:00
// @User: Standard
AP_GROUPINFO ( " FRAME_CLASS " , 30 , QuadPlane , frame_class , 0 ) ,
// @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 ) ,
2015-12-26 04:51:05 -04: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 } ,
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 + + ) {
RC_Channel_aux : : set_aux_channel_default ( ( RC_Channel_aux : : Aux_servo_function_t ) ( RC_Channel_aux : : k_motor1 + i ) , CH_5 + i ) ;
}
}
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
uint16_t mask ;
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-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
*/
2016-02-07 20:00:19 -04:00
switch ( ( enum frame_class ) frame_class . get ( ) ) {
case FRAME_CLASS_QUAD :
setup_default_channels ( 4 ) ;
2016-04-02 17:49:38 -03:00
motors = new AP_MotorsQuad ( plane . scheduler . get_loop_rate_hz ( ) ) ;
2016-02-07 20:00:19 -04:00
break ;
case FRAME_CLASS_HEXA :
setup_default_channels ( 6 ) ;
2016-04-02 17:49:38 -03:00
motors = new AP_MotorsHexa ( plane . scheduler . get_loop_rate_hz ( ) ) ;
2016-02-07 20:00:19 -04:00
break ;
case FRAME_CLASS_OCTA :
setup_default_channels ( 8 ) ;
2016-04-02 17:49:38 -03:00
motors = new AP_MotorsOcta ( plane . scheduler . get_loop_rate_hz ( ) ) ;
2016-02-07 20:00:19 -04:00
break ;
2016-03-12 19:04:25 -04:00
case FRAME_CLASS_OCTAQUAD :
setup_default_channels ( 8 ) ;
2016-04-02 17:49:38 -03:00
motors = new AP_MotorsOctaQuad ( plane . scheduler . get_loop_rate_hz ( ) ) ;
2016-03-12 19:04:25 -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 ;
}
2015-12-26 06:40:40 -04:00
if ( ! motors ) {
2016-01-06 03:17:08 -04:00
hal . console - > printf ( " Unable to allocate motors \n " ) ;
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-01-06 03:17:08 -04:00
hal . console - > printf ( " Unable to allocate attitude_control \n " ) ;
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-01-06 03:17:08 -04:00
hal . console - > printf ( " Unable to allocate pos_control \n " ) ;
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 ) ;
if ( ! pos_control ) {
2016-01-06 03:17:08 -04:00
hal . console - > printf ( " Unable to allocate wp_nav \n " ) ;
goto failed ;
2015-12-26 06:40:40 -04:00
}
AP_Param : : load_object_from_eeprom ( wp_nav , wp_nav - > var_info ) ;
2016-02-07 20:00:19 -04:00
motors - > set_frame_orientation ( frame_type ) ;
2016-01-08 16:44:45 -04:00
motors - > Init ( ) ;
2016-04-01 08:18:51 -03:00
motors - > set_throttle_range ( 100 , thr_min_pwm , thr_max_pwm ) ;
2016-01-06 00:19:57 -04:00
motors - > set_hover_throttle ( throttle_mid ) ;
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 ) ;
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-01-06 03:17:08 -04:00
mask = motors - > get_motor_mask ( ) ;
2016-01-01 22:29:05 -04:00
for ( uint8_t i = 0 ; i < 16 ; i + + ) {
if ( mask & 1U < < i ) {
RC_Channel * ch = RC_Channel : : rc_channel ( i ) ;
if ( ch ! = nullptr ) {
ch - > radio_trim = thr_min_pwm ;
}
}
}
# if CONFIG_HAL_BOARD == HAL_BOARD_PX4
// 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 ) ;
}
}
}
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
2015-12-26 06:40:40 -04:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw_smooth ( plane . nav_roll_cd ,
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-03-26 00:27:00 -03:00
float pilot_throttle_scaled = plane . channel_throttle - > 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
2015-12-26 06:40:40 -04:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw_smooth ( plane . nav_roll_cd ,
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 ;
land_state = QLAND_DESCEND ;
motors_lower_limit_start_ms = 0 ;
}
2016-01-01 06:39:36 -04:00
// helper for is_flying()
bool QuadPlane : : is_flying ( void )
{
if ( ! available ( ) ) {
return false ;
}
2016-04-01 07:54:32 -03:00
if ( motors - > get_throttle ( ) > 0.2 & & ! 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 )
{
bool motor_at_lower_limit = motors - > limit . throttle_lower & & motors - > is_throttle_mix_min ( ) ;
2016-04-01 07:54:32 -03:00
if ( motors - > get_throttle ( ) < 0.01 ) {
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 ) {
motors_lower_limit_start_ms = 0 ;
}
if ( motor_at_lower_limit & & motors_lower_limit_start_ms = = 0 ) {
motors_lower_limit_start_ms = millis ( ) ;
}
bool relax_loiter = motors_lower_limit_start_ms ! = 0 & & ( millis ( ) - motors_lower_limit_start_ms ) > 1000 ;
return relax_loiter ;
}
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
2015-12-26 06:40:40 -04:00
wp_nav - > set_pilot_desired_acceleration ( plane . channel_roll - > control_in ,
plane . channel_pitch - > 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
// call attitude controller
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-01-09 18:42:46 -04:00
get_desired_yaw_rate_cds ( ) ) ;
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-04-02 08:45:51 -03:00
float height_above_ground ;
if ( plane . g . rangefinder_landing & & plane . rangefinder_state . in_range ) {
height_above_ground = plane . rangefinder_state . height_estimate ;
} else {
height_above_ground = plane . adjusted_relative_altitude_cm ( ) * 0.01 ;
}
if ( height_above_ground < land_final_alt & & land_state < QLAND_FINAL ) {
land_state = QLAND_FINAL ;
2016-03-09 03:20:41 -04:00
}
2016-04-02 08:45:51 -03:00
float descent_rate = ( land_state = = QLAND_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 )
{
if ( plane . channel_throttle - > control_in < = 0 & & ! plane . auto_throttle_mode ) {
// 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
}
2016-01-02 21:31:13 -04:00
if ( plane . channel_throttle - > 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-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 ( ) ;
uint16_t trim = ( plane . channel_throttle - > radio_max + plane . channel_throttle - > radio_min ) / 2 ;
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 )
{
if ( plane . channel_throttle - > control_in > = 10 | |
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
climb_rate = plane . altitude_error_cm / 10 ;
} else {
// otherwise estimate from pilot input
climb_rate = plane . g . flybywire_climb_rate * ( plane . nav_pitch_cd / ( float ) plane . aparm . pitch_limit_max_cd ) ;
climb_rate * = plane . channel_throttle - > 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 ;
if ( ! ahrs . airspeed_estimate ( & aspeed ) | | aspeed < plane . aparm . airspeed_min ) {
aspeed = plane . aparm . airspeed_min ;
}
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
/*
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-02-04 00:03:39 -04:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SHUT_DOWN ) ;
motors - > output ( ) ;
2015-12-26 04:27:13 -04:00
transition_state = TRANSITION_DONE ;
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
*/
if ( have_airspeed & & aspeed < assist_speed & &
2016-01-01 03:18:53 -04:00
( plane . auto_throttle_mode | |
plane . channel_throttle - > control_in > 0 | |
plane . is_flying ( ) ) ) {
2015-12-26 06:40:40 -04:00
// the quad should provide some assistance to the plane
transition_state = TRANSITION_AIRSPEED_WAIT ;
transition_start_ms = millis ( ) ;
assisted_flight = true ;
} else {
assisted_flight = false ;
}
2016-01-09 06:50:59 -04:00
if ( transition_state < TRANSITION_TIMER ) {
// set a single loop pitch limit in TECS
2016-01-15 01:49:49 -04:00
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
}
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 ( ) ;
}
2015-12-26 06:40:40 -04: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 ( ) ) ;
attitude_control - > rate_controller_run ( ) ;
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
}
2015-12-26 04:51:05 -04:00
float throttle_scaled = last_throttle * ( transition_time_ms - ( millis ( ) - transition_start_ms ) ) / ( float ) transition_time_ms ;
2015-12-26 04:27:13 -04:00
if ( throttle_scaled < 0 ) {
throttle_scaled = 0 ;
}
2016-01-01 00:36:03 -04:00
assisted_flight = true ;
2015-12-26 04:27:13 -04:00
hold_stabilize ( throttle_scaled ) ;
2015-12-26 06:40:40 -04:00
attitude_control - > rate_controller_run ( ) ;
2016-03-24 22:11:56 -03:00
motors_output ( ) ;
2015-12-26 04:27:13 -04:00
break ;
}
case TRANSITION_DONE :
2016-04-01 03:29:51 -03:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_SHUT_DOWN ) ;
2016-02-04 00:03:39 -04:00
motors - > output ( ) ;
2015-12-26 04:27:13 -04:00
break ;
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-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
2015-12-26 06:40:40 -04:00
attitude_control - > rate_controller_run ( ) ;
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 & &
( plane . channel_throttle - > control_in > 10 | |
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
}
}
2015-12-26 06:40:40 -04:00
2016-03-24 22:11:56 -03:00
/*
output motors and do any copter needed
*/
void QuadPlane : : motors_output ( void )
{
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-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 ( ) ;
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 ;
}
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 ;
2015-12-26 06:40:40 -04:00
default :
break ;
}
return true ;
}
/*
handle a MAVLink DO_VTOL_TRANSITION
*/
bool QuadPlane : : handle_do_vtol_transition ( const mavlink_command_long_t & packet )
{
if ( ! available ( ) ) {
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " VTOL not available " ) ;
return MAV_RESULT_FAILED ;
}
if ( plane . control_mode ! = AUTO ) {
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " VTOL transition only in AUTO " ) ;
return MAV_RESULT_FAILED ;
}
switch ( ( uint8_t ) packet . param1 ) {
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 ;
return MAV_RESULT_ACCEPTED ;
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 ;
return MAV_RESULT_ACCEPTED ;
}
plane . gcs_send_text_fmt ( MAV_SEVERITY_NOTICE , " Invalid VTOL mode " ) ;
return MAV_RESULT_FAILED ;
}
2016-01-01 06:39:36 -04:00
/*
are we in a VTOL auto state ?
*/
bool QuadPlane : : in_vtol_auto ( void )
{
2016-04-09 05:48:22 -03:00
if ( ! enable | | 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 ;
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-02-20 05:20:27 -04:00
in_vtol_auto ( ) ) ;
}
2015-12-26 06:40:40 -04:00
/*
handle auto - mode when auto_state . vtol_mode is true
*/
void QuadPlane : : control_auto ( const Location & loc )
{
2016-01-09 01:26:13 -04:00
if ( ! setup ( ) ) {
return ;
}
2016-04-01 03:29:51 -03:00
motors - > set_desired_spool_state ( AP_Motors : : DESIRED_THROTTLE_UNLIMITED ) ;
2015-12-26 06:40:40 -04:00
Location origin = inertial_nav . get_origin ( ) ;
Vector2f diff2d ;
Vector3f target ;
diff2d = location_diff ( origin , loc ) ;
target . x = diff2d . x * 100 ;
target . y = diff2d . y * 100 ;
target . z = loc . alt - origin . alt ;
2016-01-02 02:55:56 -04:00
if ( ! locations_are_same ( loc , last_auto_target ) | |
loc . alt ! = last_auto_target . alt | |
millis ( ) - last_loiter_ms > 500 ) {
2015-12-26 06:40:40 -04:00
wp_nav - > set_wp_destination ( target ) ;
last_auto_target = loc ;
}
2016-01-01 03:18:53 -04:00
last_loiter_ms = millis ( ) ;
2015-12-26 06:40:40 -04:00
// 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 ) ;
2016-01-15 01:49:49 -04:00
if ( plane . mission . get_current_nav_cmd ( ) . id = = MAV_CMD_NAV_VTOL_TAKEOFF ) {
2016-01-02 02:55:56 -04:00
/*
2016-01-15 01:49:49 -04:00
for takeoff we need to use the loiter controller wpnav controller takes over the descent rate
control
2016-01-02 02:55:56 -04:00
*/
float ekfGndSpdLimit , ekfNavVelGainScaler ;
ahrs . getEkfControlLimits ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
// run loiter controller
wp_nav - > update_loiter ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
2016-01-08 19:47:41 -04:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw_smooth ( plane . nav_roll_cd ,
plane . nav_pitch_cd ,
2016-01-09 18:42:46 -04:00
get_pilot_input_yaw_rate_cds ( ) ,
2016-01-08 19:47:41 -04:00
smoothing_gain ) ;
2016-01-15 01:49:49 -04: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-01-09 18:42:46 -04:00
} else if ( plane . mission . get_current_nav_cmd ( ) . id = = MAV_CMD_NAV_VTOL_LAND & &
2016-01-15 01:49:49 -04:00
land_state > = QLAND_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
float ekfGndSpdLimit , ekfNavVelGainScaler ;
ahrs . getEkfControlLimits ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
// run loiter controller
wp_nav - > update_loiter ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
2016-01-09 18:42:46 -04:00
2016-01-15 01:49:49 -04:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw_smooth ( plane . nav_roll_cd ,
plane . nav_pitch_cd ,
get_pilot_input_yaw_rate_cds ( ) ,
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-02 06:54:01 -03:00
} else if ( plane . mission . get_current_nav_cmd ( ) . id = = MAV_CMD_NAV_VTOL_LAND & &
land_state = = QLAND_POSITION1 ) {
2016-04-10 07:36:20 -03:00
Vector2f diff_wp = location_diff ( plane . current_loc , loc ) ;
2016-04-02 06:54:01 -03:00
2016-04-10 07:36:20 -03:00
if ( land . 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 ( ) ;
// newdiff is the delta to our target if we keep going for one second
Vector2f newdiff = diff_wp - groundspeed ;
// speed towards target is the change in distance to target over one second
float speed_towards_target = diff_wp . length ( ) - newdiff . length ( ) ;
// setup land_speed_scale so at current distance we maintain speed towards target, and slow down as
// we approach
2016-04-10 07:36:20 -03:00
float distance = diff_wp . length ( ) ;
land . speed_scale = MAX ( speed_towards_target , wp_nav - > get_speed_xy ( ) * 0.01 ) / MAX ( distance , 1 ) ;
2016-04-02 06:54:01 -03:00
}
2016-04-10 07:36:20 -03:00
// run fixed wing navigation
plane . nav_controller - > update_waypoint ( plane . prev_WP_loc , plane . next_WP_loc ) ;
/*
calculate target velocity , not dropping it below 2 m / s
*/
const float final_speed = 2.0f ;
Vector2f target_speed = diff_wp * land . speed_scale ;
if ( target_speed . length ( ) < final_speed ) {
target_speed = target_speed . normalized ( ) * final_speed ;
}
pos_control - > set_desired_velocity_xy ( target_speed . x * 100 ,
target_speed . y * 100 ) ;
float ekfGndSpdLimit , ekfNavVelGainScaler ;
ahrs . getEkfControlLimits ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
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
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw_smooth ( plane . nav_roll_cd ,
plane . nav_pitch_cd ,
2016-04-10 07:36:20 -03:00
desired_auto_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-04-02 06:54:01 -03:00
land_state = QLAND_POSITION2 ;
2016-04-10 07:36:20 -03:00
wp_nav - > init_loiter_target ( ) ;
2016-04-03 20:50:15 -03:00
plane . gcs_send_text_fmt ( MAV_SEVERITY_INFO , " Land 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-01-15 01:49:49 -04:00
} else if ( plane . mission . get_current_nav_cmd ( ) . id = = MAV_CMD_NAV_VTOL_LAND ) {
/*
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
plane . nav_controller - > update_waypoint ( plane . prev_WP_loc , plane . next_WP_loc ) ;
2016-01-15 01:49:49 -04:00
pos_control - > set_xy_target ( target . x , target . y ) ;
2016-01-09 18:42:46 -04:00
2016-01-15 01:49:49 -04:00
float ekfGndSpdLimit , ekfNavVelGainScaler ;
ahrs . getEkfControlLimits ( ekfGndSpdLimit , ekfNavVelGainScaler ) ;
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-01-09 18:42:46 -04:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw_smooth ( plane . nav_roll_cd ,
plane . nav_pitch_cd ,
get_pilot_input_yaw_rate_cds ( ) ,
2016-01-15 01:49:49 -04:00
smoothing_gain ) ;
2016-01-09 18:42:46 -04:00
} else {
/*
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 ( ) ,
true ) ;
2016-01-15 01:49:49 -04: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 ( ) ;
2016-01-02 02:55:56 -04:00
}
2015-12-26 06:40:40 -04:00
2016-01-01 05:42:10 -04:00
switch ( plane . mission . get_current_nav_cmd ( ) . id ) {
case MAV_CMD_NAV_VTOL_LAND :
2016-04-02 06:54:01 -03:00
if ( land_state < = QLAND_POSITION2 ) {
2016-01-20 02:21:54 -04:00
pos_control - > set_alt_target_from_climb_rate ( 0 , plane . G_Dt , false ) ;
2016-04-02 06:54:01 -03:00
} else if ( land_state > QLAND_POSITION2 & & land_state < QLAND_FINAL ) {
2016-04-02 08:45:51 -03:00
float height_above_ground = ( plane . current_loc . alt - plane . next_WP_loc . alt ) * 0.01 ;
pos_control - > set_alt_target_from_climb_rate ( - landing_descent_rate_cms ( height_above_ground ) ,
plane . G_Dt , true ) ;
2016-01-01 06:39:36 -04:00
} else {
2016-01-20 02:21:54 -04:00
pos_control - > set_alt_target_from_climb_rate ( - land_speed_cms , plane . G_Dt , true ) ;
2016-01-01 06:39:36 -04:00
}
2016-01-01 05:42:10 -04:00
break ;
case MAV_CMD_NAV_VTOL_TAKEOFF :
2016-01-08 19:47:41 -04:00
pos_control - > set_alt_target_from_climb_rate ( 100 , plane . G_Dt , true ) ;
2016-01-01 05:42:10 -04:00
break ;
default :
pos_control - > set_alt_target_from_climb_rate_ff ( assist_climb_rate_cms ( ) , plane . G_Dt , false ) ;
break ;
}
2015-12-26 06:40:40 -04:00
pos_control - > update_z_controller ( ) ;
}
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-04-02 06:54:01 -03:00
land_state = QLAND_POSITION1 ;
2016-04-10 07:36:20 -03:00
land . speed_scale = 0 ;
wp_nav - > init_loiter_target ( ) ;
2016-01-15 01:49:49 -04:00
throttle_wait = false ;
2016-04-10 07:36:20 -03:00
land . yaw_cd = get_bearing_cd ( plane . prev_WP_loc , plane . next_WP_loc ) ;
2016-01-01 07:09:11 -04:00
motors_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-01-01 05:42:10 -04:00
return true ;
}
2016-03-09 03:20:41 -04:00
void QuadPlane : : check_land_complete ( void )
{
if ( land_state = = QLAND_FINAL & &
( motors_lower_limit_start_ms ! = 0 & &
millis ( ) - motors_lower_limit_start_ms > 5000 ) ) {
plane . disarm_motors ( ) ;
land_state = QLAND_COMPLETE ;
plane . gcs_send_text ( MAV_SEVERITY_INFO , " Land complete " ) ;
}
}
2016-01-01 05:42:10 -04:00
/*
check if a VTOL landing has completed
*/
2016-01-02 02:55:56 -04:00
bool QuadPlane : : verify_vtol_land ( const AP_Mission : : Mission_Command & cmd )
2016-01-01 05:42:10 -04:00
{
2016-01-01 07:09:11 -04:00
if ( ! available ( ) ) {
return true ;
}
2016-04-02 06:54:01 -03:00
if ( land_state = = QLAND_POSITION2 & &
2016-01-02 02:55:56 -04:00
plane . auto_state . wp_distance < 2 ) {
land_state = QLAND_DESCEND ;
plane . gcs_send_text ( MAV_SEVERITY_INFO , " Land descend started " ) ;
plane . set_next_WP ( cmd . content . location ) ;
}
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
if ( land_state = = QLAND_DESCEND & &
plane . current_loc . alt < plane . next_WP_loc . alt + land_final_alt * 100 ) {
land_state = QLAND_FINAL ;
pos_control - > set_alt_target ( inertial_nav . get_altitude ( ) ) ;
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 ) ) ;
}