3 #if MODE_SMARTRTL_ENABLED
6 * Init and run calls for Smart_RTL flight mode
8 * This code uses the SmartRTL path that is already in memory, and feeds it into WPNav, one point at a time.
9 * Once the copter is close to home, it will run a standard land controller.
12 bool ModeSmartRTL::init(bool ignore_checks
)
14 if (g2
.smart_rtl
.is_active()) {
15 // initialise waypoint and spline controller
16 wp_nav
->wp_and_spline_init();
18 // set current target to a reasonable stopping point
19 Vector3p stopping_point
;
20 pos_control
->get_stopping_point_xy_cm(stopping_point
.xy());
21 pos_control
->get_stopping_point_z_cm(stopping_point
.z
);
22 wp_nav
->set_wp_destination(stopping_point
.tofloat());
24 // initialise yaw to obey user parameter
25 auto_yaw
.set_mode_to_default(true);
27 // wait for cleanup of return path
28 smart_rtl_state
= SubMode::WAIT_FOR_PATH_CLEANUP
;
35 // perform cleanup required when leaving smart_rtl
36 void ModeSmartRTL::exit()
38 // restore last point if we hadn't reached it
39 if (smart_rtl_state
== SubMode::PATH_FOLLOW
&& !dest_NED_backup
.is_zero()) {
40 if (!g2
.smart_rtl
.add_point(dest_NED_backup
)) {
41 GCS_SEND_TEXT(MAV_SEVERITY_ERROR
, "SmartRTL: lost one point");
44 dest_NED_backup
.zero();
46 g2
.smart_rtl
.cancel_request_for_thorough_cleanup();
49 void ModeSmartRTL::run()
51 switch (smart_rtl_state
) {
52 case SubMode::WAIT_FOR_PATH_CLEANUP
:
55 case SubMode::PATH_FOLLOW
:
58 case SubMode::PRELAND_POSITION
:
59 pre_land_position_run();
61 case SubMode::DESCEND
:
62 descent_run(); // Re-using the descend method from normal rtl mode.
65 land_run(true); // Re-using the land method from normal rtl mode.
70 bool ModeSmartRTL::is_landing() const
72 return smart_rtl_state
== SubMode::LAND
;
75 void ModeSmartRTL::wait_cleanup_run()
77 // hover at current target position
78 motors
->set_desired_spool_state(AP_Motors::DesiredSpoolState::THROTTLE_UNLIMITED
);
79 wp_nav
->update_wpnav();
80 pos_control
->update_z_controller();
81 attitude_control
->input_thrust_vector_heading(pos_control
->get_thrust_vector(), auto_yaw
.get_heading());
83 // check if return path is computed and if yes, begin journey home
84 if (g2
.smart_rtl
.request_thorough_cleanup()) {
85 path_follow_last_pop_fail_ms
= 0;
86 smart_rtl_state
= SubMode::PATH_FOLLOW
;
90 void ModeSmartRTL::path_follow_run()
92 // if we are close to current target point, switch the next point to be our target.
93 if (wp_nav
->reached_wp_destination()) {
95 // clear destination backup so that it cannot be restored
96 dest_NED_backup
.zero();
98 // this pop_point can fail if the IO task currently has the
101 if (g2
.smart_rtl
.pop_point(dest_NED
)) {
102 // backup destination in case we exit smart_rtl mode and need to restore it to the path
103 dest_NED_backup
= dest_NED
;
104 path_follow_last_pop_fail_ms
= 0;
105 if (g2
.smart_rtl
.get_num_points() == 0) {
106 // this is the very last point, add 2m to the target alt and move to pre-land state
108 smart_rtl_state
= SubMode::PRELAND_POSITION
;
109 wp_nav
->set_wp_destination_NED(dest_NED
);
111 // peek at the next point. this can fail if the IO task currently has the path semaphore
112 Vector3f next_dest_NED
;
113 if (g2
.smart_rtl
.peek_point(next_dest_NED
)) {
114 wp_nav
->set_wp_destination_NED(dest_NED
);
115 if (g2
.smart_rtl
.get_num_points() == 1) {
116 // this is the very last point, add 2m to the target alt
117 next_dest_NED
.z
-= 2.0f
;
119 wp_nav
->set_wp_destination_next_NED(next_dest_NED
);
121 // this can only happen if peek failed to take the semaphore
122 // send next point anyway which will cause the vehicle to slow at the next point
123 wp_nav
->set_wp_destination_NED(dest_NED
);
124 INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control
);
127 } else if (g2
.smart_rtl
.get_num_points() == 0) {
128 // We should never get here; should always have at least
129 // two points and the "zero points left" is handled above.
130 INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control
);
131 smart_rtl_state
= SubMode::PRELAND_POSITION
;
132 } else if (path_follow_last_pop_fail_ms
== 0) {
133 // first time we've failed to pop off (ever, or after a success)
134 path_follow_last_pop_fail_ms
= AP_HAL::millis();
135 } else if (AP_HAL::millis() - path_follow_last_pop_fail_ms
> 10000) {
136 // we failed to pop a point off for 10 seconds. This is
137 // almost certainly a bug.
138 INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control
);
139 smart_rtl_state
= SubMode::PRELAND_POSITION
;
143 // update controllers
144 motors
->set_desired_spool_state(AP_Motors::DesiredSpoolState::THROTTLE_UNLIMITED
);
145 wp_nav
->update_wpnav();
146 pos_control
->update_z_controller();
148 // call attitude controller with auto yaw
149 attitude_control
->input_thrust_vector_heading(pos_control
->get_thrust_vector(), auto_yaw
.get_heading());
152 void ModeSmartRTL::pre_land_position_run()
154 // if we are close to 2m above start point, we are ready to land.
155 if (wp_nav
->reached_wp_destination()) {
156 // choose descend and hold, or land based on user parameter rtl_alt_final
157 if (g
.rtl_alt_final
<= 0 || copter
.failsafe
.radio
) {
159 smart_rtl_state
= SubMode::LAND
;
161 set_descent_target_alt(copter
.g
.rtl_alt_final
);
163 smart_rtl_state
= SubMode::DESCEND
;
167 // update controllers
168 motors
->set_desired_spool_state(AP_Motors::DesiredSpoolState::THROTTLE_UNLIMITED
);
169 wp_nav
->update_wpnav();
170 pos_control
->update_z_controller();
171 attitude_control
->input_thrust_vector_heading(pos_control
->get_thrust_vector(), auto_yaw
.get_heading());
174 // save current position for use by the smart_rtl flight mode
175 void ModeSmartRTL::save_position()
177 const bool should_save_position
= motors
->armed() && (copter
.flightmode
->mode_number() != Mode::Number::SMART_RTL
);
179 copter
.g2
.smart_rtl
.update(copter
.position_ok(), should_save_position
);
182 bool ModeSmartRTL::get_wp(Location
& destination
) const
184 // provide target in states which use wp_nav
185 switch (smart_rtl_state
) {
186 case SubMode::WAIT_FOR_PATH_CLEANUP
:
187 case SubMode::PATH_FOLLOW
:
188 case SubMode::PRELAND_POSITION
:
189 case SubMode::DESCEND
:
190 return wp_nav
->get_wp_destination_loc(destination
);
195 // we should never get here but just in case
199 uint32_t ModeSmartRTL::wp_distance() const
201 return wp_nav
->get_wp_distance_to_destination();
204 int32_t ModeSmartRTL::wp_bearing() const
206 return wp_nav
->get_wp_bearing_to_destination();
209 bool ModeSmartRTL::use_pilot_yaw() const
211 const bool land_repositioning
= g
.land_repositioning
&& (smart_rtl_state
== SubMode::DESCEND
);
212 const bool final_landing
= smart_rtl_state
== SubMode::LAND
;
213 return g2
.smart_rtl
.use_pilot_yaw() || land_repositioning
|| final_landing
;