Merge pull request #11483 from SteveCEvans/elrs_race
[betaflight.git] / src / test / unit / link_quality_unittest.cc
blob407462c551576e8ffd31955d5cbe47f9ac10f992
1 /*
2 * This file is part of Cleanflight.
4 * Cleanflight is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
9 * Cleanflight is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with Cleanflight. If not, see <http://www.gnu.org/licenses/>.
18 #include <stdint.h>
19 #include <stdbool.h>
20 #include <stdio.h>
21 #include <string.h>
23 extern "C" {
24 #include "platform.h"
25 #include "build/debug.h"
27 #include "blackbox/blackbox.h"
28 #include "blackbox/blackbox_io.h"
30 #include "common/crc.h"
31 #include "common/printf.h"
32 #include "common/streambuf.h"
33 #include "common/time.h"
34 #include "common/utils.h"
36 #include "config/config.h"
38 #include "drivers/osd_symbols.h"
39 #include "drivers/persistent.h"
40 #include "drivers/serial.h"
41 #include "drivers/system.h"
43 #include "fc/core.h"
44 #include "fc/rc_controls.h"
45 #include "fc/rc_modes.h"
46 #include "fc/runtime_config.h"
48 #include "flight/failsafe.h"
49 #include "flight/imu.h"
50 #include "flight/mixer.h"
51 #include "flight/pid.h"
53 #include "io/beeper.h"
54 #include "io/gps.h"
55 #include "io/serial.h"
57 #include "osd/osd.h"
58 #include "osd/osd_elements.h"
59 #include "osd/osd_warnings.h"
61 #include "pg/pg.h"
62 #include "pg/pg_ids.h"
63 #include "pg/rx.h"
65 #include "rx/rx.h"
67 #include "sensors/battery.h"
69 attitudeEulerAngles_t attitude;
70 float rMat[3][3];
72 pidProfile_t *currentPidProfile;
73 float rcData[MAX_SUPPORTED_RC_CHANNEL_COUNT];
74 uint8_t GPS_numSat;
75 uint16_t GPS_distanceToHome;
76 int16_t GPS_directionToHome;
77 uint32_t GPS_distanceFlownInCm;
78 int32_t GPS_coord[2];
79 gpsSolutionData_t gpsSol;
80 float motor[8];
81 acc_t acc;
82 float accAverage[XYZ_AXIS_COUNT];
84 PG_REGISTER(batteryConfig_t, batteryConfig, PG_BATTERY_CONFIG, 0);
85 PG_REGISTER(blackboxConfig_t, blackboxConfig, PG_BLACKBOX_CONFIG, 0);
86 PG_REGISTER(systemConfig_t, systemConfig, PG_SYSTEM_CONFIG, 0);
87 PG_REGISTER(pilotConfig_t, pilotConfig, PG_PILOT_CONFIG, 0);
88 PG_REGISTER(imuConfig_t, imuConfig, PG_IMU_CONFIG, 0);
89 PG_REGISTER(gpsConfig_t, gpsConfig, PG_GPS_CONFIG, 0);
90 PG_REGISTER(failsafeConfig_t, failsafeConfig, PG_FAILSAFE_CONFIG, 0);
92 timeUs_t simulationTime = 0;
94 void osdUpdate(timeUs_t currentTimeUs);
95 uint16_t updateLinkQualitySamples(uint16_t value);
96 #define LINK_QUALITY_SAMPLE_COUNT 16
99 /* #define DEBUG_OSD */
101 #include "unittest_macros.h"
102 #include "unittest_displayport.h"
103 #include "gtest/gtest.h"
105 extern "C" {
106 PG_REGISTER(flight3DConfig_t, flight3DConfig, PG_MOTOR_3D_CONFIG, 0);
108 boxBitmask_t rcModeActivationMask;
109 int16_t debug[DEBUG16_VALUE_COUNT];
110 uint8_t debugMode = 0;
112 uint16_t updateLinkQualitySamples(uint16_t value);
114 extern uint16_t applyRxChannelRangeConfiguraton(int sample, const rxChannelRangeConfig_t *range);
116 void setDefaultSimulationState()
118 setLinkQualityDirect(LINK_QUALITY_MAX_VALUE);
119 osdConfigMutable()->framerate_hz = 12;
122 * Performs a test of the OSD actions on arming.
123 * (reused throughout the test suite)
125 void doTestArm(bool testEmpty = true)
127 // given
128 // craft has been armed
129 ENABLE_ARMING_FLAG(ARMED);
131 simulationTime += 0.1e6;
132 // when
133 // sufficient OSD updates have been called
134 while (osdUpdateCheck(simulationTime, 0)) {
135 osdUpdate(simulationTime);
136 simulationTime += 10;
139 // then
140 // arming alert displayed
141 displayPortTestBufferSubstring(12, 7, "ARMED");
143 // given
144 // armed alert times out (0.5 seconds)
145 simulationTime += 0.5e6;
147 // when
148 // sufficient OSD updates have been called
149 while (osdUpdateCheck(simulationTime, 0)) {
150 osdUpdate(simulationTime);
151 simulationTime += 10;
154 // then
155 // arming alert disappears
156 #ifdef DEBUG_OSD
157 displayPortTestPrint();
158 #endif
159 if (testEmpty) {
160 displayPortTestBufferIsEmpty();
165 * Auxiliary function. Test is there're stats that must be shown
167 bool isSomeStatEnabled(void) {
168 return (osdConfigMutable()->enabled_stats != 0);
172 * Performs a test of the OSD actions on disarming.
173 * (reused throughout the test suite)
175 void doTestDisarm()
177 // given
178 // craft is disarmed after having been armed
179 DISABLE_ARMING_FLAG(ARMED);
181 // when
182 // sufficient OSD updates have been called
183 while (osdUpdateCheck(simulationTime, 0)) {
184 osdUpdate(simulationTime);
185 simulationTime += 10;
188 // then
189 // post flight statistics displayed
190 if (isSomeStatEnabled()) {
191 displayPortTestBufferSubstring(2, 2, " --- STATS ---");
196 * Tests initialisation of the OSD and the power on splash screen.
198 TEST(LQTest, TestInit)
200 // given
201 // display port is initialised
202 displayPortTestInit();
204 // and
205 // default state values are set
206 setDefaultSimulationState();
208 // and
209 // this battery configuration (used for battery voltage elements)
210 batteryConfigMutable()->vbatmincellvoltage = 330;
211 batteryConfigMutable()->vbatmaxcellvoltage = 430;
213 // when
214 // OSD is initialised
215 osdInit(&testDisplayPort, OSD_DISPLAYPORT_DEVICE_AUTO);
217 while (osdUpdateCheck(simulationTime, 0)) {
218 osdUpdate(simulationTime);
219 simulationTime += 10;
222 // then
223 // display buffer should contain splash screen
224 displayPortTestBufferSubstring(7, 8, "MENU:THR MID");
225 displayPortTestBufferSubstring(11, 9, "+ YAW LEFT");
226 displayPortTestBufferSubstring(11, 10, "+ PITCH UP");
228 // when
229 // splash screen timeout has elapsed
230 simulationTime += 4e6;
231 while (osdUpdateCheck(simulationTime, 0)) {
232 osdUpdate(simulationTime);
233 simulationTime += 10;
236 // then
237 // display buffer should be empty
238 #ifdef DEBUG_OSD
239 displayPortTestPrint();
240 #endif
241 displayPortTestBufferIsEmpty();
244 * Tests the Tests the OSD_LINK_QUALITY element updateLinkQualitySamples default LQ_SOURCE_NONE
246 TEST(LQTest, TestElement_LQ_SOURCE_NONE_SAMPLES)
248 // given
249 linkQualitySource = LQ_SOURCE_NONE;
251 osdElementConfigMutable()->item_pos[OSD_LINK_QUALITY] = OSD_POS(8, 1) | OSD_PROFILE_1_FLAG;
252 osdConfigMutable()->link_quality_alarm = 0;
254 osdAnalyzeActiveElements();
256 // when samples populated 100%
257 for (int x = 0; x < LINK_QUALITY_SAMPLE_COUNT; x++) {
258 setLinkQualityDirect(updateLinkQualitySamples(LINK_QUALITY_MAX_VALUE));
261 simulationTime += 1000000;
263 while (osdUpdateCheck(simulationTime, 0)) {
264 osdUpdate(simulationTime);
265 simulationTime += 10;
268 // then
269 displayPortTestBufferSubstring(8, 1, "%c9", SYM_LINK_QUALITY);
271 // when updateLinkQualitySamples used 50% rounds to 4
272 for (int x = 0; x < LINK_QUALITY_SAMPLE_COUNT; x++) {
273 setLinkQualityDirect(updateLinkQualitySamples(LINK_QUALITY_MAX_VALUE));
274 setLinkQualityDirect(updateLinkQualitySamples(0));
277 simulationTime += 1000000;
279 while (osdUpdateCheck(simulationTime, 0)) {
280 osdUpdate(simulationTime);
281 simulationTime += 10;
284 // then
285 displayPortTestBufferSubstring(8, 1, "%c4", SYM_LINK_QUALITY);
288 * Tests the Tests the OSD_LINK_QUALITY element values default LQ_SOURCE_NONE
290 TEST(LQTest, TestElement_LQ_SOURCE_NONE_VALUES)
292 // given
294 linkQualitySource = LQ_SOURCE_NONE;
296 osdElementConfigMutable()->item_pos[OSD_LINK_QUALITY] = OSD_POS(8, 1) | OSD_PROFILE_1_FLAG;
297 osdConfigMutable()->link_quality_alarm = 0;
299 osdAnalyzeActiveElements();
300 // when LINK_QUALITY_MAX_VALUE to 1 by 10%
301 uint16_t testscale = 0;
302 for (int testdigit = 10; testdigit > 0; testdigit--) {
303 testscale = testdigit * 102.3;
304 setLinkQualityDirect(testscale);
305 simulationTime += 100000;
306 while (osdUpdateCheck(simulationTime, 0)) {
307 osdUpdate(simulationTime);
308 simulationTime += 10;
310 #ifdef DEBUG_OSD
311 printf("%d %d\n",testscale, testdigit);
312 displayPortTestPrint();
313 #endif
314 // then
315 if (testdigit >= 10) {
316 displayPortTestBufferSubstring(8, 1,"%c9", SYM_LINK_QUALITY);
317 }else{
318 displayPortTestBufferSubstring(8, 1,"%c%1d", SYM_LINK_QUALITY, testdigit - 1);
324 * Tests the OSD_LINK_QUALITY element LQ RX_PROTOCOL_CRSF.
326 TEST(LQTest, TestElementLQ_PROTOCOL_CRSF_VALUES)
328 // given
329 linkQualitySource = LQ_SOURCE_RX_PROTOCOL_CRSF;
331 osdElementConfigMutable()->item_pos[OSD_LINK_QUALITY] = OSD_POS(8, 1) | OSD_PROFILE_1_FLAG;
332 osdConfigMutable()->link_quality_alarm = 0;
334 osdAnalyzeActiveElements();
336 simulationTime += 1000000;
337 while (osdUpdateCheck(simulationTime, 0)) {
338 osdUpdate(simulationTime);
339 simulationTime += 10;
342 // crsf setLinkQualityDirect 0-300;
344 for (uint8_t x = 0; x <= 99; x++) {
345 for (uint8_t m = 0; m <= 4; m++) {
346 // when x scaled
347 setLinkQualityDirect(x);
348 rxSetRfMode(m);
349 // then rxGetLinkQuality Osd should be x
350 // and RfMode should be m
351 simulationTime += 100000;
352 while (osdUpdateCheck(simulationTime, 0)) {
353 osdUpdate(simulationTime);
354 simulationTime += 10;
356 displayPortTestBufferSubstring(8, 1, "%c%1d:%2d", SYM_LINK_QUALITY, m, x);
361 * Tests the LQ Alarms
364 TEST(LQTest, TestLQAlarm)
366 timeUs_t startTime = simulationTime;
367 // given
368 // default state is set
369 setDefaultSimulationState();
371 linkQualitySource = LQ_SOURCE_NONE;
373 // and
374 // the following OSD elements are visible
376 osdElementConfigMutable()->item_pos[OSD_LINK_QUALITY] = OSD_POS(8, 1) | OSD_PROFILE_1_FLAG;
378 // and
379 // this set of alarm values
381 osdConfigMutable()->link_quality_alarm = 80;
382 stateFlags |= GPS_FIX | GPS_FIX_HOME;
384 osdAnalyzeActiveElements();
386 // and
387 // using the metric unit system
388 osdConfigMutable()->units = UNIT_METRIC;
390 // when
391 // the craft is armed
392 doTestArm(false);
394 for (int x = 0; x < LINK_QUALITY_SAMPLE_COUNT; x++) {
395 setLinkQualityDirect(updateLinkQualitySamples(LINK_QUALITY_MAX_VALUE));
398 // then
399 // no elements should flash as all values are out of alarm range
400 // Ensure a consistent start time for testing
401 simulationTime += 5000000;
402 simulationTime -= simulationTime % 1000000;
403 startTime = simulationTime;
404 for (int i = 0; i < 30; i++) {
405 // Check for visibility every 100ms, elements should always be visible
406 simulationTime = startTime + i*0.1e6;
407 while (osdUpdateCheck(simulationTime, 0)) {
408 osdUpdate(simulationTime);
409 simulationTime += 10;
412 #ifdef DEBUG_OSD
413 printf("%d\n", i);
414 #endif
415 displayPortTestBufferSubstring(8, 1, "%c9", SYM_LINK_QUALITY);
419 setLinkQualityDirect(512);
420 while (osdUpdateCheck(simulationTime, 0)) {
421 osdUpdate(simulationTime);
422 simulationTime += 10;
425 // then
426 // elements showing values in alarm range should flash
427 simulationTime += 1000000;
428 simulationTime -= simulationTime % 1000000;
429 startTime = simulationTime;
430 for (int i = 0; i < 15; i++) {
431 // Blinking should happen at 2Hz
432 simulationTime = startTime + i*0.25e6;
433 while (osdUpdateCheck(simulationTime, 0)) {
434 osdUpdate(simulationTime);
435 simulationTime += 10;
438 #ifdef DEBUG_OSD
439 displayPortTestPrint();
440 #endif
441 if (i % 2 == 0) {
442 displayPortTestBufferSubstring(8, 1, "%c5", SYM_LINK_QUALITY);
443 } else {
444 displayPortTestBufferIsEmpty();
448 doTestDisarm();
449 simulationTime += 1000000;
450 simulationTime -= simulationTime % 1000000;
451 while (osdUpdateCheck(simulationTime, 0)) {
452 osdUpdate(simulationTime);
453 simulationTime += 10;
457 // STUBS
458 extern "C" {
460 uint32_t micros() {
461 return simulationTime;
464 uint32_t microsISR() {
465 return micros();
468 uint32_t millis() {
469 return micros() / 1000;
472 bool featureIsEnabled(uint32_t) { return true; }
473 void beeperConfirmationBeeps(uint8_t) {}
474 bool isBeeperOn() { return false; }
475 uint8_t getCurrentPidProfileIndex() { return 0; }
476 uint8_t getCurrentControlRateProfileIndex() { return 0; }
477 batteryState_e getBatteryState() { return BATTERY_OK; }
478 uint8_t getBatteryCellCount() { return 4; }
479 uint16_t getBatteryVoltage() { return 1680; }
480 uint16_t getBatteryAverageCellVoltage() { return 420; }
481 int32_t getAmperage() { return 0; }
482 int32_t getMAhDrawn() { return 0; }
483 int32_t getEstimatedAltitudeCm() { return 0; }
484 int32_t getEstimatedVario() { return 0; }
485 int32_t blackboxGetLogNumber() { return 0; }
486 bool isBlackboxDeviceWorking() { return true; }
487 bool isBlackboxDeviceFull() { return false; }
488 serialPort_t *openSerialPort(serialPortIdentifier_e, serialPortFunction_e, serialReceiveCallbackPtr, void *, uint32_t, portMode_e, portOptions_e) {return NULL;}
489 const serialPortConfig_t *findSerialPortConfig(serialPortFunction_e ) {return NULL;}
490 bool telemetryCheckRxPortShared(const serialPortConfig_t *) {return false;}
491 bool cmsDisplayPortRegister(displayPort_t *) { return false; }
492 uint16_t getCoreTemperatureCelsius(void) { return 0; }
493 bool isFlipOverAfterCrashActive(void) { return false; }
494 float pidItermAccelerator(void) { return 1.0; }
495 uint8_t getMotorCount(void){ return 4; }
496 bool areMotorsRunning(void){ return true; }
497 bool pidOsdAntiGravityActive(void) { return false; }
498 bool failsafeIsActive(void) { return false; }
499 bool failsafeIsReceivingRxData(void) { return true; }
500 bool gpsIsHealthy(void) { return true; }
501 bool gpsRescueIsConfigured(void) { return false; }
502 int8_t calculateThrottlePercent(void) { return 0; }
503 uint32_t persistentObjectRead(persistentObjectId_e) { return 0; }
504 void persistentObjectWrite(persistentObjectId_e, uint32_t) {}
505 void failsafeOnRxSuspend(uint32_t ) {}
506 void failsafeOnRxResume(void) {}
507 uint32_t failsafeFailurePeriodMs(void) { return 400; }
508 void featureDisableImmediate(uint32_t) { }
509 bool rxMspFrameComplete(void) { return false; }
510 bool isPPMDataBeingReceived(void) { return false; }
511 bool isPWMDataBeingReceived(void) { return false; }
512 void resetPPMDataReceivedState(void){ }
513 void failsafeOnValidDataReceived(void) { }
514 void failsafeOnValidDataFailed(void) { }
515 void pinioBoxTaskControl(void) { }
516 bool taskUpdateRxMainInProgress() { return true; }
517 void schedulerIgnoreTaskStateTime(void) { }
518 void schedulerIgnoreTaskExecRate(void) { }
519 bool schedulerGetIgnoreTaskExecTime() { return false; }
520 void schedulerIgnoreTaskExecTime(void) { }
521 void schedulerSetNextStateTime(timeDelta_t) {}
523 void rxPwmInit(rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
525 UNUSED(rxRuntimeState);
526 UNUSED(callback);
529 bool sbusInit(rxConfig_t *initialRxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
531 UNUSED(initialRxConfig);
532 UNUSED(rxRuntimeState);
533 UNUSED(callback);
534 return true;
537 bool spektrumInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
539 UNUSED(rxConfig);
540 UNUSED(rxRuntimeState);
541 UNUSED(callback);
542 return true;
545 bool sumdInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
547 UNUSED(rxConfig);
548 UNUSED(rxRuntimeState);
549 UNUSED(callback);
550 return true;
553 bool sumhInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
555 UNUSED(rxConfig);
556 UNUSED(rxRuntimeState);
557 UNUSED(callback);
558 return true;
561 bool crsfRxInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback);
563 bool jetiExBusInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
565 UNUSED(rxConfig);
566 UNUSED(rxRuntimeState);
567 UNUSED(callback);
568 return true;
571 bool ibusInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
573 UNUSED(rxConfig);
574 UNUSED(rxRuntimeState);
575 UNUSED(callback);
576 return true;
579 bool xBusInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
581 UNUSED(rxConfig);
582 UNUSED(rxRuntimeState);
583 UNUSED(callback);
584 return true;
587 bool rxMspInit(rxConfig_t *rxConfig, rxRuntimeState_t *rxRuntimeState, rcReadRawDataFnPtr *callback)
589 UNUSED(rxConfig);
590 UNUSED(rxRuntimeState);
591 UNUSED(callback);
592 return true;
595 float pt1FilterGain(float f_cut, float dT)
597 UNUSED(f_cut);
598 UNUSED(dT);
599 return 0.0;
602 void pt1FilterInit(pt1Filter_t *filter, float k)
604 UNUSED(filter);
605 UNUSED(k);
608 float pt1FilterApply(pt1Filter_t *filter, float input)
610 UNUSED(filter);
611 UNUSED(input);
612 return 0.0;
615 bool isUpright(void) { return true; }
617 float getMotorOutputLow(void) { return 1000.0; }
619 float getMotorOutputHigh(void) { return 2047.0; }