mirror of
https://github.com/PX4/PX4-Autopilot.git
synced 2026-06-01 02:55:07 +08:00
Merge remote-tracking branch 'private_julian/fw_autoland_att_tecs_navigator' into fw_autoland_att_tecs_navigator
This commit is contained in:
@@ -141,6 +141,7 @@ private:
|
|||||||
int _vstatus_sub; /**< vehicle status subscription */
|
int _vstatus_sub; /**< vehicle status subscription */
|
||||||
int _params_sub; /**< notification of parameter updates */
|
int _params_sub; /**< notification of parameter updates */
|
||||||
int _mission_sub; /**< notification of mission updates */
|
int _mission_sub; /**< notification of mission updates */
|
||||||
|
int _onboard_mission_sub; /**< notification of onboard mission updates */
|
||||||
int _capabilities_sub; /**< notification of vehicle capabilities updates */
|
int _capabilities_sub; /**< notification of vehicle capabilities updates */
|
||||||
|
|
||||||
orb_advert_t _triplet_pub; /**< publish position setpoint triplet */
|
orb_advert_t _triplet_pub; /**< publish position setpoint triplet */
|
||||||
@@ -156,9 +157,12 @@ private:
|
|||||||
perf_counter_t _loop_perf; /**< loop performance counter */
|
perf_counter_t _loop_perf; /**< loop performance counter */
|
||||||
|
|
||||||
unsigned _max_mission_item_count; /**< maximum number of mission items supported */
|
unsigned _max_mission_item_count; /**< maximum number of mission items supported */
|
||||||
|
unsigned _max_onboard_mission_item_count;/**< maximum number of onboard mission items supported */
|
||||||
|
|
||||||
unsigned _mission_item_count; /** number of mission items copied */
|
unsigned _mission_item_count; /** number of mission items copied */
|
||||||
|
unsigned _onboard_mission_item_count; /** number of onboard mission items copied */
|
||||||
struct mission_item_s *_mission_item; /**< storage for mission */
|
struct mission_item_s *_mission_item; /**< storage for mission */
|
||||||
|
struct mission_item_s *_onboard_mission_item; /**< storage for onboard mission */
|
||||||
|
|
||||||
struct fence_s _fence; /**< storage for fence vertices */
|
struct fence_s _fence; /**< storage for fence vertices */
|
||||||
bool _fence_valid; /**< flag if fence is valid */
|
bool _fence_valid; /**< flag if fence is valid */
|
||||||
@@ -170,18 +174,22 @@ private:
|
|||||||
bool _waypoint_yaw_reached;
|
bool _waypoint_yaw_reached;
|
||||||
uint64_t _time_first_inside_orbit;
|
uint64_t _time_first_inside_orbit;
|
||||||
bool _mission_item_reached;
|
bool _mission_item_reached;
|
||||||
|
bool _onboard_mission_item_reached;
|
||||||
|
|
||||||
navigation_mode_t _mode;
|
navigation_mode_t _mode;
|
||||||
unsigned _current_mission_index;
|
unsigned _current_mission_index;
|
||||||
|
unsigned _current_onboard_mission_index;
|
||||||
|
|
||||||
struct {
|
struct {
|
||||||
float min_altitude;
|
float min_altitude;
|
||||||
float loiter_radius;
|
float loiter_radius;
|
||||||
|
int onboard_mission_enabled;
|
||||||
} _parameters; /**< local copies of parameters */
|
} _parameters; /**< local copies of parameters */
|
||||||
|
|
||||||
struct {
|
struct {
|
||||||
param_t min_altitude;
|
param_t min_altitude;
|
||||||
param_t loiter_radius;
|
param_t loiter_radius;
|
||||||
|
param_t onboard_mission_enabled;
|
||||||
|
|
||||||
} _parameter_handles; /**< handles for parameters */
|
} _parameter_handles; /**< handles for parameters */
|
||||||
|
|
||||||
@@ -196,6 +204,11 @@ private:
|
|||||||
*/
|
*/
|
||||||
void mission_update();
|
void mission_update();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve onboard mission.
|
||||||
|
*/
|
||||||
|
void onboard_mission_update();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Shim for calling task_main from task_create.
|
* Shim for calling task_main from task_create.
|
||||||
*/
|
*/
|
||||||
@@ -214,6 +227,10 @@ private:
|
|||||||
|
|
||||||
void set_mode(navigation_mode_t new_nav_mode);
|
void set_mode(navigation_mode_t new_nav_mode);
|
||||||
|
|
||||||
|
bool mission_possible();
|
||||||
|
|
||||||
|
bool onboard_mission_possible();
|
||||||
|
|
||||||
int set_waypoint_mission_item(unsigned mission_item_index, struct mission_item_s *new_mission_item);
|
int set_waypoint_mission_item(unsigned mission_item_index, struct mission_item_s *new_mission_item);
|
||||||
|
|
||||||
void publish_mission_item_triplet();
|
void publish_mission_item_triplet();
|
||||||
@@ -268,6 +285,7 @@ Navigator::Navigator() :
|
|||||||
_vstatus_sub(-1),
|
_vstatus_sub(-1),
|
||||||
_params_sub(-1),
|
_params_sub(-1),
|
||||||
_mission_sub(-1),
|
_mission_sub(-1),
|
||||||
|
_onboard_mission_sub(-1),
|
||||||
_capabilities_sub(-1),
|
_capabilities_sub(-1),
|
||||||
|
|
||||||
/* publications */
|
/* publications */
|
||||||
@@ -279,22 +297,28 @@ Navigator::Navigator() :
|
|||||||
_loop_perf(perf_alloc(PC_ELAPSED, "navigator")),
|
_loop_perf(perf_alloc(PC_ELAPSED, "navigator")),
|
||||||
/* states */
|
/* states */
|
||||||
_max_mission_item_count(10),
|
_max_mission_item_count(10),
|
||||||
|
_max_onboard_mission_item_count(10),
|
||||||
_mission_item_count(0),
|
_mission_item_count(0),
|
||||||
|
_onboard_mission_item_count(0),
|
||||||
_fence_valid(false),
|
_fence_valid(false),
|
||||||
_inside_fence(true),
|
_inside_fence(true),
|
||||||
_waypoint_position_reached(false),
|
_waypoint_position_reached(false),
|
||||||
_waypoint_yaw_reached(false),
|
_waypoint_yaw_reached(false),
|
||||||
_time_first_inside_orbit(0),
|
_time_first_inside_orbit(0),
|
||||||
_mission_item_reached(false),
|
_mission_item_reached(false),
|
||||||
|
_onboard_mission_item_reached(false),
|
||||||
_mode(NAVIGATION_MODE_NONE),
|
_mode(NAVIGATION_MODE_NONE),
|
||||||
_current_mission_index(0)
|
_current_mission_index(0),
|
||||||
|
_current_onboard_mission_index(0)
|
||||||
{
|
{
|
||||||
_global_pos.valid = false;
|
_global_pos.valid = false;
|
||||||
memset(&_fence, 0, sizeof(_fence));
|
memset(&_fence, 0, sizeof(_fence));
|
||||||
_parameter_handles.min_altitude = param_find("NAV_MIN_ALT");
|
_parameter_handles.min_altitude = param_find("NAV_MIN_ALT");
|
||||||
_parameter_handles.loiter_radius = param_find("NAV_LOITER_RAD");
|
_parameter_handles.loiter_radius = param_find("NAV_LOITER_RAD");
|
||||||
|
_parameter_handles.onboard_mission_enabled = param_find("NAV_ONB_MIS_EN");
|
||||||
|
|
||||||
_mission_item = (mission_item_s*)malloc(sizeof(mission_item_s) * _max_mission_item_count);
|
_mission_item = (mission_item_s*)malloc(sizeof(mission_item_s) * _max_mission_item_count);
|
||||||
|
_onboard_mission_item = (mission_item_s*)malloc(sizeof(mission_item_s) * _max_onboard_mission_item_count);
|
||||||
|
|
||||||
_mission_item_triplet.previous_valid = false;
|
_mission_item_triplet.previous_valid = false;
|
||||||
_mission_item_triplet.current_valid = false;
|
_mission_item_triplet.current_valid = false;
|
||||||
@@ -340,6 +364,7 @@ Navigator::parameters_update()
|
|||||||
|
|
||||||
param_get(_parameter_handles.min_altitude, &(_parameters.min_altitude));
|
param_get(_parameter_handles.min_altitude, &(_parameters.min_altitude));
|
||||||
param_get(_parameter_handles.loiter_radius, &(_parameters.loiter_radius));
|
param_get(_parameter_handles.loiter_radius, &(_parameters.loiter_radius));
|
||||||
|
param_get(_parameter_handles.onboard_mission_enabled, &(_parameters.onboard_mission_enabled));
|
||||||
|
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
@@ -409,6 +434,71 @@ Navigator::mission_update()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
Navigator::onboard_mission_update()
|
||||||
|
{
|
||||||
|
struct mission_s onboard_mission;
|
||||||
|
if (orb_copy(ORB_ID(onboard_mission), _onboard_mission_sub, &onboard_mission) == OK) {
|
||||||
|
// XXX this is not optimal yet, but a first prototype /
|
||||||
|
// test implementation
|
||||||
|
|
||||||
|
if (onboard_mission.count <= _max_onboard_mission_item_count) {
|
||||||
|
|
||||||
|
/* Check if first part of mission (up to _current_onboard_mission_index - 1) changed:
|
||||||
|
* if the first part changed: start again at first waypoint
|
||||||
|
* if the first part remained unchanged: continue with the (possibly changed second part)
|
||||||
|
*/
|
||||||
|
if (onboard_mission.current_index == -1 && _current_onboard_mission_index < _onboard_mission_item_count && _current_onboard_mission_index < onboard_mission.count) { //check if not finished and if the new mission is not a shorter mission
|
||||||
|
for (unsigned i = 0; i < _current_onboard_mission_index; i++) {
|
||||||
|
if (!cmp_mission_item_equivalent(_onboard_mission_item[i], onboard_mission.items[i])) {
|
||||||
|
/* set flag to restart mission next we're in auto */
|
||||||
|
_current_onboard_mission_index = 0;
|
||||||
|
mavlink_log_info(_mavlink_fd, "[navigator] Reset to onboard WP %d", _current_onboard_mission_index);
|
||||||
|
//warnx("First part of onboard mission differs i=%d", i);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// else {
|
||||||
|
// warnx("Onboard mission item is equivalent i=%d", i);
|
||||||
|
// }
|
||||||
|
}
|
||||||
|
} else if (onboard_mission.current_index >= 0 && onboard_mission.current_index < onboard_mission.count) {
|
||||||
|
/* set flag to restart mission next we're in auto */
|
||||||
|
_current_onboard_mission_index = onboard_mission.current_index;
|
||||||
|
mavlink_log_info(_mavlink_fd, "[navigator] Reset to onboard WP %d", _current_onboard_mission_index);
|
||||||
|
} else {
|
||||||
|
_current_onboard_mission_index = 0;
|
||||||
|
mavlink_log_info(_mavlink_fd, "[navigator] Reset to onboard WP %d", _current_onboard_mission_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Perform an atomic copy & state update
|
||||||
|
*/
|
||||||
|
irqstate_t flags = irqsave();
|
||||||
|
|
||||||
|
memcpy(_onboard_mission_item, onboard_mission.items, onboard_mission.count * sizeof(struct mission_item_s));
|
||||||
|
_onboard_mission_item_count = onboard_mission.count;
|
||||||
|
|
||||||
|
irqrestore(flags);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
} else {
|
||||||
|
warnx("ERROR: too many waypoints, not supported");
|
||||||
|
mavlink_log_critical(_mavlink_fd, "[navigator] too many waypoints, not supported");
|
||||||
|
_onboard_mission_item_count = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_mode == NAVIGATION_MODE_WAYPOINT) {
|
||||||
|
start_waypoint();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* TODO add checks if and how the mission has changed */
|
||||||
|
} else {
|
||||||
|
_onboard_mission_item_count = 0;
|
||||||
|
_current_onboard_mission_index = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
Navigator::task_main_trampoline(int argc, char *argv[])
|
Navigator::task_main_trampoline(int argc, char *argv[])
|
||||||
{
|
{
|
||||||
@@ -428,6 +518,7 @@ Navigator::task_main()
|
|||||||
*/
|
*/
|
||||||
_global_pos_sub = orb_subscribe(ORB_ID(vehicle_global_position));
|
_global_pos_sub = orb_subscribe(ORB_ID(vehicle_global_position));
|
||||||
_mission_sub = orb_subscribe(ORB_ID(mission));
|
_mission_sub = orb_subscribe(ORB_ID(mission));
|
||||||
|
_onboard_mission_sub = orb_subscribe(ORB_ID(onboard_mission));
|
||||||
_capabilities_sub = orb_subscribe(ORB_ID(navigation_capabilities));
|
_capabilities_sub = orb_subscribe(ORB_ID(navigation_capabilities));
|
||||||
_vstatus_sub = orb_subscribe(ORB_ID(vehicle_status));
|
_vstatus_sub = orb_subscribe(ORB_ID(vehicle_status));
|
||||||
_params_sub = orb_subscribe(ORB_ID(parameter_update));
|
_params_sub = orb_subscribe(ORB_ID(parameter_update));
|
||||||
@@ -439,6 +530,7 @@ Navigator::task_main()
|
|||||||
}
|
}
|
||||||
|
|
||||||
mission_update();
|
mission_update();
|
||||||
|
onboard_mission_update();
|
||||||
|
|
||||||
/* rate limit vehicle status updates to 5Hz */
|
/* rate limit vehicle status updates to 5Hz */
|
||||||
orb_set_interval(_vstatus_sub, 200);
|
orb_set_interval(_vstatus_sub, 200);
|
||||||
@@ -453,7 +545,7 @@ Navigator::task_main()
|
|||||||
orb_copy(ORB_ID(navigation_capabilities), _capabilities_sub, &_nav_caps);
|
orb_copy(ORB_ID(navigation_capabilities), _capabilities_sub, &_nav_caps);
|
||||||
|
|
||||||
/* wakeup source(s) */
|
/* wakeup source(s) */
|
||||||
struct pollfd fds[6];
|
struct pollfd fds[7];
|
||||||
|
|
||||||
/* Setup of loop */
|
/* Setup of loop */
|
||||||
fds[0].fd = _params_sub;
|
fds[0].fd = _params_sub;
|
||||||
@@ -466,8 +558,10 @@ Navigator::task_main()
|
|||||||
fds[3].events = POLLIN;
|
fds[3].events = POLLIN;
|
||||||
fds[4].fd = _mission_sub;
|
fds[4].fd = _mission_sub;
|
||||||
fds[4].events = POLLIN;
|
fds[4].events = POLLIN;
|
||||||
fds[5].fd = _vstatus_sub;
|
fds[5].fd = _onboard_mission_sub;
|
||||||
fds[5].events = POLLIN;
|
fds[5].events = POLLIN;
|
||||||
|
fds[6].fd = _vstatus_sub;
|
||||||
|
fds[6].events = POLLIN;
|
||||||
|
|
||||||
|
|
||||||
while (!_task_should_exit) {
|
while (!_task_should_exit) {
|
||||||
@@ -489,7 +583,7 @@ Navigator::task_main()
|
|||||||
perf_begin(_loop_perf);
|
perf_begin(_loop_perf);
|
||||||
|
|
||||||
/* only update vehicle status if it changed */
|
/* only update vehicle status if it changed */
|
||||||
if (fds[5].revents & POLLIN) {
|
if (fds[6].revents & POLLIN) {
|
||||||
/* read from param to clear updated flag */
|
/* read from param to clear updated flag */
|
||||||
orb_copy(ORB_ID(vehicle_status), _vstatus_sub, &_vstatus);
|
orb_copy(ORB_ID(vehicle_status), _vstatus_sub, &_vstatus);
|
||||||
|
|
||||||
@@ -519,8 +613,8 @@ Navigator::task_main()
|
|||||||
|
|
||||||
case NAVIGATION_STATE_AUTO_MISSION:
|
case NAVIGATION_STATE_AUTO_MISSION:
|
||||||
|
|
||||||
if (_mission_item_count > 0 && !(_current_mission_index >= _mission_item_count)) {
|
if (onboard_mission_possible() || mission_possible()) {
|
||||||
/* Start mission if there is a mission available and the last waypoint has not been reached */
|
/* Start mission or onboard mission if available */
|
||||||
set_mode(NAVIGATION_MODE_WAYPOINT);
|
set_mode(NAVIGATION_MODE_WAYPOINT);
|
||||||
} else {
|
} else {
|
||||||
/* else fallback to loiter */
|
/* else fallback to loiter */
|
||||||
@@ -570,6 +664,10 @@ Navigator::task_main()
|
|||||||
mission_update();
|
mission_update();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (fds[5].revents & POLLIN) {
|
||||||
|
onboard_mission_update();
|
||||||
|
}
|
||||||
|
|
||||||
if (fds[2].revents & POLLIN) {
|
if (fds[2].revents & POLLIN) {
|
||||||
orb_copy(ORB_ID(home_position), _home_pos_sub, &_home_pos);
|
orb_copy(ORB_ID(home_position), _home_pos_sub, &_home_pos);
|
||||||
}
|
}
|
||||||
@@ -877,24 +975,60 @@ Navigator::set_mode(navigation_mode_t new_nav_mode)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
bool
|
||||||
Navigator::set_waypoint_mission_item(unsigned mission_item_index, struct mission_item_s *new_mission_item)
|
Navigator::mission_possible()
|
||||||
{
|
{
|
||||||
if (mission_item_index < _mission_item_count) {
|
return _mission_item_count > 0 &&
|
||||||
memcpy(new_mission_item, &_mission_item[mission_item_index], sizeof(mission_item_s));
|
!(_current_mission_index >= _mission_item_count);
|
||||||
|
|
||||||
if (new_mission_item->nav_cmd == NAV_CMD_RETURN_TO_LAUNCH) {
|
|
||||||
/* if it is a RTL waypoint, append the home position */
|
|
||||||
new_mission_item->lat = (double)_home_pos.lat / 1e7;
|
|
||||||
new_mission_item->lon = (double)_home_pos.lon / 1e7;
|
|
||||||
new_mission_item->altitude = (float)_home_pos.alt / 1e3f + _parameters.min_altitude;
|
|
||||||
new_mission_item->loiter_radius = _parameters.loiter_radius; // TODO: get rid of magic number
|
|
||||||
new_mission_item->radius = 50.0f; // TODO: get rid of magic number
|
|
||||||
}
|
}
|
||||||
// warnx("added mission item: %d", mission_item_index);
|
|
||||||
|
bool
|
||||||
|
Navigator::onboard_mission_possible()
|
||||||
|
{
|
||||||
|
return _onboard_mission_item_count > 0 &&
|
||||||
|
!(_current_onboard_mission_index >= _onboard_mission_item_count) &&
|
||||||
|
_parameters.onboard_mission_enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
Navigator::set_waypoint_mission_item(unsigned index, struct mission_item_s *new_item)
|
||||||
|
{
|
||||||
|
if (onboard_mission_possible()) {
|
||||||
|
|
||||||
|
if (index < _onboard_mission_item_count) {
|
||||||
|
memcpy(new_item, &_onboard_mission_item[index], sizeof(mission_item_s));
|
||||||
|
|
||||||
|
if (new_item->nav_cmd == NAV_CMD_RETURN_TO_LAUNCH) {
|
||||||
|
/* if it is a RTL waypoint, append the home position */
|
||||||
|
new_item->lat = (double)_home_pos.lat / 1e7;
|
||||||
|
new_item->lon = (double)_home_pos.lon / 1e7;
|
||||||
|
new_item->altitude = (float)_home_pos.alt / 1e3f + _parameters.min_altitude;
|
||||||
|
new_item->loiter_radius = _parameters.loiter_radius; // TODO: get rid of magic number
|
||||||
|
new_item->radius = 50.0f; // TODO: get rid of magic number
|
||||||
|
}
|
||||||
|
// warnx("added mission item: %d", index);
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
// warnx("could not add mission item: %d", mission_item_index);
|
|
||||||
|
} else if (mission_possible()) {
|
||||||
|
|
||||||
|
if (index < _mission_item_count) {
|
||||||
|
memcpy(new_item, &_mission_item[index], sizeof(mission_item_s));
|
||||||
|
|
||||||
|
if (new_item->nav_cmd == NAV_CMD_RETURN_TO_LAUNCH) {
|
||||||
|
/* if it is a RTL waypoint, append the home position */
|
||||||
|
new_item->lat = (double)_home_pos.lat / 1e7;
|
||||||
|
new_item->lon = (double)_home_pos.lon / 1e7;
|
||||||
|
new_item->altitude = (float)_home_pos.alt / 1e3f + _parameters.min_altitude;
|
||||||
|
new_item->loiter_radius = _parameters.loiter_radius; // TODO: get rid of magic number
|
||||||
|
new_item->radius = 50.0f; // TODO: get rid of magic number
|
||||||
|
}
|
||||||
|
// warnx("added mission item: %d", index);
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// warnx("could not add mission item: %d", index);
|
||||||
return ERROR;
|
return ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -53,3 +53,4 @@
|
|||||||
|
|
||||||
PARAM_DEFINE_FLOAT(NAV_MIN_ALT, 50.0f);
|
PARAM_DEFINE_FLOAT(NAV_MIN_ALT, 50.0f);
|
||||||
PARAM_DEFINE_FLOAT(NAV_LOITER_RAD, 100.0f);
|
PARAM_DEFINE_FLOAT(NAV_LOITER_RAD, 100.0f);
|
||||||
|
PARAM_DEFINE_INT32(NAV_ONB_MIS_EN, 0);
|
||||||
@@ -128,6 +128,7 @@ ORB_DEFINE(vehicle_global_velocity_setpoint, struct vehicle_global_velocity_setp
|
|||||||
|
|
||||||
#include "topics/mission.h"
|
#include "topics/mission.h"
|
||||||
ORB_DEFINE(mission, struct mission_s);
|
ORB_DEFINE(mission, struct mission_s);
|
||||||
|
ORB_DEFINE(onboard_mission, struct mission_s);
|
||||||
|
|
||||||
#include "topics/mission_result.h"
|
#include "topics/mission_result.h"
|
||||||
ORB_DEFINE(mission_result, struct mission_result_s);
|
ORB_DEFINE(mission_result, struct mission_result_s);
|
||||||
|
|||||||
@@ -99,5 +99,6 @@ struct mission_s
|
|||||||
|
|
||||||
/* register this as object request broker structure */
|
/* register this as object request broker structure */
|
||||||
ORB_DECLARE(mission);
|
ORB_DECLARE(mission);
|
||||||
|
ORB_DECLARE(onboard_mission);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
Reference in New Issue
Block a user