mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-12 02:48:28 -04:00
aae524b9d5
This is a rather confusing problem when it happens during development, so a sanity check is good. What you see in dev is that your latitude gets corrupted between when you store and retrieve it, with no warnings that you're doing something wrong.
1982 lines
74 KiB
C++
1982 lines
74 KiB
C++
/// @file AP_Mission.cpp
|
|
/// @brief Handles the MAVLINK command mission stack. Reads and writes mission to storage.
|
|
|
|
#include "AP_Mission.h"
|
|
#include <AP_Terrain/AP_Terrain.h>
|
|
#include <GCS_MAVLink/GCS.h>
|
|
#include <AP_AHRS/AP_AHRS.h>
|
|
|
|
const AP_Param::GroupInfo AP_Mission::var_info[] = {
|
|
|
|
// @Param: TOTAL
|
|
// @DisplayName: Total mission commands
|
|
// @Description: The number of mission mission items that has been loaded by the ground station. Do not change this manually.
|
|
// @Range: 0 32766
|
|
// @Increment: 1
|
|
// @User: Advanced
|
|
// @ReadOnly: True
|
|
AP_GROUPINFO_FLAGS("TOTAL", 0, AP_Mission, _cmd_total, 0, AP_PARAM_FLAG_INTERNAL_USE_ONLY),
|
|
|
|
// @Param: RESTART
|
|
// @DisplayName: Mission Restart when entering Auto mode
|
|
// @Description: Controls mission starting point when entering Auto mode (either restart from beginning of mission or resume from last command run)
|
|
// @Values: 0:Resume Mission, 1:Restart Mission
|
|
// @User: Advanced
|
|
AP_GROUPINFO("RESTART", 1, AP_Mission, _restart, AP_MISSION_RESTART_DEFAULT),
|
|
|
|
// @Param: OPTIONS
|
|
// @DisplayName: Mission options bitmask
|
|
// @Description: Bitmask of what options to use in missions.
|
|
// @Bitmask: 0:Clear Mission on reboot
|
|
// @User: Advanced
|
|
AP_GROUPINFO("OPTIONS", 2, AP_Mission, _options, AP_MISSION_OPTIONS_DEFAULT),
|
|
|
|
AP_GROUPEND
|
|
};
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
// storage object
|
|
StorageAccess AP_Mission::_storage(StorageManager::StorageMission);
|
|
|
|
HAL_Semaphore_Recursive AP_Mission::_rsem;
|
|
|
|
///
|
|
/// public mission methods
|
|
///
|
|
|
|
/// init - initialises this library including checks the version in eeprom matches this library
|
|
void AP_Mission::init()
|
|
{
|
|
// check_eeprom_version - checks version of missions stored in eeprom matches this library
|
|
// command list will be cleared if they do not match
|
|
check_eeprom_version();
|
|
|
|
// If Mission Clear bit is set then it should clear the mission, otherwise retain the mission.
|
|
if (AP_MISSION_MASK_MISSION_CLEAR & _options) {
|
|
gcs().send_text(MAV_SEVERITY_INFO, "Clearing Mission");
|
|
clear();
|
|
}
|
|
|
|
_last_change_time_ms = AP_HAL::millis();
|
|
}
|
|
|
|
/// start - resets current commands to point to the beginning of the mission
|
|
/// To-Do: should we validate the mission first and return true/false?
|
|
void AP_Mission::start()
|
|
{
|
|
_flags.state = MISSION_RUNNING;
|
|
|
|
reset(); // reset mission to the first command, resets jump tracking
|
|
|
|
// advance to the first command
|
|
if (!advance_current_nav_cmd()) {
|
|
// on failure set mission complete
|
|
complete();
|
|
}
|
|
}
|
|
|
|
/// stop - stops mission execution. subsequent calls to update() will have no effect until the mission is started or resumed
|
|
void AP_Mission::stop()
|
|
{
|
|
_flags.state = MISSION_STOPPED;
|
|
}
|
|
|
|
/// resume - continues the mission execution from where we last left off
|
|
/// previous running commands will be re-initialized
|
|
void AP_Mission::resume()
|
|
{
|
|
// if mission had completed then start it from the first command
|
|
if (_flags.state == MISSION_COMPLETE) {
|
|
start();
|
|
return;
|
|
}
|
|
|
|
// if mission had stopped then restart it
|
|
if (_flags.state == MISSION_STOPPED) {
|
|
_flags.state = MISSION_RUNNING;
|
|
|
|
// if no valid nav command index restart from beginning
|
|
if (_nav_cmd.index == AP_MISSION_CMD_INDEX_NONE) {
|
|
start();
|
|
return;
|
|
}
|
|
}
|
|
|
|
// ensure cache coherence
|
|
if (!read_cmd_from_storage(_nav_cmd.index, _nav_cmd)) {
|
|
// if we failed to read the command from storage, then the command may have
|
|
// been from a previously loaded mission it is illogical to ever resume
|
|
// flying to a command that has been excluded from the current mission
|
|
start();
|
|
return;
|
|
}
|
|
|
|
// restart active navigation command. We run these on resume()
|
|
// regardless of whether the mission was stopped, as we may be
|
|
// re-entering AUTO mode and the nav_cmd callback needs to be run
|
|
// to setup the current target waypoint
|
|
|
|
if (_flags.do_cmd_loaded && _do_cmd.index != AP_MISSION_CMD_INDEX_NONE) {
|
|
// restart the active do command, which will also load the nav command for us
|
|
set_current_cmd(_do_cmd.index);
|
|
} else if (_flags.nav_cmd_loaded) {
|
|
// restart the active nav command
|
|
set_current_cmd(_nav_cmd.index);
|
|
}
|
|
|
|
// Note: if there is no active command then the mission must have been stopped just after the previous nav command completed
|
|
// update will take care of finding and starting the nav command
|
|
}
|
|
|
|
/// check mission starts with a takeoff command
|
|
bool AP_Mission::starts_with_takeoff_cmd()
|
|
{
|
|
Mission_Command cmd = {};
|
|
uint16_t cmd_index = _restart ? AP_MISSION_CMD_INDEX_NONE : _nav_cmd.index;
|
|
if (cmd_index == AP_MISSION_CMD_INDEX_NONE) {
|
|
cmd_index = AP_MISSION_FIRST_REAL_COMMAND;
|
|
}
|
|
|
|
// check a maximum of 16 items, remembering that missions can have
|
|
// loops in them
|
|
for (uint8_t i=0; i<16; i++, cmd_index++) {
|
|
if (!get_next_nav_cmd(cmd_index, cmd)) {
|
|
return false;
|
|
}
|
|
switch (cmd.id) {
|
|
// any of these are considered a takeoff command:
|
|
case MAV_CMD_NAV_TAKEOFF:
|
|
case MAV_CMD_NAV_TAKEOFF_LOCAL:
|
|
return true;
|
|
// any of these are considered "skippable" when determining if
|
|
// we "start with a takeoff command"
|
|
case MAV_CMD_NAV_DELAY:
|
|
continue;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/// start_or_resume - if MIS_AUTORESTART=0 this will call resume(), otherwise it will call start()
|
|
void AP_Mission::start_or_resume()
|
|
{
|
|
if (_restart) {
|
|
start();
|
|
} else {
|
|
resume();
|
|
}
|
|
}
|
|
|
|
/// reset - reset mission to the first command
|
|
void AP_Mission::reset()
|
|
{
|
|
_flags.nav_cmd_loaded = false;
|
|
_flags.do_cmd_loaded = false;
|
|
_flags.do_cmd_all_done = false;
|
|
_nav_cmd.index = AP_MISSION_CMD_INDEX_NONE;
|
|
_do_cmd.index = AP_MISSION_CMD_INDEX_NONE;
|
|
_prev_nav_cmd_index = AP_MISSION_CMD_INDEX_NONE;
|
|
_prev_nav_cmd_wp_index = AP_MISSION_CMD_INDEX_NONE;
|
|
_prev_nav_cmd_id = AP_MISSION_CMD_ID_NONE;
|
|
init_jump_tracking();
|
|
}
|
|
|
|
/// clear - clears out mission
|
|
/// returns true if mission was running so it could not be cleared
|
|
bool AP_Mission::clear()
|
|
{
|
|
// do not allow clearing the mission while it is running
|
|
if (_flags.state == MISSION_RUNNING) {
|
|
return false;
|
|
}
|
|
|
|
// remove all commands
|
|
_cmd_total.set_and_save(0);
|
|
|
|
// clear index to commands
|
|
_nav_cmd.index = AP_MISSION_CMD_INDEX_NONE;
|
|
_do_cmd.index = AP_MISSION_CMD_INDEX_NONE;
|
|
_flags.nav_cmd_loaded = false;
|
|
_flags.do_cmd_loaded = false;
|
|
|
|
// return success
|
|
return true;
|
|
}
|
|
|
|
|
|
/// trucate - truncate any mission items beyond index
|
|
void AP_Mission::truncate(uint16_t index)
|
|
{
|
|
if ((unsigned)_cmd_total > index) {
|
|
_cmd_total.set_and_save(index);
|
|
}
|
|
}
|
|
|
|
/// update - ensures the command queues are loaded with the next command and calls main programs command_init and command_verify functions to progress the mission
|
|
/// should be called at 10hz or higher
|
|
void AP_Mission::update()
|
|
{
|
|
// exit immediately if not running or no mission commands
|
|
if (_flags.state != MISSION_RUNNING || _cmd_total == 0) {
|
|
return;
|
|
}
|
|
|
|
// save persistent waypoint_num for watchdog restore
|
|
hal.util->persistent_data.waypoint_num = _nav_cmd.index;
|
|
|
|
// check if we have an active nav command
|
|
if (!_flags.nav_cmd_loaded || _nav_cmd.index == AP_MISSION_CMD_INDEX_NONE) {
|
|
// advance in mission if no active nav command
|
|
if (!advance_current_nav_cmd()) {
|
|
// failure to advance nav command means mission has completed
|
|
complete();
|
|
return;
|
|
}
|
|
}else{
|
|
// run the active nav command
|
|
if (verify_command(_nav_cmd)) {
|
|
// market _nav_cmd as complete (it will be started on the next iteration)
|
|
_flags.nav_cmd_loaded = false;
|
|
// immediately advance to the next mission command
|
|
if (!advance_current_nav_cmd()) {
|
|
// failure to advance nav command means mission has completed
|
|
complete();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// check if we have an active do command
|
|
if (!_flags.do_cmd_loaded) {
|
|
advance_current_do_cmd();
|
|
}else{
|
|
// check the active do command
|
|
if (verify_command(_do_cmd)) {
|
|
// mark _do_cmd as complete
|
|
_flags.do_cmd_loaded = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool AP_Mission::verify_command(const Mission_Command& cmd)
|
|
{
|
|
switch (cmd.id) {
|
|
// do-commands always return true for verify:
|
|
case MAV_CMD_DO_GRIPPER:
|
|
case MAV_CMD_DO_SET_SERVO:
|
|
case MAV_CMD_DO_SET_RELAY:
|
|
case MAV_CMD_DO_REPEAT_SERVO:
|
|
case MAV_CMD_DO_REPEAT_RELAY:
|
|
case MAV_CMD_DO_DIGICAM_CONFIGURE:
|
|
case MAV_CMD_DO_DIGICAM_CONTROL:
|
|
case MAV_CMD_DO_SET_CAM_TRIGG_DIST:
|
|
case MAV_CMD_DO_PARACHUTE:
|
|
return true;
|
|
default:
|
|
return _cmd_verify_fn(cmd);
|
|
}
|
|
}
|
|
|
|
bool AP_Mission::start_command(const Mission_Command& cmd)
|
|
{
|
|
gcs().send_text(MAV_SEVERITY_INFO, "Mission: %u %s", cmd.index, cmd.type());
|
|
switch (cmd.id) {
|
|
case MAV_CMD_DO_GRIPPER:
|
|
return start_command_do_gripper(cmd);
|
|
case MAV_CMD_DO_SET_SERVO:
|
|
case MAV_CMD_DO_SET_RELAY:
|
|
case MAV_CMD_DO_REPEAT_SERVO:
|
|
case MAV_CMD_DO_REPEAT_RELAY:
|
|
return start_command_do_servorelayevents(cmd);
|
|
case MAV_CMD_DO_CONTROL_VIDEO:
|
|
case MAV_CMD_DO_DIGICAM_CONFIGURE:
|
|
case MAV_CMD_DO_DIGICAM_CONTROL:
|
|
case MAV_CMD_DO_SET_CAM_TRIGG_DIST:
|
|
return start_command_camera(cmd);
|
|
case MAV_CMD_DO_PARACHUTE:
|
|
return start_command_parachute(cmd);
|
|
default:
|
|
return _cmd_start_fn(cmd);
|
|
}
|
|
}
|
|
|
|
///
|
|
/// public command methods
|
|
///
|
|
|
|
/// add_cmd - adds a command to the end of the command list and writes to storage
|
|
/// returns true if successfully added, false on failure
|
|
/// cmd.index is updated with it's new position in the mission
|
|
bool AP_Mission::add_cmd(Mission_Command& cmd)
|
|
{
|
|
// attempt to write the command to storage
|
|
bool ret = write_cmd_to_storage(_cmd_total, cmd);
|
|
|
|
if (ret) {
|
|
// update command's index
|
|
cmd.index = _cmd_total;
|
|
// increment total number of commands
|
|
_cmd_total.set_and_save(_cmd_total + 1);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/// replace_cmd - replaces the command at position 'index' in the command list with the provided cmd
|
|
/// replacing the current active command will have no effect until the command is restarted
|
|
/// returns true if successfully replaced, false on failure
|
|
bool AP_Mission::replace_cmd(uint16_t index, const Mission_Command& cmd)
|
|
{
|
|
// sanity check index
|
|
if (index >= (unsigned)_cmd_total) {
|
|
return false;
|
|
}
|
|
|
|
// attempt to write the command to storage
|
|
return write_cmd_to_storage(index, cmd);
|
|
}
|
|
|
|
/// is_nav_cmd - returns true if the command's id is a "navigation" command, false if "do" or "conditional" command
|
|
bool AP_Mission::is_nav_cmd(const Mission_Command& cmd)
|
|
{
|
|
// NAV commands all have ids below MAV_CMD_NAV_LAST except NAV_SET_YAW_SPEED
|
|
return (cmd.id <= MAV_CMD_NAV_LAST || cmd.id == MAV_CMD_NAV_SET_YAW_SPEED);
|
|
}
|
|
|
|
/// get_next_nav_cmd - gets next "navigation" command found at or after start_index
|
|
/// returns true if found, false if not found (i.e. reached end of mission command list)
|
|
/// accounts for do_jump commands but never increments the jump's num_times_run (advance_current_nav_cmd is responsible for this)
|
|
bool AP_Mission::get_next_nav_cmd(uint16_t start_index, Mission_Command& cmd)
|
|
{
|
|
// search until the end of the mission command list
|
|
for (uint16_t cmd_index = start_index; cmd_index < (unsigned)_cmd_total; cmd_index++) {
|
|
// get next command
|
|
if (!get_next_cmd(cmd_index, cmd, false)) {
|
|
// no more commands so return failure
|
|
return false;
|
|
}
|
|
// if found a "navigation" command then return it
|
|
if (is_nav_cmd(cmd)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// if we got this far we did not find a navigation command
|
|
return false;
|
|
}
|
|
|
|
/// get the ground course of the next navigation leg in centidegrees
|
|
/// from 0 36000. Return default_angle if next navigation
|
|
/// leg cannot be determined
|
|
int32_t AP_Mission::get_next_ground_course_cd(int32_t default_angle)
|
|
{
|
|
Mission_Command cmd;
|
|
if (!get_next_nav_cmd(_nav_cmd.index+1, cmd)) {
|
|
return default_angle;
|
|
}
|
|
// special handling for nav commands with no target location
|
|
if (cmd.id == MAV_CMD_NAV_GUIDED_ENABLE ||
|
|
cmd.id == MAV_CMD_NAV_DELAY) {
|
|
return default_angle;
|
|
}
|
|
if (cmd.id == MAV_CMD_NAV_SET_YAW_SPEED) {
|
|
return (_nav_cmd.content.set_yaw_speed.angle_deg * 100);
|
|
}
|
|
return _nav_cmd.content.location.get_bearing_to(cmd.content.location);
|
|
}
|
|
|
|
// set_current_cmd - jumps to command specified by index
|
|
bool AP_Mission::set_current_cmd(uint16_t index)
|
|
{
|
|
// sanity check index and that we have a mission
|
|
if (index >= (unsigned)_cmd_total || _cmd_total == 1) {
|
|
return false;
|
|
}
|
|
|
|
// stop the current running do command
|
|
_do_cmd.index = AP_MISSION_CMD_INDEX_NONE;
|
|
_flags.do_cmd_loaded = false;
|
|
_flags.do_cmd_all_done = false;
|
|
|
|
// stop current nav cmd
|
|
_flags.nav_cmd_loaded = false;
|
|
|
|
// if index is zero then the user wants to completely restart the mission
|
|
if (index == 0 || _flags.state == MISSION_COMPLETE) {
|
|
_prev_nav_cmd_id = AP_MISSION_CMD_ID_NONE;
|
|
_prev_nav_cmd_index = AP_MISSION_CMD_INDEX_NONE;
|
|
_prev_nav_cmd_wp_index = AP_MISSION_CMD_INDEX_NONE;
|
|
// reset the jump tracking to zero
|
|
init_jump_tracking();
|
|
if (index == 0) {
|
|
index = 1;
|
|
}
|
|
}
|
|
|
|
// if the mission is stopped or completed move the nav_cmd index to the specified point and set the state to stopped
|
|
// so that if the user resumes the mission it will begin at the specified index
|
|
if (_flags.state != MISSION_RUNNING) {
|
|
// search until we find next nav command or reach end of command list
|
|
while (!_flags.nav_cmd_loaded) {
|
|
// get next command
|
|
Mission_Command cmd;
|
|
if (!get_next_cmd(index, cmd, true)) {
|
|
_nav_cmd.index = AP_MISSION_CMD_INDEX_NONE;
|
|
return false;
|
|
}
|
|
|
|
// check if navigation or "do" command
|
|
if (is_nav_cmd(cmd)) {
|
|
// set current navigation command
|
|
_nav_cmd = cmd;
|
|
_flags.nav_cmd_loaded = true;
|
|
}else{
|
|
// set current do command
|
|
if (!_flags.do_cmd_loaded) {
|
|
_do_cmd = cmd;
|
|
_flags.do_cmd_loaded = true;
|
|
}
|
|
}
|
|
// move onto next command
|
|
index = cmd.index+1;
|
|
}
|
|
|
|
// if we have not found a do command then set flag to show there are no do-commands to be run before nav command completes
|
|
if (!_flags.do_cmd_loaded) {
|
|
_flags.do_cmd_all_done = true;
|
|
}
|
|
|
|
// if we got this far then the mission can safely be "resumed" from the specified index so we set the state to "stopped"
|
|
_flags.state = MISSION_STOPPED;
|
|
return true;
|
|
}
|
|
|
|
// the state must be MISSION_RUNNING, allow advance_current_nav_cmd() to manage starting the item
|
|
if (!advance_current_nav_cmd(index)) {
|
|
// on failure set mission complete
|
|
complete();
|
|
return false;
|
|
}
|
|
|
|
// if we got this far we must have successfully advanced the nav command
|
|
return true;
|
|
}
|
|
|
|
struct PACKED Packed_Location_Option_Flags {
|
|
uint8_t relative_alt : 1; // 1 if altitude is relative to home
|
|
uint8_t unused1 : 1; // unused flag (defined so that loiter_ccw uses the correct bit)
|
|
uint8_t loiter_ccw : 1; // 0 if clockwise, 1 if counter clockwise
|
|
uint8_t terrain_alt : 1; // this altitude is above terrain
|
|
uint8_t origin_alt : 1; // this altitude is above ekf origin
|
|
uint8_t loiter_xtrack : 1; // 0 to crosstrack from center of waypoint, 1 to crosstrack from tangent exit location
|
|
};
|
|
|
|
struct PACKED PackedLocation {
|
|
union {
|
|
Packed_Location_Option_Flags flags; ///< options bitmask (1<<0 = relative altitude)
|
|
uint8_t options; /// allows writing all flags to eeprom as one byte
|
|
};
|
|
// by making alt 24 bit we can make p1 in a command 16 bit,
|
|
// allowing an accurate angle in centi-degrees. This keeps the
|
|
// storage cost per mission item at 15 bytes, and allows mission
|
|
// altitudes of up to +/- 83km
|
|
int32_t alt:24; ///< param 2 - Altitude in centimeters (meters * 100) see LOCATION_ALT_MAX_M
|
|
int32_t lat; ///< param 3 - Latitude * 10**7
|
|
int32_t lng; ///< param 4 - Longitude * 10**7
|
|
};
|
|
|
|
union PackedContent {
|
|
// location
|
|
PackedLocation location; // Waypoint location
|
|
|
|
// raw bytes, for reading/writing to eeprom. Note that only 10
|
|
// bytes are available if a 16 bit command ID is used
|
|
uint8_t bytes[12];
|
|
|
|
};
|
|
|
|
assert_storage_size<PackedContent, 12> assert_storage_size_PackedContent;
|
|
|
|
/// load_cmd_from_storage - load command from storage
|
|
/// true is return if successful
|
|
bool AP_Mission::read_cmd_from_storage(uint16_t index, Mission_Command& cmd) const
|
|
{
|
|
WITH_SEMAPHORE(_rsem);
|
|
|
|
// special handling for command #0 which is home
|
|
if (index == 0) {
|
|
cmd.index = 0;
|
|
cmd.id = MAV_CMD_NAV_WAYPOINT;
|
|
cmd.p1 = 0;
|
|
cmd.content.location = AP::ahrs().get_home();
|
|
return true;
|
|
}
|
|
|
|
if (index >= (unsigned)_cmd_total) {
|
|
return false;
|
|
}
|
|
|
|
// Find out proper location in memory by using the start_byte position + the index
|
|
// we can load a command, we don't process it yet
|
|
// read WP position
|
|
const uint16_t pos_in_storage = 4 + (index * AP_MISSION_EEPROM_COMMAND_SIZE);
|
|
|
|
PackedContent packed_content {};
|
|
|
|
const uint8_t b1 = _storage.read_byte(pos_in_storage);
|
|
if (b1 == 0) {
|
|
cmd.id = _storage.read_uint16(pos_in_storage+1);
|
|
cmd.p1 = _storage.read_uint16(pos_in_storage+3);
|
|
_storage.read_block(packed_content.bytes, pos_in_storage+5, 10);
|
|
} else {
|
|
cmd.id = b1;
|
|
cmd.p1 = _storage.read_uint16(pos_in_storage+1);
|
|
_storage.read_block(packed_content.bytes, pos_in_storage+3, 12);
|
|
}
|
|
|
|
if (stored_in_location(cmd.id)) {
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
// NOTE! no 16-bit command may be stored_in_location as only
|
|
// 10 bytes are available for storage and lat/lon/alt required
|
|
// 4*sizeof(float) == 12 bytes of storage.
|
|
if (b1 == 0) {
|
|
AP_HAL::panic("May not store location for 16-bit commands");
|
|
}
|
|
#endif
|
|
// Location is not PACKED; field-wise copy it:
|
|
cmd.content.location.relative_alt = packed_content.location.flags.relative_alt;
|
|
cmd.content.location.loiter_ccw = packed_content.location.flags.loiter_ccw;
|
|
cmd.content.location.terrain_alt = packed_content.location.flags.terrain_alt;
|
|
cmd.content.location.origin_alt = packed_content.location.flags.origin_alt;
|
|
cmd.content.location.loiter_xtrack = packed_content.location.flags.loiter_xtrack;
|
|
cmd.content.location.alt = packed_content.location.alt;
|
|
cmd.content.location.lat = packed_content.location.lat;
|
|
cmd.content.location.lng = packed_content.location.lng;
|
|
} else {
|
|
// all other options in Content are assumed to be packed:
|
|
static_assert(sizeof(cmd.content) >= 12,
|
|
"content is big enough to take bytes");
|
|
// (void *) cast to specify gcc that we know that we are copy byte into a non trivial type and leaving 4 bytes untouched
|
|
memcpy((void *)&cmd.content, packed_content.bytes, 12);
|
|
}
|
|
|
|
// set command's index to it's position in eeprom
|
|
cmd.index = index;
|
|
|
|
// return success
|
|
return true;
|
|
}
|
|
|
|
bool AP_Mission::stored_in_location(uint16_t id)
|
|
{
|
|
switch (id) {
|
|
case MAV_CMD_NAV_WAYPOINT:
|
|
case MAV_CMD_NAV_LOITER_UNLIM:
|
|
case MAV_CMD_NAV_LOITER_TURNS:
|
|
case MAV_CMD_NAV_LOITER_TIME:
|
|
case MAV_CMD_NAV_LAND:
|
|
case MAV_CMD_NAV_TAKEOFF:
|
|
case MAV_CMD_NAV_CONTINUE_AND_CHANGE_ALT:
|
|
case MAV_CMD_NAV_LOITER_TO_ALT:
|
|
case MAV_CMD_NAV_SPLINE_WAYPOINT:
|
|
case MAV_CMD_NAV_GUIDED_ENABLE:
|
|
case MAV_CMD_DO_SET_HOME:
|
|
case MAV_CMD_DO_LAND_START:
|
|
case MAV_CMD_DO_GO_AROUND:
|
|
case MAV_CMD_DO_SET_ROI:
|
|
case MAV_CMD_NAV_VTOL_TAKEOFF:
|
|
case MAV_CMD_NAV_VTOL_LAND:
|
|
case MAV_CMD_NAV_PAYLOAD_PLACE:
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// write_cmd_to_storage - write a command to storage
|
|
/// index is used to calculate the storage location
|
|
/// true is returned if successful
|
|
bool AP_Mission::write_cmd_to_storage(uint16_t index, const Mission_Command& cmd)
|
|
{
|
|
WITH_SEMAPHORE(_rsem);
|
|
|
|
// range check cmd's index
|
|
if (index >= num_commands_max()) {
|
|
return false;
|
|
}
|
|
|
|
PackedContent packed {};
|
|
if (stored_in_location(cmd.id)) {
|
|
// Location is not PACKED; field-wise copy it:
|
|
packed.location.flags.relative_alt = cmd.content.location.relative_alt;
|
|
packed.location.flags.loiter_ccw = cmd.content.location.loiter_ccw;
|
|
packed.location.flags.terrain_alt = cmd.content.location.terrain_alt;
|
|
packed.location.flags.origin_alt = cmd.content.location.origin_alt;
|
|
packed.location.flags.loiter_xtrack = cmd.content.location.loiter_xtrack;
|
|
packed.location.alt = cmd.content.location.alt;
|
|
packed.location.lat = cmd.content.location.lat;
|
|
packed.location.lng = cmd.content.location.lng;
|
|
} else {
|
|
// all other options in Content are assumed to be packed:
|
|
static_assert(sizeof(packed.bytes) >= 12,
|
|
"packed.bytes is big enough to take content");
|
|
memcpy(packed.bytes, &cmd.content, 12);
|
|
}
|
|
|
|
// calculate where in storage the command should be placed
|
|
uint16_t pos_in_storage = 4 + (index * AP_MISSION_EEPROM_COMMAND_SIZE);
|
|
|
|
if (cmd.id < 256) {
|
|
_storage.write_byte(pos_in_storage, cmd.id);
|
|
_storage.write_uint16(pos_in_storage+1, cmd.p1);
|
|
_storage.write_block(pos_in_storage+3, packed.bytes, 12);
|
|
} else {
|
|
// if the command ID is above 256 we store a 0 followed by the 16 bit command ID
|
|
_storage.write_byte(pos_in_storage, 0);
|
|
_storage.write_uint16(pos_in_storage+1, cmd.id);
|
|
_storage.write_uint16(pos_in_storage+3, cmd.p1);
|
|
_storage.write_block(pos_in_storage+5, packed.bytes, 10);
|
|
}
|
|
|
|
// remember when the mission last changed
|
|
_last_change_time_ms = AP_HAL::millis();
|
|
|
|
// return success
|
|
return true;
|
|
}
|
|
|
|
/// write_home_to_storage - writes the special purpose cmd 0 (home) to storage
|
|
/// home is taken directly from ahrs
|
|
void AP_Mission::write_home_to_storage()
|
|
{
|
|
Mission_Command home_cmd = {};
|
|
home_cmd.id = MAV_CMD_NAV_WAYPOINT;
|
|
home_cmd.content.location = AP::ahrs().get_home();
|
|
write_cmd_to_storage(0,home_cmd);
|
|
}
|
|
|
|
MAV_MISSION_RESULT AP_Mission::sanity_check_params(const mavlink_mission_item_int_t& packet) {
|
|
uint8_t nan_mask;
|
|
switch (packet.command) {
|
|
case MAV_CMD_NAV_WAYPOINT:
|
|
nan_mask = ~(1 << 3); // param 4 can be nan
|
|
break;
|
|
case MAV_CMD_NAV_LAND:
|
|
nan_mask = ~(1 << 3); // param 4 can be nan
|
|
break;
|
|
case MAV_CMD_NAV_TAKEOFF:
|
|
nan_mask = ~(1 << 3); // param 4 can be nan
|
|
break;
|
|
case MAV_CMD_NAV_VTOL_TAKEOFF:
|
|
nan_mask = ~(1 << 3); // param 4 can be nan
|
|
break;
|
|
case MAV_CMD_NAV_VTOL_LAND:
|
|
nan_mask = ~((1 << 2) | (1 << 3)); // param 3 and 4 can be nan
|
|
break;
|
|
default:
|
|
nan_mask = 0xff;
|
|
break;
|
|
}
|
|
|
|
if (((nan_mask & (1 << 0)) && isnan(packet.param1)) ||
|
|
isinf(packet.param1)) {
|
|
return MAV_MISSION_INVALID_PARAM1;
|
|
}
|
|
if (((nan_mask & (1 << 1)) && isnan(packet.param2)) ||
|
|
isinf(packet.param2)) {
|
|
return MAV_MISSION_INVALID_PARAM2;
|
|
}
|
|
if (((nan_mask & (1 << 2)) && isnan(packet.param3)) ||
|
|
isinf(packet.param3)) {
|
|
return MAV_MISSION_INVALID_PARAM3;
|
|
}
|
|
if (((nan_mask & (1 << 3)) && isnan(packet.param4)) ||
|
|
isinf(packet.param4)) {
|
|
return MAV_MISSION_INVALID_PARAM4;
|
|
}
|
|
return MAV_MISSION_ACCEPTED;
|
|
}
|
|
|
|
// mavlink_int_to_mission_cmd - converts mavlink message to an AP_Mission::Mission_Command object which can be stored to eeprom
|
|
// return MAV_MISSION_ACCEPTED on success, MAV_MISSION_RESULT error on failure
|
|
MAV_MISSION_RESULT AP_Mission::mavlink_int_to_mission_cmd(const mavlink_mission_item_int_t& packet, AP_Mission::Mission_Command& cmd)
|
|
{
|
|
// command's position in mission list and mavlink id
|
|
cmd.index = packet.seq;
|
|
cmd.id = packet.command;
|
|
cmd.content.location = {};
|
|
|
|
MAV_MISSION_RESULT param_check = sanity_check_params(packet);
|
|
if (param_check != MAV_MISSION_ACCEPTED) {
|
|
return param_check;
|
|
}
|
|
|
|
// command specific conversions from mavlink packet to mission command
|
|
switch (cmd.id) {
|
|
|
|
case 0:
|
|
// this is reserved for storing 16 bit command IDs
|
|
return MAV_MISSION_INVALID;
|
|
|
|
case MAV_CMD_NAV_WAYPOINT: // MAV ID: 16
|
|
{
|
|
/*
|
|
the 15 byte limit means we can't fit both delay and radius
|
|
in the cmd structure. When we expand the mission structure
|
|
we can do this properly
|
|
*/
|
|
#if APM_BUILD_TYPE(APM_BUILD_ArduPlane)
|
|
// acceptance radius in meters and pass by distance in meters
|
|
uint16_t acp = packet.param2; // param 2 is acceptance radius in meters is held in low p1
|
|
uint16_t passby = packet.param3; // param 3 is pass by distance in meters is held in high p1
|
|
|
|
// limit to 255 so it does not wrap during the shift or mask operation
|
|
passby = MIN(0xFF,passby);
|
|
acp = MIN(0xFF,acp);
|
|
|
|
cmd.p1 = (passby << 8) | (acp & 0x00FF);
|
|
#else
|
|
// delay at waypoint in seconds (this is for copters???)
|
|
cmd.p1 = packet.param1;
|
|
#endif
|
|
}
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_UNLIM: // MAV ID: 17
|
|
cmd.p1 = fabsf(packet.param3); // store radius as 16bit since no other params are competing for space
|
|
cmd.content.location.loiter_ccw = (packet.param3 < 0); // -1 = counter clockwise, +1 = clockwise
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TURNS: // MAV ID: 18
|
|
{
|
|
uint16_t num_turns = packet.param1; // param 1 is number of times to circle is held in low p1
|
|
uint16_t radius_m = fabsf(packet.param3); // param 3 is radius in meters is held in high p1
|
|
cmd.p1 = (radius_m<<8) | (num_turns & 0x00FF); // store radius in high byte of p1, num turns in low byte of p1
|
|
cmd.content.location.loiter_ccw = (packet.param3 < 0);
|
|
cmd.content.location.loiter_xtrack = (packet.param4 > 0); // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
|
|
}
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TIME: // MAV ID: 19
|
|
cmd.p1 = packet.param1; // loiter time in seconds uses all 16 bits, 8bit seconds is too small. No room for radius.
|
|
cmd.content.location.loiter_ccw = (packet.param3 < 0);
|
|
cmd.content.location.loiter_xtrack = (packet.param4 > 0); // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
|
|
break;
|
|
|
|
case MAV_CMD_NAV_RETURN_TO_LAUNCH: // MAV ID: 20
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LAND: // MAV ID: 21
|
|
cmd.p1 = packet.param1; // abort target altitude(m) (plane only)
|
|
cmd.content.location.loiter_ccw = is_negative(packet.param4); // yaw direction, (plane deepstall only)
|
|
break;
|
|
|
|
case MAV_CMD_NAV_TAKEOFF: // MAV ID: 22
|
|
cmd.p1 = packet.param1; // minimum pitch (plane only)
|
|
break;
|
|
|
|
case MAV_CMD_NAV_CONTINUE_AND_CHANGE_ALT: // MAV ID: 30
|
|
cmd.p1 = packet.param1; // Climb/Descend
|
|
// 0 = Neutral, cmd complete at +/- 5 of indicated alt.
|
|
// 1 = Climb, cmd complete at or above indicated alt.
|
|
// 2 = Descend, cmd complete at or below indicated alt.
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TO_ALT: // MAV ID: 31
|
|
cmd.p1 = fabsf(packet.param2); // param2 is radius in meters
|
|
cmd.content.location.loiter_ccw = (packet.param2 < 0);
|
|
cmd.content.location.loiter_xtrack = (packet.param4 > 0); // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
|
|
break;
|
|
|
|
case MAV_CMD_NAV_SPLINE_WAYPOINT: // MAV ID: 82
|
|
cmd.p1 = packet.param1; // delay at waypoint in seconds
|
|
break;
|
|
|
|
case MAV_CMD_NAV_GUIDED_ENABLE: // MAV ID: 92
|
|
cmd.p1 = packet.param1; // on/off. >0.5 means "on", hand-over control to external controller
|
|
break;
|
|
|
|
case MAV_CMD_NAV_DELAY: // MAV ID: 93
|
|
cmd.content.nav_delay.seconds = packet.param1; // delay in seconds
|
|
cmd.content.nav_delay.hour_utc = packet.param2;// absolute time's hour (utc)
|
|
cmd.content.nav_delay.min_utc = packet.param3;// absolute time's min (utc)
|
|
cmd.content.nav_delay.sec_utc = packet.param4; // absolute time's second (utc)
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DELAY: // MAV ID: 112
|
|
cmd.content.delay.seconds = packet.param1; // delay in seconds
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DISTANCE: // MAV ID: 114
|
|
cmd.content.distance.meters = packet.param1; // distance in meters from next waypoint
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_YAW: // MAV ID: 115
|
|
cmd.content.yaw.angle_deg = packet.param1; // target angle in degrees
|
|
cmd.content.yaw.turn_rate_dps = packet.param2; // 0 = use default turn rate otherwise specific turn rate in deg/sec
|
|
cmd.content.yaw.direction = packet.param3; // -1 = ccw, +1 = cw
|
|
cmd.content.yaw.relative_angle = packet.param4; // lng=0: absolute angle provided, lng=1: relative angle provided
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_MODE: // MAV ID: 176
|
|
cmd.p1 = packet.param1; // flight mode identifier
|
|
break;
|
|
|
|
case MAV_CMD_DO_JUMP: // MAV ID: 177
|
|
cmd.content.jump.target = packet.param1; // jump-to command number
|
|
cmd.content.jump.num_times = packet.param2; // repeat count
|
|
break;
|
|
|
|
case MAV_CMD_DO_CHANGE_SPEED: // MAV ID: 178
|
|
cmd.content.speed.speed_type = packet.param1; // 0 = airspeed, 1 = ground speed
|
|
cmd.content.speed.target_ms = packet.param2; // target speed in m/s
|
|
cmd.content.speed.throttle_pct = packet.param3; // throttle as a percentage from 0 ~ 100%
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_HOME:
|
|
cmd.p1 = packet.param1; // p1=0 means use current location, p=1 means use provided location
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_RELAY: // MAV ID: 181
|
|
cmd.content.relay.num = packet.param1; // relay number
|
|
cmd.content.relay.state = packet.param2; // 0:off, 1:on
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_RELAY: // MAV ID: 182
|
|
cmd.content.repeat_relay.num = packet.param1; // relay number
|
|
cmd.content.repeat_relay.repeat_count = packet.param2; // count
|
|
cmd.content.repeat_relay.cycle_time = packet.param3; // time converted from seconds to milliseconds
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_SERVO: // MAV ID: 183
|
|
cmd.content.servo.channel = packet.param1; // channel
|
|
cmd.content.servo.pwm = packet.param2; // PWM
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_SERVO: // MAV ID: 184
|
|
cmd.content.repeat_servo.channel = packet.param1; // channel
|
|
cmd.content.repeat_servo.pwm = packet.param2; // PWM
|
|
cmd.content.repeat_servo.repeat_count = packet.param3; // count
|
|
cmd.content.repeat_servo.cycle_time = packet.param4; // time in seconds
|
|
break;
|
|
|
|
case MAV_CMD_DO_LAND_START: // MAV ID: 189
|
|
break;
|
|
|
|
case MAV_CMD_DO_GO_AROUND: // MAV ID: 191
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_ROI: // MAV ID: 201
|
|
cmd.p1 = packet.param1; // 0 = no roi, 1 = next waypoint, 2 = waypoint number, 3 = fixed location, 4 = given target (not supported)
|
|
break;
|
|
|
|
case MAV_CMD_DO_DIGICAM_CONFIGURE: // MAV ID: 202
|
|
cmd.content.digicam_configure.shooting_mode = packet.param1;
|
|
cmd.content.digicam_configure.shutter_speed = packet.param2;
|
|
cmd.content.digicam_configure.aperture = packet.param3;
|
|
cmd.content.digicam_configure.ISO = packet.param4;
|
|
cmd.content.digicam_configure.exposure_type = packet.x;
|
|
cmd.content.digicam_configure.cmd_id = packet.y;
|
|
cmd.content.digicam_configure.engine_cutoff_time = packet.z;
|
|
break;
|
|
|
|
case MAV_CMD_DO_DIGICAM_CONTROL: // MAV ID: 203
|
|
cmd.content.digicam_control.session = packet.param1;
|
|
cmd.content.digicam_control.zoom_pos = packet.param2;
|
|
cmd.content.digicam_control.zoom_step = packet.param3;
|
|
cmd.content.digicam_control.focus_lock = packet.param4;
|
|
cmd.content.digicam_control.shooting_cmd = packet.x;
|
|
cmd.content.digicam_control.cmd_id = packet.y;
|
|
break;
|
|
|
|
case MAV_CMD_DO_MOUNT_CONTROL: // MAV ID: 205
|
|
cmd.content.mount_control.pitch = packet.param1;
|
|
cmd.content.mount_control.roll = packet.param2;
|
|
cmd.content.mount_control.yaw = packet.param3;
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_CAM_TRIGG_DIST: // MAV ID: 206
|
|
cmd.content.cam_trigg_dist.meters = packet.param1; // distance between camera shots in meters
|
|
break;
|
|
|
|
case MAV_CMD_DO_FENCE_ENABLE: // MAV ID: 207
|
|
cmd.p1 = packet.param1; // action 0=disable, 1=enable
|
|
break;
|
|
|
|
case MAV_CMD_DO_PARACHUTE: // MAV ID: 208
|
|
cmd.p1 = packet.param1; // action 0=disable, 1=enable, 2=release. See PARACHUTE_ACTION enum
|
|
break;
|
|
|
|
case MAV_CMD_DO_INVERTED_FLIGHT: // MAV ID: 210
|
|
cmd.p1 = packet.param1; // normal=0 inverted=1
|
|
break;
|
|
|
|
case MAV_CMD_DO_GRIPPER: // MAV ID: 211
|
|
cmd.content.gripper.num = packet.param1; // gripper number
|
|
cmd.content.gripper.action = packet.param2; // action 0=release, 1=grab. See GRIPPER_ACTION enum
|
|
break;
|
|
|
|
case MAV_CMD_DO_GUIDED_LIMITS: // MAV ID: 222
|
|
cmd.p1 = packet.param1; // max time in seconds the external controller will be allowed to control the vehicle
|
|
cmd.content.guided_limits.alt_min = packet.param2; // min alt below which the command will be aborted. 0 for no lower alt limit
|
|
cmd.content.guided_limits.alt_max = packet.param3; // max alt above which the command will be aborted. 0 for no upper alt limit
|
|
cmd.content.guided_limits.horiz_max = packet.param4;// max horizontal distance the vehicle can move before the command will be aborted. 0 for no horizontal limit
|
|
break;
|
|
|
|
case MAV_CMD_DO_AUTOTUNE_ENABLE: // MAV ID: 211
|
|
cmd.p1 = packet.param1; // disable=0 enable=1
|
|
break;
|
|
|
|
case MAV_CMD_NAV_ALTITUDE_WAIT: // MAV ID: 83
|
|
cmd.content.altitude_wait.altitude = packet.param1;
|
|
cmd.content.altitude_wait.descent_rate = packet.param2;
|
|
cmd.content.altitude_wait.wiggle_time = packet.param3;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_VTOL_TAKEOFF:
|
|
break;
|
|
|
|
case MAV_CMD_NAV_VTOL_LAND:
|
|
break;
|
|
|
|
case MAV_CMD_DO_VTOL_TRANSITION:
|
|
cmd.content.do_vtol_transition.target_state = packet.param1;
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_REVERSE:
|
|
cmd.p1 = packet.param1; // 0 = forward, 1 = reverse
|
|
break;
|
|
|
|
case MAV_CMD_DO_ENGINE_CONTROL:
|
|
cmd.content.do_engine_control.start_control = (packet.param1>0);
|
|
cmd.content.do_engine_control.cold_start = (packet.param2>0);
|
|
cmd.content.do_engine_control.height_delay_cm = packet.param3*100;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_PAYLOAD_PLACE:
|
|
cmd.p1 = packet.param1*100; // copy max-descend parameter (m->cm)
|
|
break;
|
|
|
|
case MAV_CMD_NAV_SET_YAW_SPEED:
|
|
cmd.content.set_yaw_speed.angle_deg = packet.param1; // target angle in degrees
|
|
cmd.content.set_yaw_speed.speed = packet.param2; // speed in meters/second
|
|
cmd.content.set_yaw_speed.relative_angle = packet.param3; // 0 = absolute angle, 1 = relative angle
|
|
break;
|
|
|
|
case MAV_CMD_DO_WINCH: // MAV ID: 42600
|
|
cmd.content.winch.num = packet.param1; // winch number
|
|
cmd.content.winch.action = packet.param2; // action (0 = relax, 1 = length control, 2 = rate control). See WINCH_ACTION enum
|
|
cmd.content.winch.release_length = packet.param3; // cable distance to unwind in meters, negative numbers to wind in cable
|
|
cmd.content.winch.release_rate = packet.param4; // release rate in meters/second
|
|
break;
|
|
|
|
default:
|
|
// unrecognised command
|
|
return MAV_MISSION_UNSUPPORTED;
|
|
}
|
|
|
|
// copy location from mavlink to command
|
|
if (stored_in_location(cmd.id)) {
|
|
|
|
// sanity check location
|
|
if (!check_lat(packet.x)) {
|
|
return MAV_MISSION_INVALID_PARAM5_X;
|
|
}
|
|
if (!check_lng(packet.y)) {
|
|
return MAV_MISSION_INVALID_PARAM6_Y;
|
|
}
|
|
if (isnan(packet.z) || fabsf(packet.z) >= LOCATION_ALT_MAX_M) {
|
|
return MAV_MISSION_INVALID_PARAM7;
|
|
}
|
|
|
|
cmd.content.location.lat = packet.x;
|
|
cmd.content.location.lng = packet.y;
|
|
|
|
cmd.content.location.alt = packet.z * 100.0f; // convert packet's alt (m) to cmd alt (cm)
|
|
|
|
switch (packet.frame) {
|
|
|
|
case MAV_FRAME_MISSION:
|
|
case MAV_FRAME_GLOBAL:
|
|
cmd.content.location.relative_alt = 0;
|
|
break;
|
|
|
|
case MAV_FRAME_GLOBAL_RELATIVE_ALT:
|
|
cmd.content.location.relative_alt = 1;
|
|
break;
|
|
|
|
#if AP_TERRAIN_AVAILABLE
|
|
case MAV_FRAME_GLOBAL_TERRAIN_ALT:
|
|
// we mark it as a relative altitude, as it doesn't have
|
|
// home alt added
|
|
cmd.content.location.relative_alt = 1;
|
|
// mark altitude as above terrain, not above home
|
|
cmd.content.location.terrain_alt = 1;
|
|
break;
|
|
#endif
|
|
|
|
default:
|
|
return MAV_MISSION_UNSUPPORTED_FRAME;
|
|
}
|
|
}
|
|
|
|
// if we got this far then it must have been successful
|
|
return MAV_MISSION_ACCEPTED;
|
|
}
|
|
|
|
MAV_MISSION_RESULT AP_Mission::convert_MISSION_ITEM_to_MISSION_ITEM_INT(const mavlink_mission_item_t &packet,
|
|
mavlink_mission_item_int_t &mav_cmd)
|
|
{
|
|
// TODO: rename mav_cmd to mission_item_int
|
|
// TODO: rename packet to mission_item
|
|
mav_cmd.param1 = packet.param1;
|
|
mav_cmd.param2 = packet.param2;
|
|
mav_cmd.param3 = packet.param3;
|
|
mav_cmd.param4 = packet.param4;
|
|
mav_cmd.z = packet.z;
|
|
mav_cmd.seq = packet.seq;
|
|
mav_cmd.command = packet.command;
|
|
mav_cmd.target_system = packet.target_system;
|
|
mav_cmd.target_component = packet.target_component;
|
|
mav_cmd.frame = packet.frame;
|
|
mav_cmd.current = packet.current;
|
|
mav_cmd.autocontinue = packet.autocontinue;
|
|
mav_cmd.mission_type = packet.mission_type;
|
|
|
|
/*
|
|
the strategy for handling both MISSION_ITEM and MISSION_ITEM_INT
|
|
is to pass the lat/lng in MISSION_ITEM_INT straight through, and
|
|
for MISSION_ITEM multiply by 1e7 here. We need an exception for
|
|
any commands which use the x and y fields not as
|
|
latitude/longitude.
|
|
*/
|
|
switch (packet.command) {
|
|
case MAV_CMD_DO_DIGICAM_CONTROL:
|
|
case MAV_CMD_DO_DIGICAM_CONFIGURE:
|
|
mav_cmd.x = packet.x;
|
|
mav_cmd.y = packet.y;
|
|
break;
|
|
|
|
default:
|
|
// all other commands use x and y as lat/lon. We need to
|
|
// multiply by 1e7 to convert to int32_t
|
|
if (!check_lat(packet.x)) {
|
|
return MAV_MISSION_INVALID_PARAM5_X;
|
|
}
|
|
if (!check_lng(packet.y)) {
|
|
return MAV_MISSION_INVALID_PARAM6_Y;
|
|
}
|
|
mav_cmd.x = packet.x * 1.0e7f;
|
|
mav_cmd.y = packet.y * 1.0e7f;
|
|
break;
|
|
}
|
|
|
|
return MAV_MISSION_ACCEPTED;
|
|
}
|
|
|
|
MAV_MISSION_RESULT AP_Mission::convert_MISSION_ITEM_INT_to_MISSION_ITEM(const mavlink_mission_item_int_t &item_int,
|
|
mavlink_mission_item_t &item)
|
|
{
|
|
item.param1 = item_int.param1;
|
|
item.param2 = item_int.param2;
|
|
item.param3 = item_int.param3;
|
|
item.param4 = item_int.param4;
|
|
item.z = item_int.z;
|
|
item.seq = item_int.seq;
|
|
item.command = item_int.command;
|
|
item.target_system = item_int.target_system;
|
|
item.target_component = item_int.target_component;
|
|
item.frame = item_int.frame;
|
|
item.current = item_int.current;
|
|
item.autocontinue = item_int.autocontinue;
|
|
item.mission_type = item_int.mission_type;
|
|
|
|
switch (item_int.command) {
|
|
case MAV_CMD_DO_DIGICAM_CONTROL:
|
|
case MAV_CMD_DO_DIGICAM_CONFIGURE:
|
|
item.x = item_int.x;
|
|
item.y = item_int.y;
|
|
break;
|
|
|
|
default:
|
|
// all other commands use x and y as lat/lon. We need to
|
|
// multiply by 1e-7 to convert to float
|
|
item.x = item_int.x * 1.0e-7f;
|
|
item.y = item_int.y * 1.0e-7f;
|
|
if (!check_lat(item.x)) {
|
|
return MAV_MISSION_INVALID_PARAM5_X;
|
|
}
|
|
if (!check_lng(item.y)) {
|
|
return MAV_MISSION_INVALID_PARAM6_Y;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return MAV_MISSION_ACCEPTED;
|
|
}
|
|
|
|
// mavlink_cmd_long_to_mission_cmd - converts a mavlink cmd long to an AP_Mission::Mission_Command object which can be stored to eeprom
|
|
// return MAV_MISSION_ACCEPTED on success, MAV_MISSION_RESULT error on failure
|
|
MAV_MISSION_RESULT AP_Mission::mavlink_cmd_long_to_mission_cmd(const mavlink_command_long_t& packet, AP_Mission::Mission_Command& cmd)
|
|
{
|
|
mavlink_mission_item_int_t miss_item = {0};
|
|
|
|
miss_item.param1 = packet.param1;
|
|
miss_item.param2 = packet.param2;
|
|
miss_item.param3 = packet.param3;
|
|
miss_item.param4 = packet.param4;
|
|
|
|
miss_item.command = packet.command;
|
|
miss_item.target_system = packet.target_system;
|
|
miss_item.target_component = packet.target_component;
|
|
|
|
return mavlink_int_to_mission_cmd(miss_item, cmd);
|
|
}
|
|
|
|
// mission_cmd_to_mavlink_int - converts an AP_Mission::Mission_Command object to a mavlink message which can be sent to the GCS
|
|
// return true on success, false on failure
|
|
bool AP_Mission::mission_cmd_to_mavlink_int(const AP_Mission::Mission_Command& cmd, mavlink_mission_item_int_t& packet)
|
|
{
|
|
// command's position in mission list and mavlink id
|
|
packet.seq = cmd.index;
|
|
packet.command = cmd.id;
|
|
|
|
// set defaults
|
|
packet.current = 0; // 1 if we are passing back the mission command that is currently being executed
|
|
packet.param1 = 0;
|
|
packet.param2 = 0;
|
|
packet.param3 = 0;
|
|
packet.param4 = 0;
|
|
packet.autocontinue = 1;
|
|
|
|
// command specific conversions from mission command to mavlink packet
|
|
switch (cmd.id) {
|
|
case 0:
|
|
// this is reserved for 16 bit command IDs
|
|
return false;
|
|
|
|
case MAV_CMD_NAV_WAYPOINT: // MAV ID: 16
|
|
#if APM_BUILD_TYPE(APM_BUILD_ArduPlane)
|
|
// acceptance radius in meters
|
|
|
|
packet.param2 = LOWBYTE(cmd.p1); // param 2 is acceptance radius in meters is held in low p1
|
|
packet.param3 = HIGHBYTE(cmd.p1); // param 3 is pass by distance in meters is held in high p1
|
|
#else
|
|
// delay at waypoint in seconds
|
|
packet.param1 = cmd.p1;
|
|
#endif
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_UNLIM: // MAV ID: 17
|
|
packet.param3 = (float)cmd.p1;
|
|
if (cmd.content.location.loiter_ccw) {
|
|
packet.param3 *= -1;
|
|
}
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TURNS: // MAV ID: 18
|
|
packet.param1 = LOWBYTE(cmd.p1); // number of times to circle is held in low byte of p1
|
|
packet.param3 = HIGHBYTE(cmd.p1); // radius is held in high byte of p1
|
|
if (cmd.content.location.loiter_ccw) {
|
|
packet.param3 = -packet.param3;
|
|
}
|
|
packet.param4 = cmd.content.location.loiter_xtrack; // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TIME: // MAV ID: 19
|
|
packet.param1 = cmd.p1; // loiter time in seconds
|
|
if (cmd.content.location.loiter_ccw) {
|
|
packet.param3 = -1;
|
|
} else {
|
|
packet.param3 = 1;
|
|
}
|
|
packet.param4 = cmd.content.location.loiter_xtrack; // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
|
|
break;
|
|
|
|
case MAV_CMD_NAV_RETURN_TO_LAUNCH: // MAV ID: 20
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LAND: // MAV ID: 21
|
|
packet.param1 = cmd.p1; // abort target altitude(m) (plane only)
|
|
packet.param4 = cmd.content.location.loiter_ccw ? -1 : 1; // yaw direction, (plane deepstall only)
|
|
break;
|
|
|
|
case MAV_CMD_NAV_TAKEOFF: // MAV ID: 22
|
|
packet.param1 = cmd.p1; // minimum pitch (plane only)
|
|
break;
|
|
|
|
case MAV_CMD_NAV_CONTINUE_AND_CHANGE_ALT: // MAV ID: 30
|
|
packet.param1 = cmd.p1; // Climb/Descend
|
|
// 0 = Neutral, cmd complete at +/- 5 of indicated alt.
|
|
// 1 = Climb, cmd complete at or above indicated alt.
|
|
// 2 = Descend, cmd complete at or below indicated alt.
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TO_ALT: // MAV ID: 31
|
|
packet.param2 = cmd.p1; // loiter radius(m)
|
|
if (cmd.content.location.loiter_ccw) {
|
|
packet.param2 = -packet.param2;
|
|
}
|
|
packet.param4 = cmd.content.location.loiter_xtrack; // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
|
|
break;
|
|
|
|
case MAV_CMD_NAV_SPLINE_WAYPOINT: // MAV ID: 82
|
|
packet.param1 = cmd.p1; // delay at waypoint in seconds
|
|
break;
|
|
|
|
case MAV_CMD_NAV_GUIDED_ENABLE: // MAV ID: 92
|
|
packet.param1 = cmd.p1; // on/off. >0.5 means "on", hand-over control to external controller
|
|
break;
|
|
|
|
case MAV_CMD_NAV_DELAY: // MAV ID: 93
|
|
packet.param1 = cmd.content.nav_delay.seconds; // delay in seconds
|
|
packet.param2 = cmd.content.nav_delay.hour_utc; // absolute time's day of week (utc)
|
|
packet.param3 = cmd.content.nav_delay.min_utc; // absolute time's hour (utc)
|
|
packet.param4 = cmd.content.nav_delay.sec_utc; // absolute time's min (utc)
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DELAY: // MAV ID: 112
|
|
packet.param1 = cmd.content.delay.seconds; // delay in seconds
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DISTANCE: // MAV ID: 114
|
|
packet.param1 = cmd.content.distance.meters; // distance in meters from next waypoint
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_YAW: // MAV ID: 115
|
|
packet.param1 = cmd.content.yaw.angle_deg; // target angle in degrees
|
|
packet.param2 = cmd.content.yaw.turn_rate_dps; // 0 = use default turn rate otherwise specific turn rate in deg/sec
|
|
packet.param3 = cmd.content.yaw.direction; // -1 = ccw, +1 = cw
|
|
packet.param4 = cmd.content.yaw.relative_angle; // 0 = absolute angle provided, 1 = relative angle provided
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_MODE: // MAV ID: 176
|
|
packet.param1 = cmd.p1; // set flight mode identifier
|
|
break;
|
|
|
|
case MAV_CMD_DO_JUMP: // MAV ID: 177
|
|
packet.param1 = cmd.content.jump.target; // jump-to command number
|
|
packet.param2 = cmd.content.jump.num_times; // repeat count
|
|
break;
|
|
|
|
case MAV_CMD_DO_CHANGE_SPEED: // MAV ID: 178
|
|
packet.param1 = cmd.content.speed.speed_type; // 0 = airspeed, 1 = ground speed
|
|
packet.param2 = cmd.content.speed.target_ms; // speed in m/s
|
|
packet.param3 = cmd.content.speed.throttle_pct; // throttle as a percentage from 0 ~ 100%
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_HOME: // MAV ID: 179
|
|
packet.param1 = cmd.p1; // p1=0 means use current location, p=1 means use provided location
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_RELAY: // MAV ID: 181
|
|
packet.param1 = cmd.content.relay.num; // relay number
|
|
packet.param2 = cmd.content.relay.state; // 0:off, 1:on
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_RELAY: // MAV ID: 182
|
|
packet.param1 = cmd.content.repeat_relay.num; // relay number
|
|
packet.param2 = cmd.content.repeat_relay.repeat_count; // count
|
|
packet.param3 = cmd.content.repeat_relay.cycle_time; // time in seconds
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_SERVO: // MAV ID: 183
|
|
packet.param1 = cmd.content.servo.channel; // channel
|
|
packet.param2 = cmd.content.servo.pwm; // PWM
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_SERVO: // MAV ID: 184
|
|
packet.param1 = cmd.content.repeat_servo.channel; // channel
|
|
packet.param2 = cmd.content.repeat_servo.pwm; // PWM
|
|
packet.param3 = cmd.content.repeat_servo.repeat_count; // count
|
|
packet.param4 = cmd.content.repeat_servo.cycle_time; // time in milliseconds converted to seconds
|
|
break;
|
|
|
|
case MAV_CMD_DO_LAND_START: // MAV ID: 189
|
|
break;
|
|
|
|
case MAV_CMD_DO_GO_AROUND: // MAV ID: 191
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_ROI: // MAV ID: 201
|
|
packet.param1 = cmd.p1; // 0 = no roi, 1 = next waypoint, 2 = waypoint number, 3 = fixed location, 4 = given target (not supported)
|
|
break;
|
|
|
|
case MAV_CMD_DO_DIGICAM_CONFIGURE: // MAV ID: 202
|
|
packet.param1 = cmd.content.digicam_configure.shooting_mode;
|
|
packet.param2 = cmd.content.digicam_configure.shutter_speed;
|
|
packet.param3 = cmd.content.digicam_configure.aperture;
|
|
packet.param4 = cmd.content.digicam_configure.ISO;
|
|
packet.x = cmd.content.digicam_configure.exposure_type;
|
|
packet.y = cmd.content.digicam_configure.cmd_id;
|
|
packet.z = cmd.content.digicam_configure.engine_cutoff_time;
|
|
break;
|
|
|
|
case MAV_CMD_DO_DIGICAM_CONTROL: // MAV ID: 203
|
|
packet.param1 = cmd.content.digicam_control.session;
|
|
packet.param2 = cmd.content.digicam_control.zoom_pos;
|
|
packet.param3 = cmd.content.digicam_control.zoom_step;
|
|
packet.param4 = cmd.content.digicam_control.focus_lock;
|
|
packet.x = cmd.content.digicam_control.shooting_cmd;
|
|
packet.y = cmd.content.digicam_control.cmd_id;
|
|
break;
|
|
|
|
case MAV_CMD_DO_MOUNT_CONTROL: // MAV ID: 205
|
|
packet.param1 = cmd.content.mount_control.pitch;
|
|
packet.param2 = cmd.content.mount_control.roll;
|
|
packet.param3 = cmd.content.mount_control.yaw;
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_CAM_TRIGG_DIST: // MAV ID: 206
|
|
packet.param1 = cmd.content.cam_trigg_dist.meters; // distance between camera shots in meters
|
|
break;
|
|
|
|
case MAV_CMD_DO_FENCE_ENABLE: // MAV ID: 207
|
|
packet.param1 = cmd.p1; // action 0=disable, 1=enable
|
|
break;
|
|
|
|
case MAV_CMD_DO_PARACHUTE: // MAV ID: 208
|
|
packet.param1 = cmd.p1; // action 0=disable, 1=enable, 2=release. See PARACHUTE_ACTION enum
|
|
break;
|
|
|
|
case MAV_CMD_DO_INVERTED_FLIGHT: // MAV ID: 210
|
|
packet.param1 = cmd.p1; // normal=0 inverted=1
|
|
break;
|
|
|
|
case MAV_CMD_DO_GRIPPER: // MAV ID: 211
|
|
packet.param1 = cmd.content.gripper.num; // gripper number
|
|
packet.param2 = cmd.content.gripper.action; // action 0=release, 1=grab. See GRIPPER_ACTION enum
|
|
break;
|
|
|
|
case MAV_CMD_DO_GUIDED_LIMITS: // MAV ID: 222
|
|
packet.param1 = cmd.p1; // max time in seconds the external controller will be allowed to control the vehicle
|
|
packet.param2 = cmd.content.guided_limits.alt_min; // min alt below which the command will be aborted. 0 for no lower alt limit
|
|
packet.param3 = cmd.content.guided_limits.alt_max; // max alt above which the command will be aborted. 0 for no upper alt limit
|
|
packet.param4 = cmd.content.guided_limits.horiz_max;// max horizontal distance the vehicle can move before the command will be aborted. 0 for no horizontal limit
|
|
break;
|
|
|
|
case MAV_CMD_DO_AUTOTUNE_ENABLE:
|
|
packet.param1 = cmd.p1; // disable=0 enable=1
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_REVERSE:
|
|
packet.param1 = cmd.p1; // 0 = forward, 1 = reverse
|
|
break;
|
|
|
|
case MAV_CMD_NAV_ALTITUDE_WAIT: // MAV ID: 83
|
|
packet.param1 = cmd.content.altitude_wait.altitude;
|
|
packet.param2 = cmd.content.altitude_wait.descent_rate;
|
|
packet.param3 = cmd.content.altitude_wait.wiggle_time;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_VTOL_TAKEOFF:
|
|
break;
|
|
|
|
case MAV_CMD_NAV_VTOL_LAND:
|
|
break;
|
|
|
|
case MAV_CMD_DO_VTOL_TRANSITION:
|
|
packet.param1 = cmd.content.do_vtol_transition.target_state;
|
|
break;
|
|
|
|
case MAV_CMD_DO_ENGINE_CONTROL:
|
|
packet.param1 = cmd.content.do_engine_control.start_control?1:0;
|
|
packet.param2 = cmd.content.do_engine_control.cold_start?1:0;
|
|
packet.param3 = cmd.content.do_engine_control.height_delay_cm*0.01f;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_PAYLOAD_PLACE:
|
|
packet.param1 = cmd.p1/100.0f; // copy max-descend parameter (m->cm)
|
|
break;
|
|
|
|
case MAV_CMD_NAV_SET_YAW_SPEED:
|
|
packet.param1 = cmd.content.set_yaw_speed.angle_deg; // target angle in degrees
|
|
packet.param2 = cmd.content.set_yaw_speed.speed; // speed in meters/second
|
|
packet.param3 = cmd.content.set_yaw_speed.relative_angle; // 0 = absolute angle, 1 = relative angle
|
|
break;
|
|
|
|
case MAV_CMD_DO_WINCH:
|
|
packet.param1 = cmd.content.winch.num; // winch number
|
|
packet.param2 = cmd.content.winch.action; // action (0 = relax, 1 = length control, 2 = rate control). See WINCH_ACTION enum
|
|
packet.param3 = cmd.content.winch.release_length; // cable distance to unwind in meters, negative numbers to wind in cable
|
|
packet.param4 = cmd.content.winch.release_rate; // release rate in meters/second
|
|
break;
|
|
|
|
default:
|
|
// unrecognised command
|
|
return false;
|
|
}
|
|
|
|
// copy location from mavlink to command
|
|
if (stored_in_location(cmd.id)) {
|
|
packet.x = cmd.content.location.lat;
|
|
packet.y = cmd.content.location.lng;
|
|
|
|
packet.z = cmd.content.location.alt / 100.0f; // cmd alt in cm to m
|
|
if (cmd.content.location.relative_alt) {
|
|
packet.frame = MAV_FRAME_GLOBAL_RELATIVE_ALT;
|
|
}else{
|
|
packet.frame = MAV_FRAME_GLOBAL;
|
|
}
|
|
#if AP_TERRAIN_AVAILABLE
|
|
if (cmd.content.location.terrain_alt) {
|
|
// this is a above-terrain altitude
|
|
if (!cmd.content.location.relative_alt) {
|
|
// refuse to return non-relative terrain mission
|
|
// items. Internally we do have these, and they
|
|
// have home.alt added, but we should never be
|
|
// returning them to the GCS, as the GCS doesn't know
|
|
// our home.alt, so it would have no way to properly
|
|
// interpret it
|
|
return false;
|
|
}
|
|
packet.z = cmd.content.location.alt * 0.01f;
|
|
packet.frame = MAV_FRAME_GLOBAL_TERRAIN_ALT;
|
|
}
|
|
#else
|
|
// don't ever return terrain mission items if no terrain support
|
|
if (cmd.content.location.terrain_alt) {
|
|
return false;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// if we got this far then it must have been successful
|
|
return true;
|
|
}
|
|
|
|
///
|
|
/// private methods
|
|
///
|
|
|
|
/// complete - mission is marked complete and clean-up performed including calling the mission_complete_fn
|
|
void AP_Mission::complete()
|
|
{
|
|
// flag mission as complete
|
|
_flags.state = MISSION_COMPLETE;
|
|
|
|
// callback to main program's mission complete function
|
|
_mission_complete_fn();
|
|
}
|
|
|
|
/// advance_current_nav_cmd - moves current nav command forward
|
|
/// do command will also be loaded
|
|
/// accounts for do-jump commands
|
|
// returns true if command is advanced, false if failed (i.e. mission completed)
|
|
bool AP_Mission::advance_current_nav_cmd(uint16_t starting_index)
|
|
{
|
|
// exit immediately if we're not running
|
|
if (_flags.state != MISSION_RUNNING) {
|
|
return false;
|
|
}
|
|
|
|
// exit immediately if current nav command has not completed
|
|
if (_flags.nav_cmd_loaded) {
|
|
return false;
|
|
}
|
|
|
|
// stop the current running do command
|
|
_do_cmd.index = AP_MISSION_CMD_INDEX_NONE;
|
|
_flags.do_cmd_loaded = false;
|
|
_flags.do_cmd_all_done = false;
|
|
|
|
// get starting point for search
|
|
uint16_t cmd_index = starting_index > 0 ? starting_index - 1 : _nav_cmd.index;
|
|
if (cmd_index == AP_MISSION_CMD_INDEX_NONE) {
|
|
// start from beginning of the mission command list
|
|
cmd_index = AP_MISSION_FIRST_REAL_COMMAND;
|
|
}else{
|
|
// start from one position past the current nav command
|
|
cmd_index++;
|
|
}
|
|
|
|
// avoid endless loops
|
|
uint8_t max_loops = 255;
|
|
|
|
// search until we find next nav command or reach end of command list
|
|
while (!_flags.nav_cmd_loaded) {
|
|
// get next command
|
|
Mission_Command cmd;
|
|
if (!get_next_cmd(cmd_index, cmd, true)) {
|
|
return false;
|
|
}
|
|
|
|
// check if navigation or "do" command
|
|
if (is_nav_cmd(cmd)) {
|
|
// save previous nav command index
|
|
_prev_nav_cmd_id = _nav_cmd.id;
|
|
_prev_nav_cmd_index = _nav_cmd.index;
|
|
// save separate previous nav command index if it contains lat,long,alt
|
|
if (!(cmd.content.location.lat == 0 && cmd.content.location.lng == 0)) {
|
|
_prev_nav_cmd_wp_index = _nav_cmd.index;
|
|
}
|
|
// set current navigation command and start it
|
|
_nav_cmd = cmd;
|
|
if (start_command(_nav_cmd)) {
|
|
_flags.nav_cmd_loaded = true;
|
|
}
|
|
}else{
|
|
// set current do command and start it (if not already set)
|
|
if (!_flags.do_cmd_loaded) {
|
|
_do_cmd = cmd;
|
|
_flags.do_cmd_loaded = true;
|
|
start_command(_do_cmd);
|
|
} else {
|
|
// protect against endless loops of do-commands
|
|
if (max_loops-- == 0) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
// move onto next command
|
|
cmd_index = cmd.index+1;
|
|
}
|
|
|
|
// if we have not found a do command then set flag to show there are no do-commands to be run before nav command completes
|
|
if (!_flags.do_cmd_loaded) {
|
|
_flags.do_cmd_all_done = true;
|
|
}
|
|
|
|
// if we got this far we must have successfully advanced the nav command
|
|
return true;
|
|
}
|
|
|
|
/// advance_current_do_cmd - moves current do command forward
|
|
/// accounts for do-jump commands
|
|
void AP_Mission::advance_current_do_cmd()
|
|
{
|
|
// exit immediately if we're not running or we've completed all possible "do" commands
|
|
if (_flags.state != MISSION_RUNNING || _flags.do_cmd_all_done) {
|
|
return;
|
|
}
|
|
|
|
// get starting point for search
|
|
uint16_t cmd_index = _do_cmd.index;
|
|
if (cmd_index == AP_MISSION_CMD_INDEX_NONE) {
|
|
cmd_index = AP_MISSION_FIRST_REAL_COMMAND;
|
|
}else{
|
|
// start from one position past the current do command
|
|
cmd_index = _do_cmd.index + 1;
|
|
}
|
|
|
|
// find next do command
|
|
Mission_Command cmd;
|
|
if (!get_next_do_cmd(cmd_index, cmd)) {
|
|
// set flag to stop unnecessarily searching for do commands
|
|
_flags.do_cmd_all_done = true;
|
|
return;
|
|
}
|
|
|
|
// set current do command and start it
|
|
_do_cmd = cmd;
|
|
_flags.do_cmd_loaded = true;
|
|
start_command(_do_cmd);
|
|
}
|
|
|
|
/// get_next_cmd - gets next command found at or after start_index
|
|
/// returns true if found, false if not found (i.e. mission complete)
|
|
/// accounts for do_jump commands
|
|
/// increment_jump_num_times_if_found should be set to true if advancing the active navigation command
|
|
bool AP_Mission::get_next_cmd(uint16_t start_index, Mission_Command& cmd, bool increment_jump_num_times_if_found)
|
|
{
|
|
uint16_t cmd_index = start_index;
|
|
Mission_Command temp_cmd;
|
|
uint16_t jump_index = AP_MISSION_CMD_INDEX_NONE;
|
|
|
|
// search until the end of the mission command list
|
|
uint8_t max_loops = 64;
|
|
while(cmd_index < (unsigned)_cmd_total) {
|
|
// load the next command
|
|
if (!read_cmd_from_storage(cmd_index, temp_cmd)) {
|
|
// this should never happen because of check above but just in case
|
|
return false;
|
|
}
|
|
|
|
// check for do-jump command
|
|
if (temp_cmd.id == MAV_CMD_DO_JUMP) {
|
|
|
|
if (max_loops-- == 0) {
|
|
return false;
|
|
}
|
|
|
|
// check for invalid target
|
|
if ((temp_cmd.content.jump.target >= (unsigned)_cmd_total) || (temp_cmd.content.jump.target == 0)) {
|
|
// To-Do: log an error?
|
|
return false;
|
|
}
|
|
|
|
// check for endless loops
|
|
if (!increment_jump_num_times_if_found && jump_index == cmd_index) {
|
|
// we have somehow reached this jump command twice and there is no chance it will complete
|
|
// To-Do: log an error?
|
|
return false;
|
|
}
|
|
|
|
// record this command so we can check for endless loops
|
|
if (jump_index == AP_MISSION_CMD_INDEX_NONE) {
|
|
jump_index = cmd_index;
|
|
}
|
|
|
|
// check if jump command is 'repeat forever'
|
|
if (temp_cmd.content.jump.num_times == AP_MISSION_JUMP_REPEAT_FOREVER) {
|
|
// continue searching from jump target
|
|
cmd_index = temp_cmd.content.jump.target;
|
|
}else{
|
|
// get number of times jump command has already been run
|
|
int16_t jump_times_run = get_jump_times_run(temp_cmd);
|
|
if (jump_times_run < temp_cmd.content.jump.num_times) {
|
|
// update the record of the number of times run
|
|
if (increment_jump_num_times_if_found) {
|
|
increment_jump_times_run(temp_cmd);
|
|
}
|
|
// continue searching from jump target
|
|
cmd_index = temp_cmd.content.jump.target;
|
|
}else{
|
|
// jump has been run specified number of times so move search to next command in mission
|
|
cmd_index++;
|
|
}
|
|
}
|
|
}else{
|
|
// this is a non-jump command so return it
|
|
cmd = temp_cmd;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// if we got this far we did not find a navigation command
|
|
return false;
|
|
}
|
|
|
|
/// get_next_do_cmd - gets next "do" or "conditional" command after start_index
|
|
/// returns true if found, false if not found
|
|
/// stops and returns false if it hits another navigation command before it finds the first do or conditional command
|
|
/// accounts for do_jump commands but never increments the jump's num_times_run (advance_current_nav_cmd is responsible for this)
|
|
bool AP_Mission::get_next_do_cmd(uint16_t start_index, Mission_Command& cmd)
|
|
{
|
|
Mission_Command temp_cmd;
|
|
|
|
// check we have not passed the end of the mission list
|
|
if (start_index >= (unsigned)_cmd_total) {
|
|
return false;
|
|
}
|
|
|
|
// get next command
|
|
if (!get_next_cmd(start_index, temp_cmd, false)) {
|
|
// no more commands so return failure
|
|
return false;
|
|
}else if (is_nav_cmd(temp_cmd)) {
|
|
// if it's a "navigation" command then return false because we do not progress past nav commands
|
|
return false;
|
|
}else{
|
|
// this must be a "do" or "conditional" and is not a do-jump command so return it
|
|
cmd = temp_cmd;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
///
|
|
/// jump handling methods
|
|
///
|
|
|
|
// init_jump_tracking - initialise jump_tracking variables
|
|
void AP_Mission::init_jump_tracking()
|
|
{
|
|
for(uint8_t i=0; i<AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS; i++) {
|
|
_jump_tracking[i].index = AP_MISSION_CMD_INDEX_NONE;
|
|
_jump_tracking[i].num_times_run = 0;
|
|
}
|
|
}
|
|
|
|
/// get_jump_times_run - returns number of times the jump command has been run
|
|
int16_t AP_Mission::get_jump_times_run(const Mission_Command& cmd)
|
|
{
|
|
// exit immediately if cmd is not a do-jump command or target is invalid
|
|
if ((cmd.id != MAV_CMD_DO_JUMP) || (cmd.content.jump.target >= (unsigned)_cmd_total) || (cmd.content.jump.target == 0)) {
|
|
// To-Do: log an error?
|
|
return AP_MISSION_JUMP_TIMES_MAX;
|
|
}
|
|
|
|
// search through jump_tracking array for this cmd
|
|
for (uint8_t i=0; i<AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS; i++) {
|
|
if (_jump_tracking[i].index == cmd.index) {
|
|
return _jump_tracking[i].num_times_run;
|
|
}else if(_jump_tracking[i].index == AP_MISSION_CMD_INDEX_NONE) {
|
|
// we've searched through all known jump commands and haven't found it so allocate new space in _jump_tracking array
|
|
_jump_tracking[i].index = cmd.index;
|
|
_jump_tracking[i].num_times_run = 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// if we've gotten this far then the _jump_tracking array must be full
|
|
// To-Do: log an error?
|
|
return AP_MISSION_JUMP_TIMES_MAX;
|
|
}
|
|
|
|
/// increment_jump_times_run - increments the recorded number of times the jump command has been run
|
|
void AP_Mission::increment_jump_times_run(Mission_Command& cmd)
|
|
{
|
|
// exit immediately if cmd is not a do-jump command
|
|
if (cmd.id != MAV_CMD_DO_JUMP) {
|
|
// To-Do: log an error?
|
|
return;
|
|
}
|
|
|
|
// search through jump_tracking array for this cmd
|
|
for (uint8_t i=0; i<AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS; i++) {
|
|
if (_jump_tracking[i].index == cmd.index) {
|
|
_jump_tracking[i].num_times_run++;
|
|
gcs().send_text(MAV_SEVERITY_INFO, "Mission: %u Jump %i/%i", _jump_tracking[i].index, _jump_tracking[i].num_times_run, cmd.content.jump.num_times);
|
|
return;
|
|
}else if(_jump_tracking[i].index == AP_MISSION_CMD_INDEX_NONE) {
|
|
// we've searched through all known jump commands and haven't found it so allocate new space in _jump_tracking array
|
|
_jump_tracking[i].index = cmd.index;
|
|
_jump_tracking[i].num_times_run = 1;
|
|
return;
|
|
}
|
|
}
|
|
|
|
// if we've gotten this far then the _jump_tracking array must be full
|
|
// To-Do: log an error
|
|
return;
|
|
}
|
|
|
|
// check_eeprom_version - checks version of missions stored in eeprom matches this library
|
|
// command list will be cleared if they do not match
|
|
void AP_Mission::check_eeprom_version()
|
|
{
|
|
uint32_t eeprom_version = _storage.read_uint32(0);
|
|
|
|
// if eeprom version does not match, clear the command list and update the eeprom version
|
|
if (eeprom_version != AP_MISSION_EEPROM_VERSION) {
|
|
if (clear()) {
|
|
_storage.write_uint32(0, AP_MISSION_EEPROM_VERSION);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
return total number of commands that can fit in storage space
|
|
*/
|
|
uint16_t AP_Mission::num_commands_max(void) const
|
|
{
|
|
// -4 to remove space for eeprom version number
|
|
return (_storage.size() - 4) / AP_MISSION_EEPROM_COMMAND_SIZE;
|
|
}
|
|
|
|
// find the nearest landing sequence starting point (DO_LAND_START) and
|
|
// return its index. Returns 0 if no appropriate DO_LAND_START point can
|
|
// be found.
|
|
uint16_t AP_Mission::get_landing_sequence_start()
|
|
{
|
|
struct Location current_loc;
|
|
|
|
if (!AP::ahrs().get_position(current_loc)) {
|
|
return 0;
|
|
}
|
|
|
|
uint16_t landing_start_index = 0;
|
|
float min_distance = -1;
|
|
|
|
// Go through mission looking for nearest landing start command
|
|
for (uint16_t i = 1; i < num_commands(); i++) {
|
|
Mission_Command tmp;
|
|
if (!read_cmd_from_storage(i, tmp)) {
|
|
continue;
|
|
}
|
|
if (tmp.id == MAV_CMD_DO_LAND_START) {
|
|
float tmp_distance = tmp.content.location.get_distance(current_loc);
|
|
if (min_distance < 0 || tmp_distance < min_distance) {
|
|
min_distance = tmp_distance;
|
|
landing_start_index = i;
|
|
}
|
|
}
|
|
}
|
|
|
|
return landing_start_index;
|
|
}
|
|
|
|
/*
|
|
find the nearest landing sequence starting point (DO_LAND_START) and
|
|
switch to that mission item. Returns false if no DO_LAND_START
|
|
available.
|
|
*/
|
|
bool AP_Mission::jump_to_landing_sequence(void)
|
|
{
|
|
uint16_t land_idx = get_landing_sequence_start();
|
|
if (land_idx != 0 && set_current_cmd(land_idx)) {
|
|
|
|
//if the mission has ended it has to be restarted
|
|
if (state() == AP_Mission::MISSION_STOPPED) {
|
|
resume();
|
|
}
|
|
|
|
gcs().send_text(MAV_SEVERITY_INFO, "Landing sequence start");
|
|
return true;
|
|
}
|
|
|
|
gcs().send_text(MAV_SEVERITY_WARNING, "Unable to start landing sequence");
|
|
return false;
|
|
}
|
|
|
|
// jumps the mission to the closest landing abort that is planned, returns false if unable to find a valid abort
|
|
bool AP_Mission::jump_to_abort_landing_sequence(void)
|
|
{
|
|
struct Location current_loc;
|
|
|
|
uint16_t abort_index = 0;
|
|
if (AP::ahrs().get_position(current_loc)) {
|
|
float min_distance = FLT_MAX;
|
|
|
|
for (uint16_t i = 1; i < num_commands(); i++) {
|
|
Mission_Command tmp;
|
|
if (!read_cmd_from_storage(i, tmp)) {
|
|
continue;
|
|
}
|
|
if (tmp.id == MAV_CMD_DO_GO_AROUND) {
|
|
float tmp_distance = tmp.content.location.get_distance(current_loc);
|
|
if (tmp_distance < min_distance) {
|
|
min_distance = tmp_distance;
|
|
abort_index = i;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (abort_index != 0 && set_current_cmd(abort_index)) {
|
|
|
|
//if the mission has ended it has to be restarted
|
|
if (state() == AP_Mission::MISSION_STOPPED) {
|
|
resume();
|
|
}
|
|
|
|
gcs().send_text(MAV_SEVERITY_INFO, "Landing abort sequence start");
|
|
return true;
|
|
}
|
|
|
|
gcs().send_text(MAV_SEVERITY_WARNING, "Unable to start find a landing abort sequence");
|
|
return false;
|
|
}
|
|
|
|
const char *AP_Mission::Mission_Command::type() const {
|
|
switch(id) {
|
|
case MAV_CMD_NAV_WAYPOINT:
|
|
return "WP";
|
|
case MAV_CMD_NAV_SPLINE_WAYPOINT:
|
|
return "SplineWP";
|
|
case MAV_CMD_NAV_RETURN_TO_LAUNCH:
|
|
return "RTL";
|
|
case MAV_CMD_NAV_LOITER_UNLIM:
|
|
return "LoitUnlim";
|
|
case MAV_CMD_NAV_LOITER_TIME:
|
|
return "LoitTime";
|
|
case MAV_CMD_NAV_GUIDED_ENABLE:
|
|
return "GuidedEnable";
|
|
case MAV_CMD_NAV_LOITER_TURNS:
|
|
return "LoitTurns";
|
|
case MAV_CMD_NAV_LOITER_TO_ALT:
|
|
return "LoitAltitude";
|
|
case MAV_CMD_NAV_SET_YAW_SPEED:
|
|
return "SetYawSpd";
|
|
case MAV_CMD_CONDITION_DELAY:
|
|
return "CondDelay";
|
|
case MAV_CMD_CONDITION_DISTANCE:
|
|
return "CondDist";
|
|
case MAV_CMD_DO_CHANGE_SPEED:
|
|
return "ChangeSpeed";
|
|
case MAV_CMD_DO_SET_HOME:
|
|
return "SetHome";
|
|
case MAV_CMD_DO_SET_SERVO:
|
|
return "SetServo";
|
|
case MAV_CMD_DO_SET_RELAY:
|
|
return "SetRelay";
|
|
case MAV_CMD_DO_REPEAT_SERVO:
|
|
return "RepeatServo";
|
|
case MAV_CMD_DO_REPEAT_RELAY:
|
|
return "RepeatRelay";
|
|
case MAV_CMD_DO_CONTROL_VIDEO:
|
|
return "CtrlVideo";
|
|
case MAV_CMD_DO_DIGICAM_CONFIGURE:
|
|
return "DigiCamCfg";
|
|
case MAV_CMD_DO_DIGICAM_CONTROL:
|
|
return "DigiCamCtrl";
|
|
case MAV_CMD_DO_SET_CAM_TRIGG_DIST:
|
|
return "SetCamTrigDst";
|
|
case MAV_CMD_DO_SET_ROI:
|
|
return "SetROI";
|
|
case MAV_CMD_DO_SET_REVERSE:
|
|
return "SetReverse";
|
|
case MAV_CMD_DO_GUIDED_LIMITS:
|
|
return "GuidedLimits";
|
|
case MAV_CMD_NAV_TAKEOFF:
|
|
return "Takeoff";
|
|
case MAV_CMD_NAV_LAND:
|
|
return "Land";
|
|
case MAV_CMD_NAV_CONTINUE_AND_CHANGE_ALT:
|
|
return "ContinueAndChangeAlt";
|
|
case MAV_CMD_NAV_ALTITUDE_WAIT:
|
|
return "AltitudeWait";
|
|
case MAV_CMD_NAV_VTOL_TAKEOFF:
|
|
return "VTOLTakeoff";
|
|
case MAV_CMD_NAV_VTOL_LAND:
|
|
return "VTOLLand";
|
|
case MAV_CMD_DO_INVERTED_FLIGHT:
|
|
return "InvertedFlight";
|
|
case MAV_CMD_DO_FENCE_ENABLE:
|
|
return "FenceEnable";
|
|
case MAV_CMD_DO_AUTOTUNE_ENABLE:
|
|
return "AutoTuneEnable";
|
|
case MAV_CMD_DO_VTOL_TRANSITION:
|
|
return "VTOLTransition";
|
|
case MAV_CMD_DO_ENGINE_CONTROL:
|
|
return "EngineControl";
|
|
case MAV_CMD_CONDITION_YAW:
|
|
return "CondYaw";
|
|
case MAV_CMD_DO_LAND_START:
|
|
return "LandStart";
|
|
case MAV_CMD_NAV_DELAY:
|
|
return "Delay";
|
|
case MAV_CMD_DO_GRIPPER:
|
|
return "Gripper";
|
|
case MAV_CMD_NAV_PAYLOAD_PLACE:
|
|
return "PayloadPlace";
|
|
case MAV_CMD_DO_PARACHUTE:
|
|
return "Parachute";
|
|
|
|
default:
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
AP_HAL::panic("Mission command with ID %u has no string", id);
|
|
#endif
|
|
return "?";
|
|
}
|
|
}
|
|
|
|
bool AP_Mission::contains_item(MAV_CMD command) const
|
|
{
|
|
for (int i = 1; i < num_commands(); i++) {
|
|
Mission_Command tmp;
|
|
if (!read_cmd_from_storage(i, tmp)) {
|
|
continue;
|
|
}
|
|
if (tmp.id == command) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// singleton instance
|
|
AP_Mission *AP_Mission::_singleton;
|
|
|
|
namespace AP {
|
|
|
|
AP_Mission *mission()
|
|
{
|
|
return AP_Mission::get_singleton();
|
|
}
|
|
|
|
}
|