Plane: allow exiting loiter_to_alt in cases where you get stuck

- check for scenarios where updrafts can keep you from loitering down indefinitely.
This commit is contained in:
Tom Pittenger 2017-02-21 03:56:32 -08:00 committed by Tom Pittenger
parent f5c89117f6
commit 4ccd59f394
3 changed files with 46 additions and 16 deletions

View File

@ -641,9 +641,20 @@ private:
// Direction for loiter. 1 for clockwise, -1 for counter-clockwise // Direction for loiter. 1 for clockwise, -1 for counter-clockwise
int8_t direction; int8_t direction;
// when loitering and an altitude is involved, this flag is true when it has been reached at least once
bool reached_target_alt;
// check for scenarios where updrafts can keep you from loitering down indefinitely.
bool unable_to_acheive_target_alt;
// start time of the loiter. Milliseconds. // start time of the loiter. Milliseconds.
uint32_t start_time_ms; uint32_t start_time_ms;
// altitude at start of loiter loop lap. Used to detect delta alt of each lap.
// only valid when sum_cd > 36000
int32_t start_lap_alt_cm;
int32_t next_sum_lap_cd;
// The amount of time we should stay in a loiter for the Loiter Time command. Milliseconds. // The amount of time we should stay in a loiter for the Loiter Time command. Milliseconds.
uint32_t time_max_ms; uint32_t time_max_ms;
} loiter; } loiter;

View File

@ -499,12 +499,8 @@ void Plane::do_loiter_to_alt(const AP_Mission::Mission_Command& cmd)
set_next_WP(loc); set_next_WP(loc);
loiter_set_direction_wp(cmd); loiter_set_direction_wp(cmd);
// used to signify primary turns goal not yet met when non-zero // init to 0, set to 1 when altitude is reached
condition_value = next_WP_loc.alt; condition_value = 0;
if (condition_value == 0) {
// the value of 0 is used to signify it has been reached. Lets bump alt to 1 which is 10cm. Close enough!
condition_value = 1;
}
} }
/********************************************************************************/ /********************************************************************************/
@ -674,7 +670,7 @@ bool Plane::verify_loiter_time()
} }
if (result) { if (result) {
gcs_send_text(MAV_SEVERITY_WARNING,"Verify nav: LOITER time complete"); gcs_send_text(MAV_SEVERITY_INFO,"Loiter time complete");
auto_state.vtol_loiter = false; auto_state.vtol_loiter = false;
} }
return result; return result;
@ -702,7 +698,7 @@ bool Plane::verify_loiter_turns()
} }
if (result) { if (result) {
gcs_send_text(MAV_SEVERITY_WARNING,"Verify nav: LOITER orbits complete"); gcs_send_text(MAV_SEVERITY_INFO,"Loiter orbits complete");
} }
return result; return result;
} }
@ -717,12 +713,16 @@ bool Plane::verify_loiter_to_alt()
bool result = false; bool result = false;
update_loiter(mission.get_current_nav_cmd().p1); update_loiter(mission.get_current_nav_cmd().p1);
//has target altitude been reached? // condition_value == 0 means alt has never been reached
if (condition_value != 0) { if (condition_value == 0) {
// primary goal, loiter alt // primary goal, loiter to alt
if (labs(condition_value - current_loc.alt) < 500 && loiter.sum_cd > 1) { if (labs(loiter.sum_cd) > 1 && (loiter.reached_target_alt || loiter.unable_to_acheive_target_alt)) {
// primary goal completed, initialize secondary heading goal // primary goal completed, initialize secondary heading goal
condition_value = 0; if (loiter.unable_to_acheive_target_alt) {
gcs_send_text_fmt(MAV_SEVERITY_INFO,"Loiter to alt was stuck at %d", current_loc.alt/100);
}
condition_value = 1;
result = verify_loiter_heading(true); result = verify_loiter_heading(true);
} }
} else { } else {
@ -731,7 +731,7 @@ bool Plane::verify_loiter_to_alt()
} }
if (result) { if (result) {
gcs_send_text(MAV_SEVERITY_WARNING,"Verify nav: LOITER alt complete"); gcs_send_text(MAV_SEVERITY_INFO,"Loiter to alt complete");
} }
return result; return result;
} }

View File

@ -22,6 +22,8 @@ void Plane::loiter_angle_reset(void)
{ {
loiter.sum_cd = 0; loiter.sum_cd = 0;
loiter.total_cd = 0; loiter.total_cd = 0;
loiter.reached_target_alt = false;
loiter.unable_to_acheive_target_alt = false;
} }
/* /*
@ -30,21 +32,38 @@ void Plane::loiter_angle_reset(void)
*/ */
void Plane::loiter_angle_update(void) void Plane::loiter_angle_update(void)
{ {
int32_t target_bearing_cd = nav_controller->target_bearing_cd(); static const int32_t lap_check_interval_cd = 3*36000;
const int32_t target_bearing_cd = nav_controller->target_bearing_cd();
int32_t loiter_delta_cd; int32_t loiter_delta_cd;
if (loiter.sum_cd == 0 && !reached_loiter_target()) { if (loiter.sum_cd == 0 && !reached_loiter_target()) {
// we don't start summing until we are doing the real loiter // we don't start summing until we are doing the real loiter
loiter_delta_cd = 0; loiter_delta_cd = 0;
} else if (loiter.sum_cd == 0) { } else if (loiter.sum_cd == 0) {
// use 1 cd for initial delta // use 1 cd for initial delta
loiter_delta_cd = 1; loiter_delta_cd = 1;
loiter.start_lap_alt_cm = current_loc.alt;
loiter.next_sum_lap_cd = lap_check_interval_cd;
} else { } else {
loiter_delta_cd = target_bearing_cd - loiter.old_target_bearing_cd; loiter_delta_cd = target_bearing_cd - loiter.old_target_bearing_cd;
} }
loiter.old_target_bearing_cd = target_bearing_cd; loiter.old_target_bearing_cd = target_bearing_cd;
loiter_delta_cd = wrap_180_cd(loiter_delta_cd); loiter_delta_cd = wrap_180_cd(loiter_delta_cd);
loiter.sum_cd += loiter_delta_cd * loiter.direction; loiter.sum_cd += loiter_delta_cd * loiter.direction;
if (labs(current_loc.alt - next_WP_loc.alt) < 500) {
loiter.reached_target_alt = true;
loiter.unable_to_acheive_target_alt = false;
loiter.next_sum_lap_cd = loiter.sum_cd + lap_check_interval_cd;
} else if (!loiter.reached_target_alt && labs(loiter.sum_cd) >= loiter.next_sum_lap_cd) {
// check every few laps for scenario where up/downdrafts inhibit you from loitering up/down for too long
loiter.unable_to_acheive_target_alt = labs(current_loc.alt - loiter.start_lap_alt_cm) < 500;
loiter.start_lap_alt_cm = current_loc.alt;
loiter.next_sum_lap_cd += lap_check_interval_cd;
}
} }
//**************************************************************** //****************************************************************