Arducopter: New Alt control code

Uses a new method to manage altitude changes. Alt hold controller was split into a hold and a rate controller just like navigation. Changing alt is done by specifying a climb rate.
Interactive alt hold is now simplified and is an 80cm climb or descent.
This commit is contained in:
Jason Short 2012-07-18 22:39:33 -07:00
parent 806663b80b
commit f40d8f04af

View File

@ -656,7 +656,7 @@ static int32_t baro_alt;
// The climb_rate as reported by Baro in cm/s // The climb_rate as reported by Baro in cm/s
static int16_t baro_rate; static int16_t baro_rate;
// used to switch out of Manual Boost // used to switch out of Manual Boost
static boolean reset_throttle_flag; static int8_t reset_throttle_counter;
// used to track when to read sensors vs estimate alt // used to track when to read sensors vs estimate alt
static boolean alt_sensor_flag; static boolean alt_sensor_flag;
@ -685,7 +685,7 @@ static boolean takeoff_complete;
// Used to see if we have landed and if we should shut our engines - not fully implemented // Used to see if we have landed and if we should shut our engines - not fully implemented
static boolean land_complete = true; static boolean land_complete = true;
// used to manually override throttle in interactive Alt hold modes // used to manually override throttle in interactive Alt hold modes
static int16_t manual_boost; //static int16_t manual_boost;
// An additional throttle added to keep the copter at the same altitude when banking // An additional throttle added to keep the copter at the same altitude when banking
static int16_t angle_boost; static int16_t angle_boost;
// Push copter down for clean landing // Push copter down for clean landing
@ -875,12 +875,15 @@ static Vector3f accels_position;
// accels rotated to world frame // accels rotated to world frame
static Vector3f accels_rotated; static Vector3f accels_rotated;
//static Vector3f position_error;
// error correction // error correction
static Vector3f speed_error; static Vector3f speed_error;
// Manage accel drift // Manage accel drift
static Vector3f accels_offset; //static float z_offset;
//static Vector3f accels_scale;
#endif #endif
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@ -1799,65 +1802,48 @@ void update_throttle_mode(void)
case THROTTLE_HOLD: case THROTTLE_HOLD:
// allow interactive changing of atitude // allow interactive changing of atitude
adjust_altitude(); if(g.rc_3.control_in < 200){
reset_throttle_counter = 50;
nav_throttle = get_throttle_rate(-80);
g.rc_3.servo_out = g.throttle_cruise + nav_throttle + angle_boost;
break;
}else if (g.rc_3.control_in > 800){
reset_throttle_counter = 50;
nav_throttle = get_throttle_rate(80);
g.rc_3.servo_out = g.throttle_cruise + nav_throttle + angle_boost;
break;
}
// fall through if(reset_throttle_counter > 0){
reset_throttle_counter--;
if(reset_throttle_counter == 0){
force_new_altitude(max(current_loc.alt, 100));
}else{
nav_throttle = get_throttle_rate(0);
}
}
// else fall through
case THROTTLE_AUTO: case THROTTLE_AUTO:
// calculate angle boost // calculate angle boost
angle_boost = get_angle_boost(g.throttle_cruise); angle_boost = get_angle_boost(g.throttle_cruise);
// manual command up or down? // 10hz
if(manual_boost != 0){
#if FRAME_CONFIG == HELI_FRAME
throttle_out = heli_get_angle_boost(g.throttle_cruise + manual_boost);
#else
throttle_out = g.throttle_cruise + angle_boost + manual_boost;
#endif
//force a reset of the altitude change
clear_new_altitude();
/*
int16_t iterm = g.pi_alt_hold.get_integrator();
Serial.printf("tar_alt: %d, actual_alt: %d \talt_err: %d, \t manb: %d, iterm %d\n",
next_WP.alt,
current_loc.alt,
altitude_error,
manual_boost,
iterm);
//*/
// this lets us know we need to update the altitude after manual throttle control
reset_throttle_flag = true;
}else{
// we are under automatic throttle control
// ---------------------------------------
if(reset_throttle_flag) {
force_new_altitude(max(current_loc.alt, 100));
reset_throttle_flag = false;
update_throttle_cruise();
}
// 10hz, don't run up i term
if(motors.auto_armed() == true){ if(motors.auto_armed() == true){
// how far off are we // how far off are we
altitude_error = get_altitude_error(); altitude_error = get_altitude_error();
// get the AP throttle int16_t desired_speed;
nav_throttle = get_nav_throttle(altitude_error); if(alt_change_flag == REACHED_ALT){ // we are at or above the target alt
desired_speed = g.pi_alt_hold.get_p(altitude_error); // calculate desired speed from lon error
/* desired_speed = constrain(desired_speed, -250, 250);
Serial.printf("tar_alt: %d, actual_alt: %d \talt_err: %d, \tnav_thr: %d, \talt Int: %d\n", nav_throttle = get_throttle_rate(desired_speed);
next_WP.alt, }else{
current_loc.alt, desired_speed = get_desired_climb_rate(150);
altitude_error, nav_throttle = get_throttle_rate(desired_speed);
nav_throttle, }
(int16_t)g.pi_alt_hold.get_integrator());
//*/
} }
// hack to remove the influence of the ground effect // hack to remove the influence of the ground effect
@ -1866,16 +1852,11 @@ void update_throttle_mode(void)
} }
#if FRAME_CONFIG == HELI_FRAME #if FRAME_CONFIG == HELI_FRAME
throttle_out = heli_get_angle_boost(g.throttle_cruise + nav_throttle + get_z_damping() - landing_boost); throttle_out = heli_get_angle_boost(g.throttle_cruise + nav_throttle - landing_boost);
#else #else
throttle_out = g.throttle_cruise + nav_throttle + angle_boost + get_z_damping() - landing_boost; throttle_out = g.throttle_cruise + nav_throttle + angle_boost - landing_boost;
#endif #endif
}
// light filter of output
//g.rc_3.servo_out = (g.rc_3.servo_out * (THROTTLE_FILTER_SIZE - 1) + throttle_out) / THROTTLE_FILTER_SIZE;
// no filter
g.rc_3.servo_out = throttle_out; g.rc_3.servo_out = throttle_out;
break; break;
} }
@ -2119,15 +2100,19 @@ static void update_altitude()
#else #else
// This is real life // This is real life
#if INERTIAL_NAV == ENABLED
baro_rate = accels_velocity.z;
#else
// read in Actual Baro Altitude // read in Actual Baro Altitude
baro_alt = read_barometer(); baro_alt = read_barometer();
//Serial.printf("baro_alt: %d \n", baro_alt);
// calc the vertical accel rate // calc the vertical accel rate
int temp = (baro_alt - old_baro_alt) * 10; int temp = (baro_alt - old_baro_alt) * 10;
baro_rate = (temp + baro_rate) >> 1; baro_rate = (temp + baro_rate) >> 1;
baro_rate = constrain(baro_rate, -300, 300); baro_rate = constrain(baro_rate, -300, 300);
old_baro_alt = baro_alt; old_baro_alt = baro_alt;
#endif
// Note: sonar_alt is calculated in a faster loop and filtered with a mode filter // Note: sonar_alt is calculated in a faster loop and filtered with a mode filter
#endif #endif
@ -2211,6 +2196,7 @@ static void update_altitude_est()
//Serial.printf(" %d, %d, %d, %d\n", climb_rate_actual, climb_rate_error, climb_rate, current_loc.alt); //Serial.printf(" %d, %d, %d, %d\n", climb_rate_actual, climb_rate_error, climb_rate, current_loc.alt);
} }
/*
#define THROTTLE_ADJUST 225 #define THROTTLE_ADJUST 225
static void static void
adjust_altitude() adjust_altitude()
@ -2229,6 +2215,7 @@ adjust_altitude()
manual_boost = 0; manual_boost = 0;
} }
} }
*/
static void tuning(){ static void tuning(){
tuning_value = (float)g.rc_6.control_in / 1000.0; tuning_value = (float)g.rc_6.control_in / 1000.0;
@ -2416,7 +2403,7 @@ static void update_nav_wp()
// nav_lon, nav_lat is calculated // nav_lon, nav_lat is calculated
if(wp_distance > 400){ if(wp_distance > 400){
calc_nav_rate(calc_desired_speed(g.waypoint_speed_max, true)); calc_nav_rate(get_desired_speed(g.waypoint_speed_max, true));
}else{ }else{
// calc the lat and long error to the target // calc the lat and long error to the target
calc_location_error(&next_WP); calc_location_error(&next_WP);
@ -2439,7 +2426,7 @@ static void update_nav_wp()
// calc error to target // calc error to target
calc_location_error(&next_WP); calc_location_error(&next_WP);
int16_t speed = calc_desired_speed(g.waypoint_speed_max, slow_wp); int16_t speed = get_desired_speed(g.waypoint_speed_max, slow_wp);
// use error as the desired rate towards the target // use error as the desired rate towards the target
calc_nav_rate(speed); calc_nav_rate(speed);
@ -2465,7 +2452,6 @@ static void update_nav_wp()
static void update_auto_yaw() static void update_auto_yaw()
{ {
if(wp_control == CIRCLE_MODE){ if(wp_control == CIRCLE_MODE){
//trace("CIRCLE mode")
auto_yaw = get_bearing(&current_loc, &circle_WP); auto_yaw = get_bearing(&current_loc, &circle_WP);
}else if(wp_control == LOITER_MODE){ }else if(wp_control == LOITER_MODE){