From fcc4814e10d912277763670fba2a448f10b78837 Mon Sep 17 00:00:00 2001 From: Peter Barker Date: Wed, 12 May 2021 11:09:16 +1000 Subject: [PATCH 1/3] autotest: add SmartRTL test for rapid switch between smartrtl and althold --- Tools/autotest/arducopter.py | 23 ++++++++++++++++++++++- 1 file changed, 22 insertions(+), 1 deletion(-) diff --git a/Tools/autotest/arducopter.py b/Tools/autotest/arducopter.py index c94119bfe69a5d..0b57b57a4a7d0a 100644 --- a/Tools/autotest/arducopter.py +++ b/Tools/autotest/arducopter.py @@ -9026,7 +9026,7 @@ def PositionWhenGPSIsZero(self): self.wait_disarmed() self.reboot_sitl() - def SMART_RTL(self): + def SMART_pathfollow_RTL(self): '''Check SMART_RTL''' self.progress("arm the vehicle and takeoff in Guided") self.takeoff(20, mode='GUIDED') @@ -9323,6 +9323,22 @@ def GPSForYaw(self): self.print_exception_caught(e) ex = e + def test_SMART_RTL_nopoints(self): + self.context_push() + ex = None + try: + self.change_mode('AUTO') + self.wait_ready_to_arm() + self.change_mode('ALT_HOLD') + self.change_mode('SMART_RTL') + self.change_mode('ALT_HOLD') + self.change_mode('SMART_RTL') + + except Exception as e: + self.print_exception_caught(e) + ex = e + self.disarm_vehicle(force=True) + self.context_pop() self.reboot_sitl() @@ -9371,6 +9387,11 @@ def AP_Avoidance(self): self.disarm_vehicle() self.context_pop() + def SMART_RTL(self): + '''Check SMART_RTL''' + self.test_SMART_RTL_nopoints() + self.test_SMART_RTL_pathfollow() + def PAUSE_CONTINUE(self): '''Test MAV_CMD_DO_PAUSE_CONTINUE in AUTO mode''' self.load_mission(filename="copter_mission.txt", strict=False) From 2c05ea2e293d837ff8730cd4900c77af0d459382 Mon Sep 17 00:00:00 2001 From: Peter Barker Date: Wed, 12 May 2021 13:04:33 +1000 Subject: [PATCH 2/3] AP_SmartRTL: add accessor to get the path semaphore --- libraries/AP_SmartRTL/AP_SmartRTL.h | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/libraries/AP_SmartRTL/AP_SmartRTL.h b/libraries/AP_SmartRTL/AP_SmartRTL.h index 144762e512839d..251589f85278c3 100644 --- a/libraries/AP_SmartRTL/AP_SmartRTL.h +++ b/libraries/AP_SmartRTL/AP_SmartRTL.h @@ -48,6 +48,11 @@ class AP_SmartRTL { // this may fail if the IO thread has taken the path semaphore bool peek_point(Vector3f& point); + // returns the semaphore responsible for path protection: + HAL_Semaphore &path_semaphore(void) { + return _path_sem; + } + // clear return path and set return location if position_ok is true. This should be called as part of the arming procedure // if position_ok is false, SmartRTL will not be available. // example sketches use the method that allows providing vehicle position directly From 398a1eef6fc5aa847d4a071818891bce7897247f Mon Sep 17 00:00:00 2001 From: Peter Barker Date: Wed, 12 May 2021 13:00:16 +1000 Subject: [PATCH 3/3] Copter: SmartRTL: do not consume home position when following path This home position is explicitly set when set_home is called (e.g. via the mavlink interface). It doesn't make any sense to pop that home position from the list as it won't ever get reset. --- ArduCopter/mode.h | 2 +- ArduCopter/mode_smart_rtl.cpp | 63 ++++++++++++++++++----------------- 2 files changed, 34 insertions(+), 31 deletions(-) diff --git a/ArduCopter/mode.h b/ArduCopter/mode.h index 2e33f121c3026c..7dcf5063b1c4d8 100644 --- a/ArduCopter/mode.h +++ b/ArduCopter/mode.h @@ -1513,7 +1513,7 @@ class ModeSmartRTL : public ModeRTL { // keep track of how long we have failed to get another return // point while following our path home. If we take too long we // may choose to land the vehicle. - uint32_t path_follow_last_pop_fail_ms; + uint32_t path_follow_last_semtake_fail_ms; }; diff --git a/ArduCopter/mode_smart_rtl.cpp b/ArduCopter/mode_smart_rtl.cpp index dacd7cc5f29ab0..7b98727ec00b0b 100644 --- a/ArduCopter/mode_smart_rtl.cpp +++ b/ArduCopter/mode_smart_rtl.cpp @@ -74,7 +74,7 @@ void ModeSmartRTL::wait_cleanup_run() // check if return path is computed and if yes, begin journey home if (g2.smart_rtl.request_thorough_cleanup()) { - path_follow_last_pop_fail_ms = 0; + path_follow_last_semtake_fail_ms = 0; smart_rtl_state = SubMode::PATH_FOLLOW; } } @@ -83,42 +83,45 @@ void ModeSmartRTL::path_follow_run() { // if we are close to current target point, switch the next point to be our target. if (wp_nav->reached_wp_destination()) { - Vector3f dest_NED; - // this pop_point can fail if the IO task currently has the - // path semaphore. - if (g2.smart_rtl.pop_point(dest_NED)) { - path_follow_last_pop_fail_ms = 0; - if (g2.smart_rtl.get_num_points() == 0) { - // this is the very last point, add 2m to the target alt and move to pre-land state - dest_NED.z -= 2.0f; + HAL_Semaphore &sem = g2.smart_rtl.path_semaphore(); + if (sem.take_nonblocking()) { + Vector3f dest_NED; + const uint32_t num_points = g2.smart_rtl.get_num_points(); + if (num_points == 0) { + // we don't even have a home point set in + // SmartRTL... how did we manage to get here? + INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control); smart_rtl_state = SubMode::PRELAND_POSITION; - wp_nav->set_wp_destination_NED(dest_NED); - } else { - // peek at the next point. this can fail if the IO task currently has the path semaphore - Vector3f next_dest_NED; - if (g2.smart_rtl.peek_point(next_dest_NED)) { + } else if (num_points == 1) { + // we don't remove the home point + if (g2.smart_rtl.peek_point(dest_NED)) { + // this is the very last point, add 2m to the target alt + dest_NED.z -= 2.0f; + smart_rtl_state = SubMode::PRELAND_POSITION; wp_nav->set_wp_destination_NED(dest_NED); - if (g2.smart_rtl.get_num_points() == 1) { - // this is the very last point, add 2m to the target alt - next_dest_NED.z -= 2.0f; - } - wp_nav->set_wp_destination_next_NED(next_dest_NED); } else { - // this can only happen if peek failed to take the semaphore - // send next point anyway which will cause the vehicle to slow at the next point + // we're holding the semaphore lock and we checked + // the number of points - so we should *always* be + // able to get an item: + INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control); + smart_rtl_state = SubMode::PRELAND_POSITION; + } + } else { + if (g2.smart_rtl.pop_point(dest_NED)) { wp_nav->set_wp_destination_NED(dest_NED); + } else { + // we're holding the semaphore lock and we checked + // the number of points - so we should *always* be + // able to get an item: INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control); + smart_rtl_state = SubMode::PRELAND_POSITION; } } - } else if (g2.smart_rtl.get_num_points() == 0) { - // We should never get here; should always have at least - // two points and the "zero points left" is handled above. - INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control); - smart_rtl_state = SubMode::PRELAND_POSITION; - } else if (path_follow_last_pop_fail_ms == 0) { - // first time we've failed to pop off (ever, or after a success) - path_follow_last_pop_fail_ms = AP_HAL::millis(); - } else if (AP_HAL::millis() - path_follow_last_pop_fail_ms > 10000) { + sem.give(); + } else if (path_follow_last_semtake_fail_ms == 0) { + // first time we've failed to take the semaphore + path_follow_last_semtake_fail_ms = AP_HAL::millis(); + } else if (AP_HAL::millis() - path_follow_last_semtake_fail_ms > 10000) { // we failed to pop a point off for 10 seconds. This is // almost certainly a bug. INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);