Implement Stopwatch (#12623)
[betaflight.git] / src / main / telemetry / frsky_hub.c
blob6c91319f3c8e0db00812a0330340f25267d3a473
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/>.
22 * Initial FrSky Hub Telemetry implementation by silpstream @ rcgroups.
23 * Addition protocol work by airmamaf @ github.
26 #include <stddef.h>
27 #include <stdbool.h>
28 #include <stdint.h>
29 #include <math.h>
30 #include <stdlib.h>
32 #include "platform.h"
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"
41 #include "pg/pg.h"
42 #include "pg/pg_ids.h"
43 #include "pg/rx.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"
62 #include "io/gps.h"
64 #include "flight/mixer.h"
65 #include "flight/pid.h"
66 #include "flight/imu.h"
67 #include "flight/position.h"
69 #include "rx/rx.h"
71 #include "telemetry/telemetry.h"
73 #if defined(USE_ESC_SENSOR_TELEMETRY)
74 #include "sensors/esc_sensor.h"
75 #endif
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)
95 // Official data IDs
96 #define ID_GPS_ALTIDUTE_BP 0x01
97 #define ID_GPS_ALTIDUTE_AP 0x09
98 #define ID_TEMPRATURE1 0x02
99 #define ID_RPM 0x03
100 #define ID_FUEL_LEVEL 0x04
101 #define ID_TEMPRATURE2 0x05
102 #define ID_VOLT 0x06
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
109 #define ID_E_W 0x22
110 #define ID_LATITUDE_BP 0x13
111 #define ID_LATITUDE_AP 0x1B
112 #define ID_N_S 0x23
113 #define ID_COURSE_BP 0x14
114 #define ID_COURSE_AP 0x1C
115 #define ID_DATE_MONTH 0x15
116 #define ID_YEAR 0x16
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
138 enum
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
150 if (data == 0x5e) {
151 frSkyHubWriteByte(0x5d);
152 frSkyHubWriteByte(0x3e);
153 } else if (data == 0x5d) {
154 frSkyHubWriteByte(0x5d);
155 frSkyHubWriteByte(0x3d);
156 } else{
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);
180 #if defined(USE_ACC)
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));
187 #endif
189 static void sendThrottleOrBatterySizeAsRpm(void)
191 int16_t data = 0;
192 #if defined(USE_ESC_SENSOR_TELEMETRY)
193 escSensorData_t *escData = getEscSensorData(ESC_SENSOR_COMBINED);
194 if (escData) {
195 data = escData->dataAge < ESC_DATA_INVALID ? (erpmToRpm(escData->rpm) / 10) : 0;
197 #else
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)) {
202 throttleForRPM = 0;
204 data = throttleForRPM;
205 } else {
206 data = (batteryConfig()->batteryCapacity / BLADE_NUMBER_DIVIDER);
208 #endif
210 frSkyHubWriteFrame(ID_RPM, data);
213 static void sendTemperature1(void)
215 int16_t data = 0;
216 #if defined(USE_ESC_SENSOR_TELEMETRY)
217 escSensorData_t *escData = getEscSensorData(ESC_SENSOR_COMBINED);
218 if (escData) {
219 data = escData->dataAge < ESC_DATA_INVALID ? escData->temperature : 0;
221 #elif defined(USE_BARO)
222 data = lrintf(baro.temperature / 100.0f); // Airmamaf
223 #else
224 data = lrintf(gyroGetTemperature() / 10.0f);
225 #endif
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;
253 } else {
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');
274 #if defined(USE_GPS)
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)) {
281 altitudeCm = 0;
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);
294 int16_t data;
295 if (telemetryConfig()->frsky_unit == UNIT_IMPERIAL) {
296 data = lrintf((satellite - 32) / 1.8f);
297 } else {
298 data = satellite;
300 frSkyHubWriteFrame(ID_TEMPRATURE2, data);
303 static void sendSpeed(void)
305 if (!STATE(GPS_FIX)) {
306 return;
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);
322 sendLatLong(coord);
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
332 gpsFixOccured = 1;
333 coord[GPS_LATITUDE] = gpsSol.llh.lat;
334 coord[GPS_LONGITUDE] = gpsSol.llh.lon;
335 sendLatLong(coord);
336 } else {
337 // otherwise send fake lat/long in order to display compass value
338 sendFakeLatLong();
342 #endif
343 #endif
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();
357 if (cellCount) {
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);
371 } else {
372 currentCell = 0;
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);
386 currentCell++;
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);
403 } else {
404 // send in 0.2 volts resolution
405 voltage *= 110 / 21;
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)
422 int16_t data;
423 if (batteryConfig()->batteryCapacity > 0) {
424 data = (uint16_t)calculateBatteryPercentageRemaining();
425 } else {
426 data = (uint16_t)constrain(getMAhDrawn(), 0, 0xFFFF);
428 frSkyHubWriteFrame(ID_FUEL_LEVEL, data);
431 #if defined(USE_MAG)
432 static void sendFakeLatLongThatAllowsHeadingDisplay(void)
434 // Heading is only displayed on OpenTX if non-zero lat/long is also sent
435 int32_t coord[2] = {
436 1 * GPS_DEGREES_DIVIDER,
437 1 * GPS_DEGREES_DIVIDER
440 sendLatLong(coord);
443 static void sendHeading(void)
445 frSkyHubWriteFrame(ID_COURSE_BP, DECIDEGREES_TO_DEGREES(attitude.values.yaw));
446 frSkyHubWriteFrame(ID_COURSE_AP, 0);
448 #endif
450 bool initFrSkyHubTelemetry(void)
452 if (telemetryState == TELEMETRY_STATE_UNINITIALIZED) {
453 portConfig = findSerialPortConfig(FUNCTION_TELEMETRY_FRSKY_HUB);
454 if (portConfig) {
455 frSkyHubPortSharing = determinePortSharing(portConfig, FUNCTION_TELEMETRY_FRSKY_HUB);
457 frSkyHubWriteByte = frSkyHubWriteByteInternal;
459 telemetryState = TELEMETRY_STATE_INITIALIZED_SERIAL;
462 return true;
465 return false;
468 bool initFrSkyHubTelemetryExternal(frSkyHubWriteByteFn *frSkyHubWriteByteExternal)
470 if (telemetryState == TELEMETRY_STATE_UNINITIALIZED) {
471 frSkyHubWriteByte = frSkyHubWriteByteExternal;
473 telemetryState = TELEMETRY_STATE_INITIALIZED_EXTERNAL;
475 return true;
478 return false;
481 void freeFrSkyHubTelemetryPort(void)
483 closeSerialPort(frSkyHubPort);
484 frSkyHubPort = NULL;
487 static void configureFrSkyHubTelemetryPort(void)
489 if (portConfig) {
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;
501 } else {
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) {
518 return;
520 frSkyHubLastCycleTime = currentTimeUs;
522 cycleNum++;
524 #if defined(USE_ACC)
525 if (sensors(SENSOR_ACC) && telemetryIsSensorEnabled(SENSOR_ACC_X | SENSOR_ACC_Y | SENSOR_ACC_Z)) {
526 // Sent every 125ms
527 sendAccel();
529 #endif
531 #if defined(USE_BARO) || defined(USE_RANGEFINDER) || defined(USE_GPS)
532 if (sensors(SENSOR_BARO | SENSOR_RANGEFINDER) | sensors(SENSOR_GPS)) {
533 // Sent every 125ms
534 // Send vertical speed for opentx. ID_VERT_SPEED
535 // Unit is cm/s
536 #ifdef USE_VARIO
537 if (telemetryIsSensorEnabled(SENSOR_VARIO)) {
538 frSkyHubWriteFrame(ID_VERT_SPEED, getEstimatedVario());
540 #endif
542 // Sent every 500ms
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) {
549 altitudeCm = 0;
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
556 #endif
558 #if defined(USE_MAG)
559 if (sensors(SENSOR_MAG) && telemetryIsSensorEnabled(SENSOR_HEADING)) {
560 // Sent every 500ms
561 if ((cycleNum % 4) == 0) {
562 sendHeading();
565 #endif
567 // Sent every 1s
568 if ((cycleNum % 8) == 0) {
569 sendTemperature1();
570 sendThrottleOrBatterySizeAsRpm();
572 if (isBatteryVoltageConfigured()) {
573 if (telemetryIsSensorEnabled(SENSOR_VOLTAGE)) {
574 sendVoltageCells();
575 sendVoltageAmp();
578 if (isAmperageConfigured()) {
579 if (telemetryIsSensorEnabled(SENSOR_CURRENT)) {
580 sendAmperage();
582 if (telemetryIsSensorEnabled(SENSOR_FUEL)) {
583 sendFuelLevel();
588 #if defined(USE_GPS)
589 if (sensors(SENSOR_GPS)) {
590 if (telemetryIsSensorEnabled(SENSOR_GROUND_SPEED)) {
591 sendSpeed();
593 if (telemetryIsSensorEnabled(SENSOR_ALTITUDE)) {
594 sendGpsAltitude();
596 sendSatalliteSignalQualityAsTemperature2(cycleNum);
597 if (telemetryIsSensorEnabled(SENSOR_LAT_LONG)) {
598 sendGPSLatLong();
600 } else
601 #endif
602 #if defined(USE_MAG)
603 if (sensors(SENSOR_MAG)) {
604 sendFakeLatLongThatAllowsHeadingDisplay();
606 #else
608 #endif
611 // Sent every 5s
612 if (cycleNum == 40) {
613 cycleNum = 0;
614 sendTime();
617 sendTelemetryTail();
620 void handleFrSkyHubTelemetry(timeUs_t currentTimeUs)
622 if (telemetryState == TELEMETRY_STATE_INITIALIZED_SERIAL && frSkyHubPort) {
623 processFrSkyHubTelemetry(currentTimeUs);
626 #endif