mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-18 06:38:29 -04:00
Copter: add chase mode
This commit is contained in:
parent
0840f0e8ae
commit
199455dc56
@ -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
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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 {
|
||||||
|
@ -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;
|
||||||
|
@ -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
134
ArduCopter/mode_chase.cpp
Normal 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();
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user