mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-05 07:28:29 -04:00
bbcc31ecec
Of particular note this removes the partial SUMH support. This has been broken as far back as 2018, as we only called add_input() if we were in the SUMD or SUMD failsafe cases, which meant SUMH didn't work. This commits to that path, and removes all the 8 bit CRC, SUMH support. This also refactors a number of things for readability (condensing white space, simplifying guards around debug code). Many of the debugs could probably be removed, but this simplifies it. This was tested against the RCProtocol example test, and everything there reports a pass.
221 lines
6.8 KiB
C++
221 lines
6.8 KiB
C++
/*
|
|
* 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_config.h"
|
|
|
|
#if AP_RCPROTOCOL_ENABLED
|
|
|
|
#include "AP_RCProtocol.h"
|
|
#include <AP_Math/AP_Math.h>
|
|
#include <RC_Channel/RC_Channel.h>
|
|
#include <AP_Vehicle/AP_Vehicle_Type.h>
|
|
#include <AP_Logger/AP_Logger.h>
|
|
#include <AP_VideoTX/AP_VideoTX_config.h>
|
|
|
|
// for video TX configuration:
|
|
#if AP_VIDEOTX_ENABLED
|
|
#include <AP_VideoTX/AP_VideoTX.h>
|
|
#include "spm_srxl.h"
|
|
#endif
|
|
|
|
|
|
|
|
AP_RCProtocol_Backend::AP_RCProtocol_Backend(AP_RCProtocol &_frontend) :
|
|
frontend(_frontend),
|
|
rc_input_count(0),
|
|
last_rc_input_count(0),
|
|
_num_channels(0)
|
|
{}
|
|
|
|
bool AP_RCProtocol_Backend::new_input()
|
|
{
|
|
bool ret = rc_input_count != last_rc_input_count;
|
|
if (ret) {
|
|
last_rc_input_count = rc_input_count;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
uint8_t AP_RCProtocol_Backend::num_channels() const
|
|
{
|
|
return _num_channels;
|
|
}
|
|
|
|
uint16_t AP_RCProtocol_Backend::read(uint8_t chan)
|
|
{
|
|
return _pwm_values[chan];
|
|
}
|
|
|
|
void AP_RCProtocol_Backend::read(uint16_t *pwm, uint8_t n)
|
|
{
|
|
if (n >= MAX_RCIN_CHANNELS) {
|
|
n = MAX_RCIN_CHANNELS;
|
|
}
|
|
memcpy(pwm, _pwm_values, n*sizeof(pwm[0]));
|
|
}
|
|
|
|
/*
|
|
provide input from a backend
|
|
*/
|
|
void AP_RCProtocol_Backend::add_input(uint8_t num_values, uint16_t *values, bool in_failsafe, int16_t _rssi, int16_t _rx_link_quality)
|
|
{
|
|
num_values = MIN(num_values, MAX_RCIN_CHANNELS);
|
|
memcpy(_pwm_values, values, num_values*sizeof(uint16_t));
|
|
_num_channels = num_values;
|
|
rc_frame_count++;
|
|
frontend.set_failsafe_active(in_failsafe);
|
|
#if !AP_RC_CHANNEL_ENABLED
|
|
// failsafed is sorted out in AP_IOMCU.cpp
|
|
in_failsafe = false;
|
|
#else
|
|
if (rc().option_is_enabled(RC_Channels::Option::IGNORE_FAILSAFE)) {
|
|
in_failsafe = false;
|
|
}
|
|
#endif
|
|
if (!in_failsafe) {
|
|
rc_input_count++;
|
|
}
|
|
rssi = _rssi;
|
|
rx_link_quality = _rx_link_quality;
|
|
}
|
|
|
|
|
|
/*
|
|
decode channels from the standard 11bit format (used by CRSF, SBUS, FPort and FPort2)
|
|
must be used on multiples of 8 channels
|
|
*/
|
|
void AP_RCProtocol_Backend::decode_11bit_channels(const uint8_t* data, uint8_t nchannels, uint16_t *values, uint16_t mult, uint16_t div, uint16_t offset)
|
|
{
|
|
#define CHANNEL_SCALE(x) ((int32_t(x) * mult) / div + offset)
|
|
while (nchannels >= 8) {
|
|
const Channels11Bit_8Chan* channels = (const Channels11Bit_8Chan*)data;
|
|
values[0] = CHANNEL_SCALE(channels->ch0);
|
|
values[1] = CHANNEL_SCALE(channels->ch1);
|
|
values[2] = CHANNEL_SCALE(channels->ch2);
|
|
values[3] = CHANNEL_SCALE(channels->ch3);
|
|
values[4] = CHANNEL_SCALE(channels->ch4);
|
|
values[5] = CHANNEL_SCALE(channels->ch5);
|
|
values[6] = CHANNEL_SCALE(channels->ch6);
|
|
values[7] = CHANNEL_SCALE(channels->ch7);
|
|
|
|
nchannels -= 8;
|
|
data += sizeof(*channels);
|
|
values += 8;
|
|
}
|
|
}
|
|
|
|
#if AP_VIDEOTX_ENABLED
|
|
// configure the video transmitter, the input values are Spektrum-oriented
|
|
void AP_RCProtocol_Backend::configure_vtx(uint8_t band, uint8_t channel, uint8_t power, uint8_t pitmode)
|
|
{
|
|
AP_VideoTX& vtx = AP::vtx();
|
|
// VTX Band (0 = Fatshark, 1 = Raceband, 2 = E, 3 = B, 4 = A)
|
|
// map to TBS band A, B, E, Race, Airwave, LoRace
|
|
switch (band) {
|
|
case VTX_BAND_FATSHARK:
|
|
vtx.set_configured_band(AP_VideoTX::VideoBand::FATSHARK);
|
|
break;
|
|
case VTX_BAND_RACEBAND:
|
|
vtx.set_configured_band(AP_VideoTX::VideoBand::RACEBAND);
|
|
break;
|
|
case VTX_BAND_E_BAND:
|
|
vtx.set_configured_band(AP_VideoTX::VideoBand::BAND_E);
|
|
break;
|
|
case VTX_BAND_B_BAND:
|
|
vtx.set_configured_band(AP_VideoTX::VideoBand::BAND_B);
|
|
break;
|
|
case VTX_BAND_A_BAND:
|
|
vtx.set_configured_band(AP_VideoTX::VideoBand::BAND_A);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
// VTX Channel (0-7)
|
|
vtx.set_configured_channel(channel);
|
|
if (pitmode) {
|
|
vtx.set_configured_options(vtx.get_options() | uint8_t(AP_VideoTX::VideoOptions::VTX_PITMODE));
|
|
} else {
|
|
vtx.set_configured_options(vtx.get_options() & ~uint8_t(AP_VideoTX::VideoOptions::VTX_PITMODE));
|
|
}
|
|
|
|
switch (power) {
|
|
case VTX_POWER_1MW_14MW:
|
|
case VTX_POWER_15MW_25MW:
|
|
vtx.set_configured_power_mw(25);
|
|
break;
|
|
case VTX_POWER_26MW_99MW:
|
|
case VTX_POWER_100MW_299MW:
|
|
vtx.set_configured_power_mw(100);
|
|
break;
|
|
case VTX_POWER_300MW_600MW:
|
|
vtx.set_configured_power_mw(400);
|
|
break;
|
|
case VTX_POWER_601_PLUS:
|
|
vtx.set_configured_power_mw(800);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
#endif // AP_VIDEOTX_ENABLED
|
|
|
|
/*
|
|
optionally log RC input data
|
|
*/
|
|
void AP_RCProtocol_Backend::log_data(AP_RCProtocol::rcprotocol_t prot, uint32_t timestamp, const uint8_t *data, uint8_t len) const
|
|
{
|
|
#if HAL_LOGGING_ENABLED
|
|
#if (CONFIG_HAL_BOARD == HAL_BOARD_SITL || CONFIG_HAL_BOARD == HAL_BOARD_LINUX)
|
|
if (&rc() == nullptr) { // allow running without RC_Channels if we are doing the examples
|
|
return;
|
|
}
|
|
#endif
|
|
if (rc().option_is_enabled(RC_Channels::Option::LOG_RAW_DATA)) {
|
|
uint32_t u32[10] {};
|
|
if (len > sizeof(u32)) {
|
|
len = sizeof(u32);
|
|
}
|
|
memcpy(u32, data, len);
|
|
// @LoggerMessage: RCDA
|
|
// @Description: Raw RC data
|
|
// @Field: TimeUS: Time since system startup
|
|
// @Field: TS: data arrival timestamp
|
|
// @Field: Prot: Protocol currently being decoded
|
|
// @Field: Len: Number of valid bytes in message
|
|
// @Field: U0: first quartet of bytes
|
|
// @Field: U1: second quartet of bytes
|
|
// @Field: U2: third quartet of bytes
|
|
// @Field: U3: fourth quartet of bytes
|
|
// @Field: U4: fifth quartet of bytes
|
|
// @Field: U5: sixth quartet of bytes
|
|
// @Field: U6: seventh quartet of bytes
|
|
// @Field: U7: eight quartet of bytes
|
|
// @Field: U8: ninth quartet of bytes
|
|
// @Field: U9: tenth quartet of bytes
|
|
AP::logger().WriteStreaming("RCDA", "TimeUS,TS,Prot,Len,U0,U1,U2,U3,U4,U5,U6,U7,U8,U9", "QIBBIIIIIIIIII",
|
|
AP_HAL::micros64(),
|
|
timestamp,
|
|
(uint8_t)prot,
|
|
len,
|
|
u32[0], u32[1], u32[2], u32[3], u32[4],
|
|
u32[5], u32[6], u32[7], u32[8], u32[9]);
|
|
}
|
|
#endif // HAL_LOGGING_ENABLED
|
|
}
|
|
|
|
#endif // AP_RCPROTOCOL_ENABLED
|