Update actions to ubuntu-latest (#14114)
[betaflight.git] / src / main / rx / sbus.c
blob626c6586b46fb71c5c9be8d252c8a86b5d0967f6
1 /*
2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
8 * any later version.
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
21 #include <stdbool.h>
22 #include <stdint.h>
23 #include <stdlib.h>
25 #include "platform.h"
27 #ifdef USE_SERIALRX_SBUS
29 #include "build/debug.h"
31 #include "common/utils.h"
33 #include "drivers/time.h"
35 #include "io/serial.h"
37 #ifdef USE_TELEMETRY
38 #include "telemetry/telemetry.h"
39 #endif
41 #include "pg/rx.h"
43 #include "rx/rx.h"
44 #include "rx/sbus.h"
45 #include "rx/sbus_channels.h"
48 * Observations
50 * FrSky X8R
51 * time between frames: 6ms.
52 * time to send frame: 3ms.
54 * Futaba R6208SB/R6303SB
55 * time between frames: 11ms.
56 * time to send frame: 3ms.
59 #define SBUS_BAUDRATE 100000
60 #define SBUS_RX_REFRESH_RATE 11000
61 #define SBUS_TIME_NEEDED_PER_FRAME 3000
63 #define SBUS_FAST_BAUDRATE 200000
64 #define SBUS_FAST_RX_REFRESH_RATE 6000
66 #define SBUS_STATE_FAILSAFE (1 << 0)
67 #define SBUS_STATE_SIGNALLOSS (1 << 1)
69 #define SBUS_FRAME_SIZE (SBUS_CHANNEL_DATA_LENGTH + 2)
71 #define SBUS_FRAME_BEGIN_BYTE 0x0F
73 #if !defined(SBUS_PORT_OPTIONS)
74 #define SBUS_PORT_OPTIONS (SERIAL_STOPBITS_2 | SERIAL_PARITY_EVEN)
75 #endif
77 #define SBUS_DIGITAL_CHANNEL_MIN 173
78 #define SBUS_DIGITAL_CHANNEL_MAX 1812
80 enum {
81 DEBUG_SBUS_FRAME_FLAGS = 0,
82 DEBUG_SBUS_STATE_FLAGS,
83 DEBUG_SBUS_FRAME_TIME,
86 struct sbusFrame_s {
87 uint8_t syncByte;
88 sbusChannels_t channels;
89 /**
90 * The endByte is 0x00 on FrSky and some futaba RX's, on Some SBUS2 RX's the value indicates the telemetry byte that is sent after every 4th sbus frame.
92 * See https://github.com/cleanflight/cleanflight/issues/590#issuecomment-101027349
93 * and
94 * https://github.com/cleanflight/cleanflight/issues/590#issuecomment-101706023
96 uint8_t endByte;
97 } __attribute__ ((__packed__));
99 typedef union sbusFrame_u {
100 uint8_t bytes[SBUS_FRAME_SIZE];
101 struct sbusFrame_s frame;
102 } sbusFrame_t;
104 typedef struct sbusFrameData_s {
105 sbusFrame_t frame;
106 timeUs_t startAtUs;
107 uint8_t position;
108 bool done;
109 } sbusFrameData_t;
111 // Receive ISR callback
112 static void sbusDataReceive(uint16_t c, void *data)
114 sbusFrameData_t *sbusFrameData = data;
116 const timeUs_t nowUs = microsISR();
118 const timeDelta_t sbusFrameTime = cmpTimeUs(nowUs, sbusFrameData->startAtUs);
120 if (sbusFrameTime > (long)(SBUS_TIME_NEEDED_PER_FRAME + 500)) {
121 sbusFrameData->position = 0;
124 if (sbusFrameData->position == 0) {
125 if (c != SBUS_FRAME_BEGIN_BYTE) {
126 return;
128 sbusFrameData->startAtUs = nowUs;
131 if (sbusFrameData->position < SBUS_FRAME_SIZE) {
132 sbusFrameData->frame.bytes[sbusFrameData->position++] = (uint8_t)c;
133 if (sbusFrameData->position < SBUS_FRAME_SIZE) {
134 sbusFrameData->done = false;
135 } else {
136 sbusFrameData->done = true;
137 DEBUG_SET(DEBUG_SBUS, DEBUG_SBUS_FRAME_TIME, sbusFrameTime);
142 static uint8_t sbusFrameStatus(rxRuntimeState_t *rxRuntimeState)
144 sbusFrameData_t *sbusFrameData = rxRuntimeState->frameData;
145 if (!sbusFrameData->done) {
146 return RX_FRAME_PENDING;
148 sbusFrameData->done = false;
150 DEBUG_SET(DEBUG_SBUS, DEBUG_SBUS_FRAME_FLAGS, sbusFrameData->frame.frame.channels.flags);
152 const uint8_t frameStatus = sbusChannelsDecode(rxRuntimeState, &sbusFrameData->frame.frame.channels);
154 if (!(frameStatus & (RX_FRAME_FAILSAFE | RX_FRAME_DROPPED))) {
155 rxRuntimeState->lastRcFrameTimeUs = sbusFrameData->startAtUs;
158 return frameStatus;
161 bool sbusInit(const rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState)
163 static uint16_t sbusChannelData[SBUS_MAX_CHANNEL];
164 static sbusFrameData_t sbusFrameData;
165 static uint32_t sbusBaudRate;
167 rxRuntimeState->channelData = sbusChannelData;
168 rxRuntimeState->frameData = &sbusFrameData;
169 sbusChannelsInit(rxConfig, rxRuntimeState);
171 rxRuntimeState->channelCount = SBUS_MAX_CHANNEL;
173 if (rxConfig->sbus_baud_fast) {
174 sbusBaudRate = SBUS_FAST_BAUDRATE;
175 } else {
176 sbusBaudRate = SBUS_BAUDRATE;
179 rxRuntimeState->rcFrameStatusFn = sbusFrameStatus;
181 const serialPortConfig_t *portConfig = findSerialPortConfig(FUNCTION_RX_SERIAL);
182 if (!portConfig) {
183 return false;
186 #ifdef USE_TELEMETRY
187 bool portShared = telemetryCheckRxPortShared(portConfig, rxRuntimeState->serialrxProvider);
188 #else
189 bool portShared = false;
190 #endif
192 // On SBUS, SERIAL_INVERTED is default
193 const portOptions_e portOptions = SBUS_PORT_OPTIONS
194 | (rxConfig->serialrx_inverted ? SERIAL_NOT_INVERTED : SERIAL_INVERTED)
195 | (rxConfig->halfDuplex ? SERIAL_BIDIR : 0);
196 serialPort_t *sBusPort = openSerialPort(portConfig->identifier,
197 FUNCTION_RX_SERIAL,
198 sbusDataReceive,
199 &sbusFrameData,
200 sbusBaudRate,
201 portShared ? MODE_RXTX : MODE_RX,
202 portOptions
205 if (rxConfig->rssi_src_frame_errors) {
206 rssiSource = RSSI_SOURCE_FRAME_ERRORS;
209 #ifdef USE_TELEMETRY
210 if (portShared) {
211 telemetrySharedPort = sBusPort;
213 #endif
215 return sBusPort != NULL;
217 #endif