2014-01-24 02:47:42 -04:00
|
|
|
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
|
|
|
|
// counter to verify landings
|
|
|
|
static uint16_t land_detector;
|
|
|
|
static bool land_with_gps;
|
|
|
|
|
2014-07-06 06:16:27 -03:00
|
|
|
static uint32_t land_start_time;
|
|
|
|
static bool land_pause;
|
|
|
|
|
2014-01-24 02:47:42 -04:00
|
|
|
// land_init - initialise land controller
|
|
|
|
static bool land_init(bool ignore_checks)
|
|
|
|
{
|
|
|
|
// check if we have GPS and decide which LAND we're going to do
|
|
|
|
land_with_gps = GPS_ok();
|
|
|
|
if (land_with_gps) {
|
2014-01-25 00:41:17 -04:00
|
|
|
// set target to stopping point
|
|
|
|
Vector3f stopping_point;
|
|
|
|
wp_nav.get_loiter_stopping_point_xy(stopping_point);
|
2014-05-15 10:19:18 -03:00
|
|
|
wp_nav.init_loiter_target(stopping_point);
|
2014-01-24 02:47:42 -04:00
|
|
|
}
|
2014-02-15 04:37:24 -04:00
|
|
|
|
2014-04-29 23:17:59 -03:00
|
|
|
// 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());
|
2014-04-30 04:29:32 -03:00
|
|
|
|
|
|
|
// initialise altitude target to stopping point
|
|
|
|
pos_control.set_target_to_stopping_point_z();
|
|
|
|
|
2014-07-06 06:16:27 -03:00
|
|
|
land_start_time = millis();
|
|
|
|
|
|
|
|
land_pause = false;
|
|
|
|
|
2014-01-24 02:47:42 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// land_run - runs the land controller
|
|
|
|
// should be called at 100hz or more
|
|
|
|
static void land_run()
|
|
|
|
{
|
2014-01-24 03:23:33 -04:00
|
|
|
if (land_with_gps) {
|
|
|
|
land_gps_run();
|
|
|
|
}else{
|
|
|
|
land_nogps_run();
|
|
|
|
}
|
|
|
|
}
|
2014-01-24 02:47:42 -04:00
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// land_run - runs the land controller
|
|
|
|
// horizontal position controlled with loiter controller
|
|
|
|
// should be called at 100hz or more
|
|
|
|
static void land_gps_run()
|
|
|
|
{
|
|
|
|
int16_t roll_control = 0, pitch_control = 0;
|
|
|
|
float target_yaw_rate = 0;
|
|
|
|
|
|
|
|
// if not auto armed or landed set throttle to zero and exit immediately
|
|
|
|
if(!ap.auto_armed || ap.land_complete) {
|
2014-06-06 00:04:34 -03:00
|
|
|
attitude_control.relax_bf_rate_controller();
|
2014-06-06 02:45:49 -03:00
|
|
|
attitude_control.set_yaw_target_to_current_heading();
|
2014-01-24 02:47:42 -04:00
|
|
|
attitude_control.set_throttle_out(0, false);
|
2014-01-24 03:23:33 -04:00
|
|
|
wp_nav.init_loiter_target();
|
2014-01-24 02:47:42 -04:00
|
|
|
|
|
|
|
#if LAND_REQUIRE_MIN_THROTTLE_TO_DISARM == ENABLED
|
2014-01-24 03:23:33 -04:00
|
|
|
// disarm when the landing detector says we've landed and throttle is at minimum
|
|
|
|
if (ap.land_complete && (g.rc_3.control_in == 0 || failsafe.radio)) {
|
|
|
|
init_disarm_motors();
|
|
|
|
}
|
2014-01-24 02:47:42 -04:00
|
|
|
#else
|
2014-01-24 03:23:33 -04:00
|
|
|
// disarm when the landing detector says we've landed
|
|
|
|
if (ap.land_complete) {
|
|
|
|
init_disarm_motors();
|
|
|
|
}
|
|
|
|
#endif
|
2014-01-24 02:47:42 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// process pilot inputs
|
|
|
|
if (!failsafe.radio) {
|
2014-07-06 06:05:43 -03:00
|
|
|
if (g.land_repositioning) {
|
|
|
|
// apply SIMPLE mode transform to pilot inputs
|
|
|
|
update_simple_mode();
|
2014-01-24 03:23:33 -04:00
|
|
|
|
2014-07-06 06:05:43 -03:00
|
|
|
// process pilot's roll and pitch input
|
|
|
|
roll_control = g.rc_1.control_in;
|
|
|
|
pitch_control = g.rc_2.control_in;
|
|
|
|
}
|
2014-01-24 03:23:33 -04:00
|
|
|
|
|
|
|
// get pilot's desired yaw rate
|
|
|
|
target_yaw_rate = get_pilot_desired_yaw_rate(g.rc_4.control_in);
|
|
|
|
}
|
|
|
|
|
|
|
|
// process roll, pitch inputs
|
|
|
|
wp_nav.set_pilot_desired_acceleration(roll_control, pitch_control);
|
2014-01-24 02:47:42 -04:00
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// run loiter controller
|
|
|
|
wp_nav.update_loiter();
|
2014-01-24 02:47:42 -04:00
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// call attitude controller
|
2014-02-13 22:56:46 -04:00
|
|
|
attitude_control.angle_ef_roll_pitch_rate_ef_yaw(wp_nav.get_roll(), wp_nav.get_pitch(), target_yaw_rate);
|
2014-01-24 02:47:42 -04:00
|
|
|
|
2014-07-06 06:16:27 -03: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_throttle_land();
|
|
|
|
}
|
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// update altitude target and call position controller
|
2014-07-06 06:16:27 -03:00
|
|
|
pos_control.set_alt_target_from_climb_rate(cmb_rate, G_Dt);
|
2014-01-24 03:23:33 -04:00
|
|
|
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
|
|
|
|
static void land_nogps_run()
|
|
|
|
{
|
|
|
|
int16_t target_roll = 0, target_pitch = 0;
|
|
|
|
float target_yaw_rate = 0;
|
|
|
|
|
|
|
|
// if not auto armed or landed set throttle to zero and exit immediately
|
|
|
|
if(!ap.auto_armed || ap.land_complete) {
|
2014-06-06 00:04:34 -03:00
|
|
|
attitude_control.relax_bf_rate_controller();
|
2014-06-06 02:45:49 -03:00
|
|
|
attitude_control.set_yaw_target_to_current_heading();
|
2014-01-24 02:47:42 -04:00
|
|
|
attitude_control.set_throttle_out(0, false);
|
2014-01-24 03:23:33 -04:00
|
|
|
#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 && (g.rc_3.control_in == 0 || failsafe.radio)) {
|
|
|
|
init_disarm_motors();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// disarm when the landing detector says we've landed
|
|
|
|
if (ap.land_complete) {
|
|
|
|
init_disarm_motors();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// process pilot inputs
|
|
|
|
if (!failsafe.radio) {
|
2014-07-06 06:05:43 -03:00
|
|
|
if (g.land_repositioning) {
|
|
|
|
// apply SIMPLE mode transform to pilot inputs
|
|
|
|
update_simple_mode();
|
2014-01-24 02:47:42 -04:00
|
|
|
|
2014-07-06 06:05:43 -03:00
|
|
|
// get pilot desired lean angles
|
|
|
|
get_pilot_desired_lean_angles(g.rc_1.control_in, g.rc_2.control_in, target_roll, target_pitch);
|
|
|
|
}
|
2014-01-24 02:47:42 -04:00
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// get pilot's desired yaw rate
|
|
|
|
target_yaw_rate = get_pilot_desired_yaw_rate(g.rc_4.control_in);
|
2014-01-24 02:47:42 -04:00
|
|
|
}
|
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// call attitude controller
|
2014-02-19 07:52:00 -04:00
|
|
|
attitude_control.angle_ef_roll_pitch_rate_ef_yaw_smooth(target_roll, target_pitch, target_yaw_rate, get_smoothing_gain());
|
2014-01-24 03:23:33 -04:00
|
|
|
|
2014-07-06 06:16:27 -03: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_throttle_land();
|
|
|
|
}
|
|
|
|
|
2014-01-24 03:23:33 -04:00
|
|
|
// call position controller
|
2014-07-06 06:16:27 -03:00
|
|
|
pos_control.set_alt_target_from_climb_rate(cmb_rate, G_Dt);
|
2014-01-24 03:23:33 -04:00
|
|
|
pos_control.update_z_controller();
|
2014-01-24 02:47:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// get_throttle_land - 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
|
|
|
|
static float get_throttle_land()
|
|
|
|
{
|
2014-07-08 02:41:39 -03:00
|
|
|
#if CONFIG_SONAR == ENABLED
|
|
|
|
bool sonar_ok = sonar_enabled && sonar.healthy();
|
|
|
|
#else
|
|
|
|
bool sonar_ok = false;
|
|
|
|
#endif
|
2014-01-24 02:47:42 -04:00
|
|
|
// if we are above 10m and the sonar does not sense anything perform regular alt hold descent
|
2014-07-08 02:41:39 -03:00
|
|
|
if (current_loc.alt >= LAND_START_ALT && !(sonar_ok && sonar_alt_health >= SONAR_ALT_HEALTH_MAX)) {
|
2014-01-24 02:47:42 -04:00
|
|
|
return pos_control.get_speed_down();
|
|
|
|
}else{
|
|
|
|
return -abs(g.land_speed);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// update_land_detector - checks if we have landed and updates the ap.land_complete flag
|
|
|
|
// returns true if we have landed
|
|
|
|
static bool update_land_detector()
|
|
|
|
{
|
|
|
|
// detect whether we have landed by watching for low climb rate and minimum throttle
|
2014-05-28 18:59:36 -03:00
|
|
|
if (abs(climb_rate) < 40 && motors.limit.throttle_lower) {
|
2014-01-24 02:47:42 -04:00
|
|
|
if (!ap.land_complete) {
|
|
|
|
// run throttle controller if accel based throttle controller is enabled and active (active means it has been given a target)
|
|
|
|
if( land_detector < LAND_DETECTOR_TRIGGER) {
|
|
|
|
land_detector++;
|
|
|
|
}else{
|
|
|
|
set_land_complete(true);
|
|
|
|
land_detector = 0;
|
|
|
|
}
|
|
|
|
}
|
2014-07-17 06:16:55 -03:00
|
|
|
} else if ((motors.get_throttle_out() >= get_non_takeoff_throttle()) || failsafe.radio) {
|
2014-01-24 02:47:42 -04:00
|
|
|
// we've sensed movement up or down so reset land_detector
|
|
|
|
land_detector = 0;
|
|
|
|
if(ap.land_complete) {
|
|
|
|
set_land_complete(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return current state of landing
|
|
|
|
return ap.land_complete;
|
|
|
|
}
|
2014-04-27 03:08:58 -03: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
|
2014-04-27 03:38:56 -03:00
|
|
|
static void land_do_not_use_GPS()
|
2014-04-27 03:08:58 -03:00
|
|
|
{
|
|
|
|
land_with_gps = false;
|
|
|
|
}
|
2014-07-06 06:16:27 -03:00
|
|
|
|
|
|
|
// set_mode_land_with_pause - sets mode to LAND and triggers 4 second delay before descent starts
|
|
|
|
static void set_mode_land_with_pause()
|
|
|
|
{
|
|
|
|
set_mode(LAND);
|
|
|
|
land_pause = true;
|
|
|
|
}
|