mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-05 07:28:29 -04:00
Plane: cope with overshoot in POSITION1 VTOL land state
when we overshoot run a simple position controller that tries to point the nose at the landing point and aims for the position2 speed threshold
This commit is contained in:
parent
f57e3668e3
commit
a1e7072cb1
@ -2231,12 +2231,13 @@ void QuadPlane::poscontrol_init_approach(void)
|
|||||||
*/
|
*/
|
||||||
void QuadPlane::log_QPOS(void)
|
void QuadPlane::log_QPOS(void)
|
||||||
{
|
{
|
||||||
AP::logger().WriteStreaming("QPOS", "TimeUS,State,Dist,TSpd,TAcc", "QBfff",
|
AP::logger().WriteStreaming("QPOS", "TimeUS,State,Dist,TSpd,TAcc,OShoot", "QBfffB",
|
||||||
AP_HAL::micros64(),
|
AP_HAL::micros64(),
|
||||||
poscontrol.get_state(),
|
poscontrol.get_state(),
|
||||||
plane.auto_state.wp_distance,
|
plane.auto_state.wp_distance,
|
||||||
poscontrol.target_speed,
|
poscontrol.target_speed,
|
||||||
poscontrol.target_accel);
|
poscontrol.target_accel,
|
||||||
|
poscontrol.overshoot);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2274,6 +2275,7 @@ void QuadPlane::PosControlState::set_state(enum position_control_state s)
|
|||||||
state = s;
|
state = s;
|
||||||
qp.log_QPOS();
|
qp.log_QPOS();
|
||||||
last_log_ms = now;
|
last_log_ms = now;
|
||||||
|
overshoot = false;
|
||||||
}
|
}
|
||||||
last_state_change_ms = now;
|
last_state_change_ms = now;
|
||||||
|
|
||||||
@ -2495,6 +2497,7 @@ void QuadPlane::vtol_position_controller(void)
|
|||||||
const float distance = diff_wp.length();
|
const float distance = diff_wp.length();
|
||||||
const Vector2f rel_groundspeed_vector = landing_closing_velocity();
|
const Vector2f rel_groundspeed_vector = landing_closing_velocity();
|
||||||
const float rel_groundspeed_sq = rel_groundspeed_vector.length_squared();
|
const float rel_groundspeed_sq = rel_groundspeed_vector.length_squared();
|
||||||
|
const float closing_groundspeed = rel_groundspeed_vector * diff_wp.normalized();
|
||||||
|
|
||||||
// calculate speed we should be at to reach the position2
|
// calculate speed we should be at to reach the position2
|
||||||
// target speed at the position2 distance threshold, assuming
|
// target speed at the position2 distance threshold, assuming
|
||||||
@ -2530,11 +2533,31 @@ void QuadPlane::vtol_position_controller(void)
|
|||||||
|
|
||||||
Vector2f target_speed_xy_cms;
|
Vector2f target_speed_xy_cms;
|
||||||
Vector2f target_accel_cms;
|
Vector2f target_accel_cms;
|
||||||
const float target_accel = accel_needed(distance, rel_groundspeed_sq);
|
bool have_target_yaw = false;
|
||||||
|
float target_yaw_deg;
|
||||||
|
const float target_accel = accel_needed(distance, sq(closing_groundspeed));
|
||||||
if (distance > 0.1) {
|
if (distance > 0.1) {
|
||||||
Vector2f diff_wp_norm = diff_wp.normalized();
|
Vector2f diff_wp_norm = diff_wp.normalized();
|
||||||
target_speed_xy_cms = diff_wp_norm * target_speed * 100;
|
target_speed_xy_cms = diff_wp_norm * target_speed * 100;
|
||||||
target_accel_cms = diff_wp_norm * (-target_accel*100);
|
target_accel_cms = diff_wp_norm * (-target_accel*100);
|
||||||
|
target_yaw_deg = degrees(diff_wp_norm.angle());
|
||||||
|
const float yaw_err_deg = wrap_180(target_yaw_deg - degrees(plane.ahrs.yaw));
|
||||||
|
bool overshoot = (closing_groundspeed < 0 || fabsf(yaw_err_deg) > 60);
|
||||||
|
if (overshoot && !poscontrol.overshoot) {
|
||||||
|
gcs().send_text(MAV_SEVERITY_INFO,"VTOL Overshoot d=%.1f cs=%.1f yerr=%.1f",
|
||||||
|
distance, closing_groundspeed, yaw_err_deg);
|
||||||
|
poscontrol.overshoot = true;
|
||||||
|
}
|
||||||
|
if (poscontrol.overshoot) {
|
||||||
|
/* we have overshot the landing point or our nose is
|
||||||
|
off by more than 60 degrees. Zero target accel and
|
||||||
|
point nose at the landing point. Set target speed
|
||||||
|
to our position2 threshold speed
|
||||||
|
*/
|
||||||
|
target_accel_cms.zero();
|
||||||
|
target_speed_xy_cms = diff_wp_norm * position2_target_speed * 100;
|
||||||
|
have_target_yaw = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
const float target_speed_ms = target_speed_xy_cms.length() * 0.01;
|
const float target_speed_ms = target_speed_xy_cms.length() * 0.01;
|
||||||
|
|
||||||
@ -2579,10 +2602,17 @@ void QuadPlane::vtol_position_controller(void)
|
|||||||
|
|
||||||
// call attitude controller
|
// call attitude controller
|
||||||
disable_yaw_rate_time_constant();
|
disable_yaw_rate_time_constant();
|
||||||
|
if (have_target_yaw) {
|
||||||
|
attitude_control->input_euler_angle_roll_pitch_yaw(plane.nav_roll_cd,
|
||||||
|
plane.nav_pitch_cd,
|
||||||
|
target_yaw_deg*100, true);
|
||||||
|
} else {
|
||||||
attitude_control->input_euler_angle_roll_pitch_euler_rate_yaw(plane.nav_roll_cd,
|
attitude_control->input_euler_angle_roll_pitch_euler_rate_yaw(plane.nav_roll_cd,
|
||||||
plane.nav_pitch_cd,
|
plane.nav_pitch_cd,
|
||||||
desired_auto_yaw_rate_cds() + get_weathervane_yaw_rate_cds());
|
desired_auto_yaw_rate_cds() + get_weathervane_yaw_rate_cds());
|
||||||
if ((plane.auto_state.wp_distance < position2_dist_threshold) && tiltrotor.tilt_angle_achieved()) {
|
}
|
||||||
|
if ((plane.auto_state.wp_distance < position2_dist_threshold) && tiltrotor.tilt_angle_achieved() &&
|
||||||
|
fabsf(rel_groundspeed_sq) < sq(3*position2_target_speed)) {
|
||||||
// if continuous tiltrotor only advance to position 2 once tilts have finished moving
|
// if continuous tiltrotor only advance to position 2 once tilts have finished moving
|
||||||
poscontrol.set_state(QPOS_POSITION2);
|
poscontrol.set_state(QPOS_POSITION2);
|
||||||
poscontrol.pilot_correction_done = false;
|
poscontrol.pilot_correction_done = false;
|
||||||
|
@ -483,6 +483,7 @@ private:
|
|||||||
float target_speed;
|
float target_speed;
|
||||||
float target_accel;
|
float target_accel;
|
||||||
uint32_t last_pos_reset_ms;
|
uint32_t last_pos_reset_ms;
|
||||||
|
bool overshoot;
|
||||||
private:
|
private:
|
||||||
uint32_t last_state_change_ms;
|
uint32_t last_state_change_ms;
|
||||||
enum position_control_state state;
|
enum position_control_state state;
|
||||||
|
Loading…
Reference in New Issue
Block a user