64a8e66dd7
this prevents a problem where the disarm throttle will change after upgrading to 3.4
200 lines
7.3 KiB
C++
200 lines
7.3 KiB
C++
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
#include "Copter.h"
|
|
|
|
// Traditional helicopter variables and functions
|
|
|
|
#if FRAME_CONFIG == HELI_FRAME
|
|
|
|
#ifndef HELI_DYNAMIC_FLIGHT_SPEED_MIN
|
|
#define HELI_DYNAMIC_FLIGHT_SPEED_MIN 500 // we are in "dynamic flight" when the speed is over 1m/s for 2 seconds
|
|
#endif
|
|
|
|
// counter to control dynamic flight profile
|
|
static int8_t heli_dynamic_flight_counter;
|
|
|
|
// heli_init - perform any special initialisation required for the tradheli
|
|
void Copter::heli_init()
|
|
{
|
|
// helicopters are always using motor interlock
|
|
set_using_interlock(true);
|
|
|
|
/*
|
|
automatically set H_RSC_MIN and H_RSC_MAX from RC8_MIN and
|
|
RC8_MAX so that when users upgrade from tradheli version 3.3 to
|
|
3.4 they get the same throttle range as in previous versions of
|
|
the code
|
|
*/
|
|
if (!g.heli_servo_rsc.radio_min.load()) {
|
|
g.heli_servo_rsc.radio_min.set_and_save(g.rc_8.radio_min.get());
|
|
}
|
|
if (!g.heli_servo_rsc.radio_max.load()) {
|
|
g.heli_servo_rsc.radio_max.set_and_save(g.rc_8.radio_max.get());
|
|
}
|
|
}
|
|
|
|
// get_pilot_desired_collective - converts pilot input (from 0 ~ 1000) to a value that can be fed into the channel_throttle->servo_out function
|
|
int16_t Copter::get_pilot_desired_collective(int16_t control_in)
|
|
{
|
|
// return immediately if reduce collective range for manual flight has not been configured
|
|
if (g.heli_stab_col_min == 0 && g.heli_stab_col_max == 1000) {
|
|
return control_in;
|
|
}
|
|
|
|
// scale pilot input to reduced collective range
|
|
float scalar = ((float)(g.heli_stab_col_max - g.heli_stab_col_min))/1000.0f;
|
|
int16_t collective_out = g.heli_stab_col_min + control_in * scalar;
|
|
collective_out = constrain_int16(collective_out, 0, 1000);
|
|
return collective_out;
|
|
}
|
|
|
|
// heli_check_dynamic_flight - updates the dynamic_flight flag based on our horizontal velocity
|
|
// should be called at 50hz
|
|
void Copter::check_dynamic_flight(void)
|
|
{
|
|
if (!motors.armed() || !motors.rotor_runup_complete() ||
|
|
control_mode == LAND || (control_mode==RTL && rtl_state == RTL_Land) || (control_mode == AUTO && auto_mode == Auto_Land)) {
|
|
heli_dynamic_flight_counter = 0;
|
|
heli_flags.dynamic_flight = false;
|
|
return;
|
|
}
|
|
|
|
bool moving = false;
|
|
|
|
// with GPS lock use inertial nav to determine if we are moving
|
|
if (position_ok()) {
|
|
// get horizontal velocity
|
|
float velocity = inertial_nav.get_velocity_xy();
|
|
moving = (velocity >= HELI_DYNAMIC_FLIGHT_SPEED_MIN);
|
|
}else{
|
|
// with no GPS lock base it on throttle and forward lean angle
|
|
moving = (motors.get_throttle() > 800.0f || ahrs.pitch_sensor < -1500);
|
|
}
|
|
|
|
if (moving) {
|
|
// if moving for 2 seconds, set the dynamic flight flag
|
|
if (!heli_flags.dynamic_flight) {
|
|
heli_dynamic_flight_counter++;
|
|
if (heli_dynamic_flight_counter >= 100) {
|
|
heli_flags.dynamic_flight = true;
|
|
heli_dynamic_flight_counter = 100;
|
|
}
|
|
}
|
|
}else{
|
|
// if not moving for 2 seconds, clear the dynamic flight flag
|
|
if (heli_flags.dynamic_flight) {
|
|
if (heli_dynamic_flight_counter > 0) {
|
|
heli_dynamic_flight_counter--;
|
|
}else{
|
|
heli_flags.dynamic_flight = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// update_heli_control_dynamics - pushes several important factors up into AP_MotorsHeli.
|
|
// should be run between the rate controller and the servo updates.
|
|
void Copter::update_heli_control_dynamics(void)
|
|
{
|
|
// Use Leaky_I if we are not moving fast
|
|
attitude_control.use_leaky_i(!heli_flags.dynamic_flight);
|
|
|
|
// To-Do: Update dynamic phase angle of swashplate
|
|
}
|
|
|
|
// heli_update_landing_swash - sets swash plate flag so higher minimum is used when landed or landing
|
|
// should be called soon after update_land_detector in main code
|
|
void Copter::heli_update_landing_swash()
|
|
{
|
|
switch(control_mode) {
|
|
case ACRO:
|
|
case STABILIZE:
|
|
case DRIFT:
|
|
case SPORT:
|
|
// manual modes always uses full swash range
|
|
motors.set_collective_for_landing(false);
|
|
break;
|
|
|
|
case LAND:
|
|
// landing always uses limit swash range
|
|
motors.set_collective_for_landing(true);
|
|
break;
|
|
|
|
case RTL:
|
|
if (rtl_state == RTL_Land) {
|
|
motors.set_collective_for_landing(true);
|
|
}else{
|
|
motors.set_collective_for_landing(!heli_flags.dynamic_flight || ap.land_complete || !ap.auto_armed);
|
|
}
|
|
break;
|
|
|
|
case AUTO:
|
|
if (auto_mode == Auto_Land) {
|
|
motors.set_collective_for_landing(true);
|
|
}else{
|
|
motors.set_collective_for_landing(!heli_flags.dynamic_flight || ap.land_complete || !ap.auto_armed);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
// auto and hold use limited swash when landed
|
|
motors.set_collective_for_landing(!heli_flags.dynamic_flight || ap.land_complete || !ap.auto_armed);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// heli_update_rotor_speed_targets - reads pilot input and passes new rotor speed targets to heli motors object
|
|
void Copter::heli_update_rotor_speed_targets()
|
|
{
|
|
|
|
static bool rotor_runup_complete_last = false;
|
|
|
|
// get rotor control method
|
|
uint8_t rsc_control_mode = motors.get_rsc_mode();
|
|
|
|
rsc_control_deglitched = rotor_speed_deglitch_filter.apply(g.rc_8.control_in);
|
|
|
|
|
|
switch (rsc_control_mode) {
|
|
case ROTOR_CONTROL_MODE_SPEED_PASSTHROUGH:
|
|
// pass through pilot desired rotor speed if control input is higher than 10, creating a deadband at the bottom
|
|
if (rsc_control_deglitched > 10) {
|
|
motors.set_interlock(true);
|
|
motors.set_desired_rotor_speed(rsc_control_deglitched);
|
|
} else {
|
|
motors.set_interlock(false);
|
|
motors.set_desired_rotor_speed(0);
|
|
}
|
|
break;
|
|
case ROTOR_CONTROL_MODE_SPEED_SETPOINT:
|
|
case ROTOR_CONTROL_MODE_OPEN_LOOP_POWER_OUTPUT:
|
|
case ROTOR_CONTROL_MODE_CLOSED_LOOP_POWER_OUTPUT:
|
|
// pass setpoint through as desired rotor speed, this is almost pointless as the Setpoint serves no function in this mode
|
|
// other than being used to create a crude estimate of rotor speed
|
|
if (rsc_control_deglitched > 0) {
|
|
motors.set_interlock(true);
|
|
motors.set_desired_rotor_speed(motors.get_rsc_setpoint());
|
|
}else{
|
|
motors.set_interlock(false);
|
|
motors.set_desired_rotor_speed(0);
|
|
}
|
|
break;
|
|
}
|
|
|
|
// when rotor_runup_complete changes to true, log event
|
|
if (!rotor_runup_complete_last && motors.rotor_runup_complete()){
|
|
Log_Write_Event(DATA_ROTOR_RUNUP_COMPLETE);
|
|
} else if (rotor_runup_complete_last && !motors.rotor_runup_complete()){
|
|
Log_Write_Event(DATA_ROTOR_SPEED_BELOW_CRITICAL);
|
|
}
|
|
rotor_runup_complete_last = motors.rotor_runup_complete();
|
|
}
|
|
|
|
// heli_radio_passthrough send RC inputs direct into motors library for use during manual passthrough for helicopter setup
|
|
void Copter::heli_radio_passthrough()
|
|
{
|
|
motors.set_radio_passthrough(channel_roll->control_in, channel_pitch->control_in, channel_throttle->control_in, channel_yaw->control_in);
|
|
}
|
|
|
|
#endif // FRAME_CONFIG == HELI_FRAME
|