Merge branch 'navigator_new' into navigator_new_vector

This commit is contained in:
Anton Babushkin 2014-01-02 21:57:33 +04:00
commit 2dc2c2d28f
4 changed files with 251 additions and 122 deletions

View File

@ -220,7 +220,9 @@ get_mavlink_mode_and_state(uint8_t *mavlink_state, uint8_t *mavlink_base_mode, u
} else if (v_status.main_state == MAIN_STATE_AUTO) { } else if (v_status.main_state == MAIN_STATE_AUTO) {
*mavlink_base_mode |= MAV_MODE_FLAG_AUTO_ENABLED | MAV_MODE_FLAG_STABILIZE_ENABLED | MAV_MODE_FLAG_GUIDED_ENABLED; *mavlink_base_mode |= MAV_MODE_FLAG_AUTO_ENABLED | MAV_MODE_FLAG_STABILIZE_ENABLED | MAV_MODE_FLAG_GUIDED_ENABLED;
custom_mode.main_mode = PX4_CUSTOM_MAIN_MODE_AUTO; custom_mode.main_mode = PX4_CUSTOM_MAIN_MODE_AUTO;
if (control_mode.nav_state == NAV_STATE_NONE) { if (control_mode.nav_state == NAV_STATE_NONE) { // failsafe, shouldn't happen
custom_mode.sub_mode = PX4_CUSTOM_SUB_MODE_AUTO_READY;
} else if (control_mode.nav_state == NAV_STATE_READY) {
custom_mode.sub_mode = PX4_CUSTOM_SUB_MODE_AUTO_READY; custom_mode.sub_mode = PX4_CUSTOM_SUB_MODE_AUTO_READY;
} else if (control_mode.nav_state == NAV_STATE_LOITER) { } else if (control_mode.nav_state == NAV_STATE_LOITER) {
custom_mode.sub_mode = PX4_CUSTOM_SUB_MODE_AUTO_LOITER; custom_mode.sub_mode = PX4_CUSTOM_SUB_MODE_AUTO_LOITER;

View File

@ -152,7 +152,7 @@ int map_mavlink_mission_item_to_mission_item(const mavlink_mission_item_t *mavli
mission_item->loiter_direction = (mavlink_mission_item->param3 > 0) ? 1 : -1; /* 1 if positive CW, -1 if negative CCW */ mission_item->loiter_direction = (mavlink_mission_item->param3 > 0) ? 1 : -1; /* 1 if positive CW, -1 if negative CCW */
mission_item->nav_cmd = mavlink_mission_item->command; mission_item->nav_cmd = mavlink_mission_item->command;
mission_item->time_inside = mavlink_mission_item->param1 / 1e3f; /* from milliseconds to seconds */ mission_item->time_inside = mavlink_mission_item->param1;
mission_item->autocontinue = mavlink_mission_item->autocontinue; mission_item->autocontinue = mavlink_mission_item->autocontinue;
// mission_item->index = mavlink_mission_item->seq; // mission_item->index = mavlink_mission_item->seq;
mission_item->origin = ORIGIN_MAVLINK; mission_item->origin = ORIGIN_MAVLINK;
@ -184,7 +184,7 @@ int map_mission_item_to_mavlink_mission_item(const struct mission_item_s *missio
mavlink_mission_item->param4 = mission_item->yaw*M_RAD_TO_DEG_F; mavlink_mission_item->param4 = mission_item->yaw*M_RAD_TO_DEG_F;
mavlink_mission_item->param3 = mission_item->loiter_radius*(float)mission_item->loiter_direction; mavlink_mission_item->param3 = mission_item->loiter_radius*(float)mission_item->loiter_direction;
mavlink_mission_item->command = mission_item->nav_cmd; mavlink_mission_item->command = mission_item->nav_cmd;
mavlink_mission_item->param1 = mission_item->time_inside * 1e3f; /* from seconds to milliseconds */ mavlink_mission_item->param1 = mission_item->time_inside;
mavlink_mission_item->autocontinue = mission_item->autocontinue; mavlink_mission_item->autocontinue = mission_item->autocontinue;
// mavlink_mission_item->seq = mission_item->index; // mavlink_mission_item->seq = mission_item->index;

View File

@ -166,7 +166,6 @@ private:
class Mission _mission; class Mission _mission;
bool _reset_loiter_pos; /**< if true then loiter position should be set to current position */ bool _reset_loiter_pos; /**< if true then loiter position should be set to current position */
bool _rtl_done;
bool _waypoint_position_reached; bool _waypoint_position_reached;
bool _waypoint_yaw_reached; bool _waypoint_yaw_reached;
uint64_t _time_first_inside_orbit; uint64_t _time_first_inside_orbit;
@ -199,10 +198,10 @@ private:
enum Event { enum Event {
EVENT_NONE_REQUESTED, EVENT_NONE_REQUESTED,
EVENT_READY_REQUESTED,
EVENT_LOITER_REQUESTED, EVENT_LOITER_REQUESTED,
EVENT_MISSION_REQUESTED, EVENT_MISSION_REQUESTED,
EVENT_RTL_REQUESTED, EVENT_RTL_REQUESTED,
EVENT_MISSION_FINISHED,
EVENT_MISSION_CHANGED, EVENT_MISSION_CHANGED,
EVENT_HOME_POSITION_CHANGED, EVENT_HOME_POSITION_CHANGED,
MAX_EVENT MAX_EVENT
@ -214,8 +213,10 @@ private:
static StateTable::Tran const myTable[NAV_STATE_MAX][MAX_EVENT]; static StateTable::Tran const myTable[NAV_STATE_MAX][MAX_EVENT];
enum RTLState { enum RTLState {
RTL_STATE_NONE = 0,
RTL_STATE_CLIMB, RTL_STATE_CLIMB,
RTL_STATE_RETURN, RTL_STATE_RETURN,
RTL_STATE_DESCEND,
RTL_STATE_LAND RTL_STATE_LAND
}; };
@ -277,9 +278,9 @@ private:
* Functions that are triggered when a new state is entered. * Functions that are triggered when a new state is entered.
*/ */
void start_none(); void start_none();
void start_ready();
void start_loiter(); void start_loiter();
void start_mission(); void start_mission();
void finish_mission();
void start_rtl(); void start_rtl();
void finish_rtl(); void finish_rtl();
@ -308,6 +309,11 @@ private:
*/ */
void advance_mission(); void advance_mission();
/**
* Switch to next RTL state
*/
void set_rtl_item();
/** /**
* Helper function to get a loiter item * Helper function to get a loiter item
*/ */
@ -380,11 +386,11 @@ Navigator::Navigator() :
_loop_perf(perf_alloc(PC_ELAPSED, "navigator")), _loop_perf(perf_alloc(PC_ELAPSED, "navigator")),
/* states */ /* states */
_rtl_state(RTL_STATE_NONE),
_fence_valid(false), _fence_valid(false),
_inside_fence(true), _inside_fence(true),
_mission(), _mission(),
_reset_loiter_pos(true), _reset_loiter_pos(true),
_rtl_done(false),
_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),
@ -625,13 +631,16 @@ Navigator::task_main()
pub_control_mode = true; pub_control_mode = true;
/* Evaluate state machine from commander and set the navigator mode accordingly */ /* Evaluate state machine from commander and set the navigator mode accordingly */
if (_vstatus.main_state == MAIN_STATE_AUTO) { if (_vstatus.main_state == MAIN_STATE_AUTO &&
(_vstatus.arming_state == ARMING_STATE_ARMED || _vstatus.arming_state == ARMING_STATE_ARMED_ERROR)) {
bool stick_mode = false; bool stick_mode = false;
if (!_vstatus.rc_signal_lost) { if (!_vstatus.rc_signal_lost) {
/* RC signal available, use control switches to set mode */ /* RC signal available, use control switches to set mode */
/* RETURN switch, overrides MISSION switch */ /* RETURN switch, overrides MISSION switch */
if (_vstatus.return_switch == RETURN_SWITCH_RETURN) { if (_vstatus.return_switch == RETURN_SWITCH_RETURN) {
dispatch(_rtl_done ? EVENT_LOITER_REQUESTED : EVENT_RTL_REQUESTED); if (myState != NAV_STATE_READY || _rtl_state != RTL_STATE_LAND) {
dispatch(EVENT_RTL_REQUESTED);
}
stick_mode = true; stick_mode = true;
} else { } else {
/* MISSION switch */ /* MISSION switch */
@ -678,7 +687,9 @@ Navigator::task_main()
break; break;
case NAV_STATE_RTL: case NAV_STATE_RTL:
dispatch(_rtl_done ? EVENT_LOITER_REQUESTED : EVENT_RTL_REQUESTED); if (myState != NAV_STATE_READY || _rtl_state != RTL_STATE_LAND) {
dispatch(EVENT_RTL_REQUESTED);
}
break; break;
default: default:
@ -739,7 +750,7 @@ Navigator::task_main()
/* global position updated */ /* global position updated */
if (fds[1].revents & POLLIN) { if (fds[1].revents & POLLIN) {
global_position_update(); global_position_update();
/* only check if waypoint has been reached in MISSION */ /* only check if waypoint has been reached in MISSION and RTL modes */
if (myState == NAV_STATE_MISSION || myState == NAV_STATE_RTL) { if (myState == NAV_STATE_MISSION || myState == NAV_STATE_RTL) {
if (check_mission_item_reached()) { if (check_mission_item_reached()) {
on_mission_item_reached(); on_mission_item_reached();
@ -747,7 +758,6 @@ Navigator::task_main()
} }
} }
/* notify user about state changes */ /* notify user about state changes */
if (myState != prevState) { if (myState != prevState) {
mavlink_log_info(_mavlink_fd, "[navigator] nav state %d -> %d", prevState, myState); mavlink_log_info(_mavlink_fd, "[navigator] nav state %d -> %d", prevState, myState);
@ -924,40 +934,50 @@ StateTable::Tran const Navigator::myTable[NAV_STATE_MAX][MAX_EVENT] = {
{ {
/* STATE_NONE */ /* STATE_NONE */
/* EVENT_NONE_REQUESTED */ {NO_ACTION, NAV_STATE_NONE}, /* EVENT_NONE_REQUESTED */ {NO_ACTION, NAV_STATE_NONE},
/* EVENT_READY_REQUESTED */ {ACTION(&Navigator::start_ready), NAV_STATE_READY},
/* EVENT_LOITER_REQUESTED */ {ACTION(&Navigator::start_loiter), NAV_STATE_LOITER}, /* EVENT_LOITER_REQUESTED */ {ACTION(&Navigator::start_loiter), NAV_STATE_LOITER},
/* EVENT_MISSION_REQUESTED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION}, /* EVENT_MISSION_REQUESTED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION},
/* EVENT_RTL_REQUESTED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL}, /* EVENT_RTL_REQUESTED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL},
/* EVENT_MISSION_FINISHED */ {NO_ACTION, NAV_STATE_NONE},
/* EVENT_MISSION_CHANGED */ {NO_ACTION, NAV_STATE_NONE}, /* EVENT_MISSION_CHANGED */ {NO_ACTION, NAV_STATE_NONE},
/* EVENT_HOME_POSITION_CHANGED */ {NO_ACTION, NAV_STATE_NONE}, /* EVENT_HOME_POSITION_CHANGED */ {NO_ACTION, NAV_STATE_NONE},
}, },
{ {
/* STATE_READY */
/* EVENT_NONE_REQUESTED */ {NO_ACTION, NAV_STATE_NONE},
/* EVENT_READY_REQUESTED */ {NO_ACTION, NAV_STATE_READY},
/* EVENT_LOITER_REQUESTED */ {NO_ACTION, NAV_STATE_READY},
/* EVENT_MISSION_REQUESTED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION},
/* EVENT_RTL_REQUESTED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL},
/* EVENT_MISSION_CHANGED */ {NO_ACTION, NAV_STATE_READY},
/* EVENT_HOME_POSITION_CHANGED */ {NO_ACTION, NAV_STATE_READY},
},
{
/* STATE_LOITER */ /* STATE_LOITER */
/* EVENT_NONE_REQUESTED */ {ACTION(&Navigator::start_none), NAV_STATE_NONE}, /* EVENT_NONE_REQUESTED */ {ACTION(&Navigator::start_none), NAV_STATE_NONE},
/* EVENT_READY_REQUESTED */ {NO_ACTION, NAV_STATE_LOITER},
/* EVENT_LOITER_REQUESTED */ {NO_ACTION, NAV_STATE_LOITER}, /* EVENT_LOITER_REQUESTED */ {NO_ACTION, NAV_STATE_LOITER},
/* EVENT_MISSION_REQUESTED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION}, /* EVENT_MISSION_REQUESTED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION},
/* EVENT_RTL_REQUESTED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL}, /* EVENT_RTL_REQUESTED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL},
/* EVENT_MISSION_FINISHED */ {NO_ACTION, NAV_STATE_LOITER},
/* EVENT_MISSION_CHANGED */ {NO_ACTION, NAV_STATE_LOITER}, /* EVENT_MISSION_CHANGED */ {NO_ACTION, NAV_STATE_LOITER},
/* EVENT_HOME_POSITION_CHANGED */ {NO_ACTION, NAV_STATE_LOITER}, /* EVENT_HOME_POSITION_CHANGED */ {NO_ACTION, NAV_STATE_LOITER},
}, },
{ {
/* STATE_MISSION */ /* STATE_MISSION */
/* EVENT_NONE_REQUESTED */ {ACTION(&Navigator::start_none), NAV_STATE_NONE}, /* EVENT_NONE_REQUESTED */ {ACTION(&Navigator::start_none), NAV_STATE_NONE},
/* EVENT_READY_REQUESTED */ {ACTION(&Navigator::start_ready), NAV_STATE_READY},
/* EVENT_LOITER_REQUESTED */ {ACTION(&Navigator::start_loiter), NAV_STATE_LOITER}, /* EVENT_LOITER_REQUESTED */ {ACTION(&Navigator::start_loiter), NAV_STATE_LOITER},
/* EVENT_MISSION_REQUESTED */ {NO_ACTION, NAV_STATE_MISSION}, /* EVENT_MISSION_REQUESTED */ {NO_ACTION, NAV_STATE_MISSION},
/* EVENT_RTL_REQUESTED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL}, /* EVENT_RTL_REQUESTED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL},
/* EVENT_MISSION_FINISHED */ {ACTION(&Navigator::finish_mission), NAV_STATE_LOITER},
/* EVENT_MISSION_CHANGED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION}, /* EVENT_MISSION_CHANGED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION},
/* EVENT_HOME_POSITION_CHANGED */ {NO_ACTION, NAV_STATE_MISSION}, /* EVENT_HOME_POSITION_CHANGED */ {NO_ACTION, NAV_STATE_MISSION},
}, },
{ {
/* STATE_RTL */ /* STATE_RTL */
/* EVENT_NONE_REQUESTED */ {ACTION(&Navigator::start_none), NAV_STATE_NONE}, /* EVENT_NONE_REQUESTED */ {ACTION(&Navigator::start_none), NAV_STATE_NONE},
/* EVENT_READY_REQUESTED */ {ACTION(&Navigator::start_ready), NAV_STATE_READY},
/* EVENT_LOITER_REQUESTED */ {ACTION(&Navigator::start_loiter), NAV_STATE_LOITER}, /* EVENT_LOITER_REQUESTED */ {ACTION(&Navigator::start_loiter), NAV_STATE_LOITER},
/* EVENT_MISSION_REQUESTED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION}, /* EVENT_MISSION_REQUESTED */ {ACTION(&Navigator::start_mission), NAV_STATE_MISSION},
/* EVENT_RTL_REQUESTED */ {NO_ACTION, NAV_STATE_RTL}, /* EVENT_RTL_REQUESTED */ {NO_ACTION, NAV_STATE_RTL},
/* EVENT_MISSION_FINISHED */ {ACTION(&Navigator::finish_rtl), NAV_STATE_LOITER},
/* EVENT_MISSION_CHANGED */ {NO_ACTION, NAV_STATE_RTL}, /* EVENT_MISSION_CHANGED */ {NO_ACTION, NAV_STATE_RTL},
/* EVENT_HOME_POSITION_CHANGED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL}, /* EVENT_HOME_POSITION_CHANGED */ {ACTION(&Navigator::start_rtl), NAV_STATE_RTL},
}, },
@ -971,8 +991,26 @@ Navigator::start_none()
_mission_item_triplet.next_valid = false; _mission_item_triplet.next_valid = false;
_reset_loiter_pos = true; _reset_loiter_pos = true;
_rtl_done = false;
_do_takeoff = false; _do_takeoff = false;
_rtl_state = RTL_STATE_NONE;
publish_mission_item_triplet();
}
void
Navigator::start_ready()
{
_mission_item_triplet.previous_valid = false;
_mission_item_triplet.current_valid = false;
_mission_item_triplet.next_valid = false;
_reset_loiter_pos = true;
_do_takeoff = false;
// TODO check
if (_rtl_state != RTL_STATE_LAND) {
_rtl_state = RTL_STATE_NONE;
}
publish_mission_item_triplet(); publish_mission_item_triplet();
} }
@ -1015,9 +1053,9 @@ Navigator::start_loiter()
void void
Navigator::start_mission() Navigator::start_mission()
{ {
_rtl_done = false;
_need_takeoff = true; _need_takeoff = true;
mavlink_log_info(_mavlink_fd, "[navigator] mission started");
advance_mission(); advance_mission();
} }
@ -1041,6 +1079,14 @@ Navigator::advance_mission()
_mission_item_triplet.current_valid = true; _mission_item_triplet.current_valid = true;
add_home_pos_to_rtl(&_mission_item_triplet.current); add_home_pos_to_rtl(&_mission_item_triplet.current);
if (_mission_item_triplet.current.nav_cmd != NAV_CMD_RETURN_TO_LAUNCH &&
_mission_item_triplet.current.nav_cmd != NAV_CMD_LOITER_TIME_LIMIT &&
_mission_item_triplet.current.nav_cmd != NAV_CMD_LOITER_TURN_COUNT &&
_mission_item_triplet.current.nav_cmd != NAV_CMD_LOITER_UNLIMITED) {
/* don't reset RTL state on RTL or LOITER items */
_rtl_state = RTL_STATE_NONE;
}
if (_vstatus.is_rotary_wing) { if (_vstatus.is_rotary_wing) {
if (_need_takeoff && ( if (_need_takeoff && (
_mission_item_triplet.current.nav_cmd == NAV_CMD_TAKEOFF || _mission_item_triplet.current.nav_cmd == NAV_CMD_TAKEOFF ||
@ -1116,56 +1162,128 @@ Navigator::advance_mission()
publish_mission_item_triplet(); publish_mission_item_triplet();
} }
void
Navigator::finish_mission()
{
/* loiter at last waypoint */
_reset_loiter_pos = false;
mavlink_log_info(_mavlink_fd, "[navigator] mission completed");
start_loiter();
}
void void
Navigator::start_rtl() Navigator::start_rtl()
{ {
_reset_loiter_pos = true; _reset_loiter_pos = true;
_rtl_done = false;
_do_takeoff = false; _do_takeoff = false;
if (_rtl_state == RTL_STATE_NONE)
/* discard all mission item and insert RTL item */ _rtl_state = RTL_STATE_CLIMB;
_mission_item_triplet.previous_valid = false;
_mission_item_triplet.current_valid = true;
_mission_item_triplet.next_valid = false;
_mission_item_triplet.current.lat = _home_pos.lat;
_mission_item_triplet.current.lon = _home_pos.lon;
_mission_item_triplet.current.altitude = _home_pos.altitude + _parameters.rtl_alt;
_mission_item_triplet.current.altitude_is_relative = false;
_mission_item_triplet.current.yaw = 0.0f; // TODO use heading to waypoint?
_mission_item_triplet.current.nav_cmd = NAV_CMD_RETURN_TO_LAUNCH;
_mission_item_triplet.current.loiter_direction = 1;
_mission_item_triplet.current.loiter_radius = _parameters.loiter_radius;
_mission_item_triplet.current.acceptance_radius = _parameters.acceptance_radius;
_mission_item_triplet.current.autocontinue = false;
_mission_item_triplet.current_valid = true;
publish_mission_item_triplet();
mavlink_log_info(_mavlink_fd, "[navigator] RTL started"); mavlink_log_info(_mavlink_fd, "[navigator] RTL started");
} set_rtl_item();
}
void void
Navigator::finish_rtl() Navigator::set_rtl_item()
{ {
/* loiter at home position */ switch (_rtl_state) {
_reset_loiter_pos = false; case RTL_STATE_CLIMB: {
_rtl_done = true; memcpy(&_mission_item_triplet.previous, &_mission_item_triplet.current, sizeof(mission_item_s));
_mission_item_triplet.previous_valid = _mission_item_triplet.current_valid;
mavlink_log_info(_mavlink_fd, "[navigator] RTL completed"); _mission_item_triplet.current_valid = true;
_mission_item_triplet.next_valid = false;
start_loiter(); float climb_alt = _home_pos.altitude + _parameters.rtl_alt;
if (_vstatus.condition_landed)
climb_alt = fmaxf(climb_alt, _global_pos.alt + _parameters.rtl_alt);
_mission_item_triplet.current.altitude_is_relative = false;
_mission_item_triplet.current.lat = (double)_global_pos.lat / 1e7d;
_mission_item_triplet.current.lon = (double)_global_pos.lon / 1e7d;
_mission_item_triplet.current.altitude = climb_alt;
_mission_item_triplet.current.yaw = NAN;
_mission_item_triplet.current.loiter_radius = _parameters.loiter_radius;
_mission_item_triplet.current.loiter_direction = 1;
_mission_item_triplet.current.nav_cmd = NAV_CMD_TAKEOFF;
_mission_item_triplet.current.acceptance_radius = _parameters.acceptance_radius;
_mission_item_triplet.current.time_inside = 0.0f;
_mission_item_triplet.current.pitch_min = 0.0f;
_mission_item_triplet.current.autocontinue = true;
_mission_item_triplet.current.origin = ORIGIN_ONBOARD;
mavlink_log_info(_mavlink_fd, "[navigator] RTL: climb to %.1fm", climb_alt - _home_pos.altitude);
break;
}
case RTL_STATE_RETURN: {
memcpy(&_mission_item_triplet.previous, &_mission_item_triplet.current, sizeof(mission_item_s));
_mission_item_triplet.previous_valid = _mission_item_triplet.current_valid;
_mission_item_triplet.current_valid = true;
_mission_item_triplet.next_valid = false;
_mission_item_triplet.current.altitude_is_relative = false;
_mission_item_triplet.current.lat = _home_pos.lat;
_mission_item_triplet.current.lon = _home_pos.lon;
// don't change altitude setpoint
_mission_item_triplet.current.yaw = NAN;
_mission_item_triplet.current.loiter_radius = _parameters.loiter_radius;
_mission_item_triplet.current.loiter_direction = 1;
_mission_item_triplet.current.nav_cmd = NAV_CMD_WAYPOINT;
_mission_item_triplet.current.acceptance_radius = _parameters.acceptance_radius;
_mission_item_triplet.current.time_inside = 0.0f;
_mission_item_triplet.current.pitch_min = 0.0f;
_mission_item_triplet.current.autocontinue = true;
_mission_item_triplet.current.origin = ORIGIN_ONBOARD;
mavlink_log_info(_mavlink_fd, "[navigator] RTL: return");
break;
}
case RTL_STATE_DESCEND: {
memcpy(&_mission_item_triplet.previous, &_mission_item_triplet.current, sizeof(mission_item_s));
_mission_item_triplet.previous_valid = _mission_item_triplet.current_valid;
_mission_item_triplet.current_valid = true;
_mission_item_triplet.next_valid = false;
float descend_alt = _home_pos.altitude + _parameters.land_alt;
_mission_item_triplet.current.altitude_is_relative = false;
_mission_item_triplet.current.lat = _home_pos.lat;
_mission_item_triplet.current.lon = _home_pos.lon;
_mission_item_triplet.current.altitude = descend_alt;
_mission_item_triplet.current.yaw = NAN;
_mission_item_triplet.current.loiter_radius = _parameters.loiter_radius;
_mission_item_triplet.current.loiter_direction = 1;
_mission_item_triplet.current.nav_cmd = NAV_CMD_WAYPOINT;
_mission_item_triplet.current.acceptance_radius = _parameters.acceptance_radius;
_mission_item_triplet.current.time_inside = 0.0f;
_mission_item_triplet.current.pitch_min = 0.0f;
_mission_item_triplet.current.autocontinue = true;
_mission_item_triplet.current.origin = ORIGIN_ONBOARD;
mavlink_log_info(_mavlink_fd, "[navigator] RTL: descend to %.1fm", descend_alt - _home_pos.altitude);
break;
}
case RTL_STATE_LAND: {
memcpy(&_mission_item_triplet.previous, &_mission_item_triplet.current, sizeof(mission_item_s));
_mission_item_triplet.previous_valid = _mission_item_triplet.current_valid;
_mission_item_triplet.current_valid = true;
_mission_item_triplet.next_valid = false;
_mission_item_triplet.current.altitude_is_relative = false;
_mission_item_triplet.current.lat = _home_pos.lat;
_mission_item_triplet.current.lon = _home_pos.lon;
_mission_item_triplet.current.altitude = _home_pos.altitude;
_mission_item_triplet.current.yaw = NAN;
_mission_item_triplet.current.loiter_radius = _parameters.loiter_radius;
_mission_item_triplet.current.loiter_direction = 1;
_mission_item_triplet.current.nav_cmd = NAV_CMD_LAND;
_mission_item_triplet.current.acceptance_radius = _parameters.acceptance_radius;
_mission_item_triplet.current.time_inside = 0.0f;
_mission_item_triplet.current.pitch_min = 0.0f;
_mission_item_triplet.current.autocontinue = true;
_mission_item_triplet.current.origin = ORIGIN_ONBOARD;
mavlink_log_info(_mavlink_fd, "[navigator] RTL: land");
break;
}
default: {
mavlink_log_critical(_mavlink_fd, "[navigator] error: unknown RTL state: %d", _rtl_state);
start_loiter();
break;
}
}
publish_mission_item_triplet();
} }
bool bool
@ -1176,9 +1294,8 @@ Navigator::check_mission_item_reached()
return false; return false;
} }
/* don't try to reach the landing mission, just stay in that mode, XXX maybe add another state for this */
if (_mission_item_triplet.current.nav_cmd == NAV_CMD_LAND) { if (_mission_item_triplet.current.nav_cmd == NAV_CMD_LAND) {
return false; return _vstatus.condition_landed;
} }
/* XXX TODO count turns */ /* XXX TODO count turns */
@ -1191,82 +1308,66 @@ Navigator::check_mission_item_reached()
} }
uint64_t now = hrt_absolute_time(); uint64_t now = hrt_absolute_time();
float acceptance_radius;
if (_mission_item_triplet.current.nav_cmd == NAV_CMD_WAYPOINT && _mission_item_triplet.current.acceptance_radius > 0.01f) { if (!_waypoint_position_reached) {
acceptance_radius = _mission_item_triplet.current.acceptance_radius; float acceptance_radius;
} else { if (_mission_item_triplet.current.nav_cmd == NAV_CMD_WAYPOINT && _mission_item_triplet.current.acceptance_radius > 0.01f) {
acceptance_radius = _parameters.acceptance_radius; acceptance_radius = _mission_item_triplet.current.acceptance_radius;
}
// TODO add frame } else {
// int coordinate_frame = wpm->waypoints[wpm->current_active_wp_id].frame; acceptance_radius = _parameters.acceptance_radius;
float dist = -1.0f;
float dist_xy = -1.0f;
float dist_z = -1.0f;
// if (coordinate_frame == (int)MAV_FRAME_GLOBAL) {
/* current relative or AMSL altitude depending on mission item altitude_is_relative flag */
float wp_alt_amsl = _mission_item_triplet.current.altitude;
if (_mission_item_triplet.current.altitude_is_relative)
_mission_item_triplet.current.altitude += _home_pos.altitude;
dist = get_distance_to_point_global_wgs84(_mission_item_triplet.current.lat, _mission_item_triplet.current.lon, wp_alt_amsl,
(double)_global_pos.lat / 1e7d, (double)_global_pos.lon / 1e7d, _global_pos.alt,
&dist_xy, &dist_z);
// warnx("1 lat: %2.2f, lon: %2.2f, alt: %2.2f", _mission_item_triplet.current.lat, _mission_item_triplet.current.lon, _mission_item_triplet.current.altitude);
// warnx("2 lat: %2.2f, lon: %2.2f, alt: %2.2f", (double)_global_pos.lat / 1e7d, (double)_global_pos.lon / 1e7d, _global_pos.alt);
// warnx("Dist: %4.4f", dist);
// } else if (coordinate_frame == (int)MAV_FRAME_GLOBAL_RELATIVE_ALT) {
// dist = mavlink_wpm_distance_to_point_global_wgs84(wpm->current_active_wp_id, (float)global_pos->lat * 1e-7f, (float)global_pos->lon * 1e-7f, global_pos->relative_alt, &dist_xy, &dist_z);
// } else if (coordinate_frame == (int)MAV_FRAME_LOCAL_ENU || coordinate_frame == (int)MAV_FRAME_LOCAL_NED) {
// dist = mavlink_wpm_distance_to_point_local(wpm->current_active_wp_id, local_pos->x, local_pos->y, local_pos->z, &dist_xy, &dist_z);
// } else if (coordinate_frame == (int)MAV_FRAME_MISSION) {
// /* Check if conditions of mission item are satisfied */
// // XXX TODO
// }
if (_do_takeoff) {
if (_global_pos.alt > wp_alt_amsl - acceptance_radius) {
/* require only altitude for takeoff */
_waypoint_position_reached = true;
} }
} else {
if (dist >= 0.0f && dist <= acceptance_radius) { float dist = -1.0f;
_waypoint_position_reached = true; float dist_xy = -1.0f;
float dist_z = -1.0f;
/* current relative or AMSL altitude depending on mission item altitude_is_relative flag */
float wp_alt_amsl = _mission_item_triplet.current.altitude;
if (_mission_item_triplet.current.altitude_is_relative)
_mission_item_triplet.current.altitude += _home_pos.altitude;
dist = get_distance_to_point_global_wgs84(_mission_item_triplet.current.lat, _mission_item_triplet.current.lon, wp_alt_amsl,
(double)_global_pos.lat / 1e7d, (double)_global_pos.lon / 1e7d, _global_pos.alt,
&dist_xy, &dist_z);
if (_do_takeoff) {
if (_global_pos.alt > wp_alt_amsl - acceptance_radius) {
/* require only altitude for takeoff */
_waypoint_position_reached = true;
}
} else {
if (dist >= 0.0f && dist <= acceptance_radius) {
_waypoint_position_reached = true;
}
} }
} }
if (_vstatus.is_rotary_wing && !_do_takeoff) { if (!_waypoint_yaw_reached) {
/* check yow only for rotary wing except takeoff */ if (_vstatus.is_rotary_wing && !_do_takeoff && isfinite(_mission_item_triplet.current.yaw)) {
float yaw_err = _wrap_pi(_mission_item_triplet.current.yaw - _global_pos.yaw); /* check yaw if defined only for rotary wing except takeoff */
if (fabsf(yaw_err) < 0.05f) { /* XXX get rid of magic number */ float yaw_err = _wrap_pi(_mission_item_triplet.current.yaw - _global_pos.yaw);
if (fabsf(yaw_err) < 0.05f) { /* XXX get rid of magic number */
_waypoint_yaw_reached = true;
}
} else {
_waypoint_yaw_reached = true; _waypoint_yaw_reached = true;
} }
} else {
_waypoint_yaw_reached = true;
} }
/* check if the current waypoint was reached */ /* check if the current waypoint was reached */
if (_waypoint_position_reached && _waypoint_yaw_reached) { if (_waypoint_position_reached && _waypoint_yaw_reached) {
if (_time_first_inside_orbit == 0) { if (_time_first_inside_orbit == 0) {
/* XXX announcement? */
_time_first_inside_orbit = now; _time_first_inside_orbit = now;
if (_mission_item_triplet.current.time_inside > 0.01f) {
mavlink_log_info(_mavlink_fd, "[navigator] waypoint reached, wait for %.1fs", _mission_item_triplet.current.time_inside);
}
} }
/* check if the MAV was long enough inside the waypoint orbit */ /* check if the MAV was long enough inside the waypoint orbit */
if ((now - _time_first_inside_orbit >= (uint64_t)_mission_item_triplet.current.time_inside * 1e6) if ((now - _time_first_inside_orbit >= (uint64_t)_mission_item_triplet.current.time_inside * 1e6)
|| _mission_item_triplet.current.nav_cmd == NAV_CMD_TAKEOFF) { || _mission_item_triplet.current.nav_cmd == NAV_CMD_TAKEOFF) {
_time_first_inside_orbit = 0; _time_first_inside_orbit = 0;
_waypoint_yaw_reached = false; _waypoint_yaw_reached = false;
_waypoint_position_reached = false; _waypoint_position_reached = false;
@ -1294,11 +1395,26 @@ Navigator::on_mission_item_reached()
advance_mission(); advance_mission();
} else { } else {
/* if no more mission items available then finish mission */ /* if no more mission items available then finish mission */
dispatch(EVENT_MISSION_FINISHED); /* loiter at last waypoint */
_reset_loiter_pos = false;
mavlink_log_info(_mavlink_fd, "[navigator] mission completed");
if (_vstatus.condition_landed) {
dispatch(EVENT_READY_REQUESTED);
} else {
dispatch(EVENT_LOITER_REQUESTED);
}
} }
} else { } else {
/* RTL finished */ /* RTL finished */
dispatch(EVENT_MISSION_FINISHED); if (_rtl_state == RTL_STATE_LAND) {
/* landed at home position */
mavlink_log_info(_mavlink_fd, "[navigator] RTL completed, landed");
dispatch(EVENT_READY_REQUESTED);
} else {
/* next RTL step */
_rtl_state = (RTLState)(_rtl_state + 1);
set_rtl_item();
}
} }
} }
@ -1413,12 +1529,22 @@ Navigator::publish_control_mode()
case MAIN_STATE_AUTO: case MAIN_STATE_AUTO:
_control_mode.flag_control_manual_enabled = false; _control_mode.flag_control_manual_enabled = false;
_control_mode.flag_control_rates_enabled = true; if (myState == NAV_STATE_READY) {
_control_mode.flag_control_attitude_enabled = true; /* disable all controllers, armed but idle */
_control_mode.flag_control_position_enabled = true; _control_mode.flag_control_rates_enabled = false;
_control_mode.flag_control_velocity_enabled = true; _control_mode.flag_control_attitude_enabled = false;
_control_mode.flag_control_altitude_enabled = true; _control_mode.flag_control_position_enabled = false;
_control_mode.flag_control_climb_rate_enabled = true; _control_mode.flag_control_velocity_enabled = false;
_control_mode.flag_control_altitude_enabled = false;
_control_mode.flag_control_climb_rate_enabled = false;
} else {
_control_mode.flag_control_rates_enabled = true;
_control_mode.flag_control_attitude_enabled = true;
_control_mode.flag_control_position_enabled = true;
_control_mode.flag_control_velocity_enabled = true;
_control_mode.flag_control_altitude_enabled = true;
_control_mode.flag_control_climb_rate_enabled = true;
}
break; break;
default: default:

View File

@ -63,6 +63,7 @@
typedef enum { typedef enum {
NAV_STATE_NONE = 0, NAV_STATE_NONE = 0,
NAV_STATE_READY,
NAV_STATE_LOITER, NAV_STATE_LOITER,
NAV_STATE_MISSION, NAV_STATE_MISSION,
NAV_STATE_RTL, NAV_STATE_RTL,