#include "mode.h" #include "Rover.h" bool ModeSmartRTL::_enter() { // SmartRTL requires EKF (not DCM) Location ekf_origin; if (!ahrs.get_origin(ekf_origin)) { return false; } // refuse to enter SmartRTL if smart RTL's home has not been set if (!g2.smart_rtl.is_active()) { return false; } // RTL never reverses rover.set_reverse(false); // init state smart_rtl_state = SmartRTL_WaitForPathCleanup; _reached_destination = false; return true; } void ModeSmartRTL::update() { switch (smart_rtl_state) { case SmartRTL_WaitForPathCleanup: // check if return path is computed and if yes, begin journey home if (g2.smart_rtl.request_thorough_cleanup()) { smart_rtl_state = SmartRTL_PathFollow; _load_point = true; } // Note: this may lead to an unnecessary 20ms slow down of the vehicle (but it is unlikely) stop_vehicle(); break; case SmartRTL_PathFollow: // load point if required if (_load_point) { Vector3f next_point; if (!g2.smart_rtl.pop_point(next_point)) { // if not more points, we have reached home gcs().send_text(MAV_SEVERITY_INFO, "Reached destination"); smart_rtl_state = SmartRTL_StopAtHome; break; } _load_point = false; // set target destination to new point if (!set_desired_location_NED(next_point)) { gcs().send_text(MAV_SEVERITY_INFO, "SmartRTL: failed to set destination"); smart_rtl_state = SmartRTL_Failure; } } // check if we've reached the next point _distance_to_destination = get_distance(rover.current_loc, _destination); if (_distance_to_destination <= rover.g.waypoint_radius || location_passed_point(rover.current_loc, _origin, _destination)) { _load_point = true; } // continue driving towards destination calc_steering_to_waypoint(_origin, _destination); calc_throttle(calc_reduced_speed_for_turn_or_distance(_desired_speed), true); break; case SmartRTL_StopAtHome: case SmartRTL_Failure: _reached_destination = true; stop_vehicle(); break; } } // save current position for use by the smart_rtl flight mode void ModeSmartRTL::save_position(bool save_pos) { g2.smart_rtl.update(true, save_pos); }