/*
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
#pragma once
#include "AP_Vehicle_config.h"
#if AP_VEHICLE_ENABLED
/*
this header holds a parameter structure for each vehicle type for
parameters needed by multiple libraries
*/
#include "ModeReason.h" // reasons can't be defined in this header due to circular loops
#include
#include
#include
#include
#include // board configuration library
#include
#include
#include
#include
#include
#include
#include
#include // Notify library
#include
#include
#include // APM relay
#include // RSSI Library
#include
#include // Serial manager library
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
class AP_DDS_Client;
class AP_Vehicle : public AP_HAL::HAL::Callbacks {
public:
AP_Vehicle() {
if (_singleton) {
AP_HAL::panic("Too many Vehicles");
}
AP_Param::setup_object_defaults(this, var_info);
_singleton = this;
}
/* Do not allow copies */
CLASS_NO_COPY(AP_Vehicle);
static AP_Vehicle *get_singleton();
// setup() is called once during vehicle startup to initialise the
// vehicle object and the objects it contains. The
// AP_HAL_MAIN_CALLBACKS pragma creates a main(...) function
// referencing an object containing setup() and loop() functions.
// A vehicle is not expected to override setup(), but
// subclass-specific initialisation can be done in init_ardupilot
// which is called from setup().
void setup(void) override final;
// HAL::Callbacks implementation.
void loop() override final;
// set_mode *must* set control_mode_reason
virtual bool set_mode(const uint8_t new_mode, const ModeReason reason) = 0;
virtual uint8_t get_mode() const = 0;
ModeReason get_control_mode_reason() const {
return control_mode_reason;
}
virtual bool current_mode_requires_mission() const { return false; }
// perform any notifications required to indicate a mode change
// failed due to a bad mode number being supplied. This can
// happen for many reasons - bad mavlink packet and bad mode
// parameters for example.
void notify_no_such_mode(uint8_t mode_number);
void get_common_scheduler_tasks(const AP_Scheduler::Task*& tasks, uint8_t& num_tasks);
// implementations *MUST* fill in all passed-in fields or we get
// Valgrind errors
virtual void get_scheduler_tasks(const AP_Scheduler::Task *&tasks, uint8_t &task_count, uint32_t &log_bit) = 0;
/*
set the "likely flying" flag. This is not guaranteed to be
accurate, but is the vehicle codes best guess as to the whether
the vehicle is currently flying
*/
void set_likely_flying(bool b) {
if (b && !likely_flying) {
_last_flying_ms = AP_HAL::millis();
}
likely_flying = b;
}
/*
get the likely flying status. Returns true if the vehicle code
thinks we are flying at the moment. Not guaranteed to be
accurate
*/
bool get_likely_flying(void) const {
return likely_flying;
}
/*
return time in milliseconds since likely_flying was set
true. Returns zero if likely_flying is currently false
*/
uint32_t get_time_flying_ms(void) const {
if (!likely_flying) {
return 0;
}
return AP_HAL::millis() - _last_flying_ms;
}
// returns true if the vehicle has crashed
virtual bool is_crashed() const;
#if AP_EXTERNAL_CONTROL_ENABLED
// Method to control vehicle position for use by external control
virtual bool set_target_location(const Location& target_loc) { return false; }
#endif // AP_EXTERNAL_CONTROL_ENABLED
#if AP_SCRIPTING_ENABLED
/*
methods to control vehicle for use by scripting
*/
virtual bool start_takeoff(float alt) { return false; }
virtual bool set_target_pos_NED(const Vector3f& target_pos, bool use_yaw, float yaw_deg, bool use_yaw_rate, float yaw_rate_degs, bool yaw_relative, bool terrain_alt) { return false; }
virtual bool set_target_posvel_NED(const Vector3f& target_pos, const Vector3f& target_vel) { return false; }
virtual bool set_target_posvelaccel_NED(const Vector3f& target_pos, const Vector3f& target_vel, const Vector3f& target_accel, bool use_yaw, float yaw_deg, bool use_yaw_rate, float yaw_rate_degs, bool yaw_relative) { return false; }
virtual bool set_target_velocity_NED(const Vector3f& vel_ned) { return false; }
virtual bool set_target_velaccel_NED(const Vector3f& target_vel, const Vector3f& target_accel, bool use_yaw, float yaw_deg, bool use_yaw_rate, float yaw_rate_degs, bool yaw_relative) { return false; }
virtual bool set_target_angle_and_climbrate(float roll_deg, float pitch_deg, float yaw_deg, float climb_rate_ms, bool use_yaw_rate, float yaw_rate_degs) { return false; }
// command throttle percentage and roll, pitch, yaw target
// rates. For use with scripting controllers
virtual void set_target_throttle_rate_rpy(float throttle_pct, float roll_rate_dps, float pitch_rate_dps, float yaw_rate_dps) {}
virtual void set_rudder_offset(float rudder_pct, bool run_yaw_rate_controller) {}
virtual bool nav_scripting_enable(uint8_t mode) {return false;}
// get target location (for use by scripting)
virtual bool get_target_location(Location& target_loc) { return false; }
virtual bool update_target_location(const Location &old_loc, const Location &new_loc) { return false; }
// circle mode controls (only used by scripting with Copter)
virtual bool get_circle_radius(float &radius_m) { return false; }
virtual bool set_circle_rate(float rate_dps) { return false; }
// get or set steering and throttle (-1 to +1) (for use by scripting with Rover)
virtual bool set_steering_and_throttle(float steering, float throttle) { return false; }
virtual bool get_steering_and_throttle(float& steering, float& throttle) { return false; }
// set turn rate in deg/sec and speed in meters/sec (for use by scripting with Rover)
virtual bool set_desired_turn_rate_and_speed(float turn_rate, float speed) { return false; }
// set auto mode speed in meters/sec (for use by scripting with Copter/Rover)
virtual bool set_desired_speed(float speed) { return false; }
// support for NAV_SCRIPT_TIME mission command
virtual bool nav_script_time(uint16_t &id, uint8_t &cmd, float &arg1, float &arg2, int16_t &arg3, int16_t &arg4) { return false; }
virtual void nav_script_time_done(uint16_t id) {}
// allow for VTOL velocity matching of a target
virtual bool set_velocity_match(const Vector2f &velocity) { return false; }
// returns true if the EKF failsafe has triggered
virtual bool has_ekf_failsafed() const { return false; }
// allow for landing descent rate to be overridden by a script, may be -ve to climb
virtual bool set_land_descent_rate(float descent_rate) { return false; }
// control outputs enumeration
enum class ControlOutput {
Roll = 1,
Pitch = 2,
Throttle = 3,
Yaw = 4,
Lateral = 5,
MainSail = 6,
WingSail = 7,
Walking_Height = 8,
Last_ControlOutput // place new values before this
};
// get control output (for use in scripting)
// returns true on success and control_value is set to a value in the range -1 to +1
virtual bool get_control_output(AP_Vehicle::ControlOutput control_output, float &control_value) { return false; }
#endif // AP_SCRIPTING_ENABLED
// returns true if vehicle is in the process of landing
virtual bool is_landing() const { return false; }
// returns true if vehicle is in the process of taking off
virtual bool is_taking_off() const { return false; }
// zeroing the RC outputs can prevent unwanted motor movement:
virtual bool should_zero_rc_outputs_on_reboot() const { return false; }
// reboot the vehicle in an orderly manner, doing various cleanups
// and flashing LEDs as appropriate
void reboot(bool hold_in_bootloader);
/*
get the distance to next wp in meters
return false if failed or n/a
*/
virtual bool get_wp_distance_m(float &distance) const { return false; }
/*
get the current wp bearing in degrees
return false if failed or n/a
*/
virtual bool get_wp_bearing_deg(float &bearing) const { return false; }
/*
get the current wp crosstrack error in meters
return false if failed or n/a
*/
virtual bool get_wp_crosstrack_error_m(float &xtrack_error) const { return false; }
#if HAL_WITH_FRSKY_TELEM_BIDIRECTIONAL
AP_Frsky_Parameters frsky_parameters;
#endif
/*
Returns the pan and tilt for use by onvif camera in scripting
*/
virtual bool get_pan_tilt_norm(float &pan_norm, float &tilt_norm) const { return false; }
// Returns roll and pitch for OSD Horizon, Plane overrides to correct for VTOL view and fixed wing PTCH_TRIM_DEG
virtual void get_osd_roll_pitch_rad(float &roll, float &pitch) const;
/*
get the target earth-frame angular velocities in rad/s (Z-axis component used by some gimbals)
*/
virtual bool get_rate_ef_targets(Vector3f& rate_ef_targets) const { return false; }
protected:
virtual void init_ardupilot() = 0;
virtual void load_parameters() = 0;
virtual void set_control_channels() {}
// board specific config
AP_BoardConfig BoardConfig;
#if HAL_CANMANAGER_ENABLED
// board specific config for CAN bus
AP_CANManager can_mgr;
#endif
// main loop scheduler
AP_Scheduler scheduler;
// IMU variables
// Integration time; time last loop took to run
float G_Dt;
// sensor drivers
#if AP_GPS_ENABLED
AP_GPS gps;
#endif
AP_Baro barometer;
Compass compass;
AP_InertialSensor ins;
#if HAL_BUTTON_ENABLED
AP_Button button;
#endif
RangeFinder rangefinder;
#if AP_RSSI_ENABLED
AP_RSSI rssi;
#endif
#if HAL_RUNCAM_ENABLED
AP_RunCam runcam;
#endif
#if HAL_GYROFFT_ENABLED
AP_GyroFFT gyro_fft;
#endif
#if AP_VIDEOTX_ENABLED
AP_VideoTX vtx;
#endif
AP_SerialManager serial_manager;
#if AP_RELAY_ENABLED
AP_Relay relay;
#endif
#if AP_SERVORELAYEVENTS_ENABLED
AP_ServoRelayEvents ServoRelayEvents;
#endif
// notification object for LEDs, buzzers etc (parameter set to
// false disables external leds)
AP_Notify notify;
// Inertial Navigation EKF
AP_AHRS ahrs;
#if HAL_HOTT_TELEM_ENABLED
AP_Hott_Telem hott_telem;
#endif
#if HAL_VISUALODOM_ENABLED
AP_VisualOdom visual_odom;
#endif
#if HAL_WITH_ESC_TELEM
AP_ESC_Telem esc_telem;
#endif
#if AP_OPENDRONEID_ENABLED
AP_OpenDroneID opendroneid;
#endif
#if HAL_MSP_ENABLED
AP_MSP msp;
#endif
#if HAL_GENERATOR_ENABLED
AP_Generator generator;
#endif
#if HAL_EXTERNAL_AHRS_ENABLED
AP_ExternalAHRS externalAHRS;
#endif
#if AP_SMARTAUDIO_ENABLED
AP_SmartAudio smartaudio;
#endif
#if AP_TRAMP_ENABLED
AP_Tramp tramp;
#endif
#if AP_NETWORKING_ENABLED
AP_Networking networking;
#endif
#if HAL_EFI_ENABLED
// EFI Engine Monitor
AP_EFI efi;
#endif
#if AP_AIRSPEED_ENABLED
AP_Airspeed airspeed;
#endif
#if AP_AIS_ENABLED
// Automatic Identification System - for tracking sea-going vehicles
AP_AIS ais;
#endif
#if HAL_NMEA_OUTPUT_ENABLED
AP_NMEA_Output nmea;
#endif
#if AP_KDECAN_ENABLED
AP_KDECAN kdecan;
#endif
#if AP_FENCE_ENABLED
AC_Fence fence;
#endif
#if AP_TEMPERATURE_SENSOR_ENABLED
AP_TemperatureSensor temperature_sensor;
#endif
static const struct AP_Param::GroupInfo var_info[];
static const struct AP_Scheduler::Task scheduler_tasks[];
#if OSD_ENABLED
void publish_osd_info();
#endif
#if HAL_INS_ACCELCAL_ENABLED
// update accel calibration
void accel_cal_update();
#endif
// call the arming library's update function
void update_arming();
// check for motor noise at a particular frequency
void check_motor_noise();
ModeReason control_mode_reason = ModeReason::UNKNOWN;
#if AP_SIM_ENABLED
SITL::SIM sitl;
#endif
#if AP_DDS_ENABLED
// Declare the dds client for communication with ROS2 and DDS(common for all vehicles)
AP_DDS_Client *dds_client;
bool init_dds_client() WARN_IF_UNUSED;
#endif
// Check if this mode can be entered from the GCS
bool block_GCS_mode_change(uint8_t mode_num, const uint8_t *mode_list, uint8_t mode_list_length) const;
private:
// delay() callback that processing MAVLink packets
static void scheduler_delay_callback();
// if there's been a watchdog reset, notify the world via a
// statustext:
void send_watchdog_reset_statustext();
// update the harmonic notch for throttle based notch
void update_throttle_notch(AP_InertialSensor::HarmonicNotch ¬ch);
// update the harmonic notch
void update_dynamic_notch(AP_InertialSensor::HarmonicNotch ¬ch);
// run notch update at either loop rate or 200Hz
void update_dynamic_notch_at_specified_rate();
// decimation for 1Hz update
uint8_t one_Hz_counter;
void one_Hz_update();
bool likely_flying; // true if vehicle is probably flying
uint32_t _last_flying_ms; // time when likely_flying last went true
uint32_t _last_notch_update_ms[HAL_INS_NUM_HARMONIC_NOTCH_FILTERS]; // last time update_dynamic_notch() was run
static AP_Vehicle *_singleton;
#if HAL_GYROFFT_ENABLED && HAL_WITH_ESC_TELEM
LowPassFilterFloat esc_noise[ESC_TELEM_MAX_ESCS];
uint32_t last_motor_noise_ms;
#endif
bool done_safety_init;
uint32_t _last_internal_errors; // backup of AP_InternalError::internal_errors bitmask
AP_CustomRotations custom_rotations;
#if AP_FILTER_ENABLED
AP_Filters filters;
#endif
// Bitmask of modes to disable from gcs
AP_Int32 flight_mode_GCS_block;
};
namespace AP {
AP_Vehicle *vehicle();
};
extern const AP_HAL::HAL& hal;
extern const AP_Param::Info vehicle_var_info[];
#include "AP_Vehicle_Type.h"
#endif // AP_VEHICLE_ENABLED