Plane: added APIs for lua ship landing

This commit is contained in:
Andrew Tridgell 2022-01-04 15:53:45 +11:00
parent c54fb0f51b
commit 3fe9a55269
8 changed files with 147 additions and 70 deletions

View File

@ -738,6 +738,7 @@ bool Plane::get_target_location(Location& target_loc)
case Mode::Number::GUIDED: case Mode::Number::GUIDED:
case Mode::Number::AUTO: case Mode::Number::AUTO:
case Mode::Number::LOITER: case Mode::Number::LOITER:
case Mode::Number::TAKEOFF:
#if HAL_QUADPLANE_ENABLED #if HAL_QUADPLANE_ENABLED
case Mode::Number::QLOITER: case Mode::Number::QLOITER:
case Mode::Number::QLAND: case Mode::Number::QLAND:
@ -751,6 +752,39 @@ bool Plane::get_target_location(Location& target_loc)
} }
return false; return false;
} }
/*
update_target_location() works in all auto navigation modes
*/
bool Plane::update_target_location(const Location &old_loc, const Location &new_loc)
{
if (!old_loc.same_latlon_as(next_WP_loc)) {
return false;
}
ftype alt_diff;
if (!old_loc.get_alt_distance(next_WP_loc, alt_diff) ||
!is_zero(alt_diff)) {
return false;
}
next_WP_loc = new_loc;
next_WP_loc.change_alt_frame(old_loc.get_alt_frame());
return true;
}
// allow for velocity matching in VTOL
bool Plane::set_velocity_match(const Vector2f &velocity)
{
#if HAL_QUADPLANE_ENABLED
if (quadplane.in_vtol_mode() || quadplane.in_vtol_land_sequence()) {
quadplane.poscontrol.velocity_match = velocity;
quadplane.poscontrol.last_velocity_match_ms = AP_HAL::millis();
return true;
}
#endif
return false;
}
#endif // AP_SCRIPTING_ENABLED #endif // AP_SCRIPTING_ENABLED
#if OSD_ENABLED #if OSD_ENABLED

View File

@ -653,7 +653,7 @@ void GCS_MAVLINK_Plane::packetReceived(const mavlink_status_t &status,
#if HAL_ADSB_ENABLED #if HAL_ADSB_ENABLED
plane.avoidance_adsb.handle_msg(msg); plane.avoidance_adsb.handle_msg(msg);
#endif #endif
#if SHIP_LANDING_ENABLED #if AP_SCRIPTING_ENABLED
// pass message to follow library // pass message to follow library
plane.g2.follow.handle_msg(msg); plane.g2.follow.handle_msg(msg);
#endif #endif
@ -889,7 +889,7 @@ MAV_RESULT GCS_MAVLINK_Plane::handle_command_int_packet(const mavlink_command_in
return handle_command_int_guided_slew_commands(packet); return handle_command_int_guided_slew_commands(packet);
case MAV_CMD_DO_FOLLOW: case MAV_CMD_DO_FOLLOW:
#if SHIP_LANDING_ENABLED #if AP_SCRIPTING_ENABLED
// param1: sysid of target to follow // param1: sysid of target to follow
if ((packet.param1 > 0) && (packet.param1 <= 255)) { if ((packet.param1 > 0) && (packet.param1 <= 255)) {
plane.g2.follow.set_target_sysid((uint8_t)packet.param1); plane.g2.follow.set_target_sysid((uint8_t)packet.param1);
@ -1099,7 +1099,7 @@ MAV_RESULT GCS_MAVLINK_Plane::handle_command_long_packet(const mavlink_command_l
} }
return MAV_RESULT_ACCEPTED; return MAV_RESULT_ACCEPTED;
#if SHIP_LANDING_ENABLED #if AP_SCRIPTING_ENABLED
case MAV_CMD_DO_FOLLOW: case MAV_CMD_DO_FOLLOW:
// param1: sysid of target to follow // param1: sysid of target to follow
if ((packet.param1 > 0) && (packet.param1 <= 255)) { if ((packet.param1 > 0) && (packet.param1 <= 255)) {

View File

@ -23,7 +23,7 @@ void Plane::Log_Write_Attitude(void)
} else { } else {
ahrs.Write_Attitude(targets); ahrs.Write_Attitude(targets);
} }
if (quadplane.in_vtol_mode() || quadplane.in_assisted_flight()) { if (quadplane.in_vtol_mode() || quadplane.in_assisted_flight() || quadplane.in_vtol_airbrake()) {
// log quadplane PIDs separately from fixed wing PIDs // log quadplane PIDs separately from fixed wing PIDs
logger.Write_PID(LOG_PIQR_MSG, quadplane.attitude_control->get_rate_roll_pid().get_pid_info()); logger.Write_PID(LOG_PIQR_MSG, quadplane.attitude_control->get_rate_roll_pid().get_pid_info());
logger.Write_PID(LOG_PIQP_MSG, quadplane.attitude_control->get_rate_pitch_pid().get_pid_info()); logger.Write_PID(LOG_PIQP_MSG, quadplane.attitude_control->get_rate_pitch_pid().get_pid_info());

View File

@ -1230,7 +1230,7 @@ const AP_Param::GroupInfo ParametersG2::var_info[] = {
// @Bitmask: 0: Servo 1, 1: Servo 2, 2: Servo 3, 3: Servo 4, 4: Servo 5, 5: Servo 6, 6: Servo 7, 7: Servo 8, 8: Servo 9, 9: Servo 10, 10: Servo 11, 11: Servo 12, 12: Servo 13, 13: Servo 14, 14: Servo 15 // @Bitmask: 0: Servo 1, 1: Servo 2, 2: Servo 3, 3: Servo 4, 4: Servo 5, 5: Servo 6, 6: Servo 7, 7: Servo 8, 8: Servo 9, 9: Servo 10, 10: Servo 11, 11: Servo 12, 12: Servo 13, 13: Servo 14, 14: Servo 15
AP_GROUPINFO("ONESHOT_MASK", 32, ParametersG2, oneshot_mask, 0), AP_GROUPINFO("ONESHOT_MASK", 32, ParametersG2, oneshot_mask, 0),
#if SHIP_LANDING_ENABLED #if AP_SCRIPTING_ENABLED
// @Group: FOLL // @Group: FOLL
// @Path: ../libraries/AP_Follow/AP_Follow.cpp // @Path: ../libraries/AP_Follow/AP_Follow.cpp
AP_SUBGROUPINFO(follow, "FOLL", 33, ParametersG2, AP_Follow), AP_SUBGROUPINFO(follow, "FOLL", 33, ParametersG2, AP_Follow),

View File

@ -545,7 +545,7 @@ public:
AC_PID guidedHeading{5000.0, 0.0, 0.0, 0 , 10.0, 5.0, 5.0 , 5.0 , 0.2}; AC_PID guidedHeading{5000.0, 0.0, 0.0, 0 , 10.0, 5.0, 5.0 , 5.0 , 0.2};
#endif #endif
#if SHIP_LANDING_ENABLED #if AP_SCRIPTING_ENABLED
AP_Follow follow; AP_Follow follow;
#endif #endif

View File

@ -1210,6 +1210,8 @@ public:
#if AP_SCRIPTING_ENABLED #if AP_SCRIPTING_ENABLED
bool set_target_location(const Location& target_loc) override; bool set_target_location(const Location& target_loc) override;
bool get_target_location(Location& target_loc) override; bool get_target_location(Location& target_loc) override;
bool update_target_location(const Location &old_loc, const Location &new_loc) override;
bool set_velocity_match(const Vector2f &velocity) override;
#endif // AP_SCRIPTING_ENABLED #endif // AP_SCRIPTING_ENABLED
}; };

View File

@ -1714,7 +1714,7 @@ void QuadPlane::update(void)
} }
const uint32_t now = AP_HAL::millis(); const uint32_t now = AP_HAL::millis();
if (!in_vtol_mode()) { if (!in_vtol_mode() && !in_vtol_airbrake()) {
// we're in a fixed wing mode, cope with transitions and check // we're in a fixed wing mode, cope with transitions and check
// for assistance needed // for assistance needed
if (plane.control_mode == &plane.mode_manual || if (plane.control_mode == &plane.mode_manual ||
@ -1733,7 +1733,7 @@ void QuadPlane::update(void)
} else { } else {
assisted_flight = false; assisted_flight = in_vtol_airbrake();
// output to motors // output to motors
motors_output(); motors_output();
@ -2011,6 +2011,9 @@ bool QuadPlane::in_vtol_auto(void) const
/* /*
are we in a VTOL mode? This is used to decide if we run the are we in a VTOL mode? This is used to decide if we run the
transition handling code or not transition handling code or not
note that AIRBRAKE is not considered in_vtol_mode even though the
VTOL motors are running
*/ */
bool QuadPlane::in_vtol_mode(void) const bool QuadPlane::in_vtol_mode(void) const
{ {
@ -2018,7 +2021,7 @@ bool QuadPlane::in_vtol_mode(void) const
return false; return false;
} }
if (in_vtol_land_sequence()) { if (in_vtol_land_sequence()) {
return poscontrol.get_state() != QPOS_APPROACH; return poscontrol.get_state() != QPOS_APPROACH && poscontrol.get_state() != QPOS_AIRBRAKE;
} }
if (plane.control_mode->is_vtol_mode()) { if (plane.control_mode->is_vtol_mode()) {
return true; return true;
@ -2033,7 +2036,7 @@ bool QuadPlane::in_vtol_mode(void) const
return true; return true;
} }
if (in_vtol_auto()) { if (in_vtol_auto()) {
if (!plane.auto_state.vtol_loiter || poscontrol.get_state() > QPOS_APPROACH) { if (!plane.auto_state.vtol_loiter || poscontrol.get_state() > QPOS_AIRBRAKE) {
return true; return true;
} }
} }
@ -2082,7 +2085,8 @@ void QuadPlane::update_land_positioning(void)
poscontrol.target_vel_cms = Vector3f(-pitch_in, roll_in, 0) * speed_max_cms; poscontrol.target_vel_cms = Vector3f(-pitch_in, roll_in, 0) * speed_max_cms;
poscontrol.target_vel_cms.rotate_xy(ahrs_view->yaw); poscontrol.target_vel_cms.rotate_xy(ahrs_view->yaw);
poscontrol.target_cm += (poscontrol.target_vel_cms * dt).topostype(); // integrate our corrected position
poscontrol.xy_correction += poscontrol.target_vel_cms.xy() * dt * 0.01;
poscontrol.pilot_correction_active = (!is_zero(roll_in) || !is_zero(pitch_in)); poscontrol.pilot_correction_active = (!is_zero(roll_in) || !is_zero(pitch_in));
if (poscontrol.pilot_correction_active) { if (poscontrol.pilot_correction_active) {
@ -2095,9 +2099,16 @@ void QuadPlane::update_land_positioning(void)
*/ */
void QuadPlane::run_xy_controller(void) void QuadPlane::run_xy_controller(void)
{ {
float accel_cmss = wp_nav->get_wp_acceleration();
if (in_vtol_land_sequence() &&
(poscontrol.get_state() == QPOS_POSITION1 ||
poscontrol.get_state() == QPOS_POSITION2)) {
accel_cmss = MAX(accel_cmss, transition_decel*100);
}
const float speed_cms = wp_nav->get_default_speed_xy();
pos_control->set_max_speed_accel_xy(speed_cms, accel_cmss);
pos_control->set_correction_speed_accel_xy(speed_cms, accel_cmss);
if (!pos_control->is_active_xy()) { if (!pos_control->is_active_xy()) {
pos_control->set_max_speed_accel_xy(wp_nav->get_default_speed_xy(), wp_nav->get_wp_acceleration());
pos_control->set_correction_speed_accel_xy(wp_nav->get_default_speed_xy(), wp_nav->get_wp_acceleration());
pos_control->init_xy_controller(); pos_control->init_xy_controller();
} }
pos_control->update_xy_controller(); pos_control->update_xy_controller();
@ -2130,6 +2141,8 @@ void QuadPlane::poscontrol_init_approach(void)
} }
poscontrol.thrust_loss_start_ms = 0; poscontrol.thrust_loss_start_ms = 0;
} }
poscontrol.pilot_correction_done = false;
poscontrol.xy_correction.zero();
} }
/* /*
@ -2159,13 +2172,6 @@ void QuadPlane::PosControlState::set_state(enum position_control_state s)
// if it is active then the rate control should not be reset at all // if it is active then the rate control should not be reset at all
qp.attitude_control->reset_yaw_target_and_rate(false); qp.attitude_control->reset_yaw_target_and_rate(false);
pos1_start_speed = plane.ahrs.groundspeed_vector().length(); pos1_start_speed = plane.ahrs.groundspeed_vector().length();
} else if (s == QPOS_POSITION2) {
// POSITION2 changes target speed, so we need to change it
// back to normal
qp.pos_control->set_max_speed_accel_xy(qp.wp_nav->get_default_speed_xy(),
qp.wp_nav->get_wp_acceleration());
qp.pos_control->set_correction_speed_accel_xy(qp.wp_nav->get_default_speed_xy(),
qp.wp_nav->get_wp_acceleration());
} else if (s == QPOS_AIRBRAKE) { } else if (s == QPOS_AIRBRAKE) {
// start with zero integrator on vertical throttle // start with zero integrator on vertical throttle
qp.pos_control->get_accel_z_pid().set_integrator(0); qp.pos_control->get_accel_z_pid().set_integrator(0);
@ -2212,6 +2218,11 @@ void QuadPlane::vtol_position_controller(void)
// and tilt is more than tilt max // and tilt is more than tilt max
bool suppress_z_controller = false; bool suppress_z_controller = false;
Vector2f landing_velocity;
if (now_ms - poscontrol.last_velocity_match_ms < 1000) {
landing_velocity = poscontrol.velocity_match;
}
// horizontal position control // horizontal position control
switch (poscontrol.get_state()) { switch (poscontrol.get_state()) {
@ -2423,19 +2434,11 @@ void QuadPlane::vtol_position_controller(void)
if (distance > 0.1) { if (distance > 0.1) {
target_speed_xy_cms = diff_wp.normalized() * target_speed * 100; target_speed_xy_cms = diff_wp.normalized() * target_speed * 100;
} }
if (!tailsitter.enabled()) {
// this method ignores pos-control velocity/accel limtis
pos_control->set_vel_desired_xy_cms(target_speed_xy_cms);
// reset position controller xy target to current position target_speed_xy_cms += landing_velocity * 100;
// because we only want velocity control (no position control)
const Vector2f& curr_pos = inertial_nav.get_position_xy_cm(); // use input shaping and abide by accel and jerk limits
pos_control->set_pos_target_xy_cm(curr_pos.x, curr_pos.y); pos_control->input_vel_accel_xy(target_speed_xy_cms, Vector2f());
pos_control->set_accel_desired_xy_cmss(Vector2f());
} else {
// tailsitters use input shaping and abide by velocity limits
pos_control->input_vel_accel_xy(target_speed_xy_cms, Vector2f());
}
// run horizontal velocity controller // run horizontal velocity controller
run_xy_controller(); run_xy_controller();
@ -2465,35 +2468,19 @@ void QuadPlane::vtol_position_controller(void)
case QPOS_POSITION2: case QPOS_POSITION2:
case QPOS_LAND_DESCEND: { case QPOS_LAND_DESCEND: {
setup_target_position();
/* /*
for final land repositioning and descent we run the position controller for final land repositioning and descent we run the position controller
*/ */
if (poscontrol.pilot_correction_done) { Vector2f zero;
// if the pilot has repositioned the vehicle then we switch to velocity control. This prevents the vehicle Vector2f vel_cms = poscontrol.target_vel_cms.xy() + landing_velocity*100;
// shifting position in the event of GPS glitches. pos_control->input_pos_vel_accel_xy(poscontrol.target_cm.xy(), vel_cms, zero);
Vector2f zero;
pos_control->input_vel_accel_xy(poscontrol.target_vel_cms.xy(), zero);
} else {
Vector2f zero;
pos_control->input_pos_vel_accel_xy(poscontrol.target_cm.xy(), zero, zero);
}
// also run fixed wing navigation // also run fixed wing navigation
plane.nav_controller->update_waypoint(plane.current_loc, loc); plane.nav_controller->update_waypoint(plane.current_loc, loc);
update_land_positioning(); update_land_positioning();
/*
apply the same asymmetric speed limits from POSITION1, so we
don't suddenly speed up when we change to POSITION2 and
LAND_DESCEND
*/
const Vector2f diff_wp = plane.current_loc.get_distance_NE(loc);
const float scaled_wp_speed = get_scaled_wp_speed(degrees(diff_wp.angle()));
pos_control->set_max_speed_accel_xy(scaled_wp_speed*100, wp_nav->get_wp_acceleration());
pos_control->set_correction_speed_accel_xy(scaled_wp_speed*100, wp_nav->get_wp_acceleration());
run_xy_controller(); run_xy_controller();
// nav roll and pitch are controlled by position controller // nav roll and pitch are controlled by position controller
@ -2520,7 +2507,8 @@ void QuadPlane::vtol_position_controller(void)
} else { } else {
// we use velocity control in QPOS_LAND_FINAL to allow for GPS glitch handling // we use velocity control in QPOS_LAND_FINAL to allow for GPS glitch handling
Vector2f zero; Vector2f zero;
pos_control->input_vel_accel_xy(poscontrol.target_vel_cms.xy(), zero); Vector2f vel_cms = poscontrol.target_vel_cms.xy() + landing_velocity*100;
pos_control->input_vel_accel_xy(vel_cms, zero);
} }
run_xy_controller(); run_xy_controller();
@ -2681,20 +2669,12 @@ void QuadPlane::setup_target_position(void)
set_desired_spool_state(AP_Motors::DesiredSpoolState::THROTTLE_UNLIMITED); set_desired_spool_state(AP_Motors::DesiredSpoolState::THROTTLE_UNLIMITED);
} }
const Vector2f diff2d = origin.get_distance_NE(loc); Vector2f diff2d = origin.get_distance_NE(loc);
diff2d += poscontrol.xy_correction;
poscontrol.target_cm.x = diff2d.x * 100; poscontrol.target_cm.x = diff2d.x * 100;
poscontrol.target_cm.y = diff2d.y * 100; poscontrol.target_cm.y = diff2d.y * 100;
poscontrol.target_cm.z = plane.next_WP_loc.alt - origin.alt; poscontrol.target_cm.z = plane.next_WP_loc.alt - origin.alt;
const uint32_t now = AP_HAL::millis();
if (!loc.same_latlon_as(last_auto_target) ||
plane.next_WP_loc.alt != last_auto_target.alt ||
now - last_loiter_ms > 500) {
wp_nav->set_wp_destination(poscontrol.target_cm.tofloat());
last_auto_target = loc;
}
last_loiter_ms = now;
// set vertical speed and acceleration limits // set vertical speed and acceleration limits
pos_control->set_max_speed_accel_z(-get_pilot_velocity_z_max_dn(), pilot_velocity_z_max_up, pilot_accel_z); pos_control->set_max_speed_accel_z(-get_pilot_velocity_z_max_dn(), pilot_velocity_z_max_up, pilot_accel_z);
pos_control->set_correction_speed_accel_z(-get_pilot_velocity_z_max_dn(), pilot_velocity_z_max_up, pilot_accel_z); pos_control->set_correction_speed_accel_z(-get_pilot_velocity_z_max_dn(), pilot_velocity_z_max_up, pilot_accel_z);
@ -2705,18 +2685,24 @@ void QuadPlane::setup_target_position(void)
*/ */
void QuadPlane::takeoff_controller(void) void QuadPlane::takeoff_controller(void)
{ {
if (!hal.util->get_soft_armed()) {
return;
}
/* /*
for takeoff we use the position controller for takeoff we use the position controller
*/ */
setup_target_position(); setup_target_position();
// set position controller desired velocity and acceleration to zero
pos_control->set_vel_desired_xy_cms(Vector2f());
pos_control->set_accel_desired_xy_cmss(Vector2f());
// set position control target and update // set position control target and update
Vector2f zero;
pos_control->input_vel_accel_xy(zero, zero); Vector2f vel, zero;
if (AP_HAL::millis() - poscontrol.last_velocity_match_ms < 1000) {
vel = poscontrol.velocity_match * 100;
}
pos_control->set_accel_desired_xy_cmss(zero);
pos_control->set_vel_desired_xy_cms(vel);
pos_control->input_vel_accel_xy(vel, zero);
run_xy_controller(); run_xy_controller();
@ -2757,6 +2743,16 @@ void QuadPlane::waypoint_controller(void)
{ {
setup_target_position(); setup_target_position();
const Location &loc = plane.next_WP_loc;
const uint32_t now = AP_HAL::millis();
if (!loc.same_latlon_as(last_auto_target) ||
plane.next_WP_loc.alt != last_auto_target.alt ||
now - last_loiter_ms > 500) {
wp_nav->set_wp_destination(poscontrol.target_cm.tofloat());
last_auto_target = loc;
}
last_loiter_ms = now;
/* /*
this is full copter control of auto flight this is full copter control of auto flight
*/ */
@ -3086,10 +3082,18 @@ bool QuadPlane::verify_vtol_land(void)
const float dist = (inertial_nav.get_position_neu_cm().topostype() - poscontrol.target_cm).xy().length() * 0.01; const float dist = (inertial_nav.get_position_neu_cm().topostype() - poscontrol.target_cm).xy().length() * 0.01;
reached_position = dist < descend_dist_threshold; reached_position = dist < descend_dist_threshold;
} }
Vector2f target_vel;
if (AP_HAL::millis() - poscontrol.last_velocity_match_ms < 1000) {
target_vel = poscontrol.velocity_match;
}
Vector3f vel_ned;
UNUSED_RESULT(plane.ahrs.get_velocity_NED(vel_ned));
if (reached_position && if (reached_position &&
plane.ahrs.groundspeed() < descend_speed_threshold) { (vel_ned.xy() - target_vel).length() < descend_speed_threshold) {
poscontrol.set_state(QPOS_LAND_DESCEND); poscontrol.set_state(QPOS_LAND_DESCEND);
poscontrol.pilot_correction_done = false; poscontrol.pilot_correction_done = false;
poscontrol.xy_correction.zero();
#if AC_FENCE == ENABLED #if AC_FENCE == ENABLED
plane.fence.auto_disable_fence_for_landing(); plane.fence.auto_disable_fence_for_landing();
#endif #endif
@ -3642,6 +3646,23 @@ bool QuadPlane::in_vtol_land_poscontrol(void) const
return false; return false;
} }
/*
see if we are in the airbrake phase of a VTOL landing
*/
bool QuadPlane::in_vtol_airbrake(void) const
{
if (plane.control_mode == &plane.mode_qrtl &&
poscontrol.get_state() == QPOS_AIRBRAKE) {
return true;
}
if (plane.control_mode == &plane.mode_auto &&
is_vtol_land(plane.mission.get_current_nav_cmd().id) &&
poscontrol.get_state() == QPOS_AIRBRAKE) {
return true;
}
return false;
}
// return true if we should show VTOL view // return true if we should show VTOL view
bool QuadPlane::show_vtol_view() const bool QuadPlane::show_vtol_view() const
{ {
@ -3877,4 +3898,15 @@ MAV_VTOL_STATE SLT_Transition::get_mav_vtol_state() const
return MAV_VTOL_STATE_UNDEFINED; return MAV_VTOL_STATE_UNDEFINED;
} }
/*
see if we are in a VTOL takeoff
*/
bool QuadPlane::in_vtol_takeoff(void) const
{
if (in_vtol_auto() && is_vtol_takeoff(plane.mission.get_current_nav_cmd().id)) {
return true;
}
return false;
}
#endif // HAL_QUADPLANE_ENABLED #endif // HAL_QUADPLANE_ENABLED

View File

@ -108,6 +108,7 @@ public:
bool verify_vtol_land(void); bool verify_vtol_land(void);
bool in_vtol_auto(void) const; bool in_vtol_auto(void) const;
bool in_vtol_mode(void) const; bool in_vtol_mode(void) const;
bool in_vtol_takeoff(void) const;
bool in_vtol_posvel_mode(void) const; bool in_vtol_posvel_mode(void) const;
void update_throttle_hover(); void update_throttle_hover();
bool show_vtol_view() const; bool show_vtol_view() const;
@ -433,6 +434,7 @@ private:
return AP_HAL::millis() - last_state_change_ms; return AP_HAL::millis() - last_state_change_ms;
} }
Vector3p target_cm; Vector3p target_cm;
Vector2f xy_correction;
Vector3f target_vel_cms; Vector3f target_vel_cms;
bool slow_descent:1; bool slow_descent:1;
bool pilot_correction_active; bool pilot_correction_active;
@ -442,6 +444,8 @@ private:
bool reached_wp_speed; bool reached_wp_speed;
uint32_t last_run_ms; uint32_t last_run_ms;
float pos1_start_speed; float pos1_start_speed;
Vector2f velocity_match;
uint32_t last_velocity_match_ms;
private: private:
uint32_t last_state_change_ms; uint32_t last_state_change_ms;
enum position_control_state state; enum position_control_state state;
@ -564,6 +568,11 @@ private:
see if we are in the VTOL position control phase of a landing see if we are in the VTOL position control phase of a landing
*/ */
bool in_vtol_land_poscontrol(void) const; bool in_vtol_land_poscontrol(void) const;
/*
are we in the airbrake phase of a VTOL landing?
*/
bool in_vtol_airbrake(void) const;
// Q assist state, can be enabled, disabled or force. Default to enabled // Q assist state, can be enabled, disabled or force. Default to enabled
Q_ASSIST_STATE_ENUM q_assist_state = Q_ASSIST_STATE_ENUM::Q_ASSIST_ENABLED; Q_ASSIST_STATE_ENUM q_assist_state = Q_ASSIST_STATE_ENUM::Q_ASSIST_ENABLED;