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)
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/>.
22 * Initial FrSky Hub Telemetry implementation by silpstream @ rcgroups.
23 * Addition protocol work by airmamaf @ github.
34 #if defined(USE_TELEMETRY_FRSKY_HUB)
36 #include "common/maths.h"
37 #include "common/axis.h"
38 #include "common/utils.h"
40 #include "config/feature.h"
42 #include "pg/pg_ids.h"
45 #include "drivers/accgyro/accgyro.h"
46 #include "drivers/sensor.h"
47 #include "drivers/serial.h"
48 #include "drivers/time.h"
49 #include "drivers/dshot.h"
51 #include "config/config.h"
52 #include "fc/rc_controls.h"
53 #include "fc/runtime_config.h"
55 #include "sensors/sensors.h"
56 #include "sensors/acceleration.h"
57 #include "sensors/gyro.h"
58 #include "sensors/barometer.h"
59 #include "sensors/battery.h"
61 #include "io/serial.h"
64 #include "flight/mixer.h"
65 #include "flight/pid.h"
66 #include "flight/imu.h"
67 #include "flight/position.h"
71 #include "telemetry/telemetry.h"
73 #if defined(USE_ESC_SENSOR_TELEMETRY)
74 #include "sensors/esc_sensor.h"
77 #include "frsky_hub.h"
79 static serialPort_t
*frSkyHubPort
= NULL
;
80 static const serialPortConfig_t
*portConfig
= NULL
;
82 #define FRSKY_HUB_BAUDRATE 9600
83 #define FRSKY_HUB_INITIAL_PORT_MODE MODE_TX
85 static portSharing_e frSkyHubPortSharing
;
87 static frSkyHubWriteByteFn
*frSkyHubWriteByte
= NULL
;
89 #define FRSKY_HUB_CYCLETIME_US 125000
91 #define PROTOCOL_HEADER 0x5E
92 #define PROTOCOL_TAIL 0x5E
94 // Data Ids (bp = before decimal point; af = after decimal point)
96 #define ID_GPS_ALTIDUTE_BP 0x01
97 #define ID_GPS_ALTIDUTE_AP 0x09
98 #define ID_TEMPRATURE1 0x02
100 #define ID_FUEL_LEVEL 0x04
101 #define ID_TEMPRATURE2 0x05
103 #define ID_ALTITUDE_BP 0x10
104 #define ID_ALTITUDE_AP 0x21
105 #define ID_GPS_SPEED_BP 0x11
106 #define ID_GPS_SPEED_AP 0x19
107 #define ID_LONGITUDE_BP 0x12
108 #define ID_LONGITUDE_AP 0x1A
110 #define ID_LATITUDE_BP 0x13
111 #define ID_LATITUDE_AP 0x1B
113 #define ID_COURSE_BP 0x14
114 #define ID_COURSE_AP 0x1C
115 #define ID_DATE_MONTH 0x15
117 #define ID_HOUR_MINUTE 0x17
118 #define ID_SECOND 0x18
119 #define ID_ACC_X 0x24
120 #define ID_ACC_Y 0x25
121 #define ID_ACC_Z 0x26
122 #define ID_VOLTAGE_AMP 0x39
123 #define ID_VOLTAGE_AMP_BP 0x3A
124 #define ID_VOLTAGE_AMP_AP 0x3B
125 #define ID_CURRENT 0x28
126 // User defined data IDs
127 #define ID_GYRO_X 0x40
128 #define ID_GYRO_Y 0x41
129 #define ID_GYRO_Z 0x42
131 #define ID_VERT_SPEED 0x30 // opentx vario
133 #define GPS_BAD_QUALITY 300
134 #define GPS_MAX_HDOP_VAL 9999
135 #define DELAY_FOR_BARO_INITIALISATION_US 5000000
136 #define BLADE_NUMBER_DIVIDER 5 // should set 12 blades in Taranis
140 TELEMETRY_STATE_UNINITIALIZED
,
141 TELEMETRY_STATE_INITIALIZED_SERIAL
,
142 TELEMETRY_STATE_INITIALIZED_EXTERNAL
,
145 static uint8_t telemetryState
= TELEMETRY_STATE_UNINITIALIZED
;
147 static void serializeFrSkyHub(uint8_t data
)
149 // take care of byte stuffing
151 frSkyHubWriteByte(0x5d);
152 frSkyHubWriteByte(0x3e);
153 } else if (data
== 0x5d) {
154 frSkyHubWriteByte(0x5d);
155 frSkyHubWriteByte(0x3d);
157 frSkyHubWriteByte(data
);
161 static void frSkyHubWriteFrame(const uint8_t id
, const int16_t data
)
163 frSkyHubWriteByte(PROTOCOL_HEADER
);
164 frSkyHubWriteByte(id
);
166 serializeFrSkyHub((uint8_t)data
);
167 serializeFrSkyHub(data
>> 8);
170 static void sendTelemetryTail(void)
172 frSkyHubWriteByte(PROTOCOL_TAIL
);
175 static void frSkyHubWriteByteInternal(const char data
)
177 serialWrite(frSkyHubPort
, data
);
181 static void sendAccel(void)
183 for (unsigned i
= 0; i
< 3; i
++) {
184 frSkyHubWriteFrame(ID_ACC_X
+ i
, ((int16_t)(acc
.accADC
[i
] * acc
.dev
.acc_1G_rec
) * 1000));
189 static void sendThrottleOrBatterySizeAsRpm(void)
192 #if defined(USE_ESC_SENSOR_TELEMETRY)
193 escSensorData_t
*escData
= getEscSensorData(ESC_SENSOR_COMBINED
);
195 data
= escData
->dataAge
< ESC_DATA_INVALID
? (erpmToRpm(escData
->rpm
) / 10) : 0;
198 if (ARMING_FLAG(ARMED
)) {
199 const throttleStatus_e throttleStatus
= calculateThrottleStatus();
200 uint16_t throttleForRPM
= rcCommand
[THROTTLE
] / BLADE_NUMBER_DIVIDER
;
201 if (throttleStatus
== THROTTLE_LOW
&& featureIsEnabled(FEATURE_MOTOR_STOP
)) {
204 data
= throttleForRPM
;
206 data
= (batteryConfig()->batteryCapacity
/ BLADE_NUMBER_DIVIDER
);
210 frSkyHubWriteFrame(ID_RPM
, data
);
213 static void sendTemperature1(void)
216 #if defined(USE_ESC_SENSOR_TELEMETRY)
217 escSensorData_t
*escData
= getEscSensorData(ESC_SENSOR_COMBINED
);
219 data
= escData
->dataAge
< ESC_DATA_INVALID
? escData
->temperature
: 0;
221 #elif defined(USE_BARO)
222 data
= lrintf(baro
.temperature
/ 100.0f
); // Airmamaf
224 data
= lrintf(gyroGetTemperature() / 10.0f
);
226 frSkyHubWriteFrame(ID_TEMPRATURE1
, data
);
229 static void sendTime(void)
231 uint32_t seconds
= millis() / 1000;
232 uint8_t minutes
= (seconds
/ 60) % 60;
234 // if we fly for more than an hour, something's wrong anyway
235 frSkyHubWriteFrame(ID_HOUR_MINUTE
, minutes
<< 8);
236 frSkyHubWriteFrame(ID_SECOND
, seconds
% 60);
239 #if defined(USE_GPS) || defined(USE_MAG)
240 // Frsky pdf: dddmm.mmmm
241 // .mmmm is returned in decimal fraction of minutes.
242 static void GPStoDDDMM_MMMM(int32_t mwiigps
, gpsCoordinateDDDMMmmmm_t
*result
)
244 int32_t absgps
, deg
, min
;
246 absgps
= abs(mwiigps
);
247 deg
= absgps
/ GPS_DEGREES_DIVIDER
;
248 absgps
= (absgps
- deg
* GPS_DEGREES_DIVIDER
) * 60; // absgps = Minutes left * 10^7
249 min
= absgps
/ GPS_DEGREES_DIVIDER
; // minutes left
251 if (telemetryConfig()->frsky_coordinate_format
== FRSKY_FORMAT_DMS
) {
252 result
->dddmm
= deg
* 100 + min
;
254 result
->dddmm
= deg
* 60 + min
;
257 result
->mmmm
= (absgps
- min
* GPS_DEGREES_DIVIDER
) / 1000;
260 static void sendLatLong(int32_t coord
[2])
262 gpsCoordinateDDDMMmmmm_t coordinate
;
263 GPStoDDDMM_MMMM(coord
[GPS_LATITUDE
], &coordinate
);
264 frSkyHubWriteFrame(ID_LATITUDE_BP
, coordinate
.dddmm
);
265 frSkyHubWriteFrame(ID_LATITUDE_AP
, coordinate
.mmmm
);
266 frSkyHubWriteFrame(ID_N_S
, coord
[GPS_LATITUDE
] < 0 ? 'S' : 'N');
268 GPStoDDDMM_MMMM(coord
[GPS_LONGITUDE
], &coordinate
);
269 frSkyHubWriteFrame(ID_LONGITUDE_BP
, coordinate
.dddmm
);
270 frSkyHubWriteFrame(ID_LONGITUDE_AP
, coordinate
.mmmm
);
271 frSkyHubWriteFrame(ID_E_W
, coord
[GPS_LONGITUDE
] < 0 ? 'W' : 'E');
275 static void sendGpsAltitude(void)
277 int32_t altitudeCm
= gpsSol
.llh
.altCm
;
279 // Send real GPS altitude only if it's reliable (there's a GPS fix)
280 if (!STATE(GPS_FIX
)) {
283 frSkyHubWriteFrame(ID_GPS_ALTIDUTE_BP
, altitudeCm
/ 100); // meters: integer part, eg. 123 from 123.45m
284 frSkyHubWriteFrame(ID_GPS_ALTIDUTE_AP
, altitudeCm
% 100); // meters: fractional part, eg. 45 from 123.45m
287 static void sendSatalliteSignalQualityAsTemperature2(uint8_t cycleNum
)
289 uint16_t satellite
= gpsSol
.numSat
;
291 if (gpsSol
.dop
.hdop
> GPS_BAD_QUALITY
&& ( (cycleNum
% 16 ) < 8)) { // Every 1s
292 satellite
= constrain(gpsSol
.dop
.hdop
, 0, GPS_MAX_HDOP_VAL
);
295 if (telemetryConfig()->frsky_unit
== UNIT_IMPERIAL
) {
296 data
= lrintf((satellite
- 32) / 1.8f
);
300 frSkyHubWriteFrame(ID_TEMPRATURE2
, data
);
303 static void sendSpeed(void)
305 if (!STATE(GPS_FIX
)) {
308 // Speed should be sent in knots (GPS speed is in cm/s)
309 // convert to knots: 1cm/s = 0.0194384449 knots
310 frSkyHubWriteFrame(ID_GPS_SPEED_BP
, gpsSol
.groundSpeed
* 1944 / 100000);
311 frSkyHubWriteFrame(ID_GPS_SPEED_AP
, (gpsSol
.groundSpeed
* 1944 / 100) % 100);
314 static void sendFakeLatLong(void)
316 // Heading is only displayed on OpenTX if non-zero lat/long is also sent
317 int32_t coord
[2] = {0,0};
319 coord
[GPS_LATITUDE
] = ((0.01f
* telemetryConfig()->gpsNoFixLatitude
) * GPS_DEGREES_DIVIDER
);
320 coord
[GPS_LONGITUDE
] = ((0.01f
* telemetryConfig()->gpsNoFixLongitude
) * GPS_DEGREES_DIVIDER
);
325 static void sendGPSLatLong(void)
327 static uint8_t gpsFixOccured
= 0;
328 int32_t coord
[2] = {0,0};
330 if (STATE(GPS_FIX
) || gpsFixOccured
== 1) {
331 // If we have ever had a fix, send the last known lat/long
333 coord
[GPS_LATITUDE
] = gpsSol
.llh
.lat
;
334 coord
[GPS_LONGITUDE
] = gpsSol
.llh
.lon
;
337 // otherwise send fake lat/long in order to display compass value
346 * Send voltage via ID_VOLT
348 * NOTE: This sends voltage divided by batteryCellCount. To get the real
349 * battery voltage, you need to multiply the value by batteryCellCount.
351 static void sendVoltageCells(void)
353 static uint16_t currentCell
;
354 uint32_t cellVoltage
= 0;
355 const uint8_t cellCount
= getBatteryCellCount();
358 currentCell
%= cellCount
;
360 * Format for Voltage Data for single cells is like this:
362 * llll llll cccc hhhh
363 * l: Low voltage bits
364 * h: High voltage bits
365 * c: Cell number (starting at 0)
367 * The actual value sent for cell voltage has resolution of 0.002 volts
368 * Since vbat has resolution of 0.1 volts it has to be multiplied by 50
370 cellVoltage
= ((uint32_t)getBatteryVoltage() * 100 + cellCount
) / (cellCount
* 2);
375 // Cell number is at bit 9-12
376 uint16_t data
= (currentCell
<< 4);
378 // Lower voltage bits are at bit 0-8
379 data
|= ((cellVoltage
& 0x0ff) << 8);
381 // Higher voltage bits are at bits 13-15
382 data
|= ((cellVoltage
& 0xf00) >> 8);
384 frSkyHubWriteFrame(ID_VOLT
, data
);
390 * Send voltage with ID_VOLTAGE_AMP
392 static void sendVoltageAmp(void)
394 uint16_t voltage
= getLegacyBatteryVoltage();
395 const uint8_t cellCount
= getBatteryCellCount();
397 if (telemetryConfig()->frsky_vfas_precision
== FRSKY_VFAS_PRECISION_HIGH
) {
398 // Use new ID 0x39 to send voltage directly in 0.1 volts resolution
399 if (telemetryConfig()->report_cell_voltage
&& cellCount
) {
400 voltage
/= cellCount
;
402 frSkyHubWriteFrame(ID_VOLTAGE_AMP
, voltage
);
404 // send in 0.2 volts resolution
406 if (telemetryConfig()->report_cell_voltage
&& cellCount
) {
407 voltage
/= cellCount
;
410 frSkyHubWriteFrame(ID_VOLTAGE_AMP_BP
, voltage
/ 100);
411 frSkyHubWriteFrame(ID_VOLTAGE_AMP_AP
, ((voltage
% 100) + 5) / 10);
415 static void sendAmperage(void)
417 frSkyHubWriteFrame(ID_CURRENT
, (uint16_t)(getAmperage() / 10));
420 static void sendFuelLevel(void)
423 if (batteryConfig()->batteryCapacity
> 0) {
424 data
= (uint16_t)calculateBatteryPercentageRemaining();
426 data
= (uint16_t)constrain(getMAhDrawn(), 0, 0xFFFF);
428 frSkyHubWriteFrame(ID_FUEL_LEVEL
, data
);
432 static void sendFakeLatLongThatAllowsHeadingDisplay(void)
434 // Heading is only displayed on OpenTX if non-zero lat/long is also sent
436 1 * GPS_DEGREES_DIVIDER
,
437 1 * GPS_DEGREES_DIVIDER
443 static void sendHeading(void)
445 frSkyHubWriteFrame(ID_COURSE_BP
, DECIDEGREES_TO_DEGREES(attitude
.values
.yaw
));
446 frSkyHubWriteFrame(ID_COURSE_AP
, 0);
450 bool initFrSkyHubTelemetry(void)
452 if (telemetryState
== TELEMETRY_STATE_UNINITIALIZED
) {
453 portConfig
= findSerialPortConfig(FUNCTION_TELEMETRY_FRSKY_HUB
);
455 frSkyHubPortSharing
= determinePortSharing(portConfig
, FUNCTION_TELEMETRY_FRSKY_HUB
);
457 frSkyHubWriteByte
= frSkyHubWriteByteInternal
;
459 telemetryState
= TELEMETRY_STATE_INITIALIZED_SERIAL
;
468 bool initFrSkyHubTelemetryExternal(frSkyHubWriteByteFn
*frSkyHubWriteByteExternal
)
470 if (telemetryState
== TELEMETRY_STATE_UNINITIALIZED
) {
471 frSkyHubWriteByte
= frSkyHubWriteByteExternal
;
473 telemetryState
= TELEMETRY_STATE_INITIALIZED_EXTERNAL
;
481 void freeFrSkyHubTelemetryPort(void)
483 closeSerialPort(frSkyHubPort
);
487 static void configureFrSkyHubTelemetryPort(void)
490 frSkyHubPort
= openSerialPort(portConfig
->identifier
, FUNCTION_TELEMETRY_FRSKY_HUB
, NULL
, NULL
, FRSKY_HUB_BAUDRATE
, FRSKY_HUB_INITIAL_PORT_MODE
, telemetryConfig()->telemetry_inverted
? SERIAL_NOT_INVERTED
: SERIAL_INVERTED
);
494 void checkFrSkyHubTelemetryState(void)
496 if (telemetryState
== TELEMETRY_STATE_INITIALIZED_SERIAL
) {
497 if (telemetryCheckRxPortShared(portConfig
, rxRuntimeState
.serialrxProvider
)) {
498 if (frSkyHubPort
== NULL
&& telemetrySharedPort
!= NULL
) {
499 frSkyHubPort
= telemetrySharedPort
;
502 bool enableSerialTelemetry
= telemetryDetermineEnabledState(frSkyHubPortSharing
);
503 if (enableSerialTelemetry
&& !frSkyHubPort
) {
504 configureFrSkyHubTelemetryPort();
505 } else if (!enableSerialTelemetry
&& frSkyHubPort
) {
506 freeFrSkyHubTelemetryPort();
512 void processFrSkyHubTelemetry(timeUs_t currentTimeUs
)
514 static uint32_t frSkyHubLastCycleTime
= 0;
515 static uint8_t cycleNum
= 0;
517 if (cmpTimeUs(currentTimeUs
, frSkyHubLastCycleTime
) < FRSKY_HUB_CYCLETIME_US
) {
520 frSkyHubLastCycleTime
= currentTimeUs
;
525 if (sensors(SENSOR_ACC
) && telemetryIsSensorEnabled(SENSOR_ACC_X
| SENSOR_ACC_Y
| SENSOR_ACC_Z
)) {
531 #if defined(USE_BARO) || defined(USE_RANGEFINDER) || defined(USE_GPS)
532 if (sensors(SENSOR_BARO
| SENSOR_RANGEFINDER
) | sensors(SENSOR_GPS
)) {
534 // Send vertical speed for opentx. ID_VERT_SPEED
537 if (telemetryIsSensorEnabled(SENSOR_VARIO
)) {
538 frSkyHubWriteFrame(ID_VERT_SPEED
, getEstimatedVario());
543 if ((cycleNum
% 4) == 0 && telemetryIsSensorEnabled(SENSOR_ALTITUDE
)) {
544 int32_t altitudeCm
= getEstimatedAltitudeCm();
546 /* Allow 5s to boot correctly othervise send zero to prevent OpenTX
547 * sensor lost notifications after warm boot. */
548 if (frSkyHubLastCycleTime
< DELAY_FOR_BARO_INITIALISATION_US
) {
552 frSkyHubWriteFrame(ID_ALTITUDE_BP
, altitudeCm
/ 100); // meters: integer part, eg. 123 from 123.45m
553 frSkyHubWriteFrame(ID_ALTITUDE_AP
, altitudeCm
% 100); // meters: fractional part, eg. 45 from 123.45m
559 if (sensors(SENSOR_MAG
) && telemetryIsSensorEnabled(SENSOR_HEADING
)) {
561 if ((cycleNum
% 4) == 0) {
568 if ((cycleNum
% 8) == 0) {
570 sendThrottleOrBatterySizeAsRpm();
572 if (isBatteryVoltageConfigured()) {
573 if (telemetryIsSensorEnabled(SENSOR_VOLTAGE
)) {
578 if (isAmperageConfigured()) {
579 if (telemetryIsSensorEnabled(SENSOR_CURRENT
)) {
582 if (telemetryIsSensorEnabled(SENSOR_FUEL
)) {
589 if (sensors(SENSOR_GPS
)) {
590 if (telemetryIsSensorEnabled(SENSOR_GROUND_SPEED
)) {
593 if (telemetryIsSensorEnabled(SENSOR_ALTITUDE
)) {
596 sendSatalliteSignalQualityAsTemperature2(cycleNum
);
597 if (telemetryIsSensorEnabled(SENSOR_LAT_LONG
)) {
603 if (sensors(SENSOR_MAG
)) {
604 sendFakeLatLongThatAllowsHeadingDisplay();
612 if (cycleNum
== 40) {
620 void handleFrSkyHubTelemetry(timeUs_t currentTimeUs
)
622 if (telemetryState
== TELEMETRY_STATE_INITIALIZED_SERIAL
&& frSkyHubPort
) {
623 processFrSkyHubTelemetry(currentTimeUs
);