Copter: add chase mode

This commit is contained in:
Peter Barker 2017-12-27 10:09:11 +11:00 committed by Randy Mackay
parent 0840f0e8ae
commit 199455dc56
6 changed files with 188 additions and 0 deletions

View File

@ -975,6 +975,7 @@ private:
#if MODE_CIRCLE_ENABLED == ENABLED #if MODE_CIRCLE_ENABLED == ENABLED
ModeCircle mode_circle; ModeCircle mode_circle;
#endif #endif
ModeChase mode_chase;
#if MODE_DRIFT_ENABLED == ENABLED #if MODE_DRIFT_ENABLED == ENABLED
ModeDrift mode_drift; ModeDrift mode_drift;
#endif #endif

View File

@ -47,6 +47,7 @@ NOINLINE void Copter::send_heartbeat(mavlink_channel_t chan)
case RTL: case RTL:
case LOITER: case LOITER:
case AVOID_ADSB: case AVOID_ADSB:
case CHASE:
case GUIDED: case GUIDED:
case CIRCLE: case CIRCLE:
case POSHOLD: case POSHOLD:
@ -695,6 +696,7 @@ void GCS_MAVLINK_Copter::packetReceived(const mavlink_status_t &status,
copter.avoidance_adsb.handle_msg(msg); copter.avoidance_adsb.handle_msg(msg);
} }
#endif #endif
copter.mode_chase.mavlink_packet_received(msg);
GCS_MAVLINK::packetReceived(status, msg); GCS_MAVLINK::packetReceived(status, msg);
} }

View File

@ -109,6 +109,7 @@ enum control_mode_t {
GUIDED_NOGPS = 20, // guided mode but only accepts attitude and altitude GUIDED_NOGPS = 20, // guided mode but only accepts attitude and altitude
SMART_RTL = 21, // SMART_RTL returns to home by retracing its steps SMART_RTL = 21, // SMART_RTL returns to home by retracing its steps
FLOWHOLD = 22, // FLOWHOLD holds position with optical flow without rangefinder FLOWHOLD = 22, // FLOWHOLD holds position with optical flow without rangefinder
CHASE = 23, // chase attempts to follow a mavink system id
}; };
enum mode_reason_t { enum mode_reason_t {

View File

@ -56,6 +56,10 @@ Copter::Mode *Copter::mode_from_mode_num(const uint8_t mode)
break; break;
#endif #endif
case CHASE:
ret = &mode_chase;
break;
#if MODE_CIRCLE_ENABLED == ENABLED #if MODE_CIRCLE_ENABLED == ENABLED
case CIRCLE: case CIRCLE:
ret = &mode_circle; ret = &mode_circle;

View File

@ -1092,3 +1092,49 @@ protected:
private: private:
}; };
class ModeChase : public ModeGuided {
public:
ModeChase(Copter &copter) :
Copter::ModeGuided(copter) {
target_loc.flags.relative_alt = true;
}
bool init(bool ignore_checks) override;
void run() override;
bool requires_GPS() const override { return true; }
bool has_manual_throttle() const override { return false; }
bool allows_arming(bool from_gcs) const override { return false; }
bool is_autopilot() const override { return true; }
bool set_velocity(const Vector3f& velocity_neu);
void mavlink_packet_received(const mavlink_message_t &msg);
protected:
const char *name() const override { return "CHASE"; }
const char *name4() const override { return "CHAS"; }
private:
uint8_t target_srcid = 255;
Location_Class target_loc;
Vector3f target_vel;
uint32_t target_last_update_ms;
const uint16_t target_update_timeout_ms = 1000;
float sphere_radius_min = 5.0f; // no closer than this to target
float sphere_radius_max = 50.0f; // give up when further than this
const float closure_speed = 10.0f; // metres/second
const float distance_slop = 2.0f; // metres
void run_lonely_mode();
Copter::Mode *lonely_mode;
};

134
ArduCopter/mode_chase.cpp Normal file
View File

@ -0,0 +1,134 @@
#include "Copter.h"
/*
* mode_chase.cpp - chase another mavlink-enabled vehicle by system id
*
* TODO: set ROI yaw mode / point camera at target
* TODO: stick control to move around on sphere
* TODO: stick control to change sphere diameter
* TODO: "channel 7 option" to lock onto "pointed at" target
* TODO: do better in terms of loitering around the moving point; may need a PID? Maybe use loiter controller somehow?
* TODO: extrapolate target vehicle position using its velocity and acceleration
*/
#if 1
#define Debug(fmt, args ...) do {::fprintf(stderr, "%s:%d: " fmt "\n", __FUNCTION__, __LINE__, ## args); hal.scheduler->delay(1); } while(0)
#else
#define Debug(fmt, args ...)
#endif
// initialise avoid_adsb controller
bool Copter::ModeChase::init(const bool ignore_checks)
{
// re-use guided mode
return Copter::ModeGuided::init(ignore_checks);
}
bool Copter::ModeChase::set_velocity(const Vector3f& velocity_neu)
{
// check flight mode
if (_copter.flightmode != &_copter.mode_chase) {
return false;
}
return true;
}
void Copter::ModeChase::run_lonely_mode()
{
if (lonely_mode == nullptr) {
if (copter.mode_loiter.init(false)) {
lonely_mode = &copter.mode_loiter;
} else if(copter.mode_rtl.init(false)) {
lonely_mode = &copter.mode_rtl;
} else {
copter.mode_land.init(false);
lonely_mode = &copter.mode_land;
}
gcs().send_text(MAV_SEVERITY_INFO, "Chase: Lonely; %s", lonely_mode->name());
}
lonely_mode->run();
}
void Copter::ModeChase::run()
{
// if not auto armed or motor interlock not enabled set throttle to zero and exit immediately
if (!motors->armed() || !ap.auto_armed || !motors->get_interlock()) {
zero_throttle_and_relax_ac();
return;
}
// re-use guided mode's velocity controller
// Note: this is safe from interference from GCSs and companion computer's whose guided mode
// position and velocity requests will be ignored while the vehicle is not in guided mode
const uint32_t now = AP_HAL::millis();
if (now - target_last_update_ms > target_update_timeout_ms) {
return run_lonely_mode();
}
Vector3f to_vehicle = location_3d_diff_NED(_copter.current_loc, target_loc);
Debug("to_vehicle: %f %f %f", to_vehicle.x, to_vehicle.y, to_vehicle.z);
const float distance_to_vehicle = to_vehicle.length();
if (distance_to_vehicle > sphere_radius_max) {
return run_lonely_mode();
}
lonely_mode = nullptr;
const float distance_to_stop = pos_control->get_stopping_distance_xyz() * 0.01f;
const float distance_to_move = distance_to_vehicle - sphere_radius_min;
Debug("distance_to_vehicle=%f move=%f stop=%f",
distance_to_vehicle,
distance_to_move,
distance_to_stop);
to_vehicle.normalize();
if (fabsf(distance_to_move) > distance_to_stop) {
to_vehicle *= closure_speed * 100; // m/s to cm/s (which set_velocity takes)
to_vehicle.z = -to_vehicle.z; // translate to NEU
if (distance_to_move < 0) {
to_vehicle = -to_vehicle; // too close! back up!
}
} else {
to_vehicle.x = 0;
to_vehicle.y = 0;
to_vehicle.z = 0;
}
to_vehicle += target_vel;
// re-use guided mode's velocity controller (takes NEU)
Copter::ModeGuided::set_velocity(to_vehicle);
Copter::ModeGuided::run();
}
void Copter::ModeChase::mavlink_packet_received(const mavlink_message_t &msg)
{
if (copter.flightmode != &copter.mode_chase) {
return;
}
if (msg.sysid != target_srcid) {
return;
}
if (msg.msgid != MAVLINK_MSG_ID_GLOBAL_POSITION_INT) {
// handle position only for now
return;
}
mavlink_global_position_int_t packet;
mavlink_msg_global_position_int_decode(&msg, &packet);
target_loc.lat = packet.lat;
target_loc.lng = packet.lon;
target_loc.alt = packet.relative_alt / 10; // mm -> cm
target_vel.x = packet.vx/100.0f; // cm/s to m/s
target_vel.y = packet.vy/100.0f; // cm/s to m/s
target_vel.z = packet.vz/100.0f; // cm/s to m/s
target_last_update_ms = AP_HAL::millis();
}