2014-02-22 20:14:44 -04:00
/// @file AP_Mission.cpp
/// @brief Handles the MAVLINK command mission stack. Reads and writes mission to storage.
2023-12-09 00:05:08 -04:00
# include "AP_Mission_config.h"
# if AP_MISSION_ENABLED
2018-04-24 20:09:26 -03:00
# include <AP_AHRS/AP_AHRS.h>
2023-03-16 06:53:28 -03:00
# include <AP_BoardConfig/AP_BoardConfig.h>
2024-02-12 23:57:07 -04:00
# include <AP_Camera/AP_Camera.h>
# include <AP_Gripper/AP_Gripper_config.h>
# include "AP_Mission.h"
# include <AP_Scripting/AP_Scripting.h>
2023-06-06 05:05:06 -03:00
# include <AP_ServoRelayEvents/AP_ServoRelayEvents_config.h>
2024-02-12 23:57:07 -04:00
# include <AP_Terrain/AP_Terrain.h>
# include <AP_Vehicle/AP_Vehicle_Type.h>
# include <GCS_MAVLink/GCS.h>
2023-12-09 02:09:26 -04:00
# include <RC_Channel/RC_Channel_config.h>
2024-05-31 11:18:59 -03:00
# include <AC_Fence/AC_Fence.h>
2024-09-27 04:34:19 -03:00
# include <AP_Logger/AP_Logger.h>
2014-02-22 20:14:44 -04:00
2015-10-25 14:03:46 -03:00
const AP_Param : : GroupInfo AP_Mission : : var_info [ ] = {
2014-02-22 20:14:44 -04:00
// @Param: TOTAL
2014-04-28 18:53:36 -03:00
// @DisplayName: Total mission commands
2014-02-22 20:14:44 -04:00
// @Description: The number of mission mission items that has been loaded by the ground station. Do not change this manually.
2014-03-16 01:53:22 -03:00
// @Range: 0 32766
2014-02-22 20:14:44 -04:00
// @Increment: 1
// @User: Advanced
2019-06-11 01:20:25 -03:00
// @ReadOnly: True
AP_GROUPINFO_FLAGS ( " TOTAL " , 0 , AP_Mission , _cmd_total , 0 , AP_PARAM_FLAG_INTERNAL_USE_ONLY ) ,
2014-02-22 20:14:44 -04:00
2014-05-08 11:06:06 -03:00
// @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
2016-07-26 02:35:15 -03:00
// @User: Advanced
2014-05-08 11:06:06 -03:00
AP_GROUPINFO ( " RESTART " , 1 , AP_Mission , _restart , AP_MISSION_RESTART_DEFAULT ) ,
2014-04-15 13:26:45 -03:00
2017-08-27 08:12:45 -03:00
// @Param: OPTIONS
// @DisplayName: Mission options bitmask
// @Description: Bitmask of what options to use in missions.
2020-03-03 23:28:41 -04:00
// @Bitmask: 0:Clear Mission on reboot, 1:Use distance to land calc on battery failsafe,2:ContinueAfterLand
2021-03-25 21:09:42 -03:00
// @Bitmask{Copter}: 0:Clear Mission on reboot, 2:ContinueAfterLand
// @Bitmask{Rover, Sub}: 0:Clear Mission on reboot
2017-08-27 08:12:45 -03:00
// @User: Advanced
AP_GROUPINFO ( " OPTIONS " , 2 , AP_Mission , _options , AP_MISSION_OPTIONS_DEFAULT ) ,
2014-02-22 20:14:44 -04:00
AP_GROUPEND
} ;
extern const AP_HAL : : HAL & hal ;
2014-08-13 01:43:37 -03:00
// storage object
StorageAccess AP_Mission : : _storage ( StorageManager : : StorageMission ) ;
2020-01-18 17:57:26 -04:00
HAL_Semaphore AP_Mission : : _rsem ;
2018-08-19 22:37:54 -03:00
2014-02-24 01:50:24 -04:00
///
/// public mission methods
///
2014-02-22 20:14:44 -04:00
2014-03-11 00:52:23 -03:00
/// init - initialises this library including checks the version in eeprom matches this library
void AP_Mission : : init ( )
{
2023-02-14 05:39:36 -04:00
# if AP_SDCARD_STORAGE_ENABLED
// check for extra storage on microsd
const auto * bc = AP : : boardConfig ( ) ;
if ( bc ! = nullptr ) {
const auto size_kb = bc - > get_sdcard_mission_kb ( ) ;
if ( size_kb > 0 ) {
_failed_sdcard_storage = ! _storage . attach_file ( AP_MISSION_SDCARD_FILENAME , size_kb ) ;
if ( _failed_sdcard_storage ) {
// wipe mission if storage not available, but don't save. This allows sdcard error to be fixed and reboot
_cmd_total . set ( 0 ) ;
}
}
}
# endif
// work out maximum index for our storage size
if ( _storage . size ( ) > = AP_MISSION_EEPROM_COMMAND_SIZE + 4 ) {
_commands_max = ( _storage . size ( ) - 4U ) / AP_MISSION_EEPROM_COMMAND_SIZE ;
}
if ( _cmd_total . get ( ) > _commands_max ) {
// wipe mission if storage not available, but don't save. This allows sdcard error to be fixed and reboot
_cmd_total . set ( 0 ) ;
}
2014-03-11 00:52:23 -03:00
// 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 ( ) ;
2014-03-20 02:57:01 -03:00
2022-09-21 15:38:52 -03:00
// initialize the jump tracking array
init_jump_tracking ( ) ;
2017-08-27 08:12:45 -03:00
// If Mission Clear bit is set then it should clear the mission, otherwise retain the mission.
2024-07-25 21:58:16 -03:00
if ( option_is_set ( Option : : CLEAR_ON_BOOT ) ) {
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Clearing Mission " ) ;
2020-05-04 21:27:56 -03:00
clear ( ) ;
2017-08-27 08:12:45 -03:00
}
2015-11-19 23:12:21 -04:00
_last_change_time_ms = AP_HAL : : millis ( ) ;
2014-03-11 00:52:23 -03:00
}
2014-02-22 20:14:44 -04:00
/// 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 ;
2014-04-15 13:56:09 -03:00
reset ( ) ; // reset mission to the first command, resets jump tracking
2020-02-24 18:43:20 -04:00
2014-02-23 03:54:53 -04:00
// advance to the first command
2014-02-24 01:50:24 -04:00
if ( ! advance_current_nav_cmd ( ) ) {
// on failure set mission complete
complete ( ) ;
}
2014-02-22 20:14:44 -04:00
}
/// 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
2016-03-02 11:16:27 -04:00
/// previous running commands will be re-initialized
2014-02-22 20:14:44 -04:00
void AP_Mission : : resume ( )
{
2014-02-24 01:50:24 -04:00
// if mission had completed then start it from the first command
if ( _flags . state = = MISSION_COMPLETE ) {
start ( ) ;
2014-02-23 03:54:53 -04:00
return ;
}
2014-02-24 01:50:24 -04:00
// if mission had stopped then restart it
if ( _flags . state = = MISSION_STOPPED ) {
_flags . state = MISSION_RUNNING ;
2014-02-23 03:54:53 -04:00
2014-02-24 01:50:24 -04:00
// if no valid nav command index restart from beginning
if ( _nav_cmd . index = = AP_MISSION_CMD_INDEX_NONE ) {
start ( ) ;
2014-02-23 03:54:53 -04:00
return ;
}
2014-03-30 08:29:22 -03:00
}
2014-02-23 03:54:53 -04:00
2015-06-23 15:12:28 -03:00
// ensure cache coherence
2017-05-24 13:24:19 -03:00
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 ;
}
2015-06-23 15:12:28 -03:00
2020-02-24 18:43:20 -04:00
// rewind the mission wp if the repeat distance has been set via MAV_CMD_DO_SET_RESUME_REPEAT_DIST
if ( _repeat_dist > 0 & & _wp_index_history [ LAST_WP_PASSED ] ! = AP_MISSION_CMD_INDEX_NONE ) {
// if not already in a resume state calculate the position to rewind to
Mission_Command tmp_cmd ;
2020-05-04 21:27:56 -03:00
if ( ! _flags . resuming_mission & & calc_rewind_pos ( tmp_cmd ) ) {
2020-02-24 18:43:20 -04:00
_resume_cmd = tmp_cmd ;
}
// resume mission to rewound position
if ( _resume_cmd . index ! = AP_MISSION_CMD_INDEX_NONE & & start_command ( _resume_cmd ) ) {
_nav_cmd = _resume_cmd ;
_flags . nav_cmd_loaded = true ;
// set flag to prevent history being re-written
_flags . resuming_mission = true ;
return ;
}
}
2014-03-30 08:29:22 -03:00
// 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
2014-02-23 03:54:53 -04:00
2014-03-30 08:29:22 -03:00
if ( _flags . do_cmd_loaded & & _do_cmd . index ! = AP_MISSION_CMD_INDEX_NONE ) {
2018-08-22 19:38:29 -03:00
// 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 ) ;
2014-02-23 03:54:53 -04:00
}
2018-08-22 19:38:29 -03:00
// 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
2014-02-22 20:14:44 -04:00
}
2021-08-30 17:22:58 -03:00
/// check if the next nav command is a takeoff, skipping delays
bool AP_Mission : : is_takeoff_next ( uint16_t cmd_index )
2016-05-19 07:35:42 -03:00
{
Mission_Command cmd = { } ;
2018-08-08 06:10:40 -03:00
// 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:
2022-05-16 19:08:58 -03:00
case MAV_CMD_NAV_VTOL_TAKEOFF :
2018-08-08 06:10:40 -03:00
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"
2022-06-09 20:14:40 -03:00
case MAV_CMD_DO_AUX_FUNCTION :
2018-08-08 06:10:40 -03:00
case MAV_CMD_NAV_DELAY :
continue ;
default :
return false ;
}
2016-05-19 07:35:42 -03:00
}
2018-08-08 06:10:40 -03:00
return false ;
2016-05-19 07:35:42 -03:00
}
2021-08-30 17:22:58 -03:00
/// check mission starts with a takeoff command
bool AP_Mission : : starts_with_takeoff_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 ;
}
return is_takeoff_next ( cmd_index ) ;
}
/*
return true if MIS_OPTIONS is set to allow continue of mission
logic after a land and the next waypoint is a takeoff . If this
is false then after a landing is complete the vehicle should
disarm and mission logic should stop
*/
bool AP_Mission : : continue_after_land_check_for_takeoff ( )
{
if ( ! continue_after_land ( ) ) {
return false ;
}
if ( _nav_cmd . index = = AP_MISSION_CMD_INDEX_NONE ) {
return false ;
}
return is_takeoff_next ( _nav_cmd . index + 1 ) ;
}
2014-04-15 13:56:09 -03:00
/// start_or_resume - if MIS_AUTORESTART=0 this will call resume(), otherwise it will call start()
2014-04-15 13:26:45 -03:00
void AP_Mission : : start_or_resume ( )
{
2020-02-23 05:52:57 -04:00
if ( _restart = = 1 & & ! _force_resume ) {
2014-04-15 13:26:45 -03:00
start ( ) ;
} else {
resume ( ) ;
2020-02-23 05:52:57 -04:00
_force_resume = false ;
2014-04-15 13:26:45 -03:00
}
}
2014-04-15 13:56:09 -03:00
/// 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 ;
2020-02-17 19:44:53 -04:00
_flags . in_landing_sequence = false ;
2024-03-02 21:00:47 -04:00
_flags . in_return_path = false ;
2014-04-15 13:56:09 -03:00
_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 ;
2015-08-23 22:06:31 -03:00
_prev_nav_cmd_wp_index = AP_MISSION_CMD_INDEX_NONE ;
2016-05-13 20:06:25 -03:00
_prev_nav_cmd_id = AP_MISSION_CMD_ID_NONE ;
2014-04-15 13:56:09 -03:00
init_jump_tracking ( ) ;
2020-02-24 18:43:20 -04:00
reset_wp_history ( ) ;
2014-04-15 13:56:09 -03:00
}
2014-02-22 20:14:44 -04:00
/// clear - clears out mission
2023-10-25 21:09:58 -03:00
/// returns false if mission was running so it could not be cleared
2014-02-22 20:14:44 -04:00
bool AP_Mission : : clear ( )
{
2023-03-23 21:01:47 -03:00
// do not allow clearing the mission while it is running unless disarmed
if ( hal . util - > get_soft_armed ( ) & & _flags . state = = MISSION_RUNNING ) {
2014-02-22 20:14:44 -04:00
return false ;
}
// remove all commands
2022-08-04 03:19:27 -03:00
truncate ( 0 ) ;
2014-02-22 20:14:44 -04:00
// clear index to commands
_nav_cmd . index = AP_MISSION_CMD_INDEX_NONE ;
_do_cmd . index = AP_MISSION_CMD_INDEX_NONE ;
2014-02-23 03:54:53 -04:00
_flags . nav_cmd_loaded = false ;
_flags . do_cmd_loaded = false ;
2023-03-23 21:01:47 -03:00
_flags . state = MISSION_STOPPED ;
2014-02-23 03:54:53 -04:00
// return success
return true ;
2014-02-22 20:14:44 -04:00
}
2014-03-13 23:06:00 -03:00
2014-03-16 02:05:22 -03:00
/// trucate - truncate any mission items beyond index
2014-03-13 23:06:00 -03:00
void AP_Mission : : truncate ( uint16_t index )
{
2020-05-04 21:27:56 -03:00
if ( ( unsigned ) _cmd_total > index ) {
2014-03-13 23:06:00 -03:00
_cmd_total . set_and_save ( index ) ;
2022-08-04 03:19:27 -03:00
_last_change_time_ms = AP_HAL : : millis ( ) ;
2014-03-13 23:06:00 -03:00
}
}
2014-02-24 01:50:24 -04:00
/// 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 ( )
2014-02-22 20:14:44 -04:00
{
2014-02-24 01:50:24 -04:00
// exit immediately if not running or no mission commands
if ( _flags . state ! = MISSION_RUNNING | | _cmd_total = = 0 ) {
return ;
}
2020-02-24 18:43:20 -04:00
update_exit_position ( ) ;
2023-02-21 21:24:05 -04:00
// mission_change events
if ( _last_change_time_prev_ms ! = _last_change_time_ms ) {
_last_change_time_prev_ms = _last_change_time_ms ;
on_mission_timestamp_change ( ) ;
}
2019-05-09 06:23:35 -03:00
// save persistent waypoint_num for watchdog restore
2019-08-26 21:36:11 -03:00
hal . util - > persistent_data . waypoint_num = _nav_cmd . index ;
2019-05-09 06:23:35 -03:00
2014-02-24 01:50:24 -04:00
// 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 ;
}
2020-05-04 21:27:56 -03:00
} else {
2014-02-24 01:50:24 -04:00
// run the active nav command
2018-10-22 23:56:46 -03:00
if ( verify_command ( _nav_cmd ) ) {
2014-02-24 01:50:24 -04:00
// market _nav_cmd as complete (it will be started on the next iteration)
_flags . nav_cmd_loaded = false ;
2014-08-27 05:04:30 -03:00
// immediately advance to the next mission command
if ( ! advance_current_nav_cmd ( ) ) {
// failure to advance nav command means mission has completed
complete ( ) ;
return ;
}
2014-02-24 01:50:24 -04:00
}
}
// check if we have an active do command
2015-07-17 00:13:28 -03:00
if ( ! _flags . do_cmd_loaded ) {
2014-02-24 01:50:24 -04:00
advance_current_do_cmd ( ) ;
2020-05-04 21:27:56 -03:00
} else {
2018-10-22 23:56:46 -03:00
// check the active do command
if ( verify_command ( _do_cmd ) ) {
// mark _do_cmd as complete
2014-02-24 01:50:24 -04:00
_flags . do_cmd_loaded = false ;
}
}
2014-02-22 20:14:44 -04:00
}
2023-02-21 21:24:05 -04:00
// handle events for when the mission has been updated (but maybe not changed)
void AP_Mission : : on_mission_timestamp_change ( )
{
_jump_tag . age = 0 ;
}
2018-10-22 23:56:46 -03:00
bool AP_Mission : : verify_command ( const Mission_Command & cmd )
{
2018-10-23 01:45:34 -03:00
switch ( cmd . id ) {
2020-05-04 21:27:56 -03:00
// do-commands always return true for verify:
2022-09-20 04:37:48 -03:00
# if AP_GRIPPER_ENABLED
2018-10-23 01:45:34 -03:00
case MAV_CMD_DO_GRIPPER :
2022-09-20 04:37:48 -03:00
# endif
2018-10-23 03:11:47 -03:00
case MAV_CMD_DO_SET_SERVO :
case MAV_CMD_DO_SET_RELAY :
case MAV_CMD_DO_REPEAT_SERVO :
case MAV_CMD_DO_REPEAT_RELAY :
2018-10-23 04:03:50 -03:00
case MAV_CMD_DO_DIGICAM_CONFIGURE :
case MAV_CMD_DO_DIGICAM_CONTROL :
case MAV_CMD_DO_SET_CAM_TRIGG_DIST :
2019-01-31 20:16:42 -04:00
case MAV_CMD_DO_PARACHUTE :
2021-02-25 21:08:22 -04:00
case MAV_CMD_DO_SEND_SCRIPT_MESSAGE :
2020-07-02 20:04:27 -03:00
case MAV_CMD_DO_SPRAYER :
2021-02-17 21:01:41 -04:00
case MAV_CMD_DO_AUX_FUNCTION :
2020-02-24 18:43:20 -04:00
case MAV_CMD_DO_SET_RESUME_REPEAT_DIST :
2022-06-20 08:51:35 -03:00
case MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW :
2023-02-21 21:24:05 -04:00
case MAV_CMD_JUMP_TAG :
2023-03-08 07:28:43 -04:00
case MAV_CMD_IMAGE_START_CAPTURE :
2023-09-07 07:12:48 -03:00
case MAV_CMD_IMAGE_STOP_CAPTURE :
2023-03-08 07:28:43 -04:00
case MAV_CMD_SET_CAMERA_ZOOM :
case MAV_CMD_SET_CAMERA_FOCUS :
2024-02-08 04:30:08 -04:00
case MAV_CMD_SET_CAMERA_SOURCE :
2023-03-08 07:28:43 -04:00
case MAV_CMD_VIDEO_START_CAPTURE :
case MAV_CMD_VIDEO_STOP_CAPTURE :
2018-10-23 01:45:34 -03:00
return true ;
default :
return _cmd_verify_fn ( cmd ) ;
}
2018-10-22 23:56:46 -03:00
}
bool AP_Mission : : start_command ( const Mission_Command & cmd )
{
2024-09-27 04:34:19 -03:00
# if HAL_LOGGING_ENABLED
if ( log_start_mission_item_bit ! = ( uint32_t ) - 1 ) {
auto & logger = AP : : logger ( ) ;
if ( logger . should_log ( log_start_mission_item_bit ) ) {
logger . Write_MISE ( * this , cmd ) ;
}
}
# endif
2024-03-02 21:00:47 -04:00
// check for landing related commands and set flags
2020-02-17 19:44:53 -04:00
if ( is_landing_type_cmd ( cmd . id ) | | cmd . id = = MAV_CMD_DO_LAND_START ) {
2024-03-02 21:00:47 -04:00
_flags . in_landing_sequence = true ;
} else if ( cmd . id = = MAV_CMD_DO_RETURN_PATH_START ) {
_flags . in_return_path = true ;
2022-06-09 20:14:40 -03:00
} else if ( is_takeoff_type_cmd ( cmd . id ) ) {
2024-03-02 21:00:47 -04:00
// Clear landing and return path flags on takeoff
_flags . in_landing_sequence = false ;
_flags . in_return_path = false ;
2020-02-17 19:44:53 -04:00
}
2024-05-01 16:25:14 -03:00
switch ( cmd . id ) {
case MAV_CMD_DO_JUMP :
case MAV_CMD_JUMP_TAG :
case MAV_CMD_DO_JUMP_TAG :
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Mission: %u %s %u " , cmd . index , cmd . type ( ) , ( unsigned ) cmd . p1 ) ;
2024-05-01 16:25:14 -03:00
break ;
default :
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Mission: %u %s " , cmd . index , cmd . type ( ) ) ;
2024-05-01 16:25:14 -03:00
break ;
2023-08-03 15:04:19 -03:00
}
2018-10-23 01:45:34 -03:00
switch ( cmd . id ) {
2023-12-09 02:09:26 -04:00
# if AP_RC_CHANNEL_ENABLED
2021-02-17 21:01:41 -04:00
case MAV_CMD_DO_AUX_FUNCTION :
return start_command_do_aux_function ( cmd ) ;
2023-12-09 02:09:26 -04:00
# endif
2022-09-20 04:37:48 -03:00
# if AP_GRIPPER_ENABLED
2018-10-23 01:45:34 -03:00
case MAV_CMD_DO_GRIPPER :
return start_command_do_gripper ( cmd ) ;
2022-09-20 04:37:48 -03:00
# endif
2023-06-06 05:05:06 -03:00
# if AP_SERVORELAYEVENTS_ENABLED
2018-10-23 03:11:47 -03:00
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 ) ;
2023-06-06 05:05:06 -03:00
# endif
2023-03-08 06:51:41 -04:00
# if AP_CAMERA_ENABLED
2018-10-23 04:03:50 -03:00
case MAV_CMD_DO_DIGICAM_CONFIGURE :
case MAV_CMD_DO_DIGICAM_CONTROL :
case MAV_CMD_DO_SET_CAM_TRIGG_DIST :
2023-03-08 07:28:43 -04:00
case MAV_CMD_IMAGE_START_CAPTURE :
2023-09-07 07:12:48 -03:00
case MAV_CMD_IMAGE_STOP_CAPTURE :
2023-03-08 07:28:43 -04:00
case MAV_CMD_SET_CAMERA_ZOOM :
case MAV_CMD_SET_CAMERA_FOCUS :
2024-02-08 04:30:08 -04:00
case MAV_CMD_SET_CAMERA_SOURCE :
2023-03-08 07:28:43 -04:00
case MAV_CMD_VIDEO_START_CAPTURE :
case MAV_CMD_VIDEO_STOP_CAPTURE :
2018-10-23 04:03:50 -03:00
return start_command_camera ( cmd ) ;
2024-05-31 11:18:59 -03:00
# endif
# if AP_FENCE_ENABLED
case MAV_CMD_DO_FENCE_ENABLE :
return start_command_fence ( cmd ) ;
2022-06-02 05:28:26 -03:00
# endif
2019-01-31 20:16:42 -04:00
case MAV_CMD_DO_PARACHUTE :
return start_command_parachute ( cmd ) ;
2021-02-25 21:08:22 -04:00
case MAV_CMD_DO_SEND_SCRIPT_MESSAGE :
return start_command_do_scripting ( cmd ) ;
2020-07-02 20:04:27 -03:00
case MAV_CMD_DO_SPRAYER :
return start_command_do_sprayer ( cmd ) ;
2020-02-24 18:43:20 -04:00
case MAV_CMD_DO_SET_RESUME_REPEAT_DIST :
return command_do_set_repeat_dist ( cmd ) ;
2022-06-02 09:23:47 -03:00
case MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW :
return start_command_do_gimbal_manager_pitchyaw ( cmd ) ;
2023-02-21 21:24:05 -04:00
case MAV_CMD_JUMP_TAG :
_jump_tag . tag = cmd . content . jump . target ;
_jump_tag . age = 1 ;
FALLTHROUGH ; // fall through in case the vehicle handles tag events
2018-10-23 01:45:34 -03:00
default :
return _cmd_start_fn ( cmd ) ;
}
2018-10-22 23:56:46 -03:00
}
2014-02-24 01:50:24 -04:00
///
/// public command methods
///
2014-02-22 20:14:44 -04:00
/// 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 ;
}
2014-02-28 22:12:41 -04:00
/// 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
2019-01-30 05:20:20 -04:00
bool AP_Mission : : replace_cmd ( uint16_t index , const Mission_Command & cmd )
2014-02-28 22:12:41 -04:00
{
// sanity check index
2014-03-23 22:52:27 -03:00
if ( index > = ( unsigned ) _cmd_total ) {
2014-02-28 22:12:41 -04:00
return false ;
}
// attempt to write the command to storage
return write_cmd_to_storage ( index , cmd ) ;
}
2014-02-24 01:50:24 -04:00
/// 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 )
{
2021-10-25 00:59:11 -03:00
// NAV commands all have ids below MAV_CMD_NAV_LAST, plus some exceptions
return ( cmd . id < = MAV_CMD_NAV_LAST | |
cmd . id = = MAV_CMD_NAV_SET_YAW_SPEED | |
2022-05-25 01:52:01 -03:00
cmd . id = = MAV_CMD_NAV_SCRIPT_TIME | |
cmd . id = = MAV_CMD_NAV_ATTITUDE_TIME ) ;
2014-02-24 01:50:24 -04:00
}
/// 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)
2014-02-28 08:49:37 -04:00
bool AP_Mission : : get_next_nav_cmd ( uint16_t start_index , Mission_Command & cmd )
2014-02-24 01:50:24 -04:00
{
// search until the end of the mission command list
2018-08-24 20:52:34 -03:00
for ( uint16_t cmd_index = start_index ; cmd_index < ( unsigned ) _cmd_total ; cmd_index + + ) {
2014-02-24 01:50:24 -04:00
// get next command
2014-04-06 20:29:54 -03:00
if ( ! get_next_cmd ( cmd_index , cmd , false ) ) {
2014-02-24 01:50:24 -04:00
// no more commands so return failure
return false ;
}
2018-08-08 05:04:21 -03:00
// if found a "navigation" command then return it
if ( is_nav_cmd ( cmd ) ) {
return true ;
}
2014-02-24 01:50:24 -04:00
}
// if we got this far we did not find a navigation command
return false ;
}
2014-04-06 20:29:54 -03:00
/// 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 ;
}
2017-08-02 02:25:57 -03:00
// special handling for nav commands with no target location
2024-05-01 16:25:14 -03:00
switch ( cmd . id ) {
case MAV_CMD_NAV_GUIDED_ENABLE :
case MAV_CMD_NAV_DELAY :
2017-08-02 02:25:57 -03:00
return default_angle ;
2024-05-01 16:25:14 -03:00
case MAV_CMD_NAV_SET_YAW_SPEED :
2017-08-02 02:25:57 -03:00
return ( _nav_cmd . content . set_yaw_speed . angle_deg * 100 ) ;
2024-05-01 16:25:14 -03:00
default :
break ;
2017-08-02 02:25:57 -03:00
}
2024-05-01 16:25:14 -03:00
2019-04-05 10:02:43 -03:00
return _nav_cmd . content . location . get_bearing_to ( cmd . content . location ) ;
2014-04-06 20:29:54 -03:00
}
2014-02-28 21:19:38 -04:00
// set_current_cmd - jumps to command specified by index
2023-09-27 22:02:31 -03:00
bool AP_Mission : : set_current_cmd ( uint16_t index )
2014-02-28 21:19:38 -04:00
{
2024-03-02 21:00:47 -04:00
// Clear flags
_flags . in_landing_sequence = false ;
_flags . in_return_path = false ;
// read command to check for DO_LAND_START and DO_RETURN_PATH_START
2020-02-17 19:44:53 -04:00
Mission_Command cmd ;
2024-03-02 21:00:47 -04:00
if ( read_cmd_from_storage ( index , cmd ) ) {
2024-05-01 16:25:14 -03:00
switch ( cmd . id ) {
case MAV_CMD_DO_LAND_START :
2024-03-02 21:00:47 -04:00
_flags . in_landing_sequence = true ;
2024-05-01 16:25:14 -03:00
break ;
2024-03-02 21:00:47 -04:00
2024-05-01 16:25:14 -03:00
case MAV_CMD_DO_RETURN_PATH_START :
2024-03-02 21:00:47 -04:00
_flags . in_return_path = true ;
2024-05-01 16:25:14 -03:00
break ;
default :
break ;
2024-03-02 21:00:47 -04:00
}
2020-02-17 19:44:53 -04:00
}
2023-09-27 22:02:31 -03:00
// mission command has been set, don't track history.
reset_wp_history ( ) ;
2020-02-24 18:43:20 -04:00
2014-03-07 04:01:05 -04:00
// sanity check index and that we have a mission
2014-03-23 22:52:27 -03:00
if ( index > = ( unsigned ) _cmd_total | | _cmd_total = = 1 ) {
2014-02-28 21:19:38 -04:00
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 ;
2014-03-07 04:01:05 -04:00
// if index is zero then the user wants to completely restart the mission
if ( index = = 0 | | _flags . state = = MISSION_COMPLETE ) {
2016-05-13 20:06:25 -03:00
_prev_nav_cmd_id = AP_MISSION_CMD_ID_NONE ;
2014-03-07 04:01:05 -04:00
_prev_nav_cmd_index = AP_MISSION_CMD_INDEX_NONE ;
2015-08-23 22:06:31 -03:00
_prev_nav_cmd_wp_index = AP_MISSION_CMD_INDEX_NONE ;
2014-03-07 04:01:05 -04:00
// reset the jump tracking to zero
init_jump_tracking ( ) ;
2014-03-14 09:37:56 -03:00
if ( index = = 0 ) {
index = 1 ;
}
2014-03-07 04:01:05 -04:00
}
// 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
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 ;
2020-05-04 21:27:56 -03:00
} else {
2014-03-07 04:01:05 -04:00
// 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 ;
}
2015-07-17 00:13:28 -03:00
// 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 ;
}
2014-03-07 04:01:05 -04:00
// 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 ;
}
2018-08-22 19:38:29 -03:00
// 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 ;
2015-07-17 00:13:28 -03:00
}
2014-02-28 21:19:38 -04:00
// if we got this far we must have successfully advanced the nav command
return true ;
}
2021-03-05 21:26:40 -04:00
// restart current navigation command. Used to handle external changes to mission
// returns true on success, false if mission is not running or current nav command is invalid
bool AP_Mission : : restart_current_nav_cmd ( )
{
// return immediately if mission is not running
if ( _flags . state ! = MISSION_RUNNING ) {
return false ;
}
// return immediately if nav command index is invalid
const uint16_t nav_cmd_index = get_current_nav_index ( ) ;
if ( ( nav_cmd_index = = 0 ) | | ( nav_cmd_index > = num_commands ( ) ) ) {
return false ;
}
return set_current_cmd ( _nav_cmd . index ) ;
}
2020-05-04 21:27:56 -03:00
// returns false on any issue at all.
bool AP_Mission : : set_item ( uint16_t index , mavlink_mission_item_int_t & src_packet )
{
// this is the on-storage format
2023-02-14 05:39:36 -04:00
AP_Mission : : Mission_Command cmd { } ;
2020-05-04 21:27:56 -03:00
// can't handle request for anything bigger than the mission size+1...
2021-03-05 21:26:16 -04:00
if ( index > num_commands ( ) ) {
2020-05-04 21:27:56 -03:00
return false ;
}
// convert from mavlink-ish format to storage format, if we can.
if ( mavlink_int_to_mission_cmd ( src_packet , cmd ) ! = MAV_MISSION_ACCEPTED ) {
return false ;
}
// A request to set the 'next' item after the end is how we add an extra
// item to the list, thus allowing us to write entire missions if needed.
2021-03-05 21:26:16 -04:00
if ( index = = num_commands ( ) ) {
return add_cmd ( cmd ) ;
2020-05-04 21:27:56 -03:00
}
// replacing an existing mission item...
2021-03-05 21:26:16 -04:00
return AP_Mission : : replace_cmd ( index , cmd ) ;
2020-05-04 21:27:56 -03:00
}
2021-02-01 12:26:31 -04:00
bool AP_Mission : : get_item ( uint16_t index , mavlink_mission_item_int_t & ret_packet ) const
2020-05-04 21:27:56 -03:00
{
2024-09-04 01:41:47 -03:00
mavlink_mission_item_int_t tmp ;
2020-05-04 21:27:56 -03:00
// can't handle request for anything bigger than the mission size...
2021-03-05 21:26:16 -04:00
if ( index > = num_commands ( ) ) {
2020-05-04 21:27:56 -03:00
return false ;
}
// retrieve mission from eeprom
2024-09-04 01:41:47 -03:00
AP_Mission : : Mission_Command cmd { } ;
2025-01-11 09:56:59 -04:00
if ( ! read_cmd_from_storage ( index , cmd ) ) {
2020-05-04 21:27:56 -03:00
return false ;
}
// convert into mavlink-ish format for lua and friends.
2024-09-04 01:41:47 -03:00
if ( ! mission_cmd_to_mavlink_int ( cmd , tmp ) ) {
2020-05-04 21:27:56 -03:00
return false ;
}
// set packet's current field to 1 if this is the command being executed
if ( cmd . id = = ( uint16_t ) get_current_nav_cmd ( ) . index ) {
2024-09-04 01:41:47 -03:00
tmp . current = 1 ;
2020-05-04 21:27:56 -03:00
}
2024-09-04 01:41:47 -03:00
ret_packet = tmp ;
2020-05-04 21:27:56 -03:00
return true ;
}
2019-01-01 20:07:03 -04:00
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
2022-05-11 00:40:48 -03:00
uint8_t type_specific_bit_0 : 1 ; // each mission item type can use this for storing 1 bit of extra data
2024-01-12 00:14:19 -04:00
uint8_t type_specific_bit_1 : 1 ; // each mission item type can use this for storing 1 bit of extra data
2019-01-01 20:07:03 -04:00
} ;
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 ] ;
} ;
2014-02-22 20:14:44 -04:00
/// load_cmd_from_storage - load command from storage
/// true is return if successful
2014-02-28 08:49:37 -04:00
bool AP_Mission : : read_cmd_from_storage ( uint16_t index , Mission_Command & cmd ) const
2014-02-22 20:14:44 -04:00
{
2023-12-13 19:20:16 -04:00
ASSERT_STORAGE_SIZE ( PackedContent , 12 ) ;
2018-08-19 22:37:54 -03:00
WITH_SEMAPHORE ( _rsem ) ;
2019-03-13 07:45:23 -03:00
2014-02-27 22:58:27 -04:00
// special handling for command #0 which is home
if ( index = = 0 ) {
2021-03-15 00:00:15 -03:00
cmd = { } ;
2014-02-27 22:58:27 -04:00
cmd . id = MAV_CMD_NAV_WAYPOINT ;
2018-04-24 20:09:26 -03:00
cmd . content . location = AP : : ahrs ( ) . get_home ( ) ;
2019-07-30 04:34:18 -03:00
return true ;
}
2023-02-14 05:39:36 -04:00
if ( index > = ( unsigned ) _cmd_total | | index > = _commands_max ) {
2019-07-30 04:34:18 -03:00
return false ;
}
2021-03-15 00:00:15 -03:00
// ensure all bytes of cmd are zeroed
cmd = { } ;
2019-07-30 04:34:18 -03:00
// 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 ) ;
2014-02-27 09:02:12 -04:00
2019-07-30 04:34:18 -03:00
PackedContent packed_content { } ;
const uint8_t b1 = _storage . read_byte ( pos_in_storage ) ;
2022-10-18 22:44:33 -03:00
if ( b1 = = 0 | | b1 = = 1 ) {
2019-07-30 04:34:18 -03:00
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 ) ;
2022-10-18 22:44:33 -03:00
format_conversion ( b1 , cmd , packed_content ) ;
2019-07-30 04:34:18 -03:00
} else {
cmd . id = b1 ;
cmd . p1 = _storage . read_uint16 ( pos_in_storage + 1 ) ;
_storage . read_block ( packed_content . bytes , pos_in_storage + 3 , 12 ) ;
2014-02-27 22:58:27 -04:00
}
2014-02-22 20:14:44 -04:00
2019-07-30 04:34:18 -03:00
if ( stored_in_location ( cmd . id ) ) {
2019-10-17 18:49:04 -03:00
# 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
2019-07-30 04:34:18 -03:00
// 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 ;
2022-05-11 00:40:48 -03:00
if ( packed_content . location . flags . type_specific_bit_0 ) {
2024-01-12 00:14:19 -04:00
cmd . type_specific_bits | = 1U < < 0 ;
}
if ( packed_content . location . flags . type_specific_bit_1 ) {
cmd . type_specific_bits | = 1U < < 1 ;
2022-05-11 00:40:48 -03:00
}
2019-07-30 04:34:18 -03:00
} 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 ;
2014-02-22 20:14:44 -04:00
// return success
return true ;
}
2019-01-01 20:07:03 -04:00
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 :
2019-03-08 01:27:09 -04:00
case MAV_CMD_NAV_GUIDED_ENABLE :
2019-01-01 20:07:03 -04:00
case MAV_CMD_DO_SET_HOME :
2024-03-02 21:00:47 -04:00
case MAV_CMD_DO_RETURN_PATH_START :
2019-01-01 20:07:03 -04:00
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 ;
}
}
2014-02-22 20:14:44 -04:00
/// write_cmd_to_storage - write a command to storage
/// index is used to calculate the storage location
/// true is returned if successful
2019-01-30 05:20:20 -04:00
bool AP_Mission : : write_cmd_to_storage ( uint16_t index , const Mission_Command & cmd )
2014-02-22 20:14:44 -04:00
{
2018-08-19 22:37:54 -03:00
WITH_SEMAPHORE ( _rsem ) ;
2020-05-04 21:27:56 -03:00
2014-02-22 20:14:44 -04:00
// range check cmd's index
2014-08-13 01:43:37 -03:00
if ( index > = num_commands_max ( ) ) {
2014-02-22 20:14:44 -04:00
return false ;
}
2019-06-24 06:35:21 -03:00
PackedContent packed { } ;
2019-01-01 20:07:03 -04:00
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 ;
2024-01-12 00:14:19 -04:00
packed . location . flags . type_specific_bit_0 = ( cmd . type_specific_bits & ( 1U < < 0 ) ) > > 0 ;
packed . location . flags . type_specific_bit_1 = ( cmd . type_specific_bits & ( 1U < < 1 ) ) > > 1 ;
2019-01-01 20:07:03 -04:00
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:
2019-02-20 00:25:54 -04:00
static_assert ( sizeof ( packed . bytes ) > = 12 ,
" packed.bytes is big enough to take content " ) ;
memcpy ( packed . bytes , & cmd . content , 12 ) ;
2019-01-01 20:07:03 -04:00
}
2014-02-22 20:14:44 -04:00
// calculate where in storage the command should be placed
2014-08-13 01:43:37 -03:00
uint16_t pos_in_storage = 4 + ( index * AP_MISSION_EEPROM_COMMAND_SIZE ) ;
2014-02-22 20:14:44 -04:00
2016-04-22 04:58:42 -03:00
if ( cmd . id < 256 ) {
2022-10-18 22:44:33 -03:00
// for commands below 256 we store up to 12 bytes
2016-04-22 04:58:42 -03:00
_storage . write_byte ( pos_in_storage , cmd . id ) ;
_storage . write_uint16 ( pos_in_storage + 1 , cmd . p1 ) ;
2019-01-01 20:07:03 -04:00
_storage . write_block ( pos_in_storage + 3 , packed . bytes , 12 ) ;
2016-04-22 04:58:42 -03:00
} else {
2022-10-18 22:44:33 -03:00
// if the command ID is above 256 we store a tag byte followed
// by the 16 bit command ID. The tag byte is 1 for commands
// where we have changed the storage format (see
// format_conversion), 0 otherwise
uint8_t tag_byte = 0 ;
// currently the only converted structure is NAV_SCRIPT_TIME
if ( cmd . id = = MAV_CMD_NAV_SCRIPT_TIME ) {
tag_byte = 1 ;
}
_storage . write_byte ( pos_in_storage , tag_byte ) ;
2016-04-22 04:58:42 -03:00
_storage . write_uint16 ( pos_in_storage + 1 , cmd . id ) ;
_storage . write_uint16 ( pos_in_storage + 3 , cmd . p1 ) ;
2019-01-01 20:07:03 -04:00
_storage . write_block ( pos_in_storage + 5 , packed . bytes , 10 ) ;
2016-04-22 04:58:42 -03:00
}
2014-02-22 20:14:44 -04:00
2014-08-06 03:17:11 -03:00
// remember when the mission last changed
2015-11-19 23:12:21 -04:00
_last_change_time_ms = AP_HAL : : millis ( ) ;
2014-08-06 03:17:11 -03:00
2014-02-22 20:14:44 -04:00
// return success
return true ;
}
2014-03-03 01:36:37 -04:00
/// 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 ( )
{
2014-03-10 23:23:44 -03:00
Mission_Command home_cmd = { } ;
2014-03-03 01:36:37 -04:00
home_cmd . id = MAV_CMD_NAV_WAYPOINT ;
2018-04-24 20:09:26 -03:00
home_cmd . content . location = AP : : ahrs ( ) . get_home ( ) ;
2014-03-03 01:36:37 -04:00
write_cmd_to_storage ( 0 , home_cmd ) ;
}
2020-05-04 21:27:56 -03:00
MAV_MISSION_RESULT AP_Mission : : sanity_check_params ( const mavlink_mission_item_int_t & packet )
{
2018-04-26 02:46:46 -03:00
uint8_t nan_mask ;
switch ( packet . command ) {
2020-05-04 21:27:56 -03:00
case MAV_CMD_NAV_WAYPOINT :
nan_mask = ~ ( 1 < < 3 ) ; // param 4 can be nan
2024-07-07 19:01:53 -03:00
break ;
case MAV_CMD_NAV_LOITER_UNLIM :
nan_mask = ~ ( 1 < < 3 ) ; // param 4 can be nan
2020-05-04 21:27:56 -03:00
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 ;
2018-04-26 02:46:46 -03:00
}
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 ;
}
2019-01-30 19:27:22 -04:00
// mavlink_int_to_mission_cmd - converts mavlink message to an AP_Mission::Mission_Command object which can be stored to eeprom
2015-12-21 01:02:23 -04:00
// return MAV_MISSION_ACCEPTED on success, MAV_MISSION_RESULT error on failure
2016-02-28 02:47:00 -04:00
MAV_MISSION_RESULT AP_Mission : : mavlink_int_to_mission_cmd ( const mavlink_mission_item_int_t & packet , AP_Mission : : Mission_Command & cmd )
2014-02-28 02:55:02 -04:00
{
2024-01-23 00:37:37 -04:00
cmd = { } ;
2014-02-28 02:55:02 -04:00
// command's position in mission list and mavlink id
cmd . index = packet . seq ;
cmd . id = packet . command ;
2019-01-01 23:13:08 -04:00
cmd . content . location = { } ;
2014-02-28 02:55:02 -04:00
2018-04-26 02:46:46 -03:00
MAV_MISSION_RESULT param_check = sanity_check_params ( packet ) ;
if ( param_check ! = MAV_MISSION_ACCEPTED ) {
return param_check ;
}
2014-02-28 02:55:02 -04:00
// command specific conversions from mavlink packet to mission command
switch ( cmd . id ) {
2022-10-18 22:44:33 -03:00
case 0 . . . 1 :
// these are reserved for storing 16 bit command IDs
2016-04-22 04:58:42 -03:00
return MAV_MISSION_INVALID ;
2020-05-04 21:27:56 -03:00
case MAV_CMD_NAV_WAYPOINT : { // MAV ID: 16
2014-09-02 22:23:08 -03:00
/*
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
*/
2014-09-03 00:28:17 -03:00
# if APM_BUILD_TYPE(APM_BUILD_ArduPlane)
2016-12-18 10:39:44 -04:00
// 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
2017-01-06 21:29:05 -04:00
// limit to 255 so it does not wrap during the shift or mask operation
passby = MIN ( 0xFF , passby ) ;
acp = MIN ( 0xFF , acp ) ;
2016-12-18 10:39:44 -04:00
cmd . p1 = ( passby < < 8 ) | ( acp & 0x00FF ) ;
2014-09-02 22:23:08 -03:00
# else
2016-12-18 10:39:44 -04:00
// delay at waypoint in seconds (this is for copters???)
cmd . p1 = packet . param1 ;
2014-09-02 22:23:08 -03:00
# endif
2016-12-18 10:39:44 -04:00
}
2020-05-04 21:27:56 -03:00
break ;
2014-02-28 02:55:02 -04:00
case MAV_CMD_NAV_LOITER_UNLIM : // MAV ID: 17
2016-01-27 15:35:41 -04:00
cmd . p1 = fabsf ( packet . param3 ) ; // store radius as 16bit since no other params are competing for space
2019-01-01 20:07:03 -04:00
cmd . content . location . loiter_ccw = ( packet . param3 < 0 ) ; // -1 = counter clockwise, +1 = clockwise
2014-02-28 02:55:02 -04:00
break ;
2020-05-04 21:27:56 -03:00
case MAV_CMD_NAV_LOITER_TURNS : { // MAV ID: 18
2024-01-12 00:14:19 -04:00
// number of turns is stored in the lowest bits. Number of
// turns 0 < N < 1 are stored multiplied by 256 and a bit set
// in storage so that on retrieval they are divided by 256.
// Radii below 255m are stored in the top 8 bits as an 8-bit integer.
2022-05-11 00:40:48 -03:00
// Radii above 255m are stored divided by 10 and a bit set in
// storage so that on retrieval they are multiplied by 10
2024-01-12 00:14:19 -04:00
float param1_stored = packet . param1 ;
if ( param1_stored > 0 & & param1_stored < 1 ) {
param1_stored * = 256.0 ;
cmd . type_specific_bits | = ( 1U < < 1 ) ;
}
cmd . p1 = MIN ( 255 , param1_stored ) ; // store number of times to circle in low p1
2022-05-11 00:40:48 -03:00
uint8_t radius_m ;
const float abs_radius = fabsf ( packet . param3 ) ;
if ( abs_radius < = 255 ) {
radius_m = abs_radius ;
} else {
radius_m = MIN ( 255 , abs_radius * 0.1 ) ;
2024-01-12 00:14:19 -04:00
cmd . type_specific_bits | = ( 1U < < 0 ) ;
2022-05-11 00:40:48 -03:00
}
cmd . p1 | = ( radius_m < < 8 ) ; // store radius in high byte of p1
2019-01-01 20:07:03 -04:00
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
2016-03-02 11:16:27 -04:00
}
2020-05-04 21:27:56 -03:00
break ;
2014-02-28 02:55:02 -04:00
case MAV_CMD_NAV_LOITER_TIME : // MAV ID: 19
2016-01-27 15:35:41 -04:00
cmd . p1 = packet . param1 ; // loiter time in seconds uses all 16 bits, 8bit seconds is too small. No room for radius.
2019-01-01 20:07:03 -04:00
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
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_NAV_RETURN_TO_LAUNCH : // MAV ID: 20
break ;
case MAV_CMD_NAV_LAND : // MAV ID: 21
2015-08-28 22:33:48 -03:00
cmd . p1 = packet . param1 ; // abort target altitude(m) (plane only)
2021-05-13 00:11:11 -03:00
if ( ! isnan ( packet . param4 ) ) {
cmd . content . location . loiter_ccw = is_negative ( packet . param4 ) ; // yaw direction, (plane deepstall only)
}
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_NAV_TAKEOFF : // MAV ID: 22
2014-03-03 01:36:37 -04:00
cmd . p1 = packet . param1 ; // minimum pitch (plane only)
2014-02-28 02:55:02 -04:00
break ;
2014-10-21 17:45:24 -03:00
case MAV_CMD_NAV_CONTINUE_AND_CHANGE_ALT : // MAV ID: 30
2015-08-18 23:45:48 -03:00
cmd . p1 = packet . param1 ; // Climb/Descend
2020-05-04 21:27:56 -03:00
// 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.
2014-10-21 17:45:24 -03:00
break ;
2015-03-13 18:47:37 -03:00
case MAV_CMD_NAV_LOITER_TO_ALT : // MAV ID: 31
2016-03-02 11:16:27 -04:00
cmd . p1 = fabsf ( packet . param2 ) ; // param2 is radius in meters
2019-01-01 20:07:03 -04:00
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
2015-03-13 18:47:37 -03:00
break ;
2014-03-22 00:45:26 -03:00
case MAV_CMD_NAV_SPLINE_WAYPOINT : // MAV ID: 82
2021-11-18 17:20:14 -04:00
# if APM_BUILD_TYPE(APM_BUILD_ArduPlane)
return MAV_MISSION_UNSUPPORTED ;
# else
2014-03-22 00:45:26 -03:00
cmd . p1 = packet . param1 ; // delay at waypoint in seconds
break ;
2021-11-18 17:20:14 -04:00
# endif
2014-03-22 00:45:26 -03:00
2014-10-13 04:03:56 -03:00
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
2014-06-02 06:02:04 -03:00
break ;
2019-08-14 14:01:42 -03:00
case MAV_CMD_NAV_DELAY : // MAV ID: 93
2016-04-18 04:12:40 -03:00
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 ;
2014-02-28 02:55:02 -04:00
case MAV_CMD_CONDITION_DELAY : // MAV ID: 112
2014-03-17 02:14:45 -03:00
cmd . content . delay . seconds = packet . param1 ; // delay in seconds
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_CONDITION_DISTANCE : // MAV ID: 114
2014-03-17 02:14:45 -03:00
cmd . content . distance . meters = packet . param1 ; // distance in meters from next waypoint
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_CONDITION_YAW : // MAV ID: 115
2014-03-17 02:14:45 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_DO_JUMP : // MAV ID: 177
2023-02-21 21:24:05 -04:00
case MAV_CMD_DO_JUMP_TAG : // MAV ID: 601
cmd . content . jump . target = packet . param1 ; // jump-to command/tag number
2014-02-28 02:55:02 -04:00
cmd . content . jump . num_times = packet . param2 ; // repeat count
break ;
2023-02-21 21:24:05 -04:00
case MAV_CMD_JUMP_TAG : // MAV ID: 600
cmd . content . jump . target = packet . param1 ; // jump-to tag number
break ;
2014-02-28 02:55:02 -04:00
case MAV_CMD_DO_CHANGE_SPEED : // MAV ID: 178
2014-03-17 02:14:45 -03:00
cmd . content . speed . speed_type = packet . param1 ; // 0 = airspeed, 1 = ground speed
cmd . content . speed . target_ms = packet . param2 ; // target speed in m/s
2020-01-04 07:46:46 -04:00
cmd . content . speed . throttle_pct = packet . param3 ; // throttle as a percentage from 1 ~ 100%
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_DO_SET_HOME :
2014-03-17 02:14:45 -03:00
cmd . p1 = packet . param1 ; // p1=0 means use current location, p=1 means use provided location
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_DO_SET_RELAY : // MAV ID: 181
2014-03-17 02:14:45 -03:00
cmd . content . relay . num = packet . param1 ; // relay number
cmd . content . relay . state = packet . param2 ; // 0:off, 1:on
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_DO_REPEAT_RELAY : // MAV ID: 182
2014-03-18 23:13:04 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_DO_SET_SERVO : // MAV ID: 183
2014-03-17 02:14:45 -03:00
cmd . content . servo . channel = packet . param1 ; // channel
cmd . content . servo . pwm = packet . param2 ; // PWM
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_DO_REPEAT_SERVO : // MAV ID: 184
2014-03-18 23:13:04 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
2024-03-02 21:00:47 -04:00
case MAV_CMD_DO_RETURN_PATH_START : // MAV ID: 188
2014-10-17 14:34:04 -03:00
case MAV_CMD_DO_LAND_START : // MAV ID: 189
break ;
2018-12-05 19:28:11 -04:00
case MAV_CMD_DO_GO_AROUND : // MAV ID: 191
break ;
2014-02-28 02:55:02 -04:00
case MAV_CMD_DO_SET_ROI : // MAV ID: 201
2014-03-17 02:14:45 -03:00
cmd . p1 = packet . param1 ; // 0 = no roi, 1 = next waypoint, 2 = waypoint number, 3 = fixed location, 4 = given target (not supported)
2014-02-28 02:55:02 -04:00
break ;
2015-03-28 22:29:21 -03:00
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 ;
2014-02-28 02:55:02 -04:00
case MAV_CMD_DO_DIGICAM_CONTROL : // MAV ID: 203
2015-03-28 22:29:21 -03:00
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 ;
2015-03-21 08:56:50 -03:00
break ;
2014-02-28 02:55:02 -04:00
case MAV_CMD_DO_MOUNT_CONTROL : // MAV ID: 205
2024-05-15 22:27:05 -03:00
// TODO: this is only valid if packet.z == MAV_MOUNT_MODE_MAVLINK_TARGETING
2015-03-21 08:56:50 -03:00
cmd . content . mount_control . pitch = packet . param1 ;
cmd . content . mount_control . roll = packet . param2 ;
cmd . content . mount_control . yaw = packet . param3 ;
2014-02-28 02:55:02 -04:00
break ;
case MAV_CMD_DO_SET_CAM_TRIGG_DIST : // MAV ID: 206
2014-03-17 02:14:45 -03:00
cmd . content . cam_trigg_dist . meters = packet . param1 ; // distance between camera shots in meters
2020-03-23 20:55:51 -03:00
cmd . content . cam_trigg_dist . trigger = packet . param3 ; // when enabled, camera triggers once immediately
2014-02-28 02:55:02 -04:00
break ;
2014-10-25 18:19:21 -03:00
case MAV_CMD_DO_FENCE_ENABLE : // MAV ID: 207
2024-01-13 14:03:19 -04:00
cmd . p1 = packet . param1 ; // action 0=disable, 1=enable, 2=disable floor
2014-10-25 18:19:21 -03:00
break ;
2021-02-17 21:01:41 -04:00
case MAV_CMD_DO_AUX_FUNCTION :
cmd . content . auxfunction . function = packet . param1 ;
cmd . content . auxfunction . switchpos = packet . param2 ;
break ;
2014-04-03 05:49:04 -03:00
case MAV_CMD_DO_PARACHUTE : // MAV ID: 208
cmd . p1 = packet . param1 ; // action 0=disable, 1=enable, 2=release. See PARACHUTE_ACTION enum
break ;
2014-06-05 02:44:18 -03:00
case MAV_CMD_DO_INVERTED_FLIGHT : // MAV ID: 210
cmd . p1 = packet . param1 ; // normal=0 inverted=1
break ;
2022-09-20 04:37:48 -03:00
# if AP_GRIPPER_ENABLED
2014-09-17 04:21:12 -03:00
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 ;
2022-09-20 04:37:48 -03:00
# endif
2014-09-17 04:21:12 -03:00
2014-10-13 04:03:56 -03:00
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
2015-07-16 23:25:42 -03:00
break ;
2014-04-23 09:36:56 -03:00
2014-08-14 03:03:01 -03:00
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
2014-04-23 09:36:56 -03:00
cmd . content . altitude_wait . altitude = packet . param1 ;
cmd . content . altitude_wait . descent_rate = packet . param2 ;
cmd . content . altitude_wait . wiggle_time = packet . param3 ;
2014-10-13 04:03:56 -03:00
break ;
2016-01-01 05:41:52 -04:00
case MAV_CMD_NAV_VTOL_TAKEOFF :
break ;
case MAV_CMD_NAV_VTOL_LAND :
2020-07-21 17:15:40 -03:00
cmd . p1 = ( NAV_VTOL_LAND_OPTIONS ) packet . param1 ;
2016-01-01 05:41:52 -04:00
break ;
2016-04-22 05:22:12 -03:00
case MAV_CMD_DO_VTOL_TRANSITION :
cmd . content . do_vtol_transition . target_state = packet . param1 ;
break ;
2016-07-23 06:02:48 -03:00
2016-07-14 04:35:49 -03:00
case MAV_CMD_DO_SET_REVERSE :
cmd . p1 = packet . param1 ; // 0 = forward, 1 = reverse
break ;
2016-07-23 06:02:48 -03:00
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 ;
2023-09-28 15:51:37 -03:00
cmd . content . do_engine_control . allow_disarmed_start = ( ( ( uint32_t ) packet . param4 ) & ENGINE_CONTROL_OPTIONS_ALLOW_START_WHILE_DISARMED ) ! = 0 ;
2020-05-04 21:27:56 -03:00
break ;
2016-11-17 01:19:05 -04:00
2023-10-18 23:34:30 -03:00
# if AP_MISSION_NAV_PAYLOAD_PLACE_ENABLED
2016-11-17 01:19:05 -04:00
case MAV_CMD_NAV_PAYLOAD_PLACE :
cmd . p1 = packet . param1 * 100 ; // copy max-descend parameter (m->cm)
break ;
2023-10-18 23:34:30 -03:00
# endif
2016-11-17 01:19:05 -04:00
2017-08-02 02:25:57 -03:00
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 ;
2017-10-06 21:45:35 -03:00
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 ;
2020-02-24 18:43:20 -04:00
case MAV_CMD_DO_SET_RESUME_REPEAT_DIST :
cmd . p1 = packet . param1 ; // Resume repeat distance (m)
break ;
2020-07-02 20:04:27 -03:00
case MAV_CMD_DO_SPRAYER :
cmd . p1 = packet . param1 ; // action 0=disable, 1=enable
break ;
2021-02-25 21:08:22 -04:00
case MAV_CMD_DO_SEND_SCRIPT_MESSAGE :
cmd . p1 = packet . param1 ;
cmd . content . scripting . p1 = packet . param2 ;
cmd . content . scripting . p2 = packet . param3 ;
cmd . content . scripting . p3 = packet . param4 ;
break ;
2022-10-18 22:59:40 -03:00
# if AP_SCRIPTING_ENABLED
2021-10-25 00:59:11 -03:00
case MAV_CMD_NAV_SCRIPT_TIME :
cmd . content . nav_script_time . command = packet . param1 ;
cmd . content . nav_script_time . timeout_s = packet . param2 ;
2022-10-13 17:30:41 -03:00
cmd . content . nav_script_time . arg1 . set ( packet . param3 ) ;
cmd . content . nav_script_time . arg2 . set ( packet . param4 ) ;
2022-10-18 22:44:33 -03:00
cmd . content . nav_script_time . arg3 = int16_t ( packet . x ) ;
cmd . content . nav_script_time . arg4 = int16_t ( packet . y ) ;
2021-10-25 00:59:11 -03:00
break ;
2022-10-18 22:59:40 -03:00
# endif
2021-11-24 06:46:40 -04:00
2022-05-25 01:52:01 -03:00
case MAV_CMD_NAV_ATTITUDE_TIME :
cmd . content . nav_attitude_time . time_sec = constrain_float ( packet . param1 , 0 , UINT16_MAX ) ;
cmd . content . nav_attitude_time . roll_deg = ( fabsf ( packet . param2 ) < = 180 ) ? packet . param2 : 0 ;
cmd . content . nav_attitude_time . pitch_deg = ( fabsf ( packet . param3 ) < = 90 ) ? packet . param3 : 0 ;
cmd . content . nav_attitude_time . yaw_deg = ( ( packet . param4 > = - 180 ) & & ( packet . param4 < = 360 ) ) ? packet . param4 : 0 ;
cmd . content . nav_attitude_time . climb_rate = packet . x ;
break ;
2021-11-24 06:46:40 -04:00
case MAV_CMD_DO_PAUSE_CONTINUE :
cmd . p1 = packet . param1 ;
break ;
2022-06-02 09:23:47 -03:00
case MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW :
cmd . content . gimbal_manager_pitchyaw . pitch_angle_deg = packet . param1 ;
cmd . content . gimbal_manager_pitchyaw . yaw_angle_deg = packet . param2 ;
cmd . content . gimbal_manager_pitchyaw . pitch_rate_degs = packet . param3 ;
cmd . content . gimbal_manager_pitchyaw . yaw_rate_degs = packet . param4 ;
cmd . content . gimbal_manager_pitchyaw . flags = packet . x ;
cmd . content . gimbal_manager_pitchyaw . gimbal_id = packet . z ;
break ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_IMAGE_START_CAPTURE :
2023-09-07 07:05:40 -03:00
cmd . content . image_start_capture . instance = packet . param1 ;
2023-03-08 07:28:43 -04:00
cmd . content . image_start_capture . interval_s = packet . param2 ;
cmd . content . image_start_capture . total_num_images = packet . param3 ;
cmd . content . image_start_capture . start_seq_number = packet . param4 ;
break ;
2023-09-07 07:12:48 -03:00
case MAV_CMD_IMAGE_STOP_CAPTURE :
cmd . p1 = packet . param1 ;
break ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_SET_CAMERA_ZOOM :
cmd . content . set_camera_zoom . zoom_type = packet . param1 ;
cmd . content . set_camera_zoom . zoom_value = packet . param2 ;
break ;
case MAV_CMD_SET_CAMERA_FOCUS :
cmd . content . set_camera_focus . focus_type = packet . param1 ;
cmd . content . set_camera_focus . focus_value = packet . param2 ;
break ;
2024-02-08 04:30:08 -04:00
case MAV_CMD_SET_CAMERA_SOURCE :
cmd . content . set_camera_source . instance = packet . param1 ;
cmd . content . set_camera_source . primary_source = packet . param2 ;
cmd . content . set_camera_source . secondary_source = packet . param3 ;
break ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_VIDEO_START_CAPTURE :
cmd . content . video_start_capture . video_stream_id = packet . param1 ;
break ;
case MAV_CMD_VIDEO_STOP_CAPTURE :
cmd . content . video_stop_capture . video_stream_id = packet . param1 ;
break ;
2014-02-28 02:55:02 -04:00
default :
// unrecognised command
2015-12-21 01:02:23 -04:00
return MAV_MISSION_UNSUPPORTED ;
2014-02-28 02:55:02 -04:00
}
// copy location from mavlink to command
2019-01-01 20:07:03 -04:00
if ( stored_in_location ( cmd . id ) ) {
2015-08-27 02:22:05 -03:00
// sanity check location
2019-01-01 21:25:48 -04:00
if ( ! check_lat ( packet . x ) ) {
return MAV_MISSION_INVALID_PARAM5_X ;
}
if ( ! check_lng ( packet . y ) ) {
return MAV_MISSION_INVALID_PARAM6_Y ;
2015-08-27 02:22:05 -03:00
}
2018-04-26 02:46:46 -03:00
if ( isnan ( packet . z ) | | fabsf ( packet . z ) > = LOCATION_ALT_MAX_M ) {
2015-12-21 01:02:23 -04:00
return MAV_MISSION_INVALID_PARAM7 ;
2015-10-19 04:14:00 -03:00
}
2015-08-27 02:22:05 -03:00
2019-01-01 21:25:48 -04:00
cmd . content . location . lat = packet . x ;
cmd . content . location . lng = packet . y ;
2017-10-13 15:16:41 -03:00
cmd . content . location . alt = packet . z * 100.0f ; // convert packet's alt (m) to cmd alt (cm)
2014-03-03 01:36:37 -04:00
switch ( packet . frame ) {
case MAV_FRAME_MISSION :
case MAV_FRAME_GLOBAL :
2019-12-03 02:11:22 -04:00
case MAV_FRAME_GLOBAL_INT :
2019-01-01 20:07:03 -04:00
cmd . content . location . relative_alt = 0 ;
2014-03-03 01:36:37 -04:00
break ;
case MAV_FRAME_GLOBAL_RELATIVE_ALT :
2019-12-03 02:11:22 -04:00
case MAV_FRAME_GLOBAL_RELATIVE_ALT_INT :
2019-01-01 20:07:03 -04:00
cmd . content . location . relative_alt = 1 ;
2014-03-03 01:36:37 -04:00
break ;
2014-07-24 19:33:44 -03:00
# if AP_TERRAIN_AVAILABLE
case MAV_FRAME_GLOBAL_TERRAIN_ALT :
2019-12-03 02:11:22 -04:00
case MAV_FRAME_GLOBAL_TERRAIN_ALT_INT :
2014-07-24 19:33:44 -03:00
// we mark it as a relative altitude, as it doesn't have
// home alt added
2019-01-01 20:07:03 -04:00
cmd . content . location . relative_alt = 1 ;
2014-07-24 19:33:44 -03:00
// mark altitude as above terrain, not above home
2019-01-01 20:07:03 -04:00
cmd . content . location . terrain_alt = 1 ;
2014-07-24 19:33:44 -03:00
break ;
# endif
2014-03-03 01:36:37 -04:00
default :
2015-12-21 01:02:23 -04:00
return MAV_MISSION_UNSUPPORTED_FRAME ;
2014-02-28 02:55:02 -04:00
}
}
2016-05-12 13:59:11 -03:00
// if we got this far then it must have been successful
2015-12-21 01:02:23 -04:00
return MAV_MISSION_ACCEPTED ;
2014-02-28 02:55:02 -04:00
}
2019-01-30 19:27:22 -04:00
MAV_MISSION_RESULT AP_Mission : : convert_MISSION_ITEM_to_MISSION_ITEM_INT ( const mavlink_mission_item_t & packet ,
2020-05-04 21:27:56 -03:00
mavlink_mission_item_int_t & mav_cmd )
2016-02-28 02:47:00 -04:00
{
2019-01-30 19:27:22 -04:00
// TODO: rename mav_cmd to mission_item_int
// TODO: rename packet to mission_item
2016-02-28 02:47:00 -04:00
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 ;
2019-01-30 19:27:22 -04:00
mav_cmd . mission_type = packet . mission_type ;
2020-05-04 21:27:56 -03:00
2016-02-28 02:47:00 -04:00
/*
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 .
*/
2023-01-15 20:03:55 -04:00
if ( ! cmd_has_location ( packet . command ) ) {
2016-02-28 02:47:00 -04:00
mav_cmd . x = packet . x ;
mav_cmd . y = packet . y ;
2023-01-15 20:03:55 -04:00
} else {
//these commands use x and y as lat/lon. We need to
2016-02-28 02:47:00 -04:00
// multiply by 1e7 to convert to int32_t
2016-06-06 17:08:37 -03:00
if ( ! check_lat ( packet . x ) ) {
2016-06-06 05:55:22 -03:00
return MAV_MISSION_INVALID_PARAM5_X ;
}
2016-06-06 17:08:37 -03:00
if ( ! check_lng ( packet . y ) ) {
2016-06-06 05:55:22 -03:00
return MAV_MISSION_INVALID_PARAM6_Y ;
}
2016-06-06 17:08:37 -03:00
mav_cmd . x = packet . x * 1.0e7 f ;
mav_cmd . y = packet . y * 1.0e7 f ;
2016-02-28 02:47:00 -04:00
}
2019-01-30 19:27:22 -04:00
return MAV_MISSION_ACCEPTED ;
2016-02-28 02:47:00 -04:00
}
2019-01-30 19:27:22 -04:00
MAV_MISSION_RESULT AP_Mission : : convert_MISSION_ITEM_INT_to_MISSION_ITEM ( const mavlink_mission_item_int_t & item_int ,
2020-05-04 21:27:56 -03:00
mavlink_mission_item_t & item )
2016-02-28 02:47:00 -04:00
{
2019-01-30 19:27:22 -04:00
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 ;
2023-01-15 20:03:55 -04:00
if ( ! cmd_has_location ( item_int . command ) ) {
2019-01-30 19:27:22 -04:00
item . x = item_int . x ;
item . y = item_int . y ;
2016-02-28 02:47:00 -04:00
2023-01-15 20:03:55 -04:00
} else {
// These commands use x and y as lat/lon. We need to
2019-01-30 19:27:22 -04:00
// multiply by 1e-7 to convert to float
item . x = item_int . x * 1.0e-7 f ;
item . y = item_int . y * 1.0e-7 f ;
if ( ! check_lat ( item . x ) ) {
return MAV_MISSION_INVALID_PARAM5_X ;
}
if ( ! check_lng ( item . y ) ) {
return MAV_MISSION_INVALID_PARAM6_Y ;
}
2016-02-28 02:47:00 -04:00
}
2019-01-30 19:27:22 -04:00
return MAV_MISSION_ACCEPTED ;
2016-02-28 02:47:00 -04:00
}
2019-01-30 19:27:22 -04:00
// mission_cmd_to_mavlink_int - converts an AP_Mission::Mission_Command object to a mavlink message which can be sent to the GCS
2014-02-28 02:55:02 -04:00
// return true on success, false on failure
2022-09-24 19:12:12 -03:00
// NOTE: callers to this method current fill parts of "packet" in before calling this method, so do NOT attempt to zero the entire packet in here
2016-02-28 02:47:00 -04:00
bool AP_Mission : : mission_cmd_to_mavlink_int ( const AP_Mission : : Mission_Command & cmd , mavlink_mission_item_int_t & packet )
2014-02-28 02:55:02 -04:00
{
2025-01-11 09:56:59 -04:00
// Make sure return packed is zeroed
// Strictly params 1 to 4 and alt should be defaulted to NaN
// x and y should be defaulted to INT32_MAX
packet = {
seq : cmd . index ,
command : cmd . id ,
autocontinue : 1 ,
} ;
2014-02-28 02:55:02 -04:00
// command specific conversions from mission command to mavlink packet
switch ( cmd . id ) {
2016-04-22 04:58:42 -03:00
case 0 :
// this is reserved for 16 bit command IDs
return false ;
2020-05-04 21:27:56 -03:00
2014-03-17 02:14:45 -03:00
case MAV_CMD_NAV_WAYPOINT : // MAV ID: 16
2014-09-03 08:15:41 -03:00
# if APM_BUILD_TYPE(APM_BUILD_ArduPlane)
// acceptance radius in meters
2016-12-18 10:39:44 -04:00
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
2014-09-03 08:15:41 -03:00
# else
// delay at waypoint in seconds
packet . param1 = cmd . p1 ;
# endif
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_NAV_LOITER_UNLIM : // MAV ID: 17
2016-12-05 21:09:58 -04:00
packet . param3 = ( float ) cmd . p1 ;
2019-01-01 20:07:03 -04:00
if ( cmd . content . location . loiter_ccw ) {
2016-12-05 21:09:58 -04:00
packet . param3 * = - 1 ;
2014-03-18 04:32:28 -03:00
}
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_NAV_LOITER_TURNS : // MAV ID: 18
2014-06-11 04:17:19 -03:00
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
2019-01-01 20:07:03 -04:00
if ( cmd . content . location . loiter_ccw ) {
2014-06-11 04:17:19 -03:00
packet . param3 = - packet . param3 ;
2014-03-18 04:32:28 -03:00
}
2022-05-11 00:40:48 -03:00
if ( cmd . type_specific_bits & ( 1U < < 0 ) ) {
packet . param3 * = 10 ;
}
2024-01-12 00:14:19 -04:00
if ( cmd . type_specific_bits & ( 1U < < 1 ) ) {
packet . param1 / = 256.0 ;
}
2019-01-01 20:07:03 -04:00
packet . param4 = cmd . content . location . loiter_xtrack ; // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_NAV_LOITER_TIME : // MAV ID: 19
packet . param1 = cmd . p1 ; // loiter time in seconds
2019-01-01 20:07:03 -04:00
if ( cmd . content . location . loiter_ccw ) {
2014-03-18 04:32:28 -03:00
packet . param3 = - 1 ;
2016-03-02 11:16:27 -04:00
} else {
2014-03-18 04:32:28 -03:00
packet . param3 = 1 ;
}
2019-01-01 20:07:03 -04:00
packet . param4 = cmd . content . location . loiter_xtrack ; // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_NAV_RETURN_TO_LAUNCH : // MAV ID: 20
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_NAV_LAND : // MAV ID: 21
2015-08-28 22:33:48 -03:00
packet . param1 = cmd . p1 ; // abort target altitude(m) (plane only)
2019-01-01 20:07:03 -04:00
packet . param4 = cmd . content . location . loiter_ccw ? - 1 : 1 ; // yaw direction, (plane deepstall only)
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_NAV_TAKEOFF : // MAV ID: 22
2014-03-03 01:36:37 -04:00
packet . param1 = cmd . p1 ; // minimum pitch (plane only)
2014-02-28 02:55:02 -04:00
break ;
2014-10-21 17:45:24 -03:00
case MAV_CMD_NAV_CONTINUE_AND_CHANGE_ALT : // MAV ID: 30
2015-08-18 23:45:48 -03:00
packet . param1 = cmd . p1 ; // Climb/Descend
2020-05-04 21:27:56 -03:00
// 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.
2014-10-21 17:45:24 -03:00
break ;
2015-03-13 18:47:37 -03:00
case MAV_CMD_NAV_LOITER_TO_ALT : // MAV ID: 31
2016-03-02 11:16:27 -04:00
packet . param2 = cmd . p1 ; // loiter radius(m)
2019-01-01 20:07:03 -04:00
if ( cmd . content . location . loiter_ccw ) {
2015-03-13 18:47:37 -03:00
packet . param2 = - packet . param2 ;
}
2019-01-01 20:07:03 -04:00
packet . param4 = cmd . content . location . loiter_xtrack ; // 0 to xtrack from center of waypoint, 1 to xtrack from tangent exit location
2015-03-13 18:47:37 -03:00
break ;
2014-03-22 00:45:26 -03:00
case MAV_CMD_NAV_SPLINE_WAYPOINT : // MAV ID: 82
packet . param1 = cmd . p1 ; // delay at waypoint in seconds
break ;
2014-10-13 04:03:56 -03:00
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
2014-06-02 06:02:04 -03:00
break ;
2019-08-14 14:01:42 -03:00
case MAV_CMD_NAV_DELAY : // MAV ID: 93
2016-04-18 04:12:40 -03:00
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 ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_CONDITION_DELAY : // MAV ID: 112
packet . param1 = cmd . content . delay . seconds ; // delay in seconds
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_CONDITION_DISTANCE : // MAV ID: 114
packet . param1 = cmd . content . distance . meters ; // distance in meters from next waypoint
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_DO_JUMP : // MAV ID: 177
2023-02-21 21:24:05 -04:00
case MAV_CMD_DO_JUMP_TAG : // MAV ID: 601
packet . param1 = cmd . content . jump . target ; // jump-to command/tag number
2014-02-28 02:55:02 -04:00
packet . param2 = cmd . content . jump . num_times ; // repeat count
break ;
2023-02-21 21:24:05 -04:00
case MAV_CMD_JUMP_TAG : // MAV ID: 600
packet . param1 = cmd . content . jump . target ; // jump-to tag number
break ;
2014-03-17 02:14:45 -03:00
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
2020-01-04 07:46:46 -04:00
packet . param3 = cmd . content . speed . throttle_pct ; // throttle as a percentage from 1 ~ 100%
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_DO_REPEAT_RELAY : // MAV ID: 182
2014-03-18 23:13:04 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_DO_SET_SERVO : // MAV ID: 183
packet . param1 = cmd . content . servo . channel ; // channel
packet . param2 = cmd . content . servo . pwm ; // PWM
2014-02-28 02:55:02 -04:00
break ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_DO_REPEAT_SERVO : // MAV ID: 184
2014-03-18 23:13:04 -03:00
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
2014-02-28 02:55:02 -04:00
break ;
2024-03-02 21:00:47 -04:00
case MAV_CMD_DO_RETURN_PATH_START : // MAV ID: 188
2014-10-17 14:34:04 -03:00
case MAV_CMD_DO_LAND_START : // MAV ID: 189
break ;
2018-12-05 19:28:11 -04:00
case MAV_CMD_DO_GO_AROUND : // MAV ID: 191
break ;
2014-03-17 02:14:45 -03:00
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)
2014-02-28 02:55:02 -04:00
break ;
2015-03-28 22:29:21 -03:00
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 ;
2014-03-17 02:14:45 -03:00
case MAV_CMD_DO_DIGICAM_CONTROL : // MAV ID: 203
2015-03-28 22:29:21 -03:00
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 ;
2014-02-28 02:55:02 -04:00
break ;
2015-03-21 08:56:50 -03:00
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 ;
2024-05-15 22:27:05 -03:00
packet . z = MAV_MOUNT_MODE_MAVLINK_TARGETING ;
2015-03-21 08:56:50 -03:00
break ;
2014-03-17 02:14:45 -03:00
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
2020-03-23 20:55:51 -03:00
packet . param3 = cmd . content . cam_trigg_dist . trigger ; // when enabled, camera triggers once immediately
2014-02-28 02:55:02 -04:00
break ;
2014-10-25 18:19:21 -03:00
case MAV_CMD_DO_FENCE_ENABLE : // MAV ID: 207
2024-01-13 14:03:19 -04:00
packet . param1 = cmd . p1 ; // action 0=disable, 1=enable, 2=disable floor, 3=enable except floor
2014-10-25 18:19:21 -03:00
break ;
2014-04-03 05:49:04 -03:00
case MAV_CMD_DO_PARACHUTE : // MAV ID: 208
packet . param1 = cmd . p1 ; // action 0=disable, 1=enable, 2=release. See PARACHUTE_ACTION enum
break ;
2020-07-02 20:04:27 -03:00
case MAV_CMD_DO_SPRAYER :
packet . param1 = cmd . p1 ; // action 0=disable, 1=enable
break ;
2021-02-17 21:01:41 -04:00
case MAV_CMD_DO_AUX_FUNCTION :
packet . param1 = cmd . content . auxfunction . function ;
packet . param2 = cmd . content . auxfunction . switchpos ;
break ;
2014-06-05 02:44:18 -03:00
case MAV_CMD_DO_INVERTED_FLIGHT : // MAV ID: 210
packet . param1 = cmd . p1 ; // normal=0 inverted=1
break ;
2022-09-20 04:37:48 -03:00
# if AP_GRIPPER_ENABLED
2014-09-17 04:21:12 -03:00
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 ;
2022-09-20 04:37:48 -03:00
# endif
2014-09-17 04:21:12 -03:00
2014-10-13 04:03:56 -03:00
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
2015-07-16 23:25:42 -03:00
break ;
2014-04-23 09:36:56 -03:00
2014-08-14 03:03:01 -03:00
case MAV_CMD_DO_AUTOTUNE_ENABLE :
packet . param1 = cmd . p1 ; // disable=0 enable=1
break ;
2016-07-14 04:35:49 -03:00
case MAV_CMD_DO_SET_REVERSE :
packet . param1 = cmd . p1 ; // 0 = forward, 1 = reverse
break ;
2014-08-14 03:03:01 -03:00
case MAV_CMD_NAV_ALTITUDE_WAIT : // MAV ID: 83
2014-04-23 09:36:56 -03:00
packet . param1 = cmd . content . altitude_wait . altitude ;
packet . param2 = cmd . content . altitude_wait . descent_rate ;
packet . param3 = cmd . content . altitude_wait . wiggle_time ;
2014-10-13 04:03:56 -03:00
break ;
2016-01-01 05:41:52 -04:00
case MAV_CMD_NAV_VTOL_TAKEOFF :
break ;
case MAV_CMD_NAV_VTOL_LAND :
2020-07-21 17:15:40 -03:00
packet . param1 = cmd . p1 ;
2016-01-01 05:41:52 -04:00
break ;
2016-04-22 05:22:12 -03:00
case MAV_CMD_DO_VTOL_TRANSITION :
packet . param1 = cmd . content . do_vtol_transition . target_state ;
break ;
2016-07-23 06:02:48 -03:00
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 ;
2023-09-28 15:51:37 -03:00
if ( cmd . content . do_engine_control . allow_disarmed_start ) {
packet . param4 = ENGINE_CONTROL_OPTIONS_ALLOW_START_WHILE_DISARMED ;
}
2020-05-04 21:27:56 -03:00
break ;
2016-11-17 01:19:05 -04:00
2023-10-18 23:34:30 -03:00
# if AP_MISSION_NAV_PAYLOAD_PLACE_ENABLED
2016-11-17 01:19:05 -04:00
case MAV_CMD_NAV_PAYLOAD_PLACE :
2024-05-01 17:12:50 -03:00
packet . param1 = cmd . p1 * 0.01f ; // copy max-descend parameter (cm->m)
2016-11-17 01:19:05 -04:00
break ;
2023-10-18 23:34:30 -03:00
# endif
2016-11-17 01:19:05 -04:00
2017-08-02 02:25:57 -03:00
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 ;
2017-10-06 21:45:35 -03:00
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 ;
2020-02-24 18:43:20 -04:00
case MAV_CMD_DO_SET_RESUME_REPEAT_DIST :
packet . param1 = cmd . p1 ; // Resume repeat distance (m)
break ;
2021-02-25 21:08:22 -04:00
case MAV_CMD_DO_SEND_SCRIPT_MESSAGE :
packet . param1 = cmd . p1 ;
packet . param2 = cmd . content . scripting . p1 ;
packet . param3 = cmd . content . scripting . p2 ;
packet . param4 = cmd . content . scripting . p3 ;
break ;
2022-10-18 22:59:40 -03:00
# if AP_SCRIPTING_ENABLED
2021-10-25 00:59:11 -03:00
case MAV_CMD_NAV_SCRIPT_TIME :
packet . param1 = cmd . content . nav_script_time . command ;
packet . param2 = cmd . content . nav_script_time . timeout_s ;
2022-10-13 17:30:41 -03:00
packet . param3 = cmd . content . nav_script_time . arg1 . get ( ) ;
packet . param4 = cmd . content . nav_script_time . arg2 . get ( ) ;
2022-10-18 22:44:33 -03:00
packet . x = cmd . content . nav_script_time . arg3 ;
packet . y = cmd . content . nav_script_time . arg4 ;
2021-10-25 00:59:11 -03:00
break ;
2022-10-18 22:59:40 -03:00
# endif
2021-11-24 06:46:40 -04:00
2022-05-25 01:52:01 -03:00
case MAV_CMD_NAV_ATTITUDE_TIME :
packet . param1 = cmd . content . nav_attitude_time . time_sec ;
packet . param2 = cmd . content . nav_attitude_time . roll_deg ;
packet . param3 = cmd . content . nav_attitude_time . pitch_deg ;
packet . param4 = cmd . content . nav_attitude_time . yaw_deg ;
packet . x = cmd . content . nav_attitude_time . climb_rate ;
break ;
2021-11-24 06:46:40 -04:00
case MAV_CMD_DO_PAUSE_CONTINUE :
packet . param1 = cmd . p1 ;
break ;
2022-06-02 09:23:47 -03:00
case MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW :
packet . param1 = cmd . content . gimbal_manager_pitchyaw . pitch_angle_deg ;
packet . param2 = cmd . content . gimbal_manager_pitchyaw . yaw_angle_deg ;
packet . param3 = cmd . content . gimbal_manager_pitchyaw . pitch_rate_degs ;
packet . param4 = cmd . content . gimbal_manager_pitchyaw . yaw_rate_degs ;
packet . x = cmd . content . gimbal_manager_pitchyaw . flags ;
packet . z = cmd . content . gimbal_manager_pitchyaw . gimbal_id ;
break ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_IMAGE_START_CAPTURE :
2023-09-07 07:05:40 -03:00
packet . param1 = cmd . content . image_start_capture . instance ;
2023-03-08 07:28:43 -04:00
packet . param2 = cmd . content . image_start_capture . interval_s ;
packet . param3 = cmd . content . image_start_capture . total_num_images ;
packet . param4 = cmd . content . image_start_capture . start_seq_number ;
break ;
2023-09-07 07:12:48 -03:00
case MAV_CMD_IMAGE_STOP_CAPTURE :
packet . param1 = cmd . p1 ;
break ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_SET_CAMERA_ZOOM :
packet . param1 = cmd . content . set_camera_zoom . zoom_type ;
packet . param2 = cmd . content . set_camera_zoom . zoom_value ;
break ;
case MAV_CMD_SET_CAMERA_FOCUS :
packet . param1 = cmd . content . set_camera_focus . focus_type ;
packet . param2 = cmd . content . set_camera_focus . focus_value ;
break ;
2024-02-08 04:30:08 -04:00
case MAV_CMD_SET_CAMERA_SOURCE :
packet . param1 = cmd . content . set_camera_source . instance ;
packet . param2 = cmd . content . set_camera_source . primary_source ;
packet . param3 = cmd . content . set_camera_source . secondary_source ;
break ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_VIDEO_START_CAPTURE :
packet . param1 = cmd . content . video_start_capture . video_stream_id ;
break ;
case MAV_CMD_VIDEO_STOP_CAPTURE :
packet . param1 = cmd . content . video_stop_capture . video_stream_id ;
break ;
2014-02-28 02:55:02 -04:00
default :
// unrecognised command
return false ;
}
// copy location from mavlink to command
2019-01-01 20:07:03 -04:00
if ( stored_in_location ( cmd . id ) ) {
2016-02-28 02:47:00 -04:00
packet . x = cmd . content . location . lat ;
packet . y = cmd . content . location . lng ;
2019-01-01 21:25:48 -04:00
2022-03-11 13:36:41 -04:00
packet . z = cmd . content . location . alt * 0.01f ; // cmd alt in cm to m
2019-01-01 20:07:03 -04:00
if ( cmd . content . location . relative_alt ) {
2014-02-28 02:55:02 -04:00
packet . frame = MAV_FRAME_GLOBAL_RELATIVE_ALT ;
2020-05-04 21:27:56 -03:00
} else {
2014-02-28 02:55:02 -04:00
packet . frame = MAV_FRAME_GLOBAL ;
}
2014-07-24 19:33:44 -03:00
# if AP_TERRAIN_AVAILABLE
2019-01-01 20:07:03 -04:00
if ( cmd . content . location . terrain_alt ) {
2014-07-24 19:33:44 -03:00
// this is a above-terrain altitude
2019-01-01 20:07:03 -04:00
if ( ! cmd . content . location . relative_alt ) {
2014-07-24 19:33:44 -03:00
// 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
2019-01-01 20:07:03 -04:00
if ( cmd . content . location . terrain_alt ) {
2014-07-24 19:33:44 -03:00
return false ;
}
# endif
2014-02-28 02:55:02 -04:00
}
2014-03-17 02:14:45 -03:00
// if we got this far then it must have been successful
2014-02-28 02:55:02 -04:00
return true ;
}
2014-02-23 03:54:53 -04:00
///
/// private methods
///
2014-02-22 20:14:44 -04:00
2014-02-23 03:54:53 -04:00
/// complete - mission is marked complete and clean-up performed including calling the mission_complete_fn
void AP_Mission : : complete ( )
2014-02-22 20:14:44 -04:00
{
2014-02-23 03:54:53 -04:00
// flag mission as complete
_flags . state = MISSION_COMPLETE ;
2020-02-17 19:44:53 -04:00
_flags . in_landing_sequence = false ;
2024-03-02 21:00:47 -04:00
_flags . in_return_path = false ;
2014-02-23 03:54:53 -04:00
// callback to main program's mission complete function
_mission_complete_fn ( ) ;
2014-02-22 20:14:44 -04:00
}
2014-02-24 01:50:24 -04:00
/// 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)
2018-08-22 19:38:29 -03:00
bool AP_Mission : : advance_current_nav_cmd ( uint16_t starting_index )
2014-02-22 20:14:44 -04:00
{
2014-02-24 01:50:24 -04:00
// 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 ;
2014-02-28 09:37:02 -04:00
_flags . do_cmd_all_done = false ;
2014-02-24 01:50:24 -04:00
// get starting point for search
2018-08-24 23:05:58 -03:00
uint16_t cmd_index = starting_index > 0 ? starting_index - 1 : _nav_cmd . index ;
2014-02-24 01:50:24 -04:00
if ( cmd_index = = AP_MISSION_CMD_INDEX_NONE ) {
2014-03-07 04:01:05 -04:00
// start from beginning of the mission command list
2014-02-27 09:02:12 -04:00
cmd_index = AP_MISSION_FIRST_REAL_COMMAND ;
2020-05-04 21:27:56 -03:00
} else {
2014-02-24 01:50:24 -04:00
// start from one position past the current nav command
cmd_index + + ;
}
2015-06-01 04:16:04 -03:00
// avoid endless loops
uint8_t max_loops = 255 ;
2014-02-24 01:50:24 -04:00
// search until we find next nav command or reach end of command list
2021-09-20 19:36:48 -03:00
while ( ! _flags . nav_cmd_loaded & & max_loops - - > 0 ) {
2014-02-24 01:50:24 -04:00
// get next command
2018-08-24 23:05:58 -03:00
Mission_Command cmd ;
2014-02-24 01:50:24 -04:00
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
2016-05-13 20:06:25 -03:00
_prev_nav_cmd_id = _nav_cmd . id ;
2014-02-24 01:50:24 -04:00
_prev_nav_cmd_index = _nav_cmd . index ;
2015-08-23 22:06:31 -03:00
// save separate previous nav command index if it contains lat,long,alt
2015-08-27 02:42:42 -03:00
if ( ! ( cmd . content . location . lat = = 0 & & cmd . content . location . lng = = 0 ) ) {
2015-08-23 22:06:31 -03:00
_prev_nav_cmd_wp_index = _nav_cmd . index ;
}
2014-02-24 01:50:24 -04:00
// set current navigation command and start it
_nav_cmd = cmd ;
2018-10-22 23:56:46 -03:00
if ( start_command ( _nav_cmd ) ) {
2018-08-22 17:56:01 -03:00
_flags . nav_cmd_loaded = true ;
2023-02-21 21:24:05 -04:00
if ( _jump_tag . age > 0 & & _jump_tag . age < UINT16_MAX ) {
// we're tracking a tag so increase it's age on every new NAV item
_jump_tag . age + + ;
}
2018-08-22 17:56:01 -03:00
}
2020-02-24 18:43:20 -04:00
// save a loaded wp index in history array for when _repeat_dist is set via MAV_CMD_DO_SET_RESUME_REPEAT_DIST
2021-03-05 21:04:25 -04:00
// and prevent history being re-written until vehicle returns to interrupted position
2020-05-04 21:27:56 -03:00
if ( _repeat_dist > 0 & & ! _flags . resuming_mission & & _nav_cmd . index ! = AP_MISSION_CMD_INDEX_NONE & & ! ( _nav_cmd . content . location . lat = = 0 & & _nav_cmd . content . location . lng = = 0 ) ) {
2020-02-24 18:43:20 -04:00
// update mission history. last index position is always the most recent wp loaded.
for ( uint8_t i = 0 ; i < AP_MISSION_MAX_WP_HISTORY - 1 ; i + + ) {
_wp_index_history [ i ] = _wp_index_history [ i + 1 ] ;
}
_wp_index_history [ AP_MISSION_MAX_WP_HISTORY - 1 ] = _nav_cmd . index ;
}
2021-03-05 21:04:25 -04:00
// check if the vehicle is resuming and has returned to where it was interrupted
2020-02-24 18:43:20 -04:00
if ( _flags . resuming_mission & & _nav_cmd . index = = _wp_index_history [ AP_MISSION_MAX_WP_HISTORY - 1 ] ) {
// vehicle has resumed previous position
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Mission: Returned to interrupted WP " ) ;
2020-02-24 18:43:20 -04:00
_flags . resuming_mission = false ;
}
2020-05-04 21:27:56 -03:00
} else {
2014-02-24 01:50:24 -04:00
// set current do command and start it (if not already set)
if ( ! _flags . do_cmd_loaded ) {
_do_cmd = cmd ;
_flags . do_cmd_loaded = true ;
2018-10-22 23:56:46 -03:00
start_command ( _do_cmd ) ;
2014-02-24 01:50:24 -04:00
}
}
// move onto next command
cmd_index = cmd . index + 1 ;
}
2021-09-20 19:36:48 -03:00
if ( max_loops = = 0 ) {
// infinite loop. This can happen if there's a loop involving
// only nav commands (no DO commands) which won't start()
return false ;
}
2015-07-17 00:13:28 -03:00
// 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 ;
}
2014-02-24 01:50:24 -04:00
// 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 ( )
{
2014-02-28 09:37:02 -04:00
// 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 ) {
2014-02-24 01:50:24 -04:00
return ;
}
// get starting point for search
2018-08-24 23:05:58 -03:00
uint16_t cmd_index = _do_cmd . index ;
2014-02-24 01:50:24 -04:00
if ( cmd_index = = AP_MISSION_CMD_INDEX_NONE ) {
2014-02-27 09:02:12 -04:00
cmd_index = AP_MISSION_FIRST_REAL_COMMAND ;
2020-05-04 21:27:56 -03:00
} else {
2014-02-24 01:50:24 -04:00
// start from one position past the current do command
2018-08-24 23:05:58 -03:00
cmd_index = _do_cmd . index + 1 ;
2014-02-24 01:50:24 -04:00
}
// find next do command
2018-08-24 23:05:58 -03:00
Mission_Command cmd ;
2018-08-27 23:09:29 -03:00
if ( ! get_next_do_cmd ( cmd_index , cmd ) ) {
2014-02-28 09:37:02 -04:00
// set flag to stop unnecessarily searching for do commands
_flags . do_cmd_all_done = true ;
2018-08-27 23:09:29 -03:00
return ;
2014-02-24 01:50:24 -04:00
}
2018-08-27 23:09:29 -03:00
// set current do command and start it
_do_cmd = cmd ;
_flags . do_cmd_loaded = true ;
start_command ( _do_cmd ) ;
2014-02-22 20:14:44 -04:00
}
2014-02-23 03:54:53 -04:00
/// 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
2020-02-17 19:44:53 -04:00
bool AP_Mission : : get_next_cmd ( uint16_t start_index , Mission_Command & cmd , bool increment_jump_num_times_if_found , bool send_gcs_msg )
2014-02-22 20:14:44 -04:00
{
2014-02-28 08:49:37 -04:00
uint16_t cmd_index = start_index ;
2014-02-23 03:54:53 -04:00
Mission_Command temp_cmd ;
2014-02-28 08:49:37 -04:00
uint16_t jump_index = AP_MISSION_CMD_INDEX_NONE ;
2014-02-23 03:54:53 -04:00
// search until the end of the mission command list
2015-01-22 22:13:54 -04:00
uint8_t max_loops = 64 ;
2020-05-04 21:27:56 -03:00
while ( cmd_index < ( unsigned ) _cmd_total ) {
2014-02-23 03:54:53 -04:00
// load the next command
2014-03-10 23:23:44 -03:00
if ( ! read_cmd_from_storage ( cmd_index , temp_cmd ) ) {
// this should never happen because of check above but just in case
return false ;
}
2014-02-23 03:54:53 -04:00
2023-02-21 21:24:05 -04:00
// check for do-jump-tag command and convert target tag to do-jump target index and do-jump to it
if ( temp_cmd . id = = MAV_CMD_DO_JUMP_TAG ) {
// convert tmp_cmd target from a target tag to a target index
temp_cmd . content . jump . target = get_index_of_jump_tag ( temp_cmd . content . jump . target ) ;
temp_cmd . id = MAV_CMD_DO_JUMP ;
}
2014-02-23 03:54:53 -04:00
// check for do-jump command
if ( temp_cmd . id = = MAV_CMD_DO_JUMP ) {
2014-02-28 08:23:38 -04:00
2015-01-22 22:13:54 -04:00
if ( max_loops - - = = 0 ) {
return false ;
}
2014-02-28 08:23:38 -04:00
// check for invalid target
2015-04-21 08:37:09 -03:00
if ( ( temp_cmd . content . jump . target > = ( unsigned ) _cmd_total ) | | ( temp_cmd . content . jump . target = = 0 ) ) {
2014-02-28 08:23:38 -04:00
// To-Do: log an error?
return false ;
}
2014-02-23 03:54:53 -04:00
// 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 ;
}
2014-02-28 08:23:38 -04:00
// record this command so we can check for endless loops
if ( jump_index = = AP_MISSION_CMD_INDEX_NONE ) {
jump_index = cmd_index ;
}
2024-05-07 00:58:07 -03:00
// get number of times jump command has already been run
if ( temp_cmd . content . jump . num_times = = AP_MISSION_JUMP_REPEAT_FOREVER | |
get_jump_times_run ( temp_cmd ) < temp_cmd . content . jump . num_times ) {
// update the record of the number of times run
if ( increment_jump_num_times_if_found & & ! _flags . resuming_mission ) {
increment_jump_times_run ( temp_cmd , send_gcs_msg ) ;
}
2014-02-23 03:54:53 -04:00
// continue searching from jump target
cmd_index = temp_cmd . content . jump . target ;
2020-05-04 21:27:56 -03:00
} else {
2024-05-07 00:58:07 -03:00
// jump has been run specified number of times so move search to next command in mission
cmd_index + + ;
2014-02-23 03:54:53 -04:00
}
2020-05-04 21:27:56 -03:00
} else {
2014-02-23 03:54:53 -04:00
// this is a non-jump command so return it
cmd = temp_cmd ;
2014-02-22 20:14:44 -04:00
return true ;
}
}
2014-02-23 03:54:53 -04:00
// if we got this far we did not find a navigation command
2014-02-22 20:14:44 -04:00
return false ;
}
2014-02-23 03:54:53 -04:00
/// 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)
2014-02-28 08:49:37 -04:00
bool AP_Mission : : get_next_do_cmd ( uint16_t start_index , Mission_Command & cmd )
2014-02-22 20:14:44 -04:00
{
2014-02-23 03:54:53 -04:00
Mission_Command temp_cmd ;
2014-02-22 20:14:44 -04:00
2014-02-23 03:54:53 -04:00
// check we have not passed the end of the mission list
2014-03-23 22:52:27 -03:00
if ( start_index > = ( unsigned ) _cmd_total ) {
2014-02-23 03:54:53 -04:00
return false ;
}
2014-02-22 20:14:44 -04:00
2014-02-23 03:54:53 -04:00
// get next command
if ( ! get_next_cmd ( start_index , temp_cmd , false ) ) {
// no more commands so return failure
return false ;
2020-05-04 21:27:56 -03:00
} else if ( is_nav_cmd ( temp_cmd ) ) {
2014-02-23 03:54:53 -04:00
// if it's a "navigation" command then return false because we do not progress past nav commands
return false ;
2020-05-04 21:27:56 -03:00
} else {
2014-02-23 03:54:53 -04:00
// this must be a "do" or "conditional" and is not a do-jump command so return it
cmd = temp_cmd ;
return true ;
2014-02-22 20:14:44 -04:00
}
}
2014-02-23 03:54:53 -04:00
///
/// jump handling methods
///
2014-02-22 20:14:44 -04:00
2023-02-21 21:24:05 -04:00
// Set the mission index to the first JUMP_TAG with this tag.
// Returns true on success, else false if no appropriate JUMP_TAG match can be found or if setting the index failed
bool AP_Mission : : jump_to_tag ( const uint16_t tag )
{
const uint16_t index = get_index_of_jump_tag ( tag ) ;
if ( index = = 0 ) {
return false ;
}
return set_current_cmd ( index ) ;
}
// find the first JUMP_TAG with this tag and return its index.
// Returns 0 if no appropriate JUMP_TAG match can be found.
uint16_t AP_Mission : : get_index_of_jump_tag ( const uint16_t tag ) const
{
const auto count = num_commands ( ) ;
for ( uint16_t i = 1 ; i < count ; i + + ) {
2023-02-21 16:52:23 -04:00
if ( get_command_id ( i ) ! = uint16_t ( MAV_CMD_JUMP_TAG ) ) {
continue ;
}
2023-02-21 21:24:05 -04:00
Mission_Command tmp ;
if ( ! read_cmd_from_storage ( i , tmp ) ) {
continue ;
}
if ( tmp . id = = MAV_CMD_JUMP_TAG & & tmp . content . jump . target = = tag ) {
return i ;
}
}
return 0 ;
}
2023-02-21 21:47:01 -04:00
# if AP_SCRIPTING_ENABLED
bool AP_Mission : : get_last_jump_tag ( uint16_t & tag , uint16_t & age ) const
{
if ( _jump_tag . age = = 0 ) {
return false ;
}
tag = _jump_tag . tag ;
age = _jump_tag . age ;
return true ;
}
# endif
2014-02-23 03:54:53 -04:00
// init_jump_tracking - initialise jump_tracking variables
void AP_Mission : : init_jump_tracking ( )
{
2020-05-04 21:27:56 -03:00
for ( uint8_t i = 0 ; i < AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS ; i + + ) {
2014-02-23 03:54:53 -04:00
_jump_tracking [ i ] . index = AP_MISSION_CMD_INDEX_NONE ;
_jump_tracking [ i ] . num_times_run = 0 ;
2014-02-22 20:14:44 -04:00
}
}
2014-02-23 03:54:53 -04:00
/// get_jump_times_run - returns number of times the jump command has been run
2014-02-28 08:23:38 -04:00
int16_t AP_Mission : : get_jump_times_run ( const Mission_Command & cmd )
2014-02-22 20:14:44 -04:00
{
2016-05-12 13:59:11 -03:00
// exit immediately if cmd is not a do-jump command or target is invalid
2015-04-21 08:37:09 -03:00
if ( ( cmd . id ! = MAV_CMD_DO_JUMP ) | | ( cmd . content . jump . target > = ( unsigned ) _cmd_total ) | | ( cmd . content . jump . target = = 0 ) ) {
2014-02-23 03:54:53 -04:00
// To-Do: log an error?
return AP_MISSION_JUMP_TIMES_MAX ;
}
2014-02-22 20:14:44 -04:00
2014-02-23 03:54:53 -04:00
// 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 ;
2020-05-04 21:27:56 -03:00
} else if ( _jump_tracking [ i ] . index = = AP_MISSION_CMD_INDEX_NONE ) {
2014-02-23 03:54:53 -04:00
// 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 ;
2014-02-22 20:14:44 -04:00
}
}
2014-02-23 03:54:53 -04:00
// 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 ;
}
2014-02-22 20:14:44 -04:00
2014-02-23 03:54:53 -04:00
/// increment_jump_times_run - increments the recorded number of times the jump command has been run
2020-02-17 19:44:53 -04:00
void AP_Mission : : increment_jump_times_run ( Mission_Command & cmd , bool send_gcs_msg )
2014-02-22 20:14:44 -04:00
{
2014-02-24 01:50:24 -04:00
// exit immediately if cmd is not a do-jump command
2014-02-23 03:54:53 -04:00
if ( cmd . id ! = MAV_CMD_DO_JUMP ) {
// To-Do: log an error?
return ;
2014-02-22 20:14:44 -04:00
}
2014-02-23 03:54:53 -04:00
// 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 + + ;
2020-02-17 19:44:53 -04:00
if ( send_gcs_msg ) {
2024-05-07 00:58:07 -03:00
if ( cmd . content . jump . num_times = = AP_MISSION_JUMP_REPEAT_FOREVER ) {
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Mission: %u Jump %i/unlimited " , _jump_tracking [ i ] . index , _jump_tracking [ i ] . num_times_run ) ;
} else {
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 ) ;
}
2020-02-17 19:44:53 -04:00
}
2014-02-23 03:54:53 -04:00
return ;
2020-05-04 21:27:56 -03:00
} else if ( _jump_tracking [ i ] . index = = AP_MISSION_CMD_INDEX_NONE ) {
2014-02-23 03:54:53 -04:00
// 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 ;
}
2014-02-22 20:14:44 -04:00
}
2014-02-23 03:54:53 -04:00
// if we've gotten this far then the _jump_tracking array must be full
// To-Do: log an error
return ;
2014-02-22 20:14:44 -04:00
}
2014-03-11 00:52:23 -03:00
// 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 ( )
{
2014-08-13 01:43:37 -03:00
uint32_t eeprom_version = _storage . read_uint32 ( 0 ) ;
2014-03-11 00:52:23 -03:00
// if eeprom version does not match, clear the command list and update the eeprom version
if ( eeprom_version ! = AP_MISSION_EEPROM_VERSION ) {
if ( clear ( ) ) {
2014-08-13 01:43:37 -03:00
_storage . write_uint32 ( 0 , AP_MISSION_EEPROM_VERSION ) ;
2014-03-11 00:52:23 -03:00
}
}
}
2014-08-13 01:43:37 -03:00
2014-10-17 14:34:04 -03:00
// find the nearest landing sequence starting point (DO_LAND_START) and
2014-10-24 00:06:35 -03:00
// return its index. Returns 0 if no appropriate DO_LAND_START point can
// be found.
2024-03-27 14:59:52 -03:00
uint16_t AP_Mission : : get_landing_sequence_start ( const Location & current_loc )
2014-10-17 22:40:18 -03:00
{
2014-10-24 00:06:35 -03:00
uint16_t landing_start_index = 0 ;
2014-10-17 22:40:18 -03:00
float min_distance = - 1 ;
2014-10-17 14:34:04 -03:00
// Go through mission looking for nearest landing start command
2023-02-14 05:39:36 -04:00
const auto count = num_commands ( ) ;
for ( uint16_t i = 1 ; i < count ; i + + ) {
if ( get_command_id ( i ) ! = uint16_t ( MAV_CMD_DO_LAND_START ) ) {
continue ;
}
2014-10-17 22:40:18 -03:00
Mission_Command tmp ;
if ( ! read_cmd_from_storage ( i , tmp ) ) {
continue ;
}
if ( tmp . id = = MAV_CMD_DO_LAND_START ) {
2021-07-24 20:21:07 -03:00
if ( ! tmp . content . location . initialised ( ) & & ! get_next_nav_cmd ( i , tmp ) ) {
// command does not have a valid location and cannot get next valid
continue ;
}
2022-11-01 18:43:10 -03:00
2024-03-02 21:00:47 -04:00
const float tmp_distance = tmp . content . location . get_distance_NED_alt_frame ( current_loc ) . length ( ) ;
2014-10-17 22:40:18 -03:00
if ( min_distance < 0 | | tmp_distance < min_distance ) {
2014-10-17 14:34:04 -03:00
min_distance = tmp_distance ;
2014-10-17 22:40:18 -03:00
landing_start_index = i ;
2020-05-04 21:27:56 -03:00
}
2014-10-17 14:34:04 -03:00
}
}
2014-10-24 00:06:35 -03:00
return landing_start_index ;
2014-10-17 14:34:04 -03:00
}
2016-11-25 20:45:11 -04:00
/*
find the nearest landing sequence starting point ( DO_LAND_START ) and
switch to that mission item . Returns false if no DO_LAND_START
available .
*/
2024-03-27 14:59:52 -03:00
bool AP_Mission : : jump_to_landing_sequence ( const Location & current_loc )
2016-11-25 20:45:11 -04:00
{
2024-03-27 14:59:52 -03:00
uint16_t land_idx = get_landing_sequence_start ( current_loc ) ;
2016-11-25 20:45:11 -04:00
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 ( ) ;
}
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Landing sequence start " ) ;
2016-11-25 20:45:11 -04:00
return true ;
}
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_WARNING , " Unable to start landing sequence " ) ;
2016-11-25 20:45:11 -04:00
return false ;
}
2017-11-26 18:24:05 -04:00
2024-03-02 21:00:47 -04:00
/*
find the closest point on the mission after a DO_RETURN_PATH_START and before DO_LAND_START or landing
*/
bool AP_Mission : : jump_to_closest_mission_leg ( const Location & current_loc )
{
if ( _flags . state = = MISSION_RUNNING ) {
// if mission is already running don't switch away from a active landing or return path
if ( _flags . in_landing_sequence ) {
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Landing sequence active " ) ;
return true ;
} else if ( _flags . in_return_path ) {
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Return path active " ) ;
return true ;
}
}
uint16_t landing_start_index = 0 ;
float min_distance = - 1 ;
2024-11-13 15:41:44 -04:00
// This defines the maximum number of waypoints that will be searched, this limits the worst case runtime
uint16_t search_remaining = 1000 ;
2024-03-02 21:00:47 -04:00
// Go through mission and check each DO_RETURN_PATH_START
for ( uint16_t i = 1 ; i < num_commands ( ) ; i + + ) {
2024-11-13 15:41:44 -04:00
if ( get_command_id ( i ) = = uint16_t ( MAV_CMD_DO_RETURN_PATH_START ) ) {
2024-03-02 21:00:47 -04:00
uint16_t tmp_index ;
float tmp_distance ;
2024-11-13 15:41:44 -04:00
if ( distance_to_mission_leg ( i , search_remaining , tmp_distance , tmp_index , current_loc ) & & ( min_distance < 0 | | tmp_distance < = min_distance ) ) {
2024-03-02 21:00:47 -04:00
min_distance = tmp_distance ;
landing_start_index = tmp_index ;
}
2024-11-13 15:41:44 -04:00
if ( search_remaining = = 0 ) {
// Run out of time to search, stop and return the best so far
break ;
}
2024-03-02 21:00:47 -04:00
}
}
if ( landing_start_index ! = 0 & & set_current_cmd ( landing_start_index ) ) {
// if the mission has ended it has to be restarted
if ( state ( ) = = AP_Mission : : MISSION_STOPPED ) {
resume ( ) ;
}
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Return path started " ) ;
_flags . in_return_path = true ;
return true ;
}
// Failed to find do land start
return false ;
}
2018-12-05 19:28:11 -04:00
// jumps the mission to the closest landing abort that is planned, returns false if unable to find a valid abort
2024-03-27 14:59:52 -03:00
bool AP_Mission : : jump_to_abort_landing_sequence ( const Location & current_loc )
2018-12-05 19:28:11 -04:00
{
uint16_t abort_index = 0 ;
2024-03-27 14:59:52 -03:00
float min_distance = FLT_MAX ;
2018-12-05 19:28:11 -04:00
2024-03-27 14:59:52 -03:00
const auto count = num_commands ( ) ;
for ( uint16_t i = 1 ; i < count ; i + + ) {
if ( get_command_id ( i ) ! = uint16_t ( MAV_CMD_DO_GO_AROUND ) ) {
continue ;
}
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 ;
2018-12-05 19:28:11 -04:00
}
}
}
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 ( ) ;
}
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " Landing abort sequence start " ) ;
2018-12-05 19:28:11 -04:00
return true ;
}
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_WARNING , " Unable to start find a landing abort sequence " ) ;
2018-12-05 19:28:11 -04:00
return false ;
}
2020-02-17 19:44:53 -04:00
// check which is the shortest route to landing an RTL via a DO_LAND_START or continuing on the current mission plan
2024-03-27 14:59:52 -03:00
bool AP_Mission : : is_best_land_sequence ( const Location & current_loc )
2020-02-17 19:44:53 -04:00
{
// check if there is even a running mission to interupt
if ( _flags . state ! = MISSION_RUNNING ) {
return false ;
}
// check if aircraft has already jumped to a landing sequence
if ( _flags . in_landing_sequence ) {
return true ;
}
// check if MIS_OPTIONS bit set to allow distance calculation to be done
2024-07-25 21:58:16 -03:00
if ( ! option_is_set ( Option : : FAILSAFE_TO_BEST_LANDING ) ) {
2020-02-17 19:44:53 -04:00
return false ;
}
2020-05-04 21:27:56 -03:00
// The decision to allow a failsafe to interupt a potential landing approach
// is a distance travelled minimization problem. Look forward in
2020-02-17 19:44:53 -04:00
// mission to evaluate the shortest remaining distance to land.
2020-05-04 21:27:56 -03:00
// go through the mission for the nearest DO_LAND_START first as this is the most probable route
2020-02-17 19:44:53 -04:00
// to a landing with the minimum number of WP.
2024-03-27 14:59:52 -03:00
uint16_t do_land_start_index = get_landing_sequence_start ( current_loc ) ;
2020-02-17 19:44:53 -04:00
if ( do_land_start_index = = 0 ) {
// then no DO_LAND_START commands are in mission and normal failsafe behaviour should be maintained
return false ;
}
// get distance to landing if travelled to nearest DO_LAND_START via RTL
float dist_via_do_land ;
if ( ! distance_to_landing ( do_land_start_index , dist_via_do_land , current_loc ) ) {
// cant get a valid distance to landing
return false ;
}
// get distance to landing if continue along current mission path
float dist_continue_to_land ;
if ( ! distance_to_landing ( _nav_cmd . index , dist_continue_to_land , current_loc ) ) {
// cant get a valid distance to landing
return false ;
}
// compare distances
if ( dist_via_do_land > = dist_continue_to_land ) {
// then the mission should carry on uninterrupted as that is the shorter distance
2023-09-02 02:21:35 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_NOTICE , " Rejecting RTL: closer land if mis continued " ) ;
2020-02-17 19:44:53 -04:00
return true ;
} else {
// allow failsafes to interrupt the current mission
return false ;
}
}
// Approximate the distance travelled to get to a landing. DO_JUMP commands are observed in look forward.
bool AP_Mission : : distance_to_landing ( uint16_t index , float & tot_distance , Location prev_loc )
{
Mission_Command temp_cmd ;
tot_distance = 0.0f ;
2022-09-25 02:25:08 -03:00
bool ret = false ; // reached end of loop without getting to a landing
2020-02-17 19:44:53 -04:00
// back up jump tracking to reset after distance calculation
jump_tracking_struct _jump_tracking_backup [ AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS ] ;
for ( uint8_t i = 0 ; i < AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS ; i + + ) {
_jump_tracking_backup [ i ] = _jump_tracking [ i ] ;
}
// run through remainder of mission to approximate a distance to landing
2024-05-01 16:51:52 -03:00
for ( uint8_t i = 0 ; i < UINT8_MAX ; i + + ) {
2020-02-17 19:44:53 -04:00
// search until the end of the mission command list
for ( uint16_t cmd_index = index ; cmd_index < ( unsigned ) _cmd_total ; cmd_index + + ) {
// get next command
if ( ! get_next_cmd ( cmd_index , temp_cmd , true , false ) ) {
// we got to the end of the mission
goto reset_do_jump_tracking ;
}
if ( temp_cmd . id = = MAV_CMD_NAV_WAYPOINT | | temp_cmd . id = = MAV_CMD_NAV_SPLINE_WAYPOINT | | is_landing_type_cmd ( temp_cmd . id ) ) {
break ;
} else if ( is_nav_cmd ( temp_cmd ) | | temp_cmd . id = = MAV_CMD_CONDITION_DELAY ) {
// if we receive a nav command that we dont handle then give up as cant measure the distance e.g. MAV_CMD_NAV_LOITER_UNLIM
goto reset_do_jump_tracking ;
}
}
index = temp_cmd . index + 1 ;
if ( ! ( temp_cmd . content . location . lat = = 0 & & temp_cmd . content . location . lng = = 0 ) ) {
// add distance to running total
float disttemp = prev_loc . get_distance ( temp_cmd . content . location ) ;
tot_distance = tot_distance + disttemp ;
// store wp location as previous
prev_loc = temp_cmd . content . location ;
}
if ( is_landing_type_cmd ( temp_cmd . id ) ) {
// reached a landing!
ret = true ;
goto reset_do_jump_tracking ;
}
}
reset_do_jump_tracking :
for ( uint8_t i = 0 ; i < AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS ; i + + ) {
_jump_tracking [ i ] = _jump_tracking_backup [ i ] ;
}
return ret ;
}
2024-03-02 21:00:47 -04:00
// Approximate the distance travelled to return to the mission path. DO_JUMP commands are observed in look forward.
// Stop searching once reaching a landing or do-land-start
2024-11-13 15:41:44 -04:00
bool AP_Mission : : distance_to_mission_leg ( uint16_t start_index , uint16_t & search_remaining , float & rejoin_distance , uint16_t & rejoin_index , const Location & current_loc )
2024-03-02 21:00:47 -04:00
{
Location prev_loc ;
Mission_Command temp_cmd ;
rejoin_distance = - 1 ;
rejoin_index = - 1 ;
bool ret = false ;
// back up jump tracking to reset after distance calculation
jump_tracking_struct _jump_tracking_backup [ AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS ] ;
for ( uint8_t i = 0 ; i < AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS ; i + + ) {
_jump_tracking_backup [ i ] = _jump_tracking [ i ] ;
}
// run through remainder of mission to approximate a distance to landing
uint16_t index = start_index ;
2024-11-13 15:41:44 -04:00
for ( ; search_remaining > 0 ; search_remaining - - ) {
2024-03-02 21:00:47 -04:00
// search until the end of the mission command list
for ( uint16_t cmd_index = index ; cmd_index < = ( unsigned ) _cmd_total ; cmd_index + + ) {
if ( get_next_cmd ( cmd_index , temp_cmd , true , false ) ) {
break ;
} else {
// got to the end of the mission
goto reset_do_jump_tracking ;
}
}
index = temp_cmd . index + 1 ;
if ( stored_in_location ( temp_cmd . id ) & & temp_cmd . content . location . initialised ( ) ) {
if ( prev_loc . lat = = 0 & & prev_loc . lng = = 0 ) {
// Need a valid previous location to do distance to leg calculation
prev_loc = temp_cmd . content . location ;
// single point dist calc
rejoin_distance = prev_loc . get_distance_NED_alt_frame ( current_loc ) . length ( ) ;
rejoin_index = temp_cmd . index ;
ret = true ;
} else {
// Calculate the distance to rejoin
const Vector3f mission_vector = prev_loc . get_distance_NED_alt_frame ( temp_cmd . content . location ) ;
if ( ! mission_vector . is_zero ( ) ) {
Vector3f pos = prev_loc . get_distance_NED_alt_frame ( current_loc ) ;
// project pos vector on to mission vector
Vector3f p = pos . projected ( mission_vector ) ;
// constrain to mission line
p . x = constrain_float ( p . x , MIN ( 0 , mission_vector . x ) , MAX ( 0 , mission_vector . x ) ) ;
p . y = constrain_float ( p . y , MIN ( 0 , mission_vector . y ) , MAX ( 0 , mission_vector . y ) ) ;
p . z = constrain_float ( p . z , MIN ( 0 , mission_vector . z ) , MAX ( 0 , mission_vector . z ) ) ;
const float disttemp = ( p - pos ) . length ( ) ;
// store wp location as previous
prev_loc = temp_cmd . content . location ;
if ( disttemp < rejoin_distance | | is_negative ( rejoin_distance ) ) {
rejoin_distance = disttemp ;
rejoin_index = temp_cmd . index ;
}
ret = true ;
}
}
}
if ( is_landing_type_cmd ( temp_cmd . id ) | | ( temp_cmd . id = = MAV_CMD_DO_LAND_START ) ) {
// reached a landing!
goto reset_do_jump_tracking ;
}
}
// reached end of loop without getting to a landing or DO_LAND_START
ret = false ;
reset_do_jump_tracking :
for ( uint8_t i = 0 ; i < AP_MISSION_MAX_NUM_DO_JUMP_COMMANDS ; i + + ) {
_jump_tracking [ i ] = _jump_tracking_backup [ i ] ;
}
return ret ;
}
2020-02-17 19:44:53 -04:00
// check if command is a landing type command.
bool AP_Mission : : is_landing_type_cmd ( uint16_t id ) const
{
switch ( id ) {
2020-05-04 21:27:56 -03:00
case MAV_CMD_NAV_LAND :
case MAV_CMD_NAV_VTOL_LAND :
case MAV_CMD_DO_PARACHUTE :
return true ;
default :
2022-06-09 20:14:40 -03:00
return false ;
}
}
// check if command is a takeoff type command.
bool AP_Mission : : is_takeoff_type_cmd ( uint16_t id ) const
{
switch ( id ) {
case MAV_CMD_NAV_TAKEOFF :
case MAV_CMD_NAV_VTOL_TAKEOFF :
return true ;
default :
2020-05-04 21:27:56 -03:00
return false ;
2020-02-17 19:44:53 -04:00
}
}
2020-05-04 21:27:56 -03:00
const char * AP_Mission : : Mission_Command : : type ( ) const
{
switch ( id ) {
2017-11-26 18:24:05 -04:00
case MAV_CMD_NAV_WAYPOINT :
return " WP " ;
2019-04-15 22:59:21 -03:00
case MAV_CMD_NAV_SPLINE_WAYPOINT :
return " SplineWP " ;
2017-11-26 18:24:05 -04:00
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 " ;
2019-03-08 01:27:09 -04:00
case MAV_CMD_NAV_GUIDED_ENABLE :
return " GuidedEnable " ;
2017-11-26 22:13:22 -04:00
case MAV_CMD_NAV_LOITER_TURNS :
return " LoitTurns " ;
case MAV_CMD_NAV_LOITER_TO_ALT :
return " LoitAltitude " ;
2017-11-26 18:24:05 -04:00
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_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 " ;
2020-02-24 18:43:20 -04:00
case MAV_CMD_DO_SET_RESUME_REPEAT_DIST :
return " SetRepeatDist " ;
2019-03-08 04:44:58 -04:00
case MAV_CMD_DO_GUIDED_LIMITS :
return " GuidedLimits " ;
2017-11-26 22:13:22 -04:00
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 " ;
2024-03-02 21:00:47 -04:00
case MAV_CMD_DO_RETURN_PATH_START :
return " ReturnPathStart " ;
2017-11-26 22:13:22 -04:00
case MAV_CMD_DO_LAND_START :
return " LandStart " ;
case MAV_CMD_NAV_DELAY :
return " Delay " ;
2022-09-20 04:37:48 -03:00
# if AP_GRIPPER_ENABLED
2017-11-26 22:13:22 -04:00
case MAV_CMD_DO_GRIPPER :
return " Gripper " ;
2022-09-20 04:37:48 -03:00
# endif
2023-10-18 23:34:30 -03:00
# if AP_MISSION_NAV_PAYLOAD_PLACE_ENABLED
2017-11-26 22:13:22 -04:00
case MAV_CMD_NAV_PAYLOAD_PLACE :
return " PayloadPlace " ;
2023-10-18 23:34:30 -03:00
# endif
2017-11-26 22:13:22 -04:00
case MAV_CMD_DO_PARACHUTE :
return " Parachute " ;
2020-07-02 20:04:27 -03:00
case MAV_CMD_DO_SPRAYER :
return " Sprayer " ;
2021-02-17 21:01:41 -04:00
case MAV_CMD_DO_AUX_FUNCTION :
return " AuxFunction " ;
2019-12-09 18:25:15 -04:00
case MAV_CMD_DO_MOUNT_CONTROL :
return " MountControl " ;
2020-07-23 01:54:37 -03:00
case MAV_CMD_DO_WINCH :
return " Winch " ;
2021-02-25 21:08:22 -04:00
case MAV_CMD_DO_SEND_SCRIPT_MESSAGE :
return " Scripting " ;
2021-09-12 10:01:13 -03:00
case MAV_CMD_DO_JUMP :
return " Jump " ;
2023-02-21 21:24:05 -04:00
case MAV_CMD_DO_JUMP_TAG :
return " JumpToTag " ;
case MAV_CMD_JUMP_TAG :
return " Tag " ;
2021-09-12 10:01:13 -03:00
case MAV_CMD_DO_GO_AROUND :
return " Go Around " ;
2022-10-18 22:59:40 -03:00
# if AP_SCRIPTING_ENABLED
2021-10-25 00:59:11 -03:00
case MAV_CMD_NAV_SCRIPT_TIME :
return " NavScriptTime " ;
2022-10-18 22:59:40 -03:00
# endif
2022-05-25 01:52:01 -03:00
case MAV_CMD_NAV_ATTITUDE_TIME :
return " NavAttitudeTime " ;
2021-11-24 06:46:40 -04:00
case MAV_CMD_DO_PAUSE_CONTINUE :
return " PauseContinue " ;
2022-06-02 09:23:47 -03:00
case MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW :
return " GimbalPitchYaw " ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_IMAGE_START_CAPTURE :
return " ImageStartCapture " ;
2023-09-07 07:12:48 -03:00
case MAV_CMD_IMAGE_STOP_CAPTURE :
return " ImageStopCapture " ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_SET_CAMERA_ZOOM :
return " SetCameraZoom " ;
case MAV_CMD_SET_CAMERA_FOCUS :
return " SetCameraFocus " ;
2024-02-08 04:30:08 -04:00
case MAV_CMD_SET_CAMERA_SOURCE :
return " SetCameraSource " ;
2023-03-08 07:28:43 -04:00
case MAV_CMD_VIDEO_START_CAPTURE :
return " VideoStartCapture " ;
case MAV_CMD_VIDEO_STOP_CAPTURE :
return " VideoStopCapture " ;
2017-11-26 18:24:05 -04:00
default :
2017-11-26 22:13:22 -04:00
# if CONFIG_HAL_BOARD == HAL_BOARD_SITL
AP_HAL : : panic ( " Mission command with ID %u has no string " , id ) ;
# endif
2017-11-26 18:24:05 -04:00
return " ? " ;
}
}
2018-08-16 23:43:46 -03:00
2023-02-14 05:39:36 -04:00
/*
get the command ID of a mission index . Caller should have checked the index is in range
*/
uint16_t AP_Mission : : get_command_id ( uint16_t index ) const
{
const uint16_t pos_in_storage = 4 + ( index * AP_MISSION_EEPROM_COMMAND_SIZE ) ;
uint8_t b [ 3 ] { } ;
if ( ! _storage . read_block ( b , pos_in_storage , sizeof ( b ) ) ) {
return 0U ;
}
uint16_t id = 0 ;
if ( b [ 0 ] = = 0 | | b [ 0 ] = = 1 ) {
memcpy ( ( void * ) & id , ( void * ) & b [ 1 ] , 2 ) ;
} else {
id = b [ 0 ] ;
}
return id ;
}
/*
see if the mission contains a particular item
*/
2018-12-25 01:17:39 -04:00
bool AP_Mission : : contains_item ( MAV_CMD command ) const
{
2023-02-14 05:39:36 -04:00
const auto count = num_commands ( ) ;
for ( uint16_t i = 1 ; i < count ; i + + ) {
if ( get_command_id ( i ) ! = uint16_t ( command ) ) {
continue ;
}
// confirm with full read
2018-12-25 01:17:39 -04:00
Mission_Command tmp ;
if ( ! read_cmd_from_storage ( i , tmp ) ) {
continue ;
}
if ( tmp . id = = command ) {
return true ;
}
}
return false ;
}
2023-01-15 20:03:55 -04:00
/*
return true if the mission item has a location
*/
bool AP_Mission : : cmd_has_location ( const uint16_t command )
{
return stored_in_location ( command ) ;
}
2020-06-22 18:23:46 -03:00
/*
2022-08-03 22:24:43 -03:00
return true if the mission has a terrain relative item . ~ 2200u s for 530 items on H7
2020-06-22 18:23:46 -03:00
*/
2022-08-09 01:35:58 -03:00
bool AP_Mission : : contains_terrain_alt_items ( void )
2022-08-03 22:24:43 -03:00
{
if ( _last_contains_relative_calculated_ms ! = _last_change_time_ms ) {
2022-08-09 01:35:58 -03:00
_contains_terrain_alt_items = calculate_contains_terrain_alt_items ( ) ;
2022-08-03 22:24:43 -03:00
_last_contains_relative_calculated_ms = _last_change_time_ms ;
}
2022-08-09 01:35:58 -03:00
return _contains_terrain_alt_items ;
2022-08-03 22:24:43 -03:00
}
2022-08-09 01:35:58 -03:00
bool AP_Mission : : calculate_contains_terrain_alt_items ( void ) const
2020-06-22 18:23:46 -03:00
{
2023-02-14 05:39:36 -04:00
const auto count = num_commands ( ) ;
for ( uint16_t i = 1 ; i < count ; i + + ) {
if ( ! stored_in_location ( get_command_id ( i ) ) ) {
continue ;
}
2020-06-22 18:23:46 -03:00
Mission_Command tmp ;
if ( ! read_cmd_from_storage ( i , tmp ) ) {
continue ;
}
if ( stored_in_location ( tmp . id ) & & tmp . content . location . terrain_alt ) {
return true ;
}
}
return false ;
}
2020-02-24 18:43:20 -04:00
// reset the mission history to prevent recalling previous mission histories after a mission restart.
void AP_Mission : : reset_wp_history ( void )
{
for ( uint8_t i = 0 ; i < AP_MISSION_MAX_WP_HISTORY ; i + + ) {
_wp_index_history [ i ] = AP_MISSION_CMD_INDEX_NONE ;
}
_resume_cmd . index = AP_MISSION_CMD_INDEX_NONE ;
_flags . resuming_mission = false ;
_repeat_dist = 0 ;
}
// store the latest reported position incase of mission exit and rewind resume
void AP_Mission : : update_exit_position ( void )
{
2022-01-20 19:42:41 -04:00
if ( ! AP : : ahrs ( ) . get_location ( _exit_position ) ) {
2020-02-24 18:43:20 -04:00
_exit_position . lat = 0 ;
_exit_position . lng = 0 ;
}
}
// calculate the location of the _resume_cmd wp and set as current
bool AP_Mission : : calc_rewind_pos ( Mission_Command & rewind_cmd )
{
// check for a recent history
if ( _wp_index_history [ LAST_WP_PASSED ] = = AP_MISSION_CMD_INDEX_NONE ) {
// no saved history so can't rewind
return false ;
}
// check that we have a valid exit position
if ( _exit_position . lat = = 0 & & _exit_position . lng = = 0 ) {
return false ;
}
Mission_Command temp_cmd ;
float rewind_distance = _repeat_dist ; //(m)
uint16_t resume_index ;
Location prev_loc = _exit_position ;
for ( int8_t i = ( LAST_WP_PASSED ) ; i > = 0 ; i - - ) {
2020-05-04 21:27:56 -03:00
// to get this far there has to be at least one 'passed wp' stored in history. This is to check incase
2020-02-24 18:43:20 -04:00
// of history array no being completely filled with valid waypoints upon resume.
if ( _wp_index_history [ i ] = = AP_MISSION_CMD_INDEX_NONE ) {
// no more stored history
resume_index = i + 1 ;
break ;
}
if ( ! read_cmd_from_storage ( _wp_index_history [ i ] , temp_cmd ) ) {
// if read from storage failed then don't use rewind
return false ;
}
// calculate distance
float disttemp = prev_loc . get_distance ( temp_cmd . content . location ) ; //(m)
rewind_distance - = disttemp ;
resume_index = i ;
if ( rewind_distance < = 0.0f ) {
// history rewound enough distance to meet _repeat_dist requirement
rewind_cmd = temp_cmd ;
break ;
}
// store wp location as previous
prev_loc = temp_cmd . content . location ;
}
if ( rewind_distance > 0.0f ) {
// then the history array was rewound all of the way without finding a wp distance > _repeat_dist distance.
// the last read temp_cmd will be the furthest cmd back in the history array so resume to that.
rewind_cmd = temp_cmd ;
return true ;
}
// if we have got this far the desired rewind distance lies between two waypoints stored in history array.
// calculate the location for the mission to resume
// the last wp read from storage is the wp that is before the resume wp in the mission order
Location passed_wp_loc = temp_cmd . content . location ;
// fetch next destination wp
if ( ! read_cmd_from_storage ( _wp_index_history [ resume_index + 1 ] , temp_cmd ) ) {
// if read from storage failed then don't use rewind
return false ;
}
// determine the length of the mission leg that the resume wp lies in
float leg_length = passed_wp_loc . get_distance ( temp_cmd . content . location ) ; //(m)
// calculate the percentage along the leg that resume wp will be positioned
float leg_percent = fabsf ( rewind_distance ) / leg_length ;
// calculate difference vector of mission leg
Vector3f dist_vec = passed_wp_loc . get_distance_NED ( temp_cmd . content . location ) ;
// calculate the resume wp position
rewind_cmd . content . location . offset ( dist_vec . x * leg_percent , dist_vec . y * leg_percent ) ;
rewind_cmd . content . location . alt - = dist_vec . z * leg_percent * 100 ; //(cm)
// The rewind_cmd.index has the index of the 'last passed wp' from the history array. This ensures that the mission order
// continues as planned without further intervention. The resume wp is not written to memory so will not perminantely change the mission.
// if we got this far then mission rewind position was successfully calculated
return true ;
}
2022-10-18 22:44:33 -03:00
/*
handle format conversion of storage format to allow us to update
format to take advantage of new packing . This is particularly useful
for conversion to Float16 to get extra parameter space
*/
void AP_Mission : : format_conversion ( uint8_t tag_byte , const Mission_Command & cmd , PackedContent & packed_content ) const
{
// currently only one conversion needed, more can be added
2022-10-18 22:59:40 -03:00
# if AP_SCRIPTING_ENABLED
2022-10-18 22:44:33 -03:00
if ( tag_byte = = 0 & & cmd . id = = MAV_CMD_NAV_SCRIPT_TIME ) {
2022-10-18 22:59:40 -03:00
// PARAMETER_CONVERSION: conversion code added Oct 2022
2022-10-18 22:44:33 -03:00
struct nav_script_time_Command_tag0 old_fmt ;
struct nav_script_time_Command new_fmt ;
memcpy ( ( void * ) & old_fmt , packed_content . bytes , sizeof ( old_fmt ) ) ;
new_fmt . command = old_fmt . command ;
new_fmt . timeout_s = old_fmt . timeout_s ;
new_fmt . arg1 . set ( old_fmt . arg1 ) ;
new_fmt . arg2 . set ( old_fmt . arg2 ) ;
new_fmt . arg3 = 0 ;
new_fmt . arg4 = 0 ;
memcpy ( packed_content . bytes , ( void * ) & new_fmt , sizeof ( new_fmt ) ) ;
}
2022-10-18 22:59:40 -03:00
# endif
2022-10-18 22:44:33 -03:00
}
2024-03-27 14:59:52 -03:00
// Helpers to fill in location for scripting
# if AP_SCRIPTING_ENABLED
bool AP_Mission : : jump_to_landing_sequence ( void )
{
Location loc ;
if ( AP : : ahrs ( ) . get_location ( loc ) ) {
return jump_to_landing_sequence ( loc ) ;
}
GCS_SEND_TEXT ( MAV_SEVERITY_WARNING , " Unable to start landing sequence " ) ;
return false ;
}
bool AP_Mission : : jump_to_abort_landing_sequence ( void )
{
Location loc ;
if ( AP : : ahrs ( ) . get_location ( loc ) ) {
return jump_to_abort_landing_sequence ( loc ) ;
}
GCS_SEND_TEXT ( MAV_SEVERITY_WARNING , " Unable to start find a landing abort sequence " ) ;
return false ;
}
# endif // AP_SCRIPTING_ENABLED
2018-08-16 23:43:46 -03:00
// singleton instance
AP_Mission * AP_Mission : : _singleton ;
2020-05-04 21:27:56 -03:00
namespace AP
{
2018-08-16 23:43:46 -03:00
AP_Mission * mission ( )
{
return AP_Mission : : get_singleton ( ) ;
}
}
2023-12-09 00:05:08 -04:00
# endif // AP_MISSION_ENABLED