2015-12-30 18:57:56 -04:00
|
|
|
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
|
2016-01-14 15:30:56 -04:00
|
|
|
#include "Sub.h"
|
2015-12-30 18:57:56 -04:00
|
|
|
|
|
|
|
static bool land_with_gps;
|
|
|
|
|
|
|
|
static uint32_t land_start_time;
|
|
|
|
static bool land_pause;
|
|
|
|
|
|
|
|
// land_init - initialise land controller
|
2016-01-14 15:30:56 -04:00
|
|
|
bool Sub::land_init(bool ignore_checks)
|
2015-12-30 18:57:56 -04:00
|
|
|
{
|
|
|
|
// check if we have GPS and decide which LAND we're going to do
|
|
|
|
land_with_gps = position_ok();
|
|
|
|
if (land_with_gps) {
|
|
|
|
// set target to stopping point
|
|
|
|
Vector3f stopping_point;
|
|
|
|
wp_nav.get_loiter_stopping_point_xy(stopping_point);
|
|
|
|
wp_nav.init_loiter_target(stopping_point);
|
|
|
|
}
|
|
|
|
|
|
|
|
// initialize vertical speeds and leash lengths
|
|
|
|
pos_control.set_speed_z(wp_nav.get_speed_down(), wp_nav.get_speed_up());
|
|
|
|
pos_control.set_accel_z(wp_nav.get_accel_z());
|
|
|
|
|
|
|
|
// initialise altitude target to stopping point
|
|
|
|
pos_control.set_target_to_stopping_point_z();
|
|
|
|
|
|
|
|
land_start_time = millis();
|
|
|
|
|
|
|
|
land_pause = false;
|
|
|
|
|
|
|
|
// reset flag indicating if pilot has applied roll or pitch inputs during landing
|
|
|
|
ap.land_repo_active = false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// land_run - runs the land controller
|
|
|
|
// should be called at 100hz or more
|
2016-01-14 15:30:56 -04:00
|
|
|
void Sub::land_run()
|
2015-12-30 18:57:56 -04:00
|
|
|
{
|
|
|
|
if (land_with_gps) {
|
|
|
|
land_gps_run();
|
|
|
|
}else{
|
|
|
|
land_nogps_run();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// land_run - runs the land controller
|
|
|
|
// horizontal position controlled with loiter controller
|
|
|
|
// should be called at 100hz or more
|
2016-01-14 15:30:56 -04:00
|
|
|
void Sub::land_gps_run()
|
2015-12-30 18:57:56 -04:00
|
|
|
{
|
|
|
|
int16_t roll_control = 0, pitch_control = 0;
|
|
|
|
float target_yaw_rate = 0;
|
|
|
|
|
|
|
|
// if not auto armed or landed or motor interlock not enabled set throttle to zero and exit immediately
|
2016-04-05 00:17:39 -03:00
|
|
|
if (!motors.armed() || !ap.auto_armed || ap.land_complete || !motors.get_interlock()) {
|
|
|
|
motors.set_desired_spool_state(AP_Motors::DESIRED_SPIN_WHEN_ARMED);
|
2016-02-23 02:15:15 -04:00
|
|
|
// multicopters do not stabilize roll/pitch/yaw when disarmed
|
2015-12-30 18:57:56 -04:00
|
|
|
attitude_control.set_throttle_out_unstabilized(0,true,g.throttle_filt);
|
2016-02-23 02:15:15 -04:00
|
|
|
|
2015-12-30 18:57:56 -04:00
|
|
|
wp_nav.init_loiter_target();
|
|
|
|
|
|
|
|
#if LAND_REQUIRE_MIN_THROTTLE_TO_DISARM == ENABLED
|
|
|
|
// disarm when the landing detector says we've landed and throttle is at minimum
|
|
|
|
if (ap.land_complete && (ap.throttle_zero || failsafe.radio)) {
|
|
|
|
init_disarm_motors();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// disarm when the landing detector says we've landed
|
|
|
|
if (ap.land_complete) {
|
|
|
|
init_disarm_motors();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// relax loiter target if we might be landed
|
|
|
|
if (ap.land_complete_maybe) {
|
|
|
|
wp_nav.loiter_soften_for_landing();
|
|
|
|
}
|
|
|
|
|
|
|
|
// process pilot inputs
|
|
|
|
if (!failsafe.radio) {
|
2016-04-05 01:44:42 -03:00
|
|
|
if ((g.throttle_behavior & THR_BEHAVE_HIGH_THROTTLE_CANCELS_LAND) != 0 && rc_throttle_control_in_filter.get() > LAND_CANCEL_TRIGGER_THR){
|
|
|
|
Log_Write_Event(DATA_LAND_CANCELLED_BY_PILOT);
|
|
|
|
// exit land if throttle is high
|
2016-04-18 01:38:21 -03:00
|
|
|
if (!set_mode(LOITER, MODE_REASON_THROTTLE_LAND_ESCAPE)) {
|
|
|
|
set_mode(ALT_HOLD, MODE_REASON_THROTTLE_LAND_ESCAPE);
|
2016-04-05 01:44:42 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-30 18:57:56 -04:00
|
|
|
if (g.land_repositioning) {
|
|
|
|
// apply SIMPLE mode transform to pilot inputs
|
|
|
|
update_simple_mode();
|
|
|
|
|
|
|
|
// process pilot's roll and pitch input
|
|
|
|
roll_control = channel_roll->control_in;
|
|
|
|
pitch_control = channel_pitch->control_in;
|
|
|
|
|
|
|
|
// record if pilot has overriden roll or pitch
|
|
|
|
if (roll_control != 0 || pitch_control != 0) {
|
|
|
|
ap.land_repo_active = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// get pilot's desired yaw rate
|
|
|
|
target_yaw_rate = get_pilot_desired_yaw_rate(channel_yaw->control_in);
|
|
|
|
}
|
|
|
|
|
2016-04-05 00:17:39 -03:00
|
|
|
// set motors to full range
|
|
|
|
motors.set_desired_spool_state(AP_Motors::DESIRED_THROTTLE_UNLIMITED);
|
|
|
|
|
2015-12-30 18:57:56 -04:00
|
|
|
// process roll, pitch inputs
|
|
|
|
wp_nav.set_pilot_desired_acceleration(roll_control, pitch_control);
|
|
|
|
|
|
|
|
#if PRECISION_LANDING == ENABLED
|
|
|
|
// run precision landing
|
|
|
|
if (!ap.land_repo_active) {
|
|
|
|
wp_nav.shift_loiter_target(precland.get_target_shift(wp_nav.get_loiter_target()));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// run loiter controller
|
|
|
|
wp_nav.update_loiter(ekfGndSpdLimit, ekfNavVelGainScaler);
|
|
|
|
|
|
|
|
// call attitude controller
|
2016-07-05 21:18:58 -03:00
|
|
|
attitude_control.input_euler_angle_roll_pitch_euler_rate_yaw(wp_nav.get_roll(), wp_nav.get_pitch(), target_yaw_rate, get_smoothing_gain());
|
2015-12-30 18:57:56 -04:00
|
|
|
|
|
|
|
// pause 4 seconds before beginning land descent
|
|
|
|
float cmb_rate;
|
|
|
|
if(land_pause && millis()-land_start_time < 4000) {
|
|
|
|
cmb_rate = 0;
|
|
|
|
} else {
|
|
|
|
land_pause = false;
|
|
|
|
cmb_rate = get_land_descent_speed();
|
|
|
|
}
|
|
|
|
|
|
|
|
// record desired climb rate for logging
|
|
|
|
desired_climb_rate = cmb_rate;
|
|
|
|
|
|
|
|
// update altitude target and call position controller
|
|
|
|
pos_control.set_alt_target_from_climb_rate(cmb_rate, G_Dt, true);
|
|
|
|
pos_control.update_z_controller();
|
|
|
|
}
|
|
|
|
|
|
|
|
// land_nogps_run - runs the land controller
|
|
|
|
// pilot controls roll and pitch angles
|
|
|
|
// should be called at 100hz or more
|
2016-01-14 15:30:56 -04:00
|
|
|
void Sub::land_nogps_run()
|
2015-12-30 18:57:56 -04:00
|
|
|
{
|
|
|
|
float target_roll = 0.0f, target_pitch = 0.0f;
|
|
|
|
float target_yaw_rate = 0;
|
|
|
|
|
|
|
|
// process pilot inputs
|
|
|
|
if (!failsafe.radio) {
|
|
|
|
if (g.land_repositioning) {
|
|
|
|
// apply SIMPLE mode transform to pilot inputs
|
|
|
|
update_simple_mode();
|
|
|
|
|
|
|
|
// get pilot desired lean angles
|
|
|
|
get_pilot_desired_lean_angles(channel_roll->control_in, channel_pitch->control_in, target_roll, target_pitch, aparm.angle_max);
|
|
|
|
}
|
|
|
|
|
|
|
|
// get pilot's desired yaw rate
|
|
|
|
target_yaw_rate = get_pilot_desired_yaw_rate(channel_yaw->control_in);
|
|
|
|
}
|
|
|
|
|
|
|
|
// if not auto armed or landed or motor interlock not enabled set throttle to zero and exit immediately
|
2016-04-05 00:17:39 -03:00
|
|
|
if (!motors.armed() || !ap.auto_armed || ap.land_complete || !motors.get_interlock()) {
|
|
|
|
motors.set_desired_spool_state(AP_Motors::DESIRED_SPIN_WHEN_ARMED);
|
2016-02-23 02:15:15 -04:00
|
|
|
// multicopters do not stabilize roll/pitch/yaw when disarmed
|
2015-12-30 18:57:56 -04:00
|
|
|
attitude_control.set_throttle_out_unstabilized(0,true,g.throttle_filt);
|
|
|
|
|
|
|
|
#if LAND_REQUIRE_MIN_THROTTLE_TO_DISARM == ENABLED
|
|
|
|
// disarm when the landing detector says we've landed and throttle is at minimum
|
|
|
|
if (ap.land_complete && (ap.throttle_zero || failsafe.radio)) {
|
|
|
|
init_disarm_motors();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// disarm when the landing detector says we've landed
|
2016-03-06 18:39:32 -04:00
|
|
|
if (ap.at_surface) {
|
2016-04-18 01:38:21 -03:00
|
|
|
set_mode(ALT_HOLD, MODE_REASON_THROTTLE_LAND_ESCAPE);
|
2015-12-30 18:57:56 -04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-04-05 00:17:39 -03:00
|
|
|
// set motors to full range
|
|
|
|
motors.set_desired_spool_state(AP_Motors::DESIRED_THROTTLE_UNLIMITED);
|
|
|
|
|
2015-12-30 18:57:56 -04:00
|
|
|
// call attitude controller
|
2016-07-05 21:18:58 -03:00
|
|
|
attitude_control.input_euler_angle_roll_pitch_euler_rate_yaw(target_roll, target_pitch, target_yaw_rate, get_smoothing_gain());
|
2015-12-30 18:57:56 -04:00
|
|
|
|
|
|
|
// pause 4 seconds before beginning land descent
|
|
|
|
float cmb_rate;
|
|
|
|
if(land_pause && millis()-land_start_time < LAND_WITH_DELAY_MS) {
|
|
|
|
cmb_rate = 0;
|
|
|
|
} else {
|
|
|
|
land_pause = false;
|
|
|
|
cmb_rate = get_land_descent_speed();
|
|
|
|
}
|
|
|
|
|
|
|
|
// record desired climb rate for logging
|
|
|
|
desired_climb_rate = cmb_rate;
|
|
|
|
|
|
|
|
// call position controller
|
|
|
|
pos_control.set_alt_target_from_climb_rate(cmb_rate, G_Dt, true);
|
|
|
|
pos_control.update_z_controller();
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_land_descent_speed - high level landing logic
|
|
|
|
// returns climb rate (in cm/s) which should be passed to the position controller
|
|
|
|
// should be called at 100hz or higher
|
2016-01-14 15:30:56 -04:00
|
|
|
float Sub::get_land_descent_speed()
|
2015-12-30 18:57:56 -04:00
|
|
|
{
|
2016-05-22 21:50:44 -03:00
|
|
|
#if RANGEFINDER_ENABLED == ENABLED
|
|
|
|
bool rangefinder_ok = rangefinder_state.enabled && rangefinder_state.alt_healthy;
|
2015-12-30 18:57:56 -04:00
|
|
|
#else
|
2016-05-22 21:50:44 -03:00
|
|
|
bool rangefinder_ok = false;
|
2015-12-30 18:57:56 -04:00
|
|
|
#endif
|
2016-05-03 01:45:37 -03:00
|
|
|
|
|
|
|
// get position controller's target altitude above terrain
|
|
|
|
Location_Class target_loc = pos_control.get_pos_target();
|
|
|
|
int32_t target_alt_cm;
|
|
|
|
|
|
|
|
// get altitude target above home by default
|
|
|
|
target_loc.get_alt_cm(Location_Class::ALT_FRAME_ABOVE_HOME, target_alt_cm);
|
|
|
|
|
|
|
|
// try to use terrain if enabled
|
|
|
|
if (terrain_use() && !target_loc.get_alt_cm(Location_Class::ALT_FRAME_ABOVE_TERRAIN, target_alt_cm)) {
|
|
|
|
Log_Write_Error(ERROR_SUBSYSTEM_TERRAIN, ERROR_CODE_MISSING_TERRAIN_DATA);
|
|
|
|
}
|
|
|
|
|
2016-05-22 21:50:44 -03:00
|
|
|
// if we are above 10m and the rangefinder does not sense anything perform regular alt hold descent
|
|
|
|
if ((target_alt_cm >= LAND_START_ALT) && !rangefinder_ok) {
|
2016-04-03 14:19:09 -03:00
|
|
|
if (g.land_speed_high > 0) {
|
|
|
|
// user has asked for a different landing speed than normal descent rate
|
|
|
|
return -abs(g.land_speed_high);
|
|
|
|
}
|
2016-01-23 01:57:40 -04:00
|
|
|
return pos_control.get_speed_up();
|
2015-12-30 18:57:56 -04:00
|
|
|
}else{
|
2016-01-23 01:57:40 -04:00
|
|
|
return abs(g.land_speed);
|
2015-12-30 18:57:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// land_do_not_use_GPS - forces land-mode to not use the GPS but instead rely on pilot input for roll and pitch
|
|
|
|
// called during GPS failsafe to ensure that if we were already in LAND mode that we do not use the GPS
|
|
|
|
// has no effect if we are not already in LAND mode
|
2016-01-14 15:30:56 -04:00
|
|
|
void Sub::land_do_not_use_GPS()
|
2015-12-30 18:57:56 -04:00
|
|
|
{
|
|
|
|
land_with_gps = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// set_mode_land_with_pause - sets mode to LAND and triggers 4 second delay before descent starts
|
|
|
|
// this is always called from a failsafe so we trigger notification to pilot
|
2016-04-18 01:38:21 -03:00
|
|
|
void Sub::set_mode_land_with_pause(mode_reason_t reason)
|
2015-12-30 18:57:56 -04:00
|
|
|
{
|
2016-04-18 01:38:21 -03:00
|
|
|
set_mode(LAND, reason);
|
2015-12-30 18:57:56 -04:00
|
|
|
land_pause = true;
|
|
|
|
|
|
|
|
// alert pilot to mode change
|
|
|
|
AP_Notify::events.failsafe_mode_change = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// landing_with_GPS - returns true if vehicle is landing using GPS
|
2016-01-14 15:30:56 -04:00
|
|
|
bool Sub::landing_with_GPS() {
|
2015-12-30 18:57:56 -04:00
|
|
|
return (control_mode == LAND && land_with_gps);
|
|
|
|
}
|