/* 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 <http://www.gnu.org/licenses/>. */ #pragma once #include <AP_HAL/Semaphores.h> #include <AP_HAL/utility/RingBuffer.h> #include <AP_Math/AP_Math.h> #include <GCS_MAVLink/GCS_MAVLink.h> #include <AP_Common/Location.h> #define TELEM_PAYLOAD_STATUS_CAPACITY 5 // size of the message buffer queue (max number of messages waiting to be sent) // for fair scheduler #define TELEM_TIME_SLOT_MAX 20 //#define TELEM_DEBUG class AP_RCTelemetry { public: AP_RCTelemetry(uint8_t time_slots) : _time_slots(time_slots) {} virtual ~AP_RCTelemetry() {}; /* Do not allow copies */ CLASS_NO_COPY(AP_RCTelemetry); // add statustext message to message queue virtual void queue_message(MAV_SEVERITY severity, const char *text); // scheduler entry helpers void enable_scheduler_entry(const uint8_t slot) { if (slot >= TELEM_TIME_SLOT_MAX) { return; } BIT_CLEAR(_disabled_scheduler_entries_bitmask, slot); } void disable_scheduler_entry(const uint8_t slot) { if (slot >= TELEM_TIME_SLOT_MAX) { return; } BIT_SET(_disabled_scheduler_entries_bitmask, slot); } void set_scheduler_entry_min_period(const uint8_t slot, const uint32_t min_period_ms) { if (slot >= TELEM_TIME_SLOT_MAX) { return; } _scheduler.packet_min_period[slot] = min_period_ms; } bool is_scheduler_entry_enabled(const uint8_t slot) const { if (slot >= TELEM_TIME_SLOT_MAX) { return false; } return !BIT_IS_SET(_disabled_scheduler_entries_bitmask, slot); } // each derived class might provide a way to reset telemetry rates to default virtual void reset_scheduler_entry_min_periods() {} // update error mask of sensors and subsystems. The mask uses the // MAV_SYS_STATUS_* values from mavlink. If a bit is set then it // indicates that the sensor or subsystem is present but not // functioning correctly uint32_t sensor_status_flags() const; uint16_t get_avg_packet_rate() const { return _scheduler.avg_packet_rate; } uint16_t get_max_packet_rate() const { return _scheduler.max_packet_rate; } static float get_vspeed_ms(void); static float get_nav_alt_m(Location::AltFrame frame = Location::AltFrame::ABSOLUTE); protected: uint8_t run_wfq_scheduler(const bool use_shaper = true); // process a specific entry bool process_scheduler_entry(const uint8_t slot ); // set an entry in the scheduler table void set_scheduler_entry(uint8_t slot, uint32_t weight, uint32_t min_period_ms) { if (slot >= TELEM_TIME_SLOT_MAX) { return; } _scheduler.packet_weight[slot] = weight; _scheduler.packet_min_period[slot] = min_period_ms; } // add an entry to the scheduler table void add_scheduler_entry(uint32_t weight, uint32_t min_period_ms) { if (_time_slots >= TELEM_TIME_SLOT_MAX) { return; } set_scheduler_entry(_time_slots++, weight, min_period_ms); } // setup ready for passthrough operation virtual bool init(void); uint8_t _time_slots; struct { uint32_t last_poll_timer; uint32_t avg_packet_counter; uint32_t packet_timer[TELEM_TIME_SLOT_MAX]; uint32_t packet_weight[TELEM_TIME_SLOT_MAX]; uint32_t packet_min_period[TELEM_TIME_SLOT_MAX]; uint16_t avg_packet_rate; uint16_t max_packet_rate; #ifdef TELEM_DEBUG uint8_t packet_rate[TELEM_TIME_SLOT_MAX]; #endif } _scheduler; struct { HAL_Semaphore sem; ObjectBuffer<mavlink_statustext_t> queue{TELEM_PAYLOAD_STATUS_CAPACITY}; mavlink_statustext_t next; bool available; } _statustext; private: uint32_t check_sensor_status_timer; uint32_t check_ekf_status_timer; uint32_t _disabled_scheduler_entries_bitmask; // passthrough WFQ scheduler virtual void setup_wfq_scheduler() = 0; virtual bool get_next_msg_chunk(void) { return false; } virtual bool is_packet_ready(uint8_t idx, bool queue_empty) { return true; } virtual void process_packet(uint8_t idx) = 0; virtual void adjust_packet_weight(bool queue_empty) {}; bool check_scheduler_entry_time_constraints(const uint32_t now, uint8_t slot, const bool use_shaper) const { if (!use_shaper) { return true; } return ((now - _scheduler.packet_timer[slot]) >= _scheduler.packet_min_period[slot]); } void update_avg_packet_rate(); void update_max_packet_rate() { _scheduler.max_packet_rate = MAX(_scheduler.avg_packet_rate, _scheduler.max_packet_rate); } // methods to convert flight controller data to FrSky SPort Passthrough (OpenTX) format void check_sensor_status_flags(void); void check_ekf_status(void); };