/* * This file 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 file 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 . * * Code by Andrew Tridgell and Siddharth Bharat Purohit */ #include "AP_RCProtocol.h" #include "AP_RCProtocol_PPMSum.h" #include "AP_RCProtocol_DSM.h" #include "AP_RCProtocol_SBUS.h" #include "AP_RCProtocol_SBUS_NI.h" #include "AP_RCProtocol_SUMD.h" #include "AP_RCProtocol_SRXL.h" #include "AP_RCProtocol_ST24.h" // singleton AP_RCProtocol *AP_RCProtocol::instance; void AP_RCProtocol::init() { backend[AP_RCProtocol::PPM] = new AP_RCProtocol_PPMSum(*this); backend[AP_RCProtocol::SBUS] = new AP_RCProtocol_SBUS(*this); backend[AP_RCProtocol::SBUS_NI] = new AP_RCProtocol_SBUS_NI(*this); backend[AP_RCProtocol::DSM] = new AP_RCProtocol_DSM(*this); backend[AP_RCProtocol::SUMD] = new AP_RCProtocol_SUMD(*this); backend[AP_RCProtocol::SRXL] = new AP_RCProtocol_SRXL(*this); backend[AP_RCProtocol::ST24] = new AP_RCProtocol_ST24(*this); } void AP_RCProtocol::process_pulse(uint32_t width_s0, uint32_t width_s1) { if (_detected_protocol != AP_RCProtocol::NONE && _detected_with_bytes) { // we're using byte inputs, discard pulses return; } uint32_t now = AP_HAL::millis(); // first try current protocol if (_detected_protocol != AP_RCProtocol::NONE && now - _last_input_ms < 200) { backend[_detected_protocol]->process_pulse(width_s0, width_s1); if (backend[_detected_protocol]->new_input()) { _new_input = true; _last_input_ms = AP_HAL::millis(); } return; } // otherwise scan all protocols for (uint8_t i = 0; i < AP_RCProtocol::NONE; i++) { if (backend[i] != nullptr) { backend[i]->process_pulse(width_s0, width_s1); if (backend[i]->new_input()) { _new_input = true; _detected_protocol = (enum AP_RCProtocol::rcprotocol_t)i; _last_input_ms = AP_HAL::millis(); _detected_with_bytes = false; } } } } void AP_RCProtocol::process_byte(uint8_t byte) { if (_detected_protocol != AP_RCProtocol::NONE && !_detected_with_bytes) { // we're using pulse inputs, discard bytes return; } uint32_t now = AP_HAL::millis(); // first try current protocol if (_detected_protocol != AP_RCProtocol::NONE && now - _last_input_ms < 200) { backend[_detected_protocol]->process_byte(byte); if (backend[_detected_protocol]->new_input()) { _new_input = true; _last_input_ms = AP_HAL::millis(); } return; } // otherwise scan all protocols for (uint8_t i = 0; i < AP_RCProtocol::NONE; i++) { if (backend[i] != nullptr) { backend[i]->process_byte(byte); if (backend[i]->new_input()) { _new_input = true; _detected_protocol = (enum AP_RCProtocol::rcprotocol_t)i; _last_input_ms = AP_HAL::millis(); _detected_with_bytes = true; } } } } bool AP_RCProtocol::new_input() { bool ret = _new_input; _new_input = false; // run update function on backends for (uint8_t i = 0; i < AP_RCProtocol::NONE; i++) { if (backend[i] != nullptr) { backend[i]->update(); } } return ret; } uint8_t AP_RCProtocol::num_channels() { if (_detected_protocol != AP_RCProtocol::NONE) { return backend[_detected_protocol]->num_channels(); } return 0; } uint16_t AP_RCProtocol::read(uint8_t chan) { if (_detected_protocol != AP_RCProtocol::NONE) { return backend[_detected_protocol]->read(chan); } return 0; } /* ask for bind start on supported receivers (eg spektrum satellite) */ void AP_RCProtocol::start_bind(void) { for (uint8_t i = 0; i < AP_RCProtocol::NONE; i++) { if (backend[i] != nullptr) { backend[i]->start_bind(); } } }