2017-07-18 23:17:45 -03:00
# pragma once
# include <stdint.h>
2018-05-23 02:09:45 -03:00
# include <GCS_MAVLink/GCS_MAVLink.h>
# include <AP_Math/AP_Math.h>
2017-07-18 23:17:45 -03:00
# include "defines.h"
2017-08-09 22:30:35 -03:00
# define MODE_NEXT_HEADING_UNKNOWN 99999.0f // used to indicate to set_desired_location method that next leg's heading is unknown
2017-11-22 08:38:57 -04:00
// pre-define ModeRTL so Auto can appear higher in this file
class ModeRTL ;
2017-07-18 23:17:45 -03:00
class Mode
{
public :
2018-06-06 03:41:17 -03:00
// Auto Pilot modes
// ----------------
enum Number {
MANUAL = 0 ,
ACRO = 1 ,
STEERING = 3 ,
HOLD = 4 ,
LOITER = 5 ,
2018-05-24 01:47:07 -03:00
FOLLOW = 6 ,
2018-06-06 03:41:17 -03:00
AUTO = 10 ,
RTL = 11 ,
SMART_RTL = 12 ,
GUIDED = 15 ,
INITIALISING = 16
} ;
2017-07-18 23:17:45 -03:00
// Constructor
Mode ( ) ;
// enter this mode, returns false if we failed to enter
bool enter ( ) ;
// perform any cleanups required:
void exit ( ) ;
// returns a unique number specific to this mode
virtual uint32_t mode_number ( ) const = 0 ;
2017-11-28 22:35:33 -04:00
// returns short text name (up to 4 bytes)
virtual const char * name4 ( ) const = 0 ;
2017-07-18 23:17:45 -03:00
//
// methods that sub classes should override to affect movement of the vehicle in this mode
//
// convert user input to targets, implement high level control for this mode
virtual void update ( ) = 0 ;
//
// attributes of the mode
//
2018-01-10 21:54:40 -04:00
// return if in non-manual mode : Auto, Guided, RTL, SmartRTL
2017-07-18 23:17:45 -03:00
virtual bool is_autopilot_mode ( ) const { return false ; }
2018-01-10 21:54:40 -04:00
// returns true if vehicle can be armed or disarmed from the transmitter in this mode
virtual bool allows_arming_from_transmitter ( ) { return ! is_autopilot_mode ( ) ; }
2017-07-18 23:17:45 -03:00
//
// attributes for mavlink system status reporting
//
// returns true if any RC input is used
virtual bool has_manual_input ( ) const { return false ; }
// true if heading is controlled
virtual bool attitude_stabilized ( ) const { return true ; }
2018-01-22 07:00:47 -04:00
// true if mode requires position and/or velocity estimate
virtual bool requires_position ( ) const { return true ; }
virtual bool requires_velocity ( ) const { return true ; }
2017-11-29 02:30:37 -04:00
2017-08-03 05:08:09 -03:00
//
// navigation methods
//
// return distance (in meters) to destination
virtual float get_distance_to_destination ( ) const { return 0.0f ; }
2017-08-04 20:59:15 -03:00
// set desired location and speed (used in RTL, Guided, Auto)
2017-08-09 22:30:35 -03:00
// next_leg_bearing_cd should be heading to the following waypoint (used to slow the vehicle in order to make the turn)
virtual void set_desired_location ( const struct Location & destination , float next_leg_bearing_cd = MODE_NEXT_HEADING_UNKNOWN ) ;
2017-11-30 00:08:49 -04:00
// set desired location as offset from the EKF origin, return true on success
bool set_desired_location_NED ( const Vector3f & destination , float next_leg_bearing_cd = MODE_NEXT_HEADING_UNKNOWN ) ;
2017-08-03 05:08:09 -03:00
// true if vehicle has reached desired location. defaults to true because this is normally used by missions and we do not want the mission to become stuck
virtual bool reached_destination ( ) { return true ; }
2017-08-04 20:59:15 -03:00
// set desired heading and speed - supported in Auto and Guided modes
2017-08-03 05:08:09 -03:00
virtual void set_desired_heading_and_speed ( float yaw_angle_cd , float target_speed ) ;
// get speed error in m/s, returns zero for modes that do not control speed
2017-12-05 21:38:35 -04:00
float speed_error ( ) const { return _speed_error ; }
2017-08-03 05:08:09 -03:00
2017-12-05 21:41:28 -04:00
// get default speed for this mode (held in CRUISE_SPEED, WP_SPEED or RTL_SPEED)
// rtl argument should be true if called from RTL or SmartRTL modes (handled here to avoid duplication)
float get_speed_default ( bool rtl = false ) const ;
2018-06-08 22:19:41 -03:00
// set desired speed in m/s
bool set_desired_speed ( float speed ) ;
2017-12-05 21:41:28 -04:00
// restore desired speed to default from parameter values (CRUISE_SPEED or WP_SPEED)
// rtl argument should be true if called from RTL or SmartRTL modes (handled here to avoid duplication)
void set_desired_speed_to_default ( bool rtl = false ) ;
2018-08-08 01:48:38 -03:00
// execute the mission in reverse (i.e. backing up)
void set_reversed ( bool value ) ;
2017-07-18 23:17:45 -03:00
protected :
// subclasses override this to perform checks before entering the mode
virtual bool _enter ( ) { return true ; }
// subclasses override this to perform any required cleanup when exiting the mode
virtual void _exit ( ) { return ; }
2018-05-05 22:47:59 -03:00
// decode pilot steering and throttle inputs and return in steer_out and throttle_out arguments
2018-05-03 05:57:40 -03:00
// steering_out is in the range -4500 ~ +4500 with positive numbers meaning rotate clockwise
// throttle_out is in the range -100 ~ +100
2017-11-27 09:11:45 -04:00
void get_pilot_desired_steering_and_throttle ( float & steering_out , float & throttle_out ) ;
2018-05-05 22:47:59 -03:00
// decode pilot input steering and return steering_out and speed_out (in m/s)
void get_pilot_desired_steering_and_speed ( float & steering_out , float & speed_out ) ;
2018-05-10 04:10:34 -03:00
// decode pilot lateral movement input and return in lateral_out argument
void get_pilot_desired_lateral ( float & lateral_out ) ;
2017-12-23 01:54:11 -04:00
// calculate steering output to drive along line from origin to destination waypoint
void calc_steering_to_waypoint ( const struct Location & origin , const struct Location & destination , bool reversed = false ) ;
2017-07-18 23:17:45 -03:00
// calculate steering angle given a desired lateral acceleration
2017-12-23 02:17:49 -04:00
void calc_steering_from_lateral_acceleration ( float lat_accel , bool reversed = false ) ;
2017-07-18 23:17:45 -03:00
2017-12-23 01:54:35 -04:00
// calculate steering output to drive towards desired heading
2018-09-10 04:25:27 -03:00
// rate_max is a maximum turn rate in deg/s. set to zero to use default turn rate limits
void calc_steering_to_heading ( float desired_heading_cd , float rate_max_degs = 0.0f ) ;
2017-08-03 05:08:09 -03:00
// calculates the amount of throttle that should be output based
// on things like proximity to corners and current speed
2018-04-21 03:45:12 -03:00
virtual void calc_throttle ( float target_speed , bool nudge_allowed , bool avoidance_enabled ) ;
2017-07-18 23:17:45 -03:00
2017-08-10 00:06:43 -03:00
// performs a controlled stop. returns true once vehicle has stopped
bool stop_vehicle ( ) ;
2017-08-08 21:24:30 -03:00
// estimate maximum vehicle speed (in m/s)
2017-12-31 23:46:46 -04:00
// cruise_speed is in m/s, cruise_throttle should be in the range -1 to +1
float calc_speed_max ( float cruise_speed , float cruise_throttle ) const ;
2017-08-08 21:24:30 -03:00
// calculate pilot input to nudge speed up or down
// target_speed should be in meters/sec
// cruise_speed is vehicle's cruising speed, cruise_throttle is the throttle (from -1 to +1) that achieves the cruising speed
// return value is a new speed (in m/s) which up to the projected maximum speed based on the cruise speed and cruise throttle
float calc_speed_nudge ( float target_speed , float cruise_speed , float cruise_throttle ) ;
2017-07-18 04:27:05 -03:00
2017-08-03 05:08:09 -03:00
// calculated a reduced speed(in m/s) based on yaw error and lateral acceleration and/or distance to a waypoint
2017-11-27 07:57:59 -04:00
// should be called after calc_steering_to_waypoint and before calc_throttle
2017-08-03 05:08:09 -03:00
// relies on these internal members being updated: lateral_acceleration, _yaw_error_cd, _distance_to_destination
float calc_reduced_speed_for_turn_or_distance ( float desired_speed ) ;
2018-03-13 09:06:13 -03:00
// calculate vehicle stopping location using current location, velocity and maximum acceleration
void calc_stopping_location ( Location & stopping_loc ) ;
2018-05-05 22:47:59 -03:00
protected :
// decode pilot steering and throttle inputs and return in steer_out and throttle_out arguments
// steering_out is in the range -4500 ~ +4500 with positive numbers meaning rotate clockwise
// throttle_out is in the range -100 ~ +100
void get_pilot_input ( float & steering_out , float & throttle_out ) ;
2017-07-18 23:17:45 -03:00
// references to avoid code churn:
2017-08-03 05:08:09 -03:00
class AP_AHRS & ahrs ;
2017-07-18 23:17:45 -03:00
class Parameters & g ;
class ParametersG2 & g2 ;
2017-08-03 06:41:47 -03:00
class RC_Channel * & channel_steer ;
2017-07-18 23:17:45 -03:00
class RC_Channel * & channel_throttle ;
2018-05-10 04:10:34 -03:00
class RC_Channel * & channel_lateral ;
2017-07-18 23:17:45 -03:00
class AP_Mission & mission ;
2017-08-08 02:37:21 -03:00
class AR_AttitudeControl & attitude_control ;
2017-08-03 05:08:09 -03:00
// private members for waypoint navigation
Location _origin ; // origin Location (vehicle will travel from the origin to the destination)
Location _destination ; // destination Location when in Guided_WP
float _distance_to_destination ; // distance from vehicle to final destination in meters
bool _reached_destination ; // true once the vehicle has reached the destination
float _desired_yaw_cd ; // desired yaw in centi-degrees
float _yaw_error_cd ; // error between desired yaw and actual yaw in centi-degrees
float _desired_speed ; // desired speed in m/s
2017-08-09 22:30:35 -03:00
float _desired_speed_final ; // desired speed in m/s when we reach the destination
2017-08-03 05:08:09 -03:00
float _speed_error ; // ground speed error in m/s
2018-03-13 09:06:13 -03:00
uint32_t last_steer_to_wp_ms ; // system time of last call to calc_steering_to_waypoint
2018-08-08 01:48:38 -03:00
bool _reversed ; // execute the mission by backing up
2017-07-18 23:17:45 -03:00
} ;
2017-11-28 02:59:13 -04:00
class ModeAcro : public Mode
{
public :
uint32_t mode_number ( ) const override { return ACRO ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " ACRO " ; }
2017-11-28 02:59:13 -04:00
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// attributes for mavlink system status reporting
bool has_manual_input ( ) const override { return true ; }
2018-01-22 07:00:47 -04:00
2018-04-11 16:48:25 -03:00
// acro mode requires a velocity estimate for non skid-steer rovers
2018-01-22 07:00:47 -04:00
bool requires_position ( ) const override { return false ; }
2018-04-11 16:48:25 -03:00
bool requires_velocity ( ) const override ;
2017-11-28 02:59:13 -04:00
} ;
2017-07-18 23:17:45 -03:00
class ModeAuto : public Mode
{
public :
2017-11-22 08:38:57 -04:00
// constructor
ModeAuto ( ModeRTL & mode_rtl ) ;
2017-07-18 23:17:45 -03:00
uint32_t mode_number ( ) const override { return AUTO ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " AUTO " ; }
2017-07-18 23:17:45 -03:00
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
2018-04-21 03:45:12 -03:00
void calc_throttle ( float target_speed , bool nudge_allowed , bool avoidance_enabled ) ;
2017-07-18 23:17:45 -03:00
// attributes of the mode
bool is_autopilot_mode ( ) const override { return true ; }
2017-08-03 05:11:45 -03:00
// return distance (in meters) to destination
2017-12-08 23:32:53 -04:00
float get_distance_to_destination ( ) const override ;
2017-08-03 05:11:45 -03:00
// set desired location, heading and speed
2017-12-06 22:42:34 -04:00
void set_desired_location ( const struct Location & destination , float next_leg_bearing_cd = MODE_NEXT_HEADING_UNKNOWN ) ;
2017-08-03 05:11:45 -03:00
bool reached_destination ( ) override ;
// heading and speed control
void set_desired_heading_and_speed ( float yaw_angle_cd , float target_speed ) override ;
bool reached_heading ( ) ;
2017-11-22 08:38:57 -04:00
// start RTL (within auto)
void start_RTL ( ) ;
2017-07-18 23:17:45 -03:00
protected :
bool _enter ( ) override ;
void _exit ( ) override ;
2017-08-03 05:11:45 -03:00
enum AutoSubMode {
Auto_WP , // drive to a given location
2017-11-22 08:38:57 -04:00
Auto_HeadingAndSpeed , // turn to a given heading
Auto_RTL // perform RTL within auto mode
2017-08-03 05:11:45 -03:00
} _submode ;
2017-07-18 23:17:45 -03:00
private :
bool check_trigger ( void ) ;
2017-11-22 08:38:57 -04:00
// references
ModeRTL & _mode_rtl ;
2017-07-18 23:17:45 -03:00
// this is set to true when auto has been triggered to start
bool auto_triggered ;
2017-08-03 05:11:45 -03:00
bool _reached_heading ; // true when vehicle has reached desired heading in TurnToHeading sub mode
2017-07-18 23:17:45 -03:00
} ;
class ModeGuided : public Mode
{
public :
uint32_t mode_number ( ) const override { return GUIDED ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " GUID " ; }
2017-07-18 23:17:45 -03:00
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// attributes of the mode
bool is_autopilot_mode ( ) const override { return true ; }
2017-08-03 05:14:16 -03:00
// return distance (in meters) to destination
float get_distance_to_destination ( ) const override ;
// set desired location, heading and speed
2017-08-09 22:30:35 -03:00
void set_desired_location ( const struct Location & destination ) ;
2017-08-03 05:14:16 -03:00
void set_desired_heading_and_speed ( float yaw_angle_cd , float target_speed ) override ;
// set desired heading-delta, turn-rate and speed
void set_desired_heading_delta_and_speed ( float yaw_delta_cd , float target_speed ) ;
void set_desired_turn_rate_and_speed ( float turn_rate_cds , float target_speed ) ;
protected :
2017-07-18 23:17:45 -03:00
enum GuidedMode {
Guided_WP ,
2017-08-03 05:14:16 -03:00
Guided_HeadingAndSpeed ,
Guided_TurnRateAndSpeed
2017-07-18 23:17:45 -03:00
} ;
2017-08-03 05:14:16 -03:00
bool _enter ( ) override ;
2017-07-18 23:17:45 -03:00
2017-08-03 05:14:16 -03:00
GuidedMode _guided_mode ; // stores which GUIDED mode the vehicle is in
2017-07-18 23:17:45 -03:00
2017-08-03 05:14:16 -03:00
// attitude control
bool have_attitude_target ; // true if we have a valid attitude target
uint32_t _des_att_time_ms ; // system time last call to set_desired_attitude was made (used for timeout)
float _desired_yaw_rate_cds ; // target turn rate centi-degrees per second
2017-07-18 23:17:45 -03:00
} ;
class ModeHold : public Mode
{
public :
uint32_t mode_number ( ) const override { return HOLD ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " HOLD " ; }
2017-07-18 23:17:45 -03:00
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// attributes for mavlink system status reporting
bool attitude_stabilized ( ) const override { return false ; }
2017-12-05 01:30:38 -04:00
2018-01-22 07:00:47 -04:00
// hold mode does not require position or velocity estimate
bool requires_position ( ) const override { return false ; }
bool requires_velocity ( ) const override { return false ; }
2017-07-18 23:17:45 -03:00
} ;
2018-04-28 03:31:31 -03:00
class ModeLoiter : public Mode
{
public :
uint32_t mode_number ( ) const override { return LOITER ; }
const char * name4 ( ) const override { return " LOIT " ; }
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// return distance (in meters) to destination
float get_distance_to_destination ( ) const override { return _distance_to_destination ; }
protected :
bool _enter ( ) override ;
} ;
2017-07-18 23:17:45 -03:00
class ModeManual : public Mode
{
public :
uint32_t mode_number ( ) const override { return MANUAL ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " MANU " ; }
2017-07-18 23:17:45 -03:00
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// attributes for mavlink system status reporting
bool has_manual_input ( ) const override { return true ; }
bool attitude_stabilized ( ) const override { return false ; }
2017-11-29 02:30:37 -04:00
2018-01-22 07:00:47 -04:00
// manual mode does not require position or velocity estimate
bool requires_position ( ) const override { return false ; }
bool requires_velocity ( ) const override { return false ; }
2018-05-10 04:10:34 -03:00
protected :
void _exit ( ) override ;
2017-07-18 23:17:45 -03:00
} ;
class ModeRTL : public Mode
{
public :
uint32_t mode_number ( ) const override { return RTL ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " RTL " ; }
2017-07-18 23:17:45 -03:00
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// attributes of the mode
bool is_autopilot_mode ( ) const override { return true ; }
2017-08-03 05:14:52 -03:00
float get_distance_to_destination ( ) const override { return _distance_to_destination ; }
bool reached_destination ( ) override { return _reached_destination ; }
2017-07-18 23:17:45 -03:00
protected :
bool _enter ( ) override ;
} ;
2017-11-29 21:58:11 -04:00
class ModeSmartRTL : public Mode
{
public :
uint32_t mode_number ( ) const override { return SMART_RTL ; }
const char * name4 ( ) const override { return " SRTL " ; }
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// attributes of the mode
bool is_autopilot_mode ( ) const override { return true ; }
float get_distance_to_destination ( ) const override { return _distance_to_destination ; }
bool reached_destination ( ) override { return smart_rtl_state = = SmartRTL_StopAtHome ; }
// save current position for use by the smart_rtl flight mode
2018-02-12 01:16:23 -04:00
void save_position ( ) ;
2017-11-29 21:58:11 -04:00
protected :
// Safe RTL states
enum SmartRTLState {
SmartRTL_WaitForPathCleanup ,
SmartRTL_PathFollow ,
SmartRTL_StopAtHome ,
SmartRTL_Failure
} smart_rtl_state ;
bool _enter ( ) override ;
bool _load_point ;
} ;
2017-07-18 23:17:45 -03:00
class ModeSteering : public Mode
{
public :
uint32_t mode_number ( ) const override { return STEERING ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " STER " ; }
2017-07-18 23:17:45 -03:00
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
// attributes for mavlink system status reporting
bool has_manual_input ( ) const override { return true ; }
2018-01-22 07:00:47 -04:00
// steering requires velocity but not position
bool requires_position ( ) const override { return false ; }
bool requires_velocity ( ) const override { return true ; }
2017-07-18 23:17:45 -03:00
} ;
class ModeInitializing : public Mode
{
public :
uint32_t mode_number ( ) const override { return INITIALISING ; }
2017-11-28 22:35:33 -04:00
const char * name4 ( ) const override { return " INIT " ; }
2017-07-18 23:17:45 -03:00
// methods that affect movement of the vehicle in this mode
void update ( ) override { }
// attributes for mavlink system status reporting
bool has_manual_input ( ) const override { return true ; }
bool attitude_stabilized ( ) const override { return false ; }
} ;
2018-05-24 01:47:07 -03:00
2018-07-07 02:12:45 -03:00
class ModeFollow : public Mode
2018-05-24 01:47:07 -03:00
{
public :
uint32_t mode_number ( ) const override { return FOLLOW ; }
const char * name4 ( ) const override { return " FOLL " ; }
// methods that affect movement of the vehicle in this mode
void update ( ) override ;
protected :
bool _enter ( ) override ;
} ;