Copter: guided supports using wpnav to reach position targets

This commit is contained in:
Randy Mackay 2021-09-07 21:30:25 +09:00 committed by Andrew Tridgell
parent 53e767ee77
commit c71b64af9d
4 changed files with 146 additions and 11 deletions

View File

@ -138,6 +138,7 @@ void GCS_MAVLINK_Copter::send_position_target_local_ned()
return; return;
case ModeGuided::SubMode::TakeOff: case ModeGuided::SubMode::TakeOff:
case ModeGuided::SubMode::WP: case ModeGuided::SubMode::WP:
case ModeGuided::SubMode::Pos:
type_mask = POSITION_TARGET_TYPEMASK_VX_IGNORE | POSITION_TARGET_TYPEMASK_VY_IGNORE | POSITION_TARGET_TYPEMASK_VZ_IGNORE | type_mask = POSITION_TARGET_TYPEMASK_VX_IGNORE | POSITION_TARGET_TYPEMASK_VY_IGNORE | POSITION_TARGET_TYPEMASK_VZ_IGNORE |
POSITION_TARGET_TYPEMASK_AX_IGNORE | POSITION_TARGET_TYPEMASK_AY_IGNORE | POSITION_TARGET_TYPEMASK_AZ_IGNORE | POSITION_TARGET_TYPEMASK_AX_IGNORE | POSITION_TARGET_TYPEMASK_AY_IGNORE | POSITION_TARGET_TYPEMASK_AZ_IGNORE |
POSITION_TARGET_TYPEMASK_FORCE_SET | POSITION_TARGET_TYPEMASK_YAW_IGNORE| POSITION_TARGET_TYPEMASK_YAW_RATE_IGNORE; // ignore everything except position POSITION_TARGET_TYPEMASK_FORCE_SET | POSITION_TARGET_TYPEMASK_YAW_IGNORE| POSITION_TARGET_TYPEMASK_YAW_RATE_IGNORE; // ignore everything except position

View File

@ -1017,7 +1017,7 @@ const AP_Param::GroupInfo ParametersG2::var_info[] = {
// @Param: GUID_OPTIONS // @Param: GUID_OPTIONS
// @DisplayName: Guided mode options // @DisplayName: Guided mode options
// @Description: Options that can be applied to change guided mode behaviour // @Description: Options that can be applied to change guided mode behaviour
// @Bitmask: 0:Allow Arming from Transmitter,2:Ignore pilot yaw,3:SetAttitudeTarget interprets Thrust As Thrust,4:Do not stabilize PositionXY,5:Do not stabilize VelocityXY // @Bitmask: 0:Allow Arming from Transmitter,2:Ignore pilot yaw,3:SetAttitudeTarget interprets Thrust As Thrust,4:Do not stabilize PositionXY,5:Do not stabilize VelocityXY,6:Waypoint navigation used for position targets
// @User: Advanced // @User: Advanced
AP_GROUPINFO("GUID_OPTIONS", 41, ParametersG2, guided_options, 0), AP_GROUPINFO("GUID_OPTIONS", 41, ParametersG2, guided_options, 0),
#endif #endif

View File

@ -866,6 +866,7 @@ public:
bool set_attitude_target_provides_thrust() const; bool set_attitude_target_provides_thrust() const;
bool stabilizing_pos_xy() const; bool stabilizing_pos_xy() const;
bool stabilizing_vel_xy() const; bool stabilizing_vel_xy() const;
bool use_wpnav_for_position_control() const;
void limit_clear(); void limit_clear();
void limit_init_time_and_pos(); void limit_init_time_and_pos();
@ -879,6 +880,7 @@ public:
enum class SubMode { enum class SubMode {
TakeOff, TakeOff,
WP, WP,
Pos,
PosVelAccel, PosVelAccel,
VelAccel, VelAccel,
Accel, Accel,
@ -912,8 +914,13 @@ private:
SetAttitudeTarget_ThrustAsThrust = (1U << 3), SetAttitudeTarget_ThrustAsThrust = (1U << 3),
DoNotStabilizePositionXY = (1U << 4), DoNotStabilizePositionXY = (1U << 4),
DoNotStabilizeVelocityXY = (1U << 5), DoNotStabilizeVelocityXY = (1U << 5),
WPNavUsedForPosControl = (1U << 6),
}; };
// wp controller
void wp_control_start();
void wp_control_run();
void pva_control_start(); void pva_control_start();
void pos_control_start(); void pos_control_start();
void accel_control_start(); void accel_control_start();

View File

@ -61,14 +61,19 @@ void ModeGuided::run()
break; break;
case SubMode::WP: case SubMode::WP:
// run position controller // run waypoint controller
pos_control_run(); wp_control_run();
if (send_notification && wp_nav->reached_wp_destination()) { if (send_notification && wp_nav->reached_wp_destination()) {
send_notification = false; send_notification = false;
gcs().send_mission_item_reached_message(0); gcs().send_mission_item_reached_message(0);
} }
break; break;
case SubMode::Pos:
// run position controller
pos_control_run();
break;
case SubMode::Accel: case SubMode::Accel:
accel_control_run(); accel_control_run();
break; break;
@ -136,6 +141,68 @@ bool ModeGuided::do_user_takeoff_start(float takeoff_alt_cm)
return true; return true;
} }
// initialise guided mode's waypoint navigation controller
void ModeGuided::wp_control_start()
{
// set to position control mode
guided_mode = SubMode::WP;
// initialise waypoint and spline controller
wp_nav->wp_and_spline_init();
// initialise wpnav to stopping point
Vector3f stopping_point;
wp_nav->get_wp_stopping_point(stopping_point);
// no need to check return status because terrain data is not used
wp_nav->set_wp_destination(stopping_point, false);
// initialise yaw
auto_yaw.set_mode_to_default(false);
}
// run guided mode's waypoint navigation controller
void ModeGuided::wp_control_run()
{
// process pilot's yaw input
float target_yaw_rate = 0;
if (!copter.failsafe.radio && use_pilot_yaw()) {
// get pilot's desired yaw rate
target_yaw_rate = get_pilot_desired_yaw_rate(channel_yaw->get_control_in());
if (!is_zero(target_yaw_rate)) {
auto_yaw.set_mode(AUTO_YAW_HOLD);
}
}
// if not armed set throttle to zero and exit immediately
if (is_disarmed_or_landed()) {
// do not spool down tradheli when on the ground with motor interlock enabled
make_safe_ground_handling(copter.is_tradheli() && motors->get_interlock());
return;
}
// set motors to full range
motors->set_desired_spool_state(AP_Motors::DesiredSpoolState::THROTTLE_UNLIMITED);
// run waypoint controller
copter.failsafe_terrain_set_status(wp_nav->update_wpnav());
// call z-axis position controller (wpnav should have already updated it's alt target)
pos_control->update_z_controller();
// call attitude controller
if (auto_yaw.mode() == AUTO_YAW_HOLD) {
// roll & pitch from waypoint controller, yaw rate from pilot
attitude_control->input_thrust_vector_rate_heading(wp_nav->get_thrust_vector(), target_yaw_rate);
} else if (auto_yaw.mode() == AUTO_YAW_RATE) {
// roll & pitch from waypoint controller, yaw rate from mavlink command or mission item
attitude_control->input_thrust_vector_rate_heading(wp_nav->get_thrust_vector(), auto_yaw.rate_cds());
} else {
// roll, pitch from waypoint controller, yaw heading from GCS or auto_heading()
attitude_control->input_thrust_vector_heading(wp_nav->get_thrust_vector(), auto_yaw.yaw());
}
}
// initialise position controller // initialise position controller
void ModeGuided::pva_control_start() void ModeGuided::pva_control_start()
{ {
@ -162,7 +229,7 @@ void ModeGuided::pva_control_start()
void ModeGuided::pos_control_start() void ModeGuided::pos_control_start()
{ {
// set to position control mode // set to position control mode
guided_mode = SubMode::WP; guided_mode = SubMode::Pos;
// initialise position controller // initialise position controller
pva_control_start(); pva_control_start();
@ -246,8 +313,28 @@ bool ModeGuided::set_destination(const Vector3f& destination, bool use_yaw, floa
} }
#endif #endif
// if configured to use wpnav for position control
if (use_wpnav_for_position_control()) {
// ensure we are in position control mode
if (guided_mode != SubMode::WP) {
wp_control_start();
}
// set yaw state
set_yaw_state(use_yaw, yaw_cd, use_yaw_rate, yaw_rate_cds, relative_yaw);
// no need to check return status because terrain data is not used
wp_nav->set_wp_destination(destination, terrain_alt);
// log target
copter.Log_Write_GuidedTarget(guided_mode, destination, terrain_alt, Vector3f(), Vector3f());
send_notification = true;
return true;
}
// if configured to use position controller for position control
// ensure we are in position control mode // ensure we are in position control mode
if (guided_mode != SubMode::WP) { if (guided_mode != SubMode::Pos) {
pos_control_start(); pos_control_start();
} }
@ -289,11 +376,18 @@ bool ModeGuided::set_destination(const Vector3f& destination, bool use_yaw, floa
bool ModeGuided::get_wp(Location& destination) bool ModeGuided::get_wp(Location& destination)
{ {
if (guided_mode != SubMode::WP) { switch (guided_mode) {
case SubMode::WP:
return wp_nav->get_oa_wp_destination(destination);
case SubMode::Pos:
destination = Location(guided_pos_target_cm.tofloat(), guided_pos_terrain_alt ? Location::AltFrame::ABOVE_TERRAIN : Location::AltFrame::ABOVE_ORIGIN);
return true;
default:
return false; return false;
} }
destination = Location(guided_pos_target_cm.tofloat(), guided_pos_terrain_alt ? Location::AltFrame::ABOVE_TERRAIN : Location::AltFrame::ABOVE_ORIGIN);
return true; // should never get here but just in case
return false;
} }
// sets guided mode's target from a Location object // sets guided mode's target from a Location object
@ -311,8 +405,31 @@ bool ModeGuided::set_destination(const Location& dest_loc, bool use_yaw, float y
} }
#endif #endif
// if using wpnav for position control
if (use_wpnav_for_position_control()) {
if (guided_mode != SubMode::WP) {
wp_control_start();
}
if (!wp_nav->set_wp_destination_loc(dest_loc)) {
// failure to set destination can only be because of missing terrain data
AP::logger().Write_Error(LogErrorSubsystem::NAVIGATION, LogErrorCode::FAILED_TO_SET_DESTINATION);
// failure is propagated to GCS with NAK
return false;
}
// set yaw state
set_yaw_state(use_yaw, yaw_cd, use_yaw_rate, yaw_rate_cds, relative_yaw);
// log target
copter.Log_Write_GuidedTarget(guided_mode, Vector3f(dest_loc.lat, dest_loc.lng, dest_loc.alt), (dest_loc.get_alt_frame() == Location::AltFrame::ABOVE_TERRAIN), Vector3f(), Vector3f());
send_notification = true;
return true;
}
// if configured to use position controller for position control
// ensure we are in position control mode // ensure we are in position control mode
if (guided_mode != SubMode::WP) { if (guided_mode != SubMode::Pos) {
pos_control_start(); pos_control_start();
} }
@ -467,6 +584,12 @@ bool ModeGuided::stabilizing_vel_xy() const
return !((copter.g2.guided_options.get() & uint32_t(Options::DoNotStabilizeVelocityXY)) != 0); return !((copter.g2.guided_options.get() & uint32_t(Options::DoNotStabilizeVelocityXY)) != 0);
} }
// returns true if GUIDED_OPTIONS param specifies waypoint navigation should be used for position control (allow path planning to be used but updates must be slower)
bool ModeGuided::use_wpnav_for_position_control() const
{
return ((copter.g2.guided_options.get() & uint32_t(Options::WPNavUsedForPosControl)) != 0);
}
// set guided mode angle target and climbrate // set guided mode angle target and climbrate
void ModeGuided::set_angle(const Quaternion &q, float climb_rate_cms_or_thrust, bool use_yaw_rate, float yaw_rate_rads, bool use_thrust) void ModeGuided::set_angle(const Quaternion &q, float climb_rate_cms_or_thrust, bool use_yaw_rate, float yaw_rate_rads, bool use_thrust)
{ {
@ -988,8 +1111,9 @@ uint32_t ModeGuided::wp_distance() const
{ {
switch(guided_mode) { switch(guided_mode) {
case SubMode::WP: case SubMode::WP:
return wp_nav->get_wp_distance_to_destination();
case SubMode::Pos:
return norm(guided_pos_target_cm.x - inertial_nav.get_position().x, guided_pos_target_cm.y - inertial_nav.get_position().y); return norm(guided_pos_target_cm.x - inertial_nav.get_position().x, guided_pos_target_cm.y - inertial_nav.get_position().y);
break;
case SubMode::PosVelAccel: case SubMode::PosVelAccel:
return pos_control->get_pos_error_xy_cm(); return pos_control->get_pos_error_xy_cm();
break; break;
@ -1002,8 +1126,9 @@ int32_t ModeGuided::wp_bearing() const
{ {
switch(guided_mode) { switch(guided_mode) {
case SubMode::WP: case SubMode::WP:
return wp_nav->get_wp_bearing_to_destination();
case SubMode::Pos:
return get_bearing_cd(inertial_nav.get_position(), guided_pos_target_cm.tofloat()); return get_bearing_cd(inertial_nav.get_position(), guided_pos_target_cm.tofloat());
break;
case SubMode::PosVelAccel: case SubMode::PosVelAccel:
return pos_control->get_bearing_to_target_cd(); return pos_control->get_bearing_to_target_cd();
break; break;
@ -1022,6 +1147,8 @@ float ModeGuided::crosstrack_error() const
{ {
switch (guided_mode) { switch (guided_mode) {
case SubMode::WP: case SubMode::WP:
return wp_nav->crosstrack_error();
case SubMode::Pos:
case SubMode::TakeOff: case SubMode::TakeOff:
case SubMode::Accel: case SubMode::Accel:
case SubMode::VelAccel: case SubMode::VelAccel: