123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 |
- /*
- * 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 <http://www.gnu.org/licenses/>.
- *
- * 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_IBUS.h"
- #include "AP_RCProtocol_SBUS.h"
- #include "AP_RCProtocol_SUMD.h"
- #include "AP_RCProtocol_SRXL.h"
- #include "AP_RCProtocol_ST24.h"
- #include <AP_Math/AP_Math.h>
- void AP_RCProtocol::init()
- {
- backend[AP_RCProtocol::PPM] = new AP_RCProtocol_PPMSum(*this);
- backend[AP_RCProtocol::IBUS] = new AP_RCProtocol_IBUS(*this);
- backend[AP_RCProtocol::SBUS] = new AP_RCProtocol_SBUS(*this, true);
- backend[AP_RCProtocol::SBUS_NI] = new AP_RCProtocol_SBUS(*this, false);
- 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);
- }
- AP_RCProtocol::~AP_RCProtocol()
- {
- for (uint8_t i = 0; i < AP_RCProtocol::NONE; i++) {
- if (backend[i] != nullptr) {
- delete backend[i];
- backend[i] = nullptr;
- }
- }
- }
- void AP_RCProtocol::process_pulse(uint32_t width_s0, uint32_t width_s1)
- {
- uint32_t now = AP_HAL::millis();
- bool searching = (now - _last_input_ms >= 200);
- if (_detected_protocol != AP_RCProtocol::NONE && _detected_with_bytes && !searching) {
- // we're using byte inputs, discard pulses
- return;
- }
- // first try current protocol
- if (_detected_protocol != AP_RCProtocol::NONE && !searching) {
- backend[_detected_protocol]->process_pulse(width_s0, width_s1);
- if (backend[_detected_protocol]->new_input()) {
- _new_input = true;
- _last_input_ms = now;
- }
- return;
- }
- // otherwise scan all protocols
- for (uint8_t i = 0; i < AP_RCProtocol::NONE; i++) {
- if (_disabled_for_pulses & (1U << i)) {
- // this protocol is disabled for pulse input
- continue;
- }
- if (backend[i] != nullptr) {
- uint32_t frame_count = backend[i]->get_rc_frame_count();
- uint32_t input_count = backend[i]->get_rc_input_count();
- backend[i]->process_pulse(width_s0, width_s1);
- if (frame_count != backend[i]->get_rc_frame_count()) {
- _good_frames[i]++;
- if (requires_3_frames((rcprotocol_t)i) && _good_frames[i] < 3) {
- continue;
- }
- _new_input = (input_count != backend[i]->get_rc_input_count());
- _detected_protocol = (enum AP_RCProtocol::rcprotocol_t)i;
- memset(_good_frames, 0, sizeof(_good_frames));
- _last_input_ms = now;
- _detected_with_bytes = false;
- break;
- }
- }
- }
- }
- /*
- process an array of pulses. n must be even
- */
- void AP_RCProtocol::process_pulse_list(const uint32_t *widths, uint16_t n, bool need_swap)
- {
- if (n & 1) {
- return;
- }
- while (n) {
- uint32_t widths0 = widths[0];
- uint32_t widths1 = widths[1];
- if (need_swap) {
- uint32_t tmp = widths1;
- widths1 = widths0;
- widths0 = tmp;
- }
- widths1 -= widths0;
- process_pulse(widths0, widths1);
- widths += 2;
- n -= 2;
- }
- }
- void AP_RCProtocol::process_byte(uint8_t byte, uint32_t baudrate)
- {
- uint32_t now = AP_HAL::millis();
- bool searching = (now - _last_input_ms >= 200);
- if (_detected_protocol != AP_RCProtocol::NONE && !_detected_with_bytes && !searching) {
- // we're using pulse inputs, discard bytes
- return;
- }
- // first try current protocol
- if (_detected_protocol != AP_RCProtocol::NONE && !searching) {
- backend[_detected_protocol]->process_byte(byte, baudrate);
- if (backend[_detected_protocol]->new_input()) {
- _new_input = true;
- _last_input_ms = now;
- }
- return;
- }
- // otherwise scan all protocols
- for (uint8_t i = 0; i < AP_RCProtocol::NONE; i++) {
- if (backend[i] != nullptr) {
- uint32_t frame_count = backend[i]->get_rc_frame_count();
- uint32_t input_count = backend[i]->get_rc_input_count();
- backend[i]->process_byte(byte, baudrate);
- if (frame_count != backend[i]->get_rc_frame_count()) {
- _good_frames[i]++;
- if (requires_3_frames((rcprotocol_t)i) && _good_frames[i] < 3) {
- continue;
- }
- _new_input = (input_count != backend[i]->get_rc_input_count());
- _detected_protocol = (enum AP_RCProtocol::rcprotocol_t)i;
- memset(_good_frames, 0, sizeof(_good_frames));
- _last_input_ms = now;
- _detected_with_bytes = true;
- break;
- }
- }
- }
- }
- /*
- check for bytes from an additional uart. This is used to support RC
- protocols from SERIALn_PROTOCOL
- */
- void AP_RCProtocol::check_added_uart(void)
- {
- if (!added.uart) {
- return;
- }
- uint32_t now = AP_HAL::millis();
- bool searching = (now - _last_input_ms >= 200);
- if (!searching && !_detected_with_bytes) {
- // not using this uart
- return;
- }
- if (!added.opened) {
- added.uart->begin(added.baudrate, 128, 128);
- added.opened = true;
- if (added.baudrate == 100000) {
- // assume SBUS settings, even parity, 2 stop bits
- added.uart->configure_parity(2);
- added.uart->set_stop_bits(2);
- added.uart->set_options(added.uart->get_options() | AP_HAL::UARTDriver::OPTION_RXINV);
- } else {
- // setup for 115200 protocols
- added.uart->configure_parity(0);
- added.uart->set_stop_bits(1);
- added.uart->set_options(added.uart->get_options() & ~AP_HAL::UARTDriver::OPTION_RXINV);
- }
- added.last_baud_change_ms = AP_HAL::millis();
- }
- uint32_t n = added.uart->available();
- n = MIN(n, 255U);
- for (uint8_t i=0; i<n; i++) {
- int16_t b = added.uart->read();
- if (b >= 0) {
- process_byte(uint8_t(b), added.baudrate);
- }
- }
- if (!_detected_with_bytes) {
- if (now - added.last_baud_change_ms > 1000) {
- // flip baudrates if not detected once a second
- added.baudrate = (added.baudrate==100000)?115200:100000;
- added.opened = false;
- }
- }
- }
- void AP_RCProtocol::update()
- {
- check_added_uart();
- }
- bool AP_RCProtocol::new_input()
- {
- bool ret = _new_input;
- _new_input = false;
- // if we have an extra UART from a SERIALn_PROTOCOL then check it for data
- check_added_uart();
- // 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();
- }
- }
- }
- /*
- return protocol name
- */
- const char *AP_RCProtocol::protocol_name_from_protocol(rcprotocol_t protocol)
- {
- switch (protocol) {
- case PPM:
- return "PPM";
- case IBUS:
- return "IBUS";
- case SBUS:
- case SBUS_NI:
- return "SBUS";
- case DSM:
- return "DSM";
- case SUMD:
- return "SUMD";
- case SRXL:
- return "SRXL";
- case ST24:
- return "ST24";
- case NONE:
- break;
- }
- return nullptr;
- }
- /*
- return protocol name
- */
- const char *AP_RCProtocol::protocol_name(void) const
- {
- return protocol_name_from_protocol(_detected_protocol);
- }
- /*
- add a uart to decode
- */
- void AP_RCProtocol::add_uart(AP_HAL::UARTDriver* uart)
- {
- added.uart = uart;
- // start with DSM
- added.baudrate = 115200U;
- }
- namespace AP {
- AP_RCProtocol &RC()
- {
- static AP_RCProtocol rcprot;
- return rcprot;
- }
- };
|