mirror of
https://gitee.com/mirrors_PX4/PX4-Autopilot.git
synced 2026-04-14 10:07:39 +08:00
713 lines
23 KiB
C++
713 lines
23 KiB
C++
/****************************************************************************
|
|
*
|
|
* Copyright (c) 2013-2014 PX4 Development Team. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* 3. Neither the name PX4 nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************/
|
|
/**
|
|
* @file navigator_mission.cpp
|
|
*
|
|
* Helper class to access missions
|
|
*
|
|
* @author Julian Oes <julian@oes.ch>
|
|
* @author Thomas Gubler <thomasgubler@gmail.com>
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <float.h>
|
|
|
|
#include <drivers/drv_hrt.h>
|
|
|
|
#include <dataman/dataman.h>
|
|
#include <mavlink/mavlink_log.h>
|
|
#include <systemlib/err.h>
|
|
#include <geo/geo.h>
|
|
#include <lib/mathlib/mathlib.h>
|
|
|
|
#include <uORB/uORB.h>
|
|
#include <uORB/topics/mission.h>
|
|
#include <uORB/topics/mission_result.h>
|
|
|
|
#include "mission.h"
|
|
#include "navigator.h"
|
|
|
|
Mission::Mission(Navigator *navigator, const char *name) :
|
|
MissionBlock(navigator, name),
|
|
_param_onboard_enabled(this, "ONBOARD_EN"),
|
|
_param_takeoff_alt(this, "TAKEOFF_ALT"),
|
|
_param_dist_1wp(this, "DIST_1WP"),
|
|
_param_altmode(this, "ALTMODE"),
|
|
_onboard_mission({0}),
|
|
_offboard_mission({0}),
|
|
_current_onboard_mission_index(-1),
|
|
_current_offboard_mission_index(-1),
|
|
_need_takeoff(true),
|
|
_takeoff(false),
|
|
_mission_result_pub(-1),
|
|
_mission_result({0}),
|
|
_mission_type(MISSION_TYPE_NONE),
|
|
_inited(false),
|
|
_dist_1wp_ok(false),
|
|
_missionFeasiblityChecker(),
|
|
_min_current_sp_distance_xy(FLT_MAX),
|
|
_mission_item_previous_alt(NAN),
|
|
_distance_current_previous(0.0f)
|
|
{
|
|
/* load initial params */
|
|
updateParams();
|
|
}
|
|
|
|
Mission::~Mission()
|
|
{
|
|
}
|
|
|
|
void
|
|
Mission::on_inactive()
|
|
{
|
|
if (_inited) {
|
|
/* check if missions have changed so that feedback to ground station is given */
|
|
bool onboard_updated = false;
|
|
orb_check(_navigator->get_onboard_mission_sub(), &onboard_updated);
|
|
if (onboard_updated) {
|
|
update_onboard_mission();
|
|
}
|
|
|
|
bool offboard_updated = false;
|
|
orb_check(_navigator->get_offboard_mission_sub(), &offboard_updated);
|
|
if (offboard_updated) {
|
|
update_offboard_mission();
|
|
}
|
|
|
|
} else {
|
|
/* read mission topics on initialization */
|
|
_inited = true;
|
|
|
|
update_onboard_mission();
|
|
update_offboard_mission();
|
|
}
|
|
|
|
if (!_navigator->get_can_loiter_at_sp() || _navigator->get_vstatus()->condition_landed) {
|
|
_need_takeoff = true;
|
|
}
|
|
}
|
|
|
|
void
|
|
Mission::on_activation()
|
|
{
|
|
set_mission_items();
|
|
}
|
|
|
|
void
|
|
Mission::on_active()
|
|
{
|
|
/* check if anything has changed */
|
|
bool onboard_updated = false;
|
|
orb_check(_navigator->get_onboard_mission_sub(), &onboard_updated);
|
|
if (onboard_updated) {
|
|
update_onboard_mission();
|
|
}
|
|
|
|
bool offboard_updated = false;
|
|
orb_check(_navigator->get_offboard_mission_sub(), &offboard_updated);
|
|
if (offboard_updated) {
|
|
update_offboard_mission();
|
|
}
|
|
|
|
/* reset mission items if needed */
|
|
if (onboard_updated || offboard_updated) {
|
|
set_mission_items();
|
|
}
|
|
|
|
/* lets check if we reached the current mission item */
|
|
if (_mission_type != MISSION_TYPE_NONE && is_mission_item_reached()) {
|
|
advance_mission();
|
|
set_mission_items();
|
|
|
|
} else if (_mission_type != MISSION_TYPE_NONE &&_param_altmode.get() == MISSION_ALTMODE_FOH) {
|
|
altitude_sp_foh_update();
|
|
} else {
|
|
/* if waypoint position reached allow loiter on the setpoint */
|
|
if (_waypoint_position_reached && _mission_item.nav_cmd != NAV_CMD_IDLE) {
|
|
_navigator->set_can_loiter_at_sp(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
Mission::update_onboard_mission()
|
|
{
|
|
if (orb_copy(ORB_ID(onboard_mission), _navigator->get_onboard_mission_sub(), &_onboard_mission) == OK) {
|
|
/* accept the current index set by the onboard mission if it is within bounds */
|
|
if (_onboard_mission.current_seq >=0
|
|
&& _onboard_mission.current_seq < (int)_onboard_mission.count) {
|
|
_current_onboard_mission_index = _onboard_mission.current_seq;
|
|
} else {
|
|
/* if less WPs available, reset to first WP */
|
|
if (_current_onboard_mission_index >= (int)_onboard_mission.count) {
|
|
_current_onboard_mission_index = 0;
|
|
/* if not initialized, set it to 0 */
|
|
} else if (_current_onboard_mission_index < 0) {
|
|
_current_onboard_mission_index = 0;
|
|
}
|
|
/* otherwise, just leave it */
|
|
}
|
|
|
|
} else {
|
|
_onboard_mission.count = 0;
|
|
_onboard_mission.current_seq = 0;
|
|
_current_onboard_mission_index = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
Mission::update_offboard_mission()
|
|
{
|
|
if (orb_copy(ORB_ID(offboard_mission), _navigator->get_offboard_mission_sub(), &_offboard_mission) == OK) {
|
|
warnx("offboard mission updated: dataman_id=%d, count=%d, current_seq=%d", _offboard_mission.dataman_id, _offboard_mission.count, _offboard_mission.current_seq);
|
|
/* determine current index */
|
|
if (_offboard_mission.current_seq >= 0 && _offboard_mission.current_seq < (int)_offboard_mission.count) {
|
|
_current_offboard_mission_index = _offboard_mission.current_seq;
|
|
} else {
|
|
/* if less items available, reset to first item */
|
|
if (_current_offboard_mission_index >= (int)_offboard_mission.count) {
|
|
_current_offboard_mission_index = 0;
|
|
|
|
/* if not initialized, set it to 0 */
|
|
} else if (_current_offboard_mission_index < 0) {
|
|
_current_offboard_mission_index = 0;
|
|
}
|
|
/* otherwise, just leave it */
|
|
}
|
|
|
|
/* Check mission feasibility, for now do not handle the return value,
|
|
* however warnings are issued to the gcs via mavlink from inside the MissionFeasiblityChecker */
|
|
dm_item_t dm_current = DM_KEY_WAYPOINTS_OFFBOARD(_offboard_mission.dataman_id);
|
|
|
|
_missionFeasiblityChecker.checkMissionFeasible(_navigator->get_vstatus()->is_rotary_wing,
|
|
dm_current, (size_t) _offboard_mission.count, _navigator->get_geofence(),
|
|
_navigator->get_home_position()->alt);
|
|
|
|
} else {
|
|
warnx("offboard mission update failed");
|
|
_offboard_mission.count = 0;
|
|
_offboard_mission.current_seq = 0;
|
|
_current_offboard_mission_index = 0;
|
|
}
|
|
|
|
report_current_offboard_mission_item();
|
|
}
|
|
|
|
|
|
void
|
|
Mission::advance_mission()
|
|
{
|
|
if (_takeoff) {
|
|
_takeoff = false;
|
|
|
|
} else {
|
|
switch (_mission_type) {
|
|
case MISSION_TYPE_ONBOARD:
|
|
_current_onboard_mission_index++;
|
|
break;
|
|
|
|
case MISSION_TYPE_OFFBOARD:
|
|
_current_offboard_mission_index++;
|
|
break;
|
|
|
|
case MISSION_TYPE_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
Mission::check_dist_1wp()
|
|
{
|
|
if (_dist_1wp_ok) {
|
|
/* always return true after at least one successful check */
|
|
return true;
|
|
}
|
|
|
|
/* check if first waypoint is not too far from home */
|
|
if (_param_dist_1wp.get() > 0.0f) {
|
|
if (_navigator->get_vstatus()->condition_home_position_valid) {
|
|
struct mission_item_s mission_item;
|
|
|
|
/* find first waypoint (with lat/lon) item in datamanager */
|
|
for (unsigned i = 0; i < _offboard_mission.count; i++) {
|
|
if (dm_read(DM_KEY_WAYPOINTS_OFFBOARD(_offboard_mission.dataman_id), i,
|
|
&mission_item, sizeof(mission_item_s)) == sizeof(mission_item_s)) {
|
|
|
|
/* check only items with valid lat/lon */
|
|
if ( mission_item.nav_cmd == NAV_CMD_WAYPOINT ||
|
|
mission_item.nav_cmd == NAV_CMD_LOITER_TIME_LIMIT ||
|
|
mission_item.nav_cmd == NAV_CMD_LOITER_TURN_COUNT ||
|
|
mission_item.nav_cmd == NAV_CMD_LOITER_UNLIMITED ||
|
|
mission_item.nav_cmd == NAV_CMD_TAKEOFF ||
|
|
mission_item.nav_cmd == NAV_CMD_PATHPLANNING) {
|
|
|
|
/* check distance from home to item */
|
|
float dist_to_1wp = get_distance_to_next_waypoint(
|
|
mission_item.lat, mission_item.lon,
|
|
_navigator->get_home_position()->lat, _navigator->get_home_position()->lon);
|
|
|
|
if (dist_to_1wp < _param_dist_1wp.get()) {
|
|
_dist_1wp_ok = true;
|
|
if (dist_to_1wp > ((_param_dist_1wp.get() * 3) / 2)) {
|
|
/* allow at 2/3 distance, but warn */
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "Warning: First waypoint very far: %d m", (int)dist_to_1wp);
|
|
}
|
|
return true;
|
|
|
|
} else {
|
|
/* item is too far from home */
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "Waypoint too far: %d m,[MIS_DIST_1WP=%d]", (int)dist_to_1wp, (int)_param_dist_1wp.get());
|
|
return false;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
/* error reading, mission is invalid */
|
|
mavlink_log_info(_navigator->get_mavlink_fd(), "error reading offboard mission");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/* no waypoints found in mission, then we will not fly far away */
|
|
_dist_1wp_ok = true;
|
|
return true;
|
|
|
|
} else {
|
|
mavlink_log_info(_navigator->get_mavlink_fd(), "no home position");
|
|
return false;
|
|
}
|
|
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
void
|
|
Mission::set_mission_items()
|
|
{
|
|
/* make sure param is up to date */
|
|
updateParams();
|
|
|
|
/* reset the altitude foh logic, if altitude foh is enabled (param) a new foh element starts now */
|
|
altitude_sp_foh_reset();
|
|
|
|
struct position_setpoint_triplet_s *pos_sp_triplet = _navigator->get_position_setpoint_triplet();
|
|
|
|
/* set previous position setpoint to current */
|
|
set_previous_pos_setpoint();
|
|
|
|
/* Copy previous mission item altitude (can be extended to a copy of the full mission item if needed) */
|
|
if (pos_sp_triplet->previous.valid) {
|
|
_mission_item_previous_alt = _mission_item.altitude;
|
|
}
|
|
|
|
/* get home distance state */
|
|
bool home_dist_ok = check_dist_1wp();
|
|
/* the home dist check provides user feedback, so we initialize it to this */
|
|
bool user_feedback_done = !home_dist_ok;
|
|
|
|
/* try setting onboard mission item */
|
|
if (_param_onboard_enabled.get() && read_mission_item(true, true, &_mission_item)) {
|
|
/* if mission type changed, notify */
|
|
if (_mission_type != MISSION_TYPE_ONBOARD) {
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "onboard mission now running");
|
|
}
|
|
_mission_type = MISSION_TYPE_ONBOARD;
|
|
|
|
/* try setting offboard mission item */
|
|
} else if (home_dist_ok && read_mission_item(false, true, &_mission_item)) {
|
|
/* if mission type changed, notify */
|
|
if (_mission_type != MISSION_TYPE_OFFBOARD) {
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "offboard mission now running");
|
|
}
|
|
_mission_type = MISSION_TYPE_OFFBOARD;
|
|
|
|
} else {
|
|
/* no mission available or mission finished, switch to loiter */
|
|
if (_mission_type != MISSION_TYPE_NONE) {
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "mission finished, loitering");
|
|
} else if (!user_feedback_done) {
|
|
/* only tell users that we got no mission if there has not been any
|
|
* better, more specific feedback yet
|
|
*/
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "no valid mission available");
|
|
}
|
|
_mission_type = MISSION_TYPE_NONE;
|
|
|
|
/* set loiter mission item */
|
|
set_loiter_item(&_mission_item);
|
|
|
|
/* update position setpoint triplet */
|
|
pos_sp_triplet->previous.valid = false;
|
|
mission_item_to_position_setpoint(&_mission_item, &pos_sp_triplet->current);
|
|
pos_sp_triplet->next.valid = false;
|
|
|
|
_navigator->set_can_loiter_at_sp(pos_sp_triplet->current.type == SETPOINT_TYPE_LOITER);
|
|
|
|
reset_mission_item_reached();
|
|
report_mission_finished();
|
|
|
|
_navigator->set_position_setpoint_triplet_updated();
|
|
return;
|
|
}
|
|
|
|
/* do takeoff on first waypoint for rotary wing vehicles */
|
|
if (_navigator->get_vstatus()->is_rotary_wing) {
|
|
/* force takeoff if landed (additional protection) */
|
|
if (!_takeoff && _navigator->get_vstatus()->condition_landed) {
|
|
_need_takeoff = true;
|
|
}
|
|
|
|
/* new current mission item set, check if we need takeoff */
|
|
if (_need_takeoff && (
|
|
_mission_item.nav_cmd == NAV_CMD_TAKEOFF ||
|
|
_mission_item.nav_cmd == NAV_CMD_WAYPOINT ||
|
|
_mission_item.nav_cmd == NAV_CMD_LOITER_TIME_LIMIT ||
|
|
_mission_item.nav_cmd == NAV_CMD_LOITER_TURN_COUNT ||
|
|
_mission_item.nav_cmd == NAV_CMD_LOITER_UNLIMITED ||
|
|
_mission_item.nav_cmd == NAV_CMD_RETURN_TO_LAUNCH)) {
|
|
_takeoff = true;
|
|
_need_takeoff = false;
|
|
}
|
|
}
|
|
|
|
if (_takeoff) {
|
|
/* do takeoff before going to setpoint */
|
|
/* set mission item as next position setpoint */
|
|
mission_item_to_position_setpoint(&_mission_item, &pos_sp_triplet->next);
|
|
|
|
/* calculate takeoff altitude */
|
|
float takeoff_alt = _mission_item.altitude;
|
|
if (_mission_item.altitude_is_relative) {
|
|
takeoff_alt += _navigator->get_home_position()->alt;
|
|
}
|
|
|
|
/* perform takeoff at least to NAV_TAKEOFF_ALT above home/ground, even if first waypoint is lower */
|
|
if (_navigator->get_vstatus()->condition_landed) {
|
|
takeoff_alt = fmaxf(takeoff_alt, _navigator->get_global_position()->alt + _param_takeoff_alt.get());
|
|
|
|
} else {
|
|
takeoff_alt = fmaxf(takeoff_alt, _navigator->get_home_position()->alt + _param_takeoff_alt.get());
|
|
}
|
|
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "takeoff to %.1f meters above home", (double)(takeoff_alt - _navigator->get_home_position()->alt));
|
|
|
|
_mission_item.lat = _navigator->get_global_position()->lat;
|
|
_mission_item.lon = _navigator->get_global_position()->lon;
|
|
_mission_item.altitude = takeoff_alt;
|
|
_mission_item.altitude_is_relative = false;
|
|
|
|
mission_item_to_position_setpoint(&_mission_item, &pos_sp_triplet->current);
|
|
|
|
} else {
|
|
/* set current position setpoint from mission item */
|
|
mission_item_to_position_setpoint(&_mission_item, &pos_sp_triplet->current);
|
|
|
|
/* require takeoff after landing or idle */
|
|
if (pos_sp_triplet->current.type == SETPOINT_TYPE_LAND || pos_sp_triplet->current.type == SETPOINT_TYPE_IDLE) {
|
|
_need_takeoff = true;
|
|
}
|
|
|
|
_navigator->set_can_loiter_at_sp(false);
|
|
reset_mission_item_reached();
|
|
|
|
if (_mission_type == MISSION_TYPE_OFFBOARD) {
|
|
report_current_offboard_mission_item();
|
|
}
|
|
// TODO: report onboard mission item somehow
|
|
|
|
/* try to read next mission item */
|
|
struct mission_item_s mission_item_next;
|
|
|
|
if (read_mission_item(_mission_type == MISSION_TYPE_ONBOARD, false, &mission_item_next)) {
|
|
/* got next mission item, update setpoint triplet */
|
|
mission_item_to_position_setpoint(&mission_item_next, &pos_sp_triplet->next);
|
|
|
|
} else {
|
|
/* next mission item is not available */
|
|
pos_sp_triplet->next.valid = false;
|
|
}
|
|
}
|
|
|
|
/* Save the distance between the current sp and the previous one */
|
|
if (pos_sp_triplet->current.valid && pos_sp_triplet->previous.valid) {
|
|
_distance_current_previous = get_distance_to_next_waypoint(pos_sp_triplet->current.lat,
|
|
pos_sp_triplet->current.lon,
|
|
pos_sp_triplet->previous.lat,
|
|
pos_sp_triplet->previous.lon);
|
|
}
|
|
|
|
_navigator->set_position_setpoint_triplet_updated();
|
|
}
|
|
|
|
void
|
|
Mission::altitude_sp_foh_update()
|
|
{
|
|
struct position_setpoint_triplet_s *pos_sp_triplet = _navigator->get_position_setpoint_triplet();
|
|
|
|
/* Don't change setpoint if last and current waypoint are not valid */
|
|
if (!pos_sp_triplet->previous.valid || !pos_sp_triplet->current.valid ||
|
|
!isfinite(_mission_item_previous_alt)) {
|
|
return;
|
|
}
|
|
|
|
/* Do not try to find a solution if the last waypoint is inside the acceptance radius of the current one */
|
|
if (_distance_current_previous - _mission_item.acceptance_radius < 0.0f) {
|
|
return;
|
|
}
|
|
|
|
/* Don't do FOH for landing and takeoff waypoints, the ground may be near
|
|
* and the FW controller has a custom landing logic */
|
|
if (_mission_item.nav_cmd == NAV_CMD_LAND || _mission_item.nav_cmd == NAV_CMD_TAKEOFF) {
|
|
return;
|
|
}
|
|
|
|
|
|
/* Calculate distance to current waypoint */
|
|
float d_current = get_distance_to_next_waypoint(_mission_item.lat, _mission_item.lon,
|
|
_navigator->get_global_position()->lat, _navigator->get_global_position()->lon);
|
|
|
|
/* Save distance to waypoint if it is the smallest ever achieved, however make sure that
|
|
* _min_current_sp_distance_xy is never larger than the distance between the current and the previous wp */
|
|
_min_current_sp_distance_xy = math::min(math::min(d_current, _min_current_sp_distance_xy),
|
|
_distance_current_previous);
|
|
|
|
/* if the minimal distance is smaller then the acceptance radius, we should be at waypoint alt
|
|
* navigator will soon switch to the next waypoint item (if there is one) as soon as we reach this altitude */
|
|
if (_min_current_sp_distance_xy < _mission_item.acceptance_radius) {
|
|
pos_sp_triplet->current.alt = _mission_item.altitude;
|
|
} else {
|
|
/* update the altitude sp of the 'current' item in the sp triplet, but do not update the altitude sp
|
|
* of the mission item as it is used to check if the mission item is reached
|
|
* The setpoint is set linearly and such that the system reaches the current altitude at the acceptance
|
|
* radius around the current waypoint
|
|
**/
|
|
float delta_alt = (_mission_item.altitude - _mission_item_previous_alt);
|
|
float grad = -delta_alt/(_distance_current_previous - _mission_item.acceptance_radius);
|
|
float a = _mission_item_previous_alt - grad * _distance_current_previous;
|
|
pos_sp_triplet->current.alt = a + grad * _min_current_sp_distance_xy;
|
|
|
|
}
|
|
|
|
_navigator->set_position_setpoint_triplet_updated();
|
|
}
|
|
|
|
void
|
|
Mission::altitude_sp_foh_reset()
|
|
{
|
|
_min_current_sp_distance_xy = FLT_MAX;
|
|
}
|
|
|
|
bool
|
|
Mission::read_mission_item(bool onboard, bool is_current, struct mission_item_s *mission_item)
|
|
{
|
|
/* select onboard/offboard mission */
|
|
int *mission_index_ptr;
|
|
struct mission_s *mission;
|
|
dm_item_t dm_item;
|
|
int mission_index_next;
|
|
|
|
if (onboard) {
|
|
/* onboard mission */
|
|
mission_index_next = _current_onboard_mission_index + 1;
|
|
mission_index_ptr = is_current ? &_current_onboard_mission_index : &mission_index_next;
|
|
|
|
mission = &_onboard_mission;
|
|
|
|
dm_item = DM_KEY_WAYPOINTS_ONBOARD;
|
|
|
|
} else {
|
|
/* offboard mission */
|
|
mission_index_next = _current_offboard_mission_index + 1;
|
|
mission_index_ptr = is_current ? &_current_offboard_mission_index : &mission_index_next;
|
|
|
|
mission = &_offboard_mission;
|
|
|
|
dm_item = DM_KEY_WAYPOINTS_OFFBOARD(_offboard_mission.dataman_id);
|
|
}
|
|
|
|
if (*mission_index_ptr < 0 || *mission_index_ptr >= (int)mission->count) {
|
|
/* mission item index out of bounds */
|
|
return false;
|
|
}
|
|
|
|
/* repeat several to get the mission item because we might have to follow multiple DO_JUMPS */
|
|
for (int i = 0; i < 10; i++) {
|
|
const ssize_t len = sizeof(struct mission_item_s);
|
|
|
|
/* read mission item to temp storage first to not overwrite current mission item if data damaged */
|
|
struct mission_item_s mission_item_tmp;
|
|
|
|
/* read mission item from datamanager */
|
|
if (dm_read(dm_item, *mission_index_ptr, &mission_item_tmp, len) != len) {
|
|
/* not supposed to happen unless the datamanager can't access the SD card, etc. */
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(),
|
|
"ERROR waypoint could not be read");
|
|
return false;
|
|
}
|
|
|
|
/* check for DO_JUMP item, and whether it hasn't not already been repeated enough times */
|
|
if (mission_item_tmp.nav_cmd == NAV_CMD_DO_JUMP) {
|
|
|
|
/* do DO_JUMP as many times as requested */
|
|
if (mission_item_tmp.do_jump_current_count < mission_item_tmp.do_jump_repeat_count) {
|
|
|
|
/* only raise the repeat count if this is for the current mission item
|
|
* but not for the next mission item */
|
|
if (is_current) {
|
|
(mission_item_tmp.do_jump_current_count)++;
|
|
/* save repeat count */
|
|
if (dm_write(dm_item, *mission_index_ptr, DM_PERSIST_IN_FLIGHT_RESET, &mission_item_tmp, len) != len) {
|
|
/* not supposed to happen unless the datamanager can't access the
|
|
* dataman */
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(),
|
|
"ERROR DO JUMP waypoint could not be written");
|
|
return false;
|
|
}
|
|
}
|
|
/* set new mission item index and repeat
|
|
* we don't have to validate here, if it's invalid, we should realize this later .*/
|
|
*mission_index_ptr = mission_item_tmp.do_jump_mission_index;
|
|
|
|
} else {
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(),
|
|
"DO JUMP repetitions completed");
|
|
/* no more DO_JUMPS, therefore just try to continue with next mission item */
|
|
(*mission_index_ptr)++;
|
|
}
|
|
|
|
} else {
|
|
/* if it's not a DO_JUMP, then we were successful */
|
|
memcpy(mission_item, &mission_item_tmp, sizeof(struct mission_item_s));
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/* we have given up, we don't want to cycle forever */
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(),
|
|
"ERROR DO JUMP is cycling, giving up");
|
|
return false;
|
|
}
|
|
|
|
void
|
|
Mission::save_offboard_mission_state()
|
|
{
|
|
mission_s mission_state;
|
|
|
|
/* lock MISSION_STATE item */
|
|
dm_lock(DM_KEY_MISSION_STATE);
|
|
|
|
/* read current state */
|
|
int read_res = dm_read(DM_KEY_MISSION_STATE, 0, &mission_state, sizeof(mission_s));
|
|
|
|
if (read_res == sizeof(mission_s)) {
|
|
/* data read successfully, check dataman ID and items count */
|
|
if (mission_state.dataman_id == _offboard_mission.dataman_id && mission_state.count == _offboard_mission.count) {
|
|
/* navigator may modify only sequence, write modified state only if it changed */
|
|
if (mission_state.current_seq != _current_offboard_mission_index) {
|
|
if (dm_write(DM_KEY_MISSION_STATE, 0, DM_PERSIST_POWER_ON_RESET, &mission_state, sizeof(mission_s)) != sizeof(mission_s)) {
|
|
warnx("ERROR: can't save mission state");
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "ERROR: can't save mission state");
|
|
}
|
|
}
|
|
}
|
|
|
|
} else {
|
|
/* invalid data, this must not happen and indicates error in offboard_mission publisher */
|
|
mission_state.dataman_id = _offboard_mission.dataman_id;
|
|
mission_state.count = _offboard_mission.count;
|
|
mission_state.current_seq = _current_offboard_mission_index;
|
|
|
|
warnx("ERROR: invalid mission state");
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "ERROR: invalid mission state");
|
|
|
|
/* write modified state only if changed */
|
|
if (dm_write(DM_KEY_MISSION_STATE, 0, DM_PERSIST_POWER_ON_RESET, &mission_state, sizeof(mission_s)) != sizeof(mission_s)) {
|
|
warnx("ERROR: can't save mission state");
|
|
mavlink_log_critical(_navigator->get_mavlink_fd(), "ERROR: can't save mission state");
|
|
}
|
|
}
|
|
|
|
/* unlock MISSION_STATE item */
|
|
dm_unlock(DM_KEY_MISSION_STATE);
|
|
}
|
|
|
|
void
|
|
Mission::report_mission_item_reached()
|
|
{
|
|
if (_mission_type == MISSION_TYPE_OFFBOARD) {
|
|
_mission_result.reached = true;
|
|
_mission_result.seq_reached = _current_offboard_mission_index;
|
|
}
|
|
publish_mission_result();
|
|
}
|
|
|
|
void
|
|
Mission::report_current_offboard_mission_item()
|
|
{
|
|
warnx("current offboard mission index: %d", _current_offboard_mission_index);
|
|
_mission_result.seq_current = _current_offboard_mission_index;
|
|
publish_mission_result();
|
|
|
|
save_offboard_mission_state();
|
|
}
|
|
|
|
void
|
|
Mission::report_mission_finished()
|
|
{
|
|
_mission_result.finished = true;
|
|
publish_mission_result();
|
|
}
|
|
|
|
void
|
|
Mission::publish_mission_result()
|
|
{
|
|
/* lazily publish the mission result only once available */
|
|
if (_mission_result_pub > 0) {
|
|
/* publish mission result */
|
|
orb_publish(ORB_ID(mission_result), _mission_result_pub, &_mission_result);
|
|
|
|
} else {
|
|
/* advertise and publish */
|
|
_mission_result_pub = orb_advertise(ORB_ID(mission_result), &_mission_result);
|
|
}
|
|
/* reset reached bool */
|
|
_mission_result.reached = false;
|
|
_mission_result.finished = false;
|
|
}
|