0c92d60406
Once more eeprom space is allocated this will allow more than 255 commands
867 lines
32 KiB
C++
867 lines
32 KiB
C++
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
/// @file AP_Mission.cpp
|
|
/// @brief Handles the MAVLINK command mission stack. Reads and writes mission to storage.
|
|
|
|
#include "AP_Mission.h"
|
|
|
|
const AP_Param::GroupInfo AP_Mission::var_info[] PROGMEM = {
|
|
|
|
// @Param: TOTAL
|
|
// @DisplayName: Total number of commands in the mission
|
|
// @Description: The number of mission mission items that has been loaded by the ground station. Do not change this manually.
|
|
// @Range: 0 255
|
|
// @Increment: 1
|
|
// @User: Advanced
|
|
AP_GROUPINFO("TOTAL", 0, AP_Mission, _cmd_total, 0),
|
|
|
|
AP_GROUPEND
|
|
};
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
///
|
|
/// public mission methods
|
|
///
|
|
|
|
/// 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;
|
|
_flags.nav_cmd_loaded = false;
|
|
_flags.do_cmd_loaded = 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;
|
|
init_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-initialised
|
|
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;
|
|
}
|
|
|
|
// restart active navigation command
|
|
// 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
|
|
if (_flags.nav_cmd_loaded) {
|
|
_cmd_start_fn(_nav_cmd);
|
|
}
|
|
|
|
// restart active do command
|
|
if (_flags.do_cmd_loaded && _do_cmd.index != AP_MISSION_CMD_INDEX_NONE) {
|
|
_cmd_start_fn(_do_cmd);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// 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;
|
|
}
|
|
|
|
/// 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;
|
|
}
|
|
|
|
// 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 (_cmd_verify_fn(_nav_cmd)) {
|
|
// market _nav_cmd as complete (it will be started on the next iteration)
|
|
_flags.nav_cmd_loaded = false;
|
|
}
|
|
}
|
|
|
|
// check if we have an active do command
|
|
if (!_flags.do_cmd_loaded || _do_cmd.index == AP_MISSION_CMD_INDEX_NONE) {
|
|
advance_current_do_cmd();
|
|
}else{
|
|
// run the active do command
|
|
if (_cmd_verify_fn(_do_cmd)) {
|
|
// market _nav_cmd as complete (it will be started on the next iteration)
|
|
_flags.do_cmd_loaded = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
///
|
|
/// 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;
|
|
}
|
|
|
|
/// 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)
|
|
{
|
|
return (cmd.id <= MAV_CMD_NAV_LAST);
|
|
}
|
|
|
|
/// 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)
|
|
{
|
|
uint16_t cmd_index = start_index;
|
|
Mission_Command temp_cmd;
|
|
|
|
// search until the end of the mission command list
|
|
while(cmd_index < _cmd_total) {
|
|
// get next command
|
|
if (!get_next_cmd(cmd_index, temp_cmd, false)) {
|
|
// no more commands so return failure
|
|
return false;
|
|
}else{
|
|
// if found a "navigation" command then return it
|
|
if (is_nav_cmd(temp_cmd)) {
|
|
cmd = temp_cmd;
|
|
return true;
|
|
}else{
|
|
// move on in list
|
|
cmd_index++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// if we got this far we did not find a navigation command
|
|
return false;
|
|
}
|
|
|
|
/// 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
|
|
{
|
|
uint16_t pos_in_storage; // position in storage from where we will read the next byte
|
|
|
|
// exit immediately if index is beyond last command but we always let cmd #0 (i.e. home) be read
|
|
if (index > _cmd_total && index != 0) {
|
|
// return a command with a blank id
|
|
cmd.id = AP_MISSION_CMD_ID_NONE;
|
|
return false;
|
|
}
|
|
|
|
// special handling for command #0 which is home
|
|
if (index == 0) {
|
|
cmd.index = 0;
|
|
cmd.id = MAV_CMD_NAV_WAYPOINT;
|
|
cmd.content.location = _ahrs.get_home();
|
|
}else{
|
|
// 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
|
|
pos_in_storage = (AP_MISSION_EEPROM_START_BYTE) + (index * AP_MISSION_EEPROM_COMMAND_SIZE);
|
|
|
|
cmd.content.location.id = hal.storage->read_byte(pos_in_storage);
|
|
pos_in_storage++;
|
|
|
|
cmd.content.location.options = hal.storage->read_byte(pos_in_storage);
|
|
pos_in_storage++;
|
|
|
|
cmd.content.location.p1 = hal.storage->read_byte(pos_in_storage);
|
|
pos_in_storage++;
|
|
|
|
cmd.content.location.alt = hal.storage->read_dword(pos_in_storage);
|
|
pos_in_storage += 4;
|
|
|
|
cmd.content.location.lat = hal.storage->read_dword(pos_in_storage);
|
|
pos_in_storage += 4;
|
|
|
|
cmd.content.location.lng = hal.storage->read_dword(pos_in_storage);
|
|
|
|
// set command's index to it's position in eeprom
|
|
cmd.index = index;
|
|
|
|
// set command from location's command
|
|
// To-Do: remove id (and p1?) from Location structure
|
|
cmd.id = cmd.content.location.id;
|
|
}
|
|
|
|
// return success
|
|
return true;
|
|
}
|
|
|
|
/// 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, Mission_Command& cmd)
|
|
{
|
|
// range check cmd's index
|
|
if (index >= AP_MISSION_MAX_COMMANDS) {
|
|
return false;
|
|
}
|
|
|
|
// calculate where in storage the command should be placed
|
|
uint16_t pos_in_storage = AP_MISSION_EEPROM_START_BYTE + (index * AP_MISSION_EEPROM_COMMAND_SIZE);
|
|
|
|
cmd.content.location.id = cmd.id;
|
|
hal.storage->write_byte(pos_in_storage, cmd.content.location.id);
|
|
|
|
pos_in_storage++;
|
|
hal.storage->write_byte(pos_in_storage, cmd.content.location.options);
|
|
|
|
pos_in_storage++;
|
|
hal.storage->write_byte(pos_in_storage, cmd.content.location.p1);
|
|
|
|
pos_in_storage++;
|
|
hal.storage->write_dword(pos_in_storage, cmd.content.location.alt);
|
|
|
|
pos_in_storage += 4;
|
|
hal.storage->write_dword(pos_in_storage, cmd.content.location.lat);
|
|
|
|
pos_in_storage += 4;
|
|
hal.storage->write_dword(pos_in_storage, cmd.content.location.lng);
|
|
|
|
// return success
|
|
return true;
|
|
}
|
|
|
|
// mavlink_to_mission_cmd - converts mavlink message to an AP_Mission::Mission_Command object which can be stored to eeprom
|
|
// return true on success, false on failure
|
|
bool AP_Mission::mavlink_to_mission_cmd(const mavlink_mission_item_t& packet, AP_Mission::Mission_Command& cmd)
|
|
{
|
|
bool copy_location = false;
|
|
|
|
// command's position in mission list and mavlink id
|
|
cmd.index = packet.seq;
|
|
cmd.id = packet.command;
|
|
|
|
// command specific conversions from mavlink packet to mission command
|
|
switch (cmd.id) {
|
|
|
|
case MAV_CMD_NAV_WAYPOINT: // MAV ID: 16
|
|
copy_location = true;
|
|
cmd.content.location.p1 = packet.param1; // delay at waypoint in seconds
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_UNLIM: // MAV ID: 17
|
|
copy_location = true;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TURNS: // MAV ID: 18
|
|
copy_location = true;
|
|
cmd.content.location.p1 = packet.param1; // number of times to circle is held in location.p1
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TIME: // MAV ID: 19
|
|
copy_location = true;
|
|
cmd.content.location.p1 = packet.param1; // loiter time in seconds
|
|
break;
|
|
|
|
case MAV_CMD_NAV_RETURN_TO_LAUNCH: // MAV ID: 20
|
|
copy_location = true;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LAND: // MAV ID: 21
|
|
copy_location = true;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_TAKEOFF: // MAV ID: 22
|
|
copy_location = true; // only altitude is used
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DELAY: // MAV ID: 112
|
|
cmd.content.location.lat = packet.param1; // delay in seconds
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_CHANGE_ALT: // MAV ID: 113
|
|
copy_location = true; // only altitude is used
|
|
cmd.content.location.p1 = packet.param1 * 100; // climb/descent rate converted from m/s to cm/s
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DISTANCE: // MAV ID: 114
|
|
cmd.content.location.lat = packet.param1; // distance in meters from next waypoint
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_YAW: // MAV ID: 115
|
|
cmd.content.location.alt = packet.param1; // target angle in degrees
|
|
cmd.content.location.lat = packet.param2; // lat=0: use default turn rate otherwise specific turn rate in deg/sec
|
|
cmd.content.location.p1 = packet.param3; // -1 = ccw, +1 = cw
|
|
cmd.content.location.lng = packet.param4; // lng=0: absolute angle provided, lng=1: relative angle provided
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_MODE: // MAV ID: 176
|
|
cmd.content.location.p1 = packet.param1; // set flight mode. To-Do: make mapping function from MAVLINK defined flight modes to AC/AP/AR flight modes
|
|
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.location.p1 = packet.param1; // 0 = airspeed, 1 = ground speed
|
|
cmd.content.location.alt = packet.param2; // speed in m/s
|
|
cmd.content.location.lat = packet.param3; // throttle as a percentage from 0 ~ 100%
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_HOME:
|
|
copy_location = true;
|
|
cmd.content.location.p1 = packet.param1; // p1=0 means use current location, p=1 means use provided location
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_PARAMETER: // MAV ID: 180
|
|
cmd.content.location.p1 = packet.param1; // parameter number
|
|
cmd.content.location.alt = packet.param2; // parameter value
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_RELAY: // MAV ID: 181
|
|
cmd.content.location.p1 = packet.param1; // relay number
|
|
cmd.content.location.alt = packet.param2; // 0:off, 1:on
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_RELAY: // MAV ID: 182
|
|
cmd.content.location.p1 = packet.param1; // relay number
|
|
cmd.content.location.alt = packet.param2; // count
|
|
cmd.content.location.lat = packet.param3*1000; // time
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_SERVO: // MAV ID: 183
|
|
cmd.content.location.p1 = packet.param1; // channel
|
|
cmd.content.location.alt = packet.param2; // PWM
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_SERVO: // MAV ID: 184
|
|
cmd.content.location.p1 = packet.param1; // channel
|
|
cmd.content.location.alt = packet.param2; // PWM
|
|
cmd.content.location.lat = packet.param3; // count
|
|
cmd.content.location.lng = packet.param4*1000; // time in seconds converted to milliseconds
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_ROI: // MAV ID: 201
|
|
copy_location = true;
|
|
cmd.content.location.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_CONTROL: // MAV ID: 203
|
|
case MAV_CMD_DO_MOUNT_CONTROL: // MAV ID: 205
|
|
// these commands takes no parameters
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_CAM_TRIGG_DIST: // MAV ID: 206
|
|
cmd.content.location.alt = packet.param1; // use alt so we can support 32 bit values
|
|
break;
|
|
|
|
default:
|
|
// unrecognised command
|
|
return false;
|
|
break;
|
|
}
|
|
|
|
// copy location from mavlink to command
|
|
if (copy_location) {
|
|
cmd.content.location.lat = 1.0e7f * packet.x; // floating point latitude to int32_t
|
|
cmd.content.location.lng = 1.0e7f * packet.y; // floating point longitude to int32_t
|
|
cmd.content.location.alt = packet.z * 100.0f; // convert packet's alt (m) to cmd alt (cm)
|
|
if (packet.frame == MAV_FRAME_GLOBAL_RELATIVE_ALT) { // convert rel/abs alt to packet's frame
|
|
cmd.content.location.options = 1; // To-Do: check this never overwrites any other uses of 'options'
|
|
}else{
|
|
cmd.content.location.options = 0;
|
|
}
|
|
}
|
|
|
|
// if we got this far then it must have been succesful
|
|
return true;
|
|
}
|
|
|
|
// mission_cmd_to_mavlink - 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(const AP_Mission::Mission_Command& cmd, mavlink_mission_item_t& packet)
|
|
{
|
|
bool copy_location = false;
|
|
|
|
// 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 MAV_CMD_NAV_WAYPOINT: // MAV ID: 16
|
|
copy_location = true;
|
|
packet.param1 = cmd.content.location.p1; // delay at waypoint in seconds
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_UNLIM: // MAV ID: 17
|
|
copy_location = true;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TURNS: // MAV ID: 18
|
|
copy_location = true;
|
|
packet.param1 = cmd.content.location.p1; // number of times to circle is held in location.p1
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LOITER_TIME: // MAV ID: 19
|
|
copy_location = true;
|
|
packet.param1 = cmd.content.location.p1; // loiter time in seconds
|
|
break;
|
|
|
|
case MAV_CMD_NAV_RETURN_TO_LAUNCH: // MAV ID: 20
|
|
copy_location = true;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_LAND: // MAV ID: 21
|
|
copy_location = true;
|
|
break;
|
|
|
|
case MAV_CMD_NAV_TAKEOFF: // MAV ID: 22
|
|
copy_location = true; // only altitude is used
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DELAY: // MAV ID: 112
|
|
packet.param1 = cmd.content.location.lat; // delay in seconds
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_CHANGE_ALT: // MAV ID: 113
|
|
copy_location = true; // only altitude is used
|
|
packet.param1 = cmd.content.location.p1/100; // climb/descent rate converted from m/s to cm/s
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_DISTANCE: // MAV ID: 114
|
|
packet.param1 = cmd.content.location.lat; // distance in meters from next waypoint
|
|
break;
|
|
|
|
case MAV_CMD_CONDITION_YAW: // MAV ID: 115
|
|
packet.param1 = cmd.content.location.alt; // target angle in degrees
|
|
packet.param2 = cmd.content.location.lat; // lat=0: use default turn rate otherwise specific turn rate in deg/sec
|
|
packet.param3 = cmd.content.location.p1; // -1 = ccw, +1 = cw
|
|
packet.param4 = cmd.content.location.lng; // lng=0: absolute angle provided, lng=1: relative angle provided
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_MODE: // MAV ID: 176
|
|
packet.param1 = cmd.content.location.p1; // set flight mode. To-Do: make mapping function from MAVLINK defined flight modes to AC/AP/AR flight modes
|
|
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.location.p1; // 0 = airspeed, 1 = ground speed
|
|
packet.param2 = cmd.content.location.alt; // speed in m/s
|
|
packet.param3 = cmd.content.location.lat; // throttle as a percentage from 0 ~ 100%
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_HOME: // MAV ID: 179
|
|
copy_location = true;
|
|
packet.param1 = cmd.content.location.p1; // p1=0 means use current location, p=1 means use provided location
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_PARAMETER: // MAV ID: 180
|
|
packet.param1 = cmd.content.location.p1; // parameter number
|
|
packet.param2 = cmd.content.location.alt; // parameter value
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_RELAY: // MAV ID: 181
|
|
packet.param1 = cmd.content.location.p1; // relay number
|
|
packet.param2 = cmd.content.location.alt; // 0:off, 1:on
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_RELAY: // MAV ID: 182
|
|
packet.param1 = cmd.content.location.p1; // relay number
|
|
packet.param2 = cmd.content.location.alt; // count
|
|
packet.param3 = cmd.content.location.lat/1000; // time
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_SERVO: // MAV ID: 183
|
|
packet.param1 = cmd.content.location.p1; // channel
|
|
packet.param2 = cmd.content.location.alt; // PWM
|
|
break;
|
|
|
|
case MAV_CMD_DO_REPEAT_SERVO: // MAV ID: 184
|
|
packet.param1 = cmd.content.location.p1; // channel
|
|
packet.param2 = cmd.content.location.alt; // PWM
|
|
packet.param3 = cmd.content.location.lat; // count
|
|
packet.param4 = cmd.content.location.lng/1000; // time in seconds converted to milliseconds
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_ROI: // MAV ID: 201
|
|
copy_location = true;
|
|
packet.param1 = cmd.content.location.p1; // 0 = no roi, 1 = next waypoint, 2 = waypoint number, 3 = fixed location, 4 = given target (not supported)
|
|
break;
|
|
|
|
case MAV_CMD_DO_DIGICAM_CONTROL: // MAV ID: 203
|
|
case MAV_CMD_DO_MOUNT_CONTROL: // MAV ID: 205
|
|
// these commands takes no parameters
|
|
break;
|
|
|
|
case MAV_CMD_DO_SET_CAM_TRIGG_DIST: // MAV ID: 206
|
|
packet.param1 = cmd.content.location.alt; // use alt so we can support 32 bit values
|
|
break;
|
|
|
|
default:
|
|
// unrecognised command
|
|
return false;
|
|
break;
|
|
}
|
|
|
|
// copy location from mavlink to command
|
|
if (copy_location) {
|
|
packet.x = cmd.content.location.lat / 1.0e7f; // int32_t latitude to float
|
|
packet.y = cmd.content.location.lng / 1.0e7f; // int32_t longitude to float
|
|
packet.z = cmd.content.location.alt / 100.0f; // cmd alt in cm to m
|
|
if (cmd.content.location.options & LOCATION_MASK_OPTIONS_RELATIVE_ALT) {
|
|
packet.frame = MAV_FRAME_GLOBAL_RELATIVE_ALT;
|
|
}else{
|
|
packet.frame = MAV_FRAME_GLOBAL;
|
|
}
|
|
}
|
|
|
|
// if we got this far then it must have been succesful
|
|
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()
|
|
{
|
|
Mission_Command cmd;
|
|
uint16_t cmd_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;
|
|
|
|
// get starting point for search
|
|
cmd_index = _nav_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 nav command
|
|
cmd_index++;
|
|
}
|
|
|
|
// search until we find next nav command or reach end of command list
|
|
while (!_flags.nav_cmd_loaded) {
|
|
// get next command
|
|
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_index = _nav_cmd.index;
|
|
// set current navigation command and start it
|
|
_nav_cmd = cmd;
|
|
_flags.nav_cmd_loaded = true;
|
|
_cmd_start_fn(_nav_cmd);
|
|
}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;
|
|
_cmd_start_fn(_do_cmd);
|
|
}
|
|
}
|
|
// move onto next command
|
|
cmd_index = cmd.index+1;
|
|
}
|
|
|
|
// 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()
|
|
{
|
|
Mission_Command cmd;
|
|
uint16_t cmd_index;
|
|
|
|
// exit immediately if we're not running
|
|
if (_flags.state != MISSION_RUNNING) {
|
|
return;
|
|
}
|
|
|
|
// get starting point for search
|
|
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++;
|
|
}
|
|
|
|
// check if we've reached end of mission
|
|
if (cmd_index >= _cmd_total) {
|
|
// To-Do: set a flag to stop us from attempting to look for do-commands over and over?
|
|
return;
|
|
}
|
|
|
|
// find next do command
|
|
if (get_next_do_cmd(cmd_index, cmd)) {
|
|
// set current do command and start it
|
|
_do_cmd = cmd;
|
|
_cmd_start_fn(_do_cmd);
|
|
}else{
|
|
// To-Do: set a flag to stop us from attempting to look for do-commands over and over?
|
|
// if added this flag should be reset once we advance the nav command
|
|
}
|
|
}
|
|
|
|
/// 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
|
|
while(cmd_index < _cmd_total) {
|
|
// load the next command
|
|
read_cmd_from_storage(cmd_index, temp_cmd);
|
|
|
|
// check for do-jump command
|
|
if (temp_cmd.id == MAV_CMD_DO_JUMP) {
|
|
|
|
// check for invalid target
|
|
if (temp_cmd.content.jump.target >= _cmd_total) {
|
|
// 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 >= _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 immediatley if cmd is not a do-jump command or target is invalid
|
|
if (cmd.id != MAV_CMD_DO_JUMP || cmd.content.jump.target >= _cmd_total) {
|
|
// 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++;
|
|
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;
|
|
}
|