mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-19 23:28:32 -04:00
61c34ea98c
RC_Channel: move handling of Rover's functions into RC_Channel RC_Channel: move auxsw handling for camera to base RC_Channel RC_Channel: add responsibility for servorelay events RC_Channel: move mode switch handling to base class
327 lines
13 KiB
C++
327 lines
13 KiB
C++
/// @file RC_Channel.h
|
|
/// @brief RC_Channel manager, with EEPROM-backed storage of constants.
|
|
#pragma once
|
|
|
|
#include <AP_Common/AP_Common.h>
|
|
#include <AP_Param/AP_Param.h>
|
|
|
|
#define RC_CHANNEL_TYPE_ANGLE 0
|
|
#define RC_CHANNEL_TYPE_RANGE 1
|
|
|
|
#define NUM_RC_CHANNELS 16
|
|
|
|
/// @class RC_Channel
|
|
/// @brief Object managing one RC channel
|
|
class RC_Channel {
|
|
public:
|
|
friend class SRV_Channels;
|
|
friend class RC_Channels;
|
|
// Constructor
|
|
RC_Channel(void);
|
|
|
|
// used to get min/max/trim limit value based on _reverse
|
|
enum LimitValue {
|
|
RC_CHANNEL_LIMIT_TRIM,
|
|
RC_CHANNEL_LIMIT_MIN,
|
|
RC_CHANNEL_LIMIT_MAX
|
|
};
|
|
|
|
enum InputIgnore {
|
|
RC_IGNORE_RECEIVER = (1 << 0), // RC reciever modules
|
|
RC_IGNORE_OVERRIDES = (1 << 1), // MAVLink overrides
|
|
};
|
|
|
|
// setup the control preferences
|
|
void set_range(uint16_t high);
|
|
void set_angle(uint16_t angle);
|
|
bool get_reverse(void) const;
|
|
void set_default_dead_zone(int16_t dzone);
|
|
uint16_t get_dead_zone(void) const { return dead_zone; }
|
|
|
|
// get the center stick position expressed as a control_in value
|
|
int16_t get_control_mid() const;
|
|
|
|
// read input from hal.rcin - create a control_in value
|
|
bool update(void);
|
|
void recompute_pwm_no_deadzone();
|
|
|
|
// calculate an angle given dead_zone and trim. This is used by the quadplane code
|
|
// for hover throttle
|
|
int16_t pwm_to_angle_dz_trim(uint16_t dead_zone, uint16_t trim);
|
|
|
|
/*
|
|
return a normalised input for a channel, in range -1 to 1,
|
|
centered around the channel trim. Ignore deadzone.
|
|
*/
|
|
float norm_input();
|
|
|
|
/*
|
|
return a normalised input for a channel, in range -1 to 1,
|
|
centered around the channel trim. Take into account the deadzone
|
|
*/
|
|
float norm_input_dz();
|
|
|
|
uint8_t percent_input();
|
|
int16_t pwm_to_range();
|
|
int16_t pwm_to_range_dz(uint16_t dead_zone);
|
|
|
|
static const struct AP_Param::GroupInfo var_info[];
|
|
|
|
// return true if input is within deadzone of trim
|
|
bool in_trim_dz();
|
|
|
|
int16_t get_radio_in() const { return radio_in;}
|
|
void set_radio_in(int16_t val) {radio_in = val;}
|
|
|
|
int16_t get_control_in() const { return control_in;}
|
|
void set_control_in(int16_t val) { control_in = val;}
|
|
|
|
void clear_override();
|
|
void set_override(const uint16_t v, const uint32_t timestamp_us=0);
|
|
bool has_override() const;
|
|
|
|
// get control input with zero deadzone
|
|
int16_t get_control_in_zero_dz(void);
|
|
|
|
int16_t get_radio_min() const {return radio_min.get();}
|
|
void set_radio_min(int16_t val) { radio_min = val;}
|
|
|
|
int16_t get_radio_max() const {return radio_max.get();}
|
|
void set_radio_max(int16_t val) {radio_max = val;}
|
|
|
|
int16_t get_radio_trim() const { return radio_trim.get();}
|
|
void set_radio_trim(int16_t val) { radio_trim.set(val);}
|
|
void save_radio_trim() { radio_trim.save();}
|
|
|
|
void set_and_save_trim() { radio_trim.set_and_save_ifchanged(radio_in);}
|
|
|
|
// set and save trim if changed
|
|
void set_and_save_radio_trim(int16_t val) { radio_trim.set_and_save_ifchanged(val);}
|
|
|
|
bool min_max_configured() const;
|
|
|
|
AP_Int16 option; // e.g. activate EPM gripper / enable fence
|
|
|
|
// auxillary switch support:
|
|
void init_aux();
|
|
void read_aux();
|
|
|
|
// Aux Switch enumeration
|
|
enum aux_func {
|
|
DO_NOTHING = 0, // aux switch disabled
|
|
FLIP = 2, // flip
|
|
SIMPLE_MODE = 3, // change to simple mode
|
|
RTL = 4, // change to RTL flight mode
|
|
SAVE_TRIM = 5, // save current position as level
|
|
SAVE_WP = 7, // save mission waypoint or RTL if in auto mode
|
|
CAMERA_TRIGGER = 9, // trigger camera servo or relay
|
|
RANGEFINDER = 10, // allow enabling or disabling rangefinder in flight which helps avoid surface tracking when you are far above the ground
|
|
FENCE = 11, // allow enabling or disabling fence in flight
|
|
RESETTOARMEDYAW = 12, // UNUSED
|
|
SUPERSIMPLE_MODE = 13, // change to simple mode in middle, super simple at top
|
|
ACRO_TRAINER = 14, // low = disabled, middle = leveled, high = leveled and limited
|
|
SPRAYER = 15, // enable/disable the crop sprayer
|
|
AUTO = 16, // change to auto flight mode
|
|
AUTOTUNE = 17, // auto tune
|
|
LAND = 18, // change to LAND flight mode
|
|
GRIPPER = 19, // Operate cargo grippers low=off, middle=neutral, high=on
|
|
PARACHUTE_ENABLE = 21, // Parachute enable/disable
|
|
PARACHUTE_RELEASE = 22, // Parachute release
|
|
PARACHUTE_3POS = 23, // Parachute disable, enable, release with 3 position switch
|
|
MISSION_RESET = 24, // Reset auto mission to start from first command
|
|
ATTCON_FEEDFWD = 25, // enable/disable the roll and pitch rate feed forward
|
|
ATTCON_ACCEL_LIM = 26, // enable/disable the roll, pitch and yaw accel limiting
|
|
RETRACT_MOUNT = 27, // Retract Mount
|
|
RELAY = 28, // Relay pin on/off (only supports first relay)
|
|
LANDING_GEAR = 29, // Landing gear controller
|
|
LOST_COPTER_SOUND = 30, // Play lost copter sound
|
|
MOTOR_ESTOP = 31, // Emergency Stop Switch
|
|
MOTOR_INTERLOCK = 32, // Motor On/Off switch
|
|
BRAKE = 33, // Brake flight mode
|
|
RELAY2 = 34, // Relay2 pin on/off (in Mission planner set RC8_OPTION = 34)
|
|
RELAY3 = 35, // Relay3 pin on/off (in Mission planner set RC9_OPTION = 35)
|
|
RELAY4 = 36, // Relay4 pin on/off (in Mission planner set RC10_OPTION = 36)
|
|
THROW = 37, // change to THROW flight mode
|
|
AVOID_ADSB = 38, // enable AP_Avoidance library
|
|
PRECISION_LOITER = 39, // enable precision loiter
|
|
AVOID_PROXIMITY = 40, // enable object avoidance using proximity sensors (ie. horizontal lidar)
|
|
ARMDISARM = 41, // arm or disarm vehicle
|
|
SMART_RTL = 42, // change to SmartRTL flight mode
|
|
INVERTED = 43, // enable inverted flight
|
|
WINCH_ENABLE = 44, // winch enable/disable
|
|
WINCH_CONTROL = 45, // winch control
|
|
RC_OVERRIDE_ENABLE = 46, // enable RC Override
|
|
USER_FUNC1 = 47, // user function #1
|
|
USER_FUNC2 = 48, // user function #2
|
|
USER_FUNC3 = 49, // user function #3
|
|
LEARN_CRUISE = 50, // learn cruise throttle (Rover)
|
|
MANUAL = 51, // manual mode
|
|
ACRO = 52, // acro mode
|
|
STEERING = 53, // steering mode
|
|
HOLD = 54, // hold mode
|
|
GUIDED = 55, // guided mode
|
|
LOITER = 56, // loiter mode
|
|
FOLLOW = 57, // follow mode
|
|
// if you add something here, make sure to update the documentation of the parameter in RC_Channel.cpp!
|
|
// also, if you add an option >255, you will need to fix duplicate_options_exist
|
|
};
|
|
typedef enum aux_func aux_func_t;
|
|
|
|
protected:
|
|
|
|
// auxillary switch handling:
|
|
enum aux_switch_pos {
|
|
LOW, // indicates auxiliary switch is in the low position (pwm <1200)
|
|
MIDDLE, // indicates auxiliary switch is in the middle position (pwm >1200, <1800)
|
|
HIGH // indicates auxiliary switch is in the high position (pwm >1800)
|
|
};
|
|
|
|
typedef enum aux_switch_pos aux_switch_pos_t;
|
|
|
|
virtual void init_aux_function(aux_func_t ch_option, aux_switch_pos_t);
|
|
virtual void do_aux_function(aux_func_t ch_option, aux_switch_pos_t);
|
|
|
|
void do_aux_function_relay(uint8_t relay, bool val);
|
|
void do_aux_function_camera_trigger(const aux_switch_pos_t ch_flag);
|
|
|
|
typedef int8_t modeswitch_pos_t;
|
|
virtual void mode_switch_changed(modeswitch_pos_t new_pos) {
|
|
// no action by default (e.g. Tracker, Sub, who do their own thing)
|
|
};
|
|
|
|
|
|
private:
|
|
|
|
// pwm is stored here
|
|
int16_t radio_in;
|
|
|
|
// value generated from PWM normalised to configured scale
|
|
int16_t control_in;
|
|
|
|
AP_Int16 radio_min;
|
|
AP_Int16 radio_trim;
|
|
AP_Int16 radio_max;
|
|
|
|
AP_Int8 reversed;
|
|
AP_Int16 dead_zone;
|
|
|
|
uint8_t type_in;
|
|
int16_t high_in;
|
|
|
|
// the input channel this corresponds to
|
|
uint8_t ch_in;
|
|
|
|
// overrides
|
|
uint16_t override_value;
|
|
uint32_t last_override_time;
|
|
|
|
// bits set when channel has been identified as configured
|
|
static uint32_t configured_mask;
|
|
|
|
int16_t pwm_to_angle();
|
|
int16_t pwm_to_angle_dz(uint16_t dead_zone);
|
|
|
|
// pwm value above which the option will be invoked:
|
|
static const uint16_t AUX_PWM_TRIGGER_HIGH = 1800;
|
|
// pwm value below which the option will be disabled:
|
|
static const uint16_t AUX_PWM_TRIGGER_LOW = 1200;
|
|
aux_switch_pos_t read_3pos_switch() const;
|
|
|
|
//Documentation of Aux Switch Flags:
|
|
// 0 is low or false, 1 is center or true, 2 is high
|
|
// pairs of bits in old_switch_positions give the old switch position for an RC input.
|
|
static uint32_t old_switch_positions;
|
|
|
|
aux_switch_pos_t old_switch_position() const {
|
|
return (aux_switch_pos_t)((old_switch_positions >> (ch_in*2)) & 0x3);
|
|
}
|
|
void set_old_switch_position(const RC_Channel::aux_switch_pos_t value) {
|
|
old_switch_positions &= ~(0x3 << (ch_in*2));
|
|
old_switch_positions |= (value << (ch_in*2));
|
|
}
|
|
|
|
// Structure used to detect changes in the flight mode control switch
|
|
// static since we should only ever have one mode switch!
|
|
typedef struct {
|
|
modeswitch_pos_t debounced_position; // currently used position
|
|
modeswitch_pos_t last_position; // position in previous iteration
|
|
uint32_t last_edge_time_ms; // system time that position was last changed
|
|
} modeswitch_state_t;
|
|
static modeswitch_state_t mode_switch_state;
|
|
|
|
// de-bounce counters
|
|
typedef struct {
|
|
uint8_t count;
|
|
uint8_t new_position;
|
|
} debounce_state_t;
|
|
debounce_state_t debounce;
|
|
|
|
void reset_mode_switch();
|
|
void read_mode_switch();
|
|
};
|
|
|
|
|
|
/*
|
|
class RC_Channels. Hold the full set of RC_Channel objects
|
|
*/
|
|
class RC_Channels {
|
|
public:
|
|
friend class SRV_Channels;
|
|
friend class RC_Channel;
|
|
// constructor
|
|
RC_Channels(void);
|
|
|
|
void init(void);
|
|
|
|
// get singleton instance
|
|
static RC_Channels *get_singleton() {
|
|
return _singleton;
|
|
}
|
|
|
|
static const struct AP_Param::GroupInfo var_info[];
|
|
|
|
virtual RC_Channel *channel(uint8_t chan) = 0;
|
|
|
|
uint8_t get_radio_in(uint16_t *chans, const uint8_t num_channels); // reads a block of chanel radio_in values starting from channel 0
|
|
// returns the number of valid channels
|
|
|
|
static uint8_t get_valid_channel_count(void); // returns the number of valid channels in the last read
|
|
static int16_t get_receiver_rssi(void); // returns [0, 255] for receiver RSSI (0 is no link) if present, otherwise -1
|
|
bool read_input(void); // returns true if new input has been read in
|
|
static void clear_overrides(void); // clears any active overrides
|
|
static bool receiver_bind(const int dsmMode); // puts the reciever in bind mode if present, returns true if success
|
|
static void set_override(const uint8_t chan, const int16_t value, const uint32_t timestamp_ms = 0); // set a channels override value
|
|
static bool has_active_overrides(void); // returns true if there are overrides applied that are valid
|
|
|
|
class RC_Channel *find_channel_for_option(const RC_Channel::aux_func_t option);
|
|
bool duplicate_options_exist();
|
|
|
|
void init_aux_all();
|
|
virtual void read_aux_all();
|
|
|
|
// mode switch handling
|
|
void reset_mode_switch();
|
|
virtual void read_mode_switch();
|
|
|
|
virtual bool in_rc_failsafe() const = 0;
|
|
|
|
private:
|
|
static RC_Channels *_singleton;
|
|
// this static arrangement is to avoid static pointers in AP_Param tables
|
|
static RC_Channel *channels;
|
|
|
|
static bool has_new_overrides;
|
|
static AP_Float *override_timeout;
|
|
static AP_Int32 *options;
|
|
RC_Channel obj_channels[NUM_RC_CHANNELS];
|
|
AP_Float _override_timeout;
|
|
AP_Int32 _options;
|
|
|
|
// flight_mode_channel_number must be overridden:
|
|
virtual int8_t flight_mode_channel_number() const = 0;
|
|
RC_Channel *flight_mode_channel();
|
|
|
|
};
|
|
|
|
RC_Channels &rc();
|