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/>.
30 #include "common/utils.h"
32 #include "build/version.h"
33 #include "build/debug.h"
35 #include "build/build_config.h"
37 #include "drivers/bus.h"
38 #include "drivers/display.h"
39 #include "drivers/display_ug2864hsweg01.h"
40 #include "drivers/time.h"
44 #include "common/printf.h"
45 #include "common/maths.h"
46 #include "common/axis.h"
47 #include "common/typeconversion.h"
49 #include "config/feature.h"
50 #include "pg/dashboard.h"
53 #include "config/config.h"
54 #include "fc/controlrate_profile.h"
55 #include "fc/rc_controls.h"
56 #include "fc/runtime_config.h"
58 #include "flight/pid.h"
59 #include "flight/imu.h"
60 #include "flight/failsafe.h"
63 #include "io/dashboard.h"
64 #include "io/displayport_oled.h"
66 #include "blackbox/blackbox_io.h"
67 #include "blackbox/blackbox.h"
71 #include "scheduler/scheduler.h"
73 #include "sensors/acceleration.h"
74 #include "sensors/battery.h"
75 #include "sensors/compass.h"
76 #include "sensors/gyro.h"
77 #include "sensors/sensors.h"
79 #define MICROSECONDS_IN_A_SECOND (1000 * 1000)
81 #define DISPLAY_UPDATE_FREQUENCY (MICROSECONDS_IN_A_SECOND / 5)
82 #define PAGE_CYCLE_FREQUENCY (MICROSECONDS_IN_A_SECOND * 5)
84 static extDevice_t
*dev
;
86 static uint32_t nextDisplayUpdateAt
= 0;
87 static bool dashboardPresent
= false;
89 static displayPort_t
*displayPort
;
91 #define PAGE_TITLE_LINE_COUNT 1
93 static char lineBuffer
[SCREEN_CHARACTER_COLUMN_COUNT
+ 1];
95 #define HALF_SCREEN_CHARACTER_COLUMN_COUNT (SCREEN_CHARACTER_COLUMN_COUNT / 2)
96 #define IS_SCREEN_CHARACTER_COLUMN_COUNT_ODD (SCREEN_CHARACTER_COLUMN_COUNT & 1)
98 typedef void (*pageFnPtr
)(void);
100 #define PAGE_FLAGS_NONE 0
101 #define PAGE_FLAGS_SKIP_CYCLING (1 << 0)
103 typedef struct pageEntry_s
{
110 static const char tickerCharacters
[] = "|/-\\"; // use 2/4/8 characters so that the divide is optimal.
111 #define TICKER_CHARACTER_COUNT (sizeof(tickerCharacters) / sizeof(char))
114 PAGE_STATE_FLAG_NONE
= 0,
115 PAGE_STATE_FLAG_CYCLE_ENABLED
= (1 << 0),
116 PAGE_STATE_FLAG_FORCE_PAGE_CHANGE
= (1 << 1),
119 typedef struct pageState_s
{
121 const pageEntry_t
*page
;
127 static pageState_t pageState
;
129 static void resetDisplay(void)
131 dashboardPresent
= ug2864hsweg01InitI2C(dev
);
134 void LCDprint(uint8_t i
)
136 i2c_OLED_send_char(dev
, i
);
139 static void padLineBuffer(void)
141 uint8_t length
= strlen(lineBuffer
);
142 while (length
< sizeof(lineBuffer
) - 1) {
143 lineBuffer
[length
++] = ' ';
145 lineBuffer
[length
] = 0;
149 static void padHalfLineBuffer(void)
151 uint8_t halfLineIndex
= sizeof(lineBuffer
) / 2;
152 uint8_t length
= strlen(lineBuffer
);
153 while (length
< halfLineIndex
- 1) {
154 lineBuffer
[length
++] = ' ';
156 lineBuffer
[length
] = 0;
160 // LCDbar(n,v) : draw a bar graph - n number of chars for width, v value in % to display
161 static void drawHorizonalPercentageBar(uint8_t width
,uint8_t percent
)
168 j
= (width
* percent
) / 100;
170 for (i
= 0; i
< j
; i
++)
171 LCDprint(159); // full
174 LCDprint(154 + (percent
* width
* 5 / 100 - 5 * j
)); // partial fill
176 for (i
= j
+ 1; i
< width
; i
++)
177 LCDprint(154); // empty
181 static void fillScreenWithCharacters(void)
183 for (uint8_t row
= 0; row
< SCREEN_CHARACTER_ROW_COUNT
; row
++) {
184 for (uint8_t column
= 0; column
< SCREEN_CHARACTER_COLUMN_COUNT
; column
++) {
185 i2c_OLED_set_xy(dev
, column
, row
);
186 i2c_OLED_send_char(dev
, 'A' + column
);
193 static void updateTicker(void)
195 static uint8_t tickerIndex
= 0;
196 i2c_OLED_set_xy(dev
, SCREEN_CHARACTER_COLUMN_COUNT
- 1, 0);
197 i2c_OLED_send_char(dev
, tickerCharacters
[tickerIndex
]);
199 tickerIndex
= tickerIndex
% TICKER_CHARACTER_COUNT
;
202 static void updateRxStatus(void)
204 i2c_OLED_set_xy(dev
, SCREEN_CHARACTER_COLUMN_COUNT
- 2, 0);
206 if (rxIsReceivingSignal()) {
208 } if (rxAreFlightChannelsValid()) {
211 i2c_OLED_send_char(dev
, rxStatus
);
214 static void updateFailsafeStatus(void)
216 char failsafeIndicator
= '?';
217 switch (failsafePhase()) {
219 failsafeIndicator
= '-';
221 case FAILSAFE_RX_LOSS_DETECTED
:
222 failsafeIndicator
= 'R';
224 case FAILSAFE_LANDING
:
225 failsafeIndicator
= 'l';
227 case FAILSAFE_LANDED
:
228 failsafeIndicator
= 'L';
230 case FAILSAFE_RX_LOSS_MONITORING
:
231 failsafeIndicator
= 'M';
233 case FAILSAFE_RX_LOSS_RECOVERED
:
234 failsafeIndicator
= 'r';
236 case FAILSAFE_GPS_RESCUE
:
237 failsafeIndicator
= 'G';
240 i2c_OLED_set_xy(dev
, SCREEN_CHARACTER_COLUMN_COUNT
- 3, 0);
241 i2c_OLED_send_char(dev
, failsafeIndicator
);
244 static void showTitle(void)
246 i2c_OLED_set_line(dev
, 0);
247 i2c_OLED_send_string(dev
, pageState
.page
->title
);
250 static void handlePageChange(void)
252 i2c_OLED_clear_display_quick(dev
);
256 static void drawRxChannel(uint8_t channelIndex
, uint8_t width
)
258 LCDprint(rcChannelLetters
[channelIndex
]);
260 const uint32_t percentage
= (constrain(rcData
[channelIndex
], PWM_RANGE_MIN
, PWM_RANGE_MAX
) - PWM_RANGE_MIN
) * 100 / (PWM_RANGE_MAX
- PWM_RANGE_MIN
);
261 drawHorizonalPercentageBar(width
- 1, percentage
);
264 #define RX_CHANNELS_PER_PAGE_COUNT 14
265 static void showRxPage(void)
267 for (int channelIndex
= 0; channelIndex
< rxRuntimeState
.channelCount
&& channelIndex
< RX_CHANNELS_PER_PAGE_COUNT
; channelIndex
+= 2) {
268 i2c_OLED_set_line(dev
, (channelIndex
/ 2) + PAGE_TITLE_LINE_COUNT
);
270 drawRxChannel(channelIndex
, HALF_SCREEN_CHARACTER_COLUMN_COUNT
);
272 if (channelIndex
>= rxRuntimeState
.channelCount
) {
276 if (IS_SCREEN_CHARACTER_COLUMN_COUNT_ODD
) {
280 drawRxChannel(channelIndex
+ PAGE_TITLE_LINE_COUNT
, HALF_SCREEN_CHARACTER_COLUMN_COUNT
);
284 static void showWelcomePage(void)
286 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
288 tfp_sprintf(lineBuffer
, "v%s (%s)", FC_VERSION_STRING
, shortGitRevision
);
289 i2c_OLED_set_line(dev
, rowIndex
++);
290 i2c_OLED_send_string(dev
, lineBuffer
);
292 i2c_OLED_set_line(dev
, rowIndex
++);
293 i2c_OLED_send_string(dev
, targetName
);
296 static void showArmedPage(void)
300 static void showProfilePage(void)
302 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
304 tfp_sprintf(lineBuffer
, "Profile: %d", getCurrentPidProfileIndex());
305 i2c_OLED_set_line(dev
, rowIndex
++);
306 i2c_OLED_send_string(dev
, lineBuffer
);
308 static const char* const axisTitles
[3] = {"ROL", "PIT", "YAW"};
309 const pidProfile_t
*pidProfile
= currentPidProfile
;
310 for (int axis
= 0; axis
< 3; ++axis
) {
311 tfp_sprintf(lineBuffer
, "%s P:%3d I:%3d D:%3d",
313 pidProfile
->pid
[axis
].P
,
314 pidProfile
->pid
[axis
].I
,
315 pidProfile
->pid
[axis
].D
318 i2c_OLED_set_line(dev
, rowIndex
++);
319 i2c_OLED_send_string(dev
, lineBuffer
);
323 static void showRateProfilePage(void)
325 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
327 const uint8_t currentRateProfileIndex
= getCurrentControlRateProfileIndex();
328 tfp_sprintf(lineBuffer
, "Rate profile: %d", currentRateProfileIndex
);
329 i2c_OLED_set_line(dev
, rowIndex
++);
330 i2c_OLED_send_string(dev
, lineBuffer
);
332 const controlRateConfig_t
*controlRateConfig
= controlRateProfiles(currentRateProfileIndex
);
334 tfp_sprintf(lineBuffer
, " R P Y");
336 i2c_OLED_set_line(dev
, rowIndex
++);
337 i2c_OLED_send_string(dev
, lineBuffer
);
339 tfp_sprintf(lineBuffer
, "RcRate %3d %3d %3d",
340 controlRateConfig
->rcRates
[FD_ROLL
],
341 controlRateConfig
->rcRates
[FD_PITCH
],
342 controlRateConfig
->rcRates
[FD_YAW
]
345 i2c_OLED_set_line(dev
, rowIndex
++);
346 i2c_OLED_send_string(dev
, lineBuffer
);
348 tfp_sprintf(lineBuffer
, "Super %3d %3d %3d",
349 controlRateConfig
->rates
[FD_ROLL
],
350 controlRateConfig
->rates
[FD_PITCH
],
351 controlRateConfig
->rates
[FD_YAW
]
354 i2c_OLED_set_line(dev
, rowIndex
++);
355 i2c_OLED_send_string(dev
, lineBuffer
);
357 tfp_sprintf(lineBuffer
, "Expo %3d %3d %3d",
358 controlRateConfig
->rcExpo
[FD_ROLL
],
359 controlRateConfig
->rcExpo
[FD_PITCH
],
360 controlRateConfig
->rcExpo
[FD_YAW
]
363 i2c_OLED_set_line(dev
, rowIndex
++);
364 i2c_OLED_send_string(dev
, lineBuffer
);
367 #define SATELLITE_COUNT (sizeof(GPS_svinfo_cno) / sizeof(GPS_svinfo_cno[0]))
368 #define SATELLITE_GRAPH_LEFT_OFFSET ((SCREEN_CHARACTER_COLUMN_COUNT - SATELLITE_COUNT) / 2)
371 static void showGpsPage(void)
373 if (!featureIsEnabled(FEATURE_GPS
)) {
374 pageState
.pageFlags
|= PAGE_STATE_FLAG_FORCE_PAGE_CHANGE
;
378 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
380 static uint8_t gpsTicker
= 0;
381 static uint32_t lastGPSSvInfoReceivedCount
= 0;
382 if (GPS_svInfoReceivedCount
!= lastGPSSvInfoReceivedCount
) {
383 lastGPSSvInfoReceivedCount
= GPS_svInfoReceivedCount
;
385 gpsTicker
= gpsTicker
% TICKER_CHARACTER_COUNT
;
388 i2c_OLED_set_xy(dev
, 0, rowIndex
);
389 i2c_OLED_send_char(dev
, tickerCharacters
[gpsTicker
]);
391 i2c_OLED_set_xy(dev
, MAX(0, (uint8_t)SATELLITE_GRAPH_LEFT_OFFSET
), rowIndex
++);
394 for (index
= 0; index
< SATELLITE_COUNT
&& index
< SCREEN_CHARACTER_COLUMN_COUNT
; index
++) {
395 uint8_t bargraphOffset
= ((uint16_t) GPS_svinfo_cno
[index
] * VERTICAL_BARGRAPH_CHARACTER_COUNT
) / (GPS_DBHZ_MAX
- 1);
396 bargraphOffset
= MIN(bargraphOffset
, VERTICAL_BARGRAPH_CHARACTER_COUNT
- 1);
397 i2c_OLED_send_char(dev
, VERTICAL_BARGRAPH_ZERO_CHARACTER
+ bargraphOffset
);
401 char fixChar
= STATE(GPS_FIX
) ? 'Y' : 'N';
402 tfp_sprintf(lineBuffer
, "Sats: %d Fix: %c", gpsSol
.numSat
, fixChar
);
404 i2c_OLED_set_line(dev
, rowIndex
++);
405 i2c_OLED_send_string(dev
, lineBuffer
);
407 tfp_sprintf(lineBuffer
, "La/Lo: %d/%d", gpsSol
.llh
.lat
/ GPS_DEGREES_DIVIDER
, gpsSol
.llh
.lon
/ GPS_DEGREES_DIVIDER
);
409 i2c_OLED_set_line(dev
, rowIndex
++);
410 i2c_OLED_send_string(dev
, lineBuffer
);
412 tfp_sprintf(lineBuffer
, "Spd: %d", gpsSol
.groundSpeed
);
414 i2c_OLED_set_line(dev
, rowIndex
);
415 i2c_OLED_send_string(dev
, lineBuffer
);
417 tfp_sprintf(lineBuffer
, "GC: %d", gpsSol
.groundCourse
);
419 i2c_OLED_set_xy(dev
, HALF_SCREEN_CHARACTER_COLUMN_COUNT
, rowIndex
++);
420 i2c_OLED_send_string(dev
, lineBuffer
);
422 tfp_sprintf(lineBuffer
, "RX: %d", GPS_packetCount
);
424 i2c_OLED_set_line(dev
, rowIndex
);
425 i2c_OLED_send_string(dev
, lineBuffer
);
427 tfp_sprintf(lineBuffer
, "ERRs: %d", gpsData
.errors
);
429 i2c_OLED_set_xy(dev
, HALF_SCREEN_CHARACTER_COLUMN_COUNT
, rowIndex
++);
430 i2c_OLED_send_string(dev
, lineBuffer
);
432 tfp_sprintf(lineBuffer
, "Dt: %d", gpsData
.lastMessage
- gpsData
.lastLastMessage
);
434 i2c_OLED_set_line(dev
, rowIndex
);
435 i2c_OLED_send_string(dev
, lineBuffer
);
437 tfp_sprintf(lineBuffer
, "TOs: %d", gpsData
.timeouts
);
439 i2c_OLED_set_xy(dev
, HALF_SCREEN_CHARACTER_COLUMN_COUNT
, rowIndex
++);
440 i2c_OLED_send_string(dev
, lineBuffer
);
442 strncpy(lineBuffer
, gpsPacketLog
, GPS_PACKET_LOG_ENTRY_COUNT
);
444 i2c_OLED_set_line(dev
, rowIndex
++);
445 i2c_OLED_send_string(dev
, lineBuffer
);
449 static void showBatteryPage(void)
451 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
453 if (batteryConfig()->voltageMeterSource
!= VOLTAGE_METER_NONE
) {
454 tfp_sprintf(lineBuffer
, "Volts: %d.%02d Cells: %d", getBatteryVoltage() / 100, getBatteryVoltage() % 100, getBatteryCellCount());
456 i2c_OLED_set_line(dev
, rowIndex
++);
457 i2c_OLED_send_string(dev
, lineBuffer
);
459 uint8_t batteryPercentage
= calculateBatteryPercentageRemaining();
460 i2c_OLED_set_line(dev
, rowIndex
++);
461 drawHorizonalPercentageBar(SCREEN_CHARACTER_COLUMN_COUNT
, batteryPercentage
);
464 if (batteryConfig()->currentMeterSource
!= CURRENT_METER_NONE
) {
466 int32_t amperage
= getAmperage();
467 // 123456789012345678901
468 // Amp: DDD.D mAh: DDDDD
469 tfp_sprintf(lineBuffer
, "Amp: %d.%d mAh: %d", amperage
/ 100, (amperage
% 100) / 10, getMAhDrawn());
471 i2c_OLED_set_line(dev
, rowIndex
++);
472 i2c_OLED_send_string(dev
, lineBuffer
);
474 uint8_t capacityPercentage
= calculateBatteryPercentageRemaining();
475 i2c_OLED_set_line(dev
, rowIndex
++);
476 drawHorizonalPercentageBar(SCREEN_CHARACTER_COLUMN_COUNT
, capacityPercentage
);
480 static void showSensorsPage(void)
482 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
483 static const char *format
= "%s %5d %5d %5d";
485 i2c_OLED_set_line(dev
, rowIndex
++);
486 i2c_OLED_send_string(dev
, " X Y Z");
489 if (sensors(SENSOR_ACC
)) {
490 tfp_sprintf(lineBuffer
, format
, "ACC", lrintf(acc
.accADC
[X
]), lrintf(acc
.accADC
[Y
]), lrintf(acc
.accADC
[Z
]));
492 i2c_OLED_set_line(dev
, rowIndex
++);
493 i2c_OLED_send_string(dev
, lineBuffer
);
497 if (sensors(SENSOR_GYRO
)) {
498 tfp_sprintf(lineBuffer
, format
, "GYR", lrintf(gyro
.gyroADCf
[X
]), lrintf(gyro
.gyroADCf
[Y
]), lrintf(gyro
.gyroADCf
[Z
]));
500 i2c_OLED_set_line(dev
, rowIndex
++);
501 i2c_OLED_send_string(dev
, lineBuffer
);
505 if (sensors(SENSOR_MAG
)) {
506 tfp_sprintf(lineBuffer
, format
, "MAG", lrintf(mag
.magADC
[X
]), lrintf(mag
.magADC
[Y
]), lrintf(mag
.magADC
[Z
]));
508 i2c_OLED_set_line(dev
, rowIndex
++);
509 i2c_OLED_send_string(dev
, lineBuffer
);
513 tfp_sprintf(lineBuffer
, format
, "I&H", attitude
.values
.roll
, attitude
.values
.pitch
, DECIDEGREES_TO_DEGREES(attitude
.values
.yaw
));
515 i2c_OLED_set_line(dev
, rowIndex
++);
516 i2c_OLED_send_string(dev
, lineBuffer
);
521 ftoa(EstG.A[X], lineBuffer);
522 length = strlen(lineBuffer);
523 while (length < HALF_SCREEN_CHARACTER_COLUMN_COUNT) {
524 lineBuffer[length++] = ' ';
525 lineBuffer[length+1] = 0;
527 ftoa(EstG.A[Y], lineBuffer + length);
529 i2c_OLED_set_line(dev, rowIndex++);
530 i2c_OLED_send_string(dev, lineBuffer);
532 ftoa(EstG.A[Z], lineBuffer);
533 length = strlen(lineBuffer);
534 while (length < HALF_SCREEN_CHARACTER_COLUMN_COUNT) {
535 lineBuffer[length++] = ' ';
536 lineBuffer[length+1] = 0;
538 ftoa(smallAngle, lineBuffer + length);
540 i2c_OLED_set_line(dev, rowIndex++);
541 i2c_OLED_send_string(dev, lineBuffer);
546 static void showTasksPage(void)
548 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
549 static const char *format
= "%2d%6d%5d%4d%4d";
551 i2c_OLED_set_line(dev
, rowIndex
++);
552 i2c_OLED_send_string(dev
, "Task max avg mx% av%");
554 for (taskId_e taskId
= 0; taskId
< TASK_COUNT
; ++taskId
) {
555 getTaskInfo(taskId
, &taskInfo
);
556 if (taskInfo
.isEnabled
&& taskId
!= TASK_SERIAL
) {// don't waste a line of the display showing serial taskInfo
557 const int taskFrequency
= (int)(1000000.0f
/ ((float)taskInfo
.latestDeltaTimeUs
));
558 const int maxLoad
= taskInfo
.maxExecutionTimeUs
== 0 ? 0 : (taskInfo
.maxExecutionTimeUs
* taskFrequency
) / 1000;
559 const int averageLoad
= taskInfo
.averageExecutionTime10thUs
== 0 ? 0 : (taskInfo
.averageExecutionTime10thUs
* taskFrequency
) / 10000;
560 tfp_sprintf(lineBuffer
, format
, taskId
, taskInfo
.maxExecutionTimeUs
, taskInfo
.averageExecutionTime10thUs
/ 10, maxLoad
, averageLoad
);
562 i2c_OLED_set_line(dev
, rowIndex
++);
563 i2c_OLED_send_string(dev
, lineBuffer
);
564 if (rowIndex
> SCREEN_CHARACTER_ROW_COUNT
) {
572 static void showBBPage(void)
574 uint8_t rowIndex
= PAGE_TITLE_LINE_COUNT
;
576 if (isBlackboxDeviceWorking()) {
577 switch (blackboxConfig()->device
) {
578 case BLACKBOX_DEVICE_SDCARD
:
579 fileNo
= blackboxGetLogFileNo();
581 tfp_sprintf(lineBuffer
, "File no: %d", fileNo
);
583 tfp_sprintf(lineBuffer
, "Not ready yet");
587 tfp_sprintf(lineBuffer
, "Not supp. dev.");
591 tfp_sprintf(lineBuffer
, "BB not working");
595 i2c_OLED_set_line(dev
, rowIndex
++);
596 i2c_OLED_send_string(dev
, lineBuffer
);
600 #ifdef ENABLE_DEBUG_DASHBOARD_PAGE
602 static void showDebugPage(void)
604 for (int rowIndex
= 0; rowIndex
< 4; rowIndex
++) {
605 tfp_sprintf(lineBuffer
, "%d = %5d", rowIndex
, debug
[rowIndex
]);
607 i2c_OLED_set_line(dev
, rowIndex
+ PAGE_TITLE_LINE_COUNT
);
608 i2c_OLED_send_string(dev
, lineBuffer
);
614 static const pageEntry_t pages
[PAGE_COUNT
] = {
615 { PAGE_WELCOME
, FC_FIRMWARE_NAME
, showWelcomePage
, PAGE_FLAGS_SKIP_CYCLING
},
616 { PAGE_ARMED
, "ARMED", showArmedPage
, PAGE_FLAGS_SKIP_CYCLING
},
617 { PAGE_PROFILE
, "PROFILE", showProfilePage
, PAGE_FLAGS_NONE
},
618 { PAGE_RPROF
, "RATE PROFILE", showRateProfilePage
,PAGE_FLAGS_NONE
},
620 { PAGE_GPS
, "GPS", showGpsPage
, PAGE_FLAGS_NONE
},
622 { PAGE_RX
, "RX", showRxPage
, PAGE_FLAGS_NONE
},
623 { PAGE_BATTERY
, "BATTERY", showBatteryPage
, PAGE_FLAGS_NONE
},
624 { PAGE_SENSORS
, "SENSORS", showSensorsPage
, PAGE_FLAGS_NONE
},
625 { PAGE_TASKS
, "TASKS", showTasksPage
, PAGE_FLAGS_NONE
},
627 { PAGE_BB
, "BLACK BOX", showBBPage
, PAGE_FLAGS_NONE
},
629 #ifdef ENABLE_DEBUG_DASHBOARD_PAGE
630 { PAGE_DEBUG
, "DEBUG", showDebugPage
, PAGE_FLAGS_NONE
},
635 void dashboardSetPage(pageId_e pageId
)
637 for (int i
= 0; i
< PAGE_COUNT
; i
++) {
638 const pageEntry_t
*candidatePage
= &pages
[i
];
639 if (candidatePage
->id
== pageId
) {
640 pageState
.page
= candidatePage
;
643 pageState
.pageFlags
|= PAGE_STATE_FLAG_FORCE_PAGE_CHANGE
;
646 void dashboardUpdate(timeUs_t currentTimeUs
)
648 static uint8_t previousArmedState
= 0;
651 if (displayIsGrabbed(displayPort
)) {
656 const bool updateNow
= (int32_t)(currentTimeUs
- nextDisplayUpdateAt
) >= 0L;
661 nextDisplayUpdateAt
= currentTimeUs
+ DISPLAY_UPDATE_FREQUENCY
;
663 bool armedState
= ARMING_FLAG(ARMED
) ? true : false;
664 bool armedStateChanged
= armedState
!= previousArmedState
;
665 previousArmedState
= armedState
;
668 if (!armedStateChanged
) {
671 dashboardSetPage(PAGE_ARMED
);
672 pageState
.pageChanging
= true;
674 if (armedStateChanged
) {
675 pageState
.pageFlags
|= PAGE_STATE_FLAG_FORCE_PAGE_CHANGE
;
678 pageState
.pageChanging
= (pageState
.pageFlags
& PAGE_STATE_FLAG_FORCE_PAGE_CHANGE
) ||
679 (((int32_t)(currentTimeUs
- pageState
.nextPageAt
) >= 0L && (pageState
.pageFlags
& PAGE_STATE_FLAG_CYCLE_ENABLED
)));
680 if (pageState
.pageChanging
&& (pageState
.pageFlags
& PAGE_STATE_FLAG_CYCLE_ENABLED
)) {
683 pageState
.cycleIndex
++;
684 pageState
.cycleIndex
= pageState
.cycleIndex
% PAGE_COUNT
;
685 pageState
.page
= &pages
[pageState
.cycleIndex
];
686 } while (pageState
.page
->flags
& PAGE_FLAGS_SKIP_CYCLING
);
690 if (pageState
.pageChanging
) {
691 pageState
.pageFlags
&= ~PAGE_STATE_FLAG_FORCE_PAGE_CHANGE
;
692 pageState
.nextPageAt
= currentTimeUs
+ PAGE_CYCLE_FREQUENCY
;
694 // Some OLED displays do not respond on the first initialisation so refresh the display
695 // when the page changes in the hopes the hardware responds. This also allows the
696 // user to power off/on the display or connect it while powered.
699 if (!dashboardPresent
) {
705 if (!dashboardPresent
) {
709 pageState
.page
->drawFn();
712 updateFailsafeStatus();
719 void dashboardInit(void)
721 static extDevice_t dashBoardDev
;
722 // TODO Use bus singleton
723 static busDevice_t dashBoardBus
;
725 dashBoardDev
.bus
= &dashBoardBus
;
727 dashBoardBus
.busType_u
.i2c
.device
= I2C_CFG_TO_DEV(dashboardConfig()->device
);
728 dashBoardDev
.busType_u
.i2c
.address
= dashboardConfig()->address
;
735 displayPort
= displayPortOledInit(dev
);
737 if (dashboardPresent
) {
738 cmsDisplayPortRegister(displayPort
);
742 memset(&pageState
, 0, sizeof(pageState
));
743 dashboardSetPage(PAGE_WELCOME
);
745 uint32_t now
= micros();
746 dashboardUpdate(now
);
748 dashboardSetNextPageChangeAt(now
+ PAGE_CYCLE_FREQUENCY
);
751 void dashboardShowFixedPage(pageId_e pageId
)
753 dashboardSetPage(pageId
);
754 dashboardDisablePageCycling();
757 void dashboardSetNextPageChangeAt(timeUs_t futureMicros
)
759 pageState
.nextPageAt
= futureMicros
;
762 void dashboardEnablePageCycling(void)
764 pageState
.pageFlags
|= PAGE_STATE_FLAG_CYCLE_ENABLED
;
767 void dashboardResetPageCycling(void)
769 pageState
.cycleIndex
= PAGE_COUNT
- 1; // start at first page
773 void dashboardDisablePageCycling(void)
775 pageState
.pageFlags
&= ~PAGE_STATE_FLAG_CYCLE_ENABLED
;
777 #endif // USE_DASHBOARD