|
44 | 44 | #include "supervisor/spi_flash_api.h"
|
45 | 45 | #endif
|
46 | 46 |
|
| 47 | +// The default indicates no primary display |
| 48 | +static int primary_display_number = -1; |
| 49 | + |
47 | 50 | primary_display_bus_t display_buses[CIRCUITPY_DISPLAY_LIMIT];
|
48 | 51 | primary_display_t displays[CIRCUITPY_DISPLAY_LIMIT];
|
49 | 52 |
|
@@ -108,10 +111,16 @@ void displayio_background(void) {
|
108 | 111 |
|
109 | 112 | }
|
110 | 113 |
|
111 |
| -void common_hal_displayio_release_displays(void) { |
| 114 | +static void common_hal_displayio_release_displays_impl(bool keep_primary) { |
112 | 115 | // Release displays before busses so that they can send any final commands to turn the display
|
113 | 116 | // off properly.
|
| 117 | + if (!keep_primary) { |
| 118 | + primary_display_number = -1; |
| 119 | + } |
114 | 120 | for (uint8_t i = 0; i < CIRCUITPY_DISPLAY_LIMIT; i++) {
|
| 121 | + if (i == primary_display_number) { |
| 122 | + continue; |
| 123 | + } |
115 | 124 | mp_const_obj_t display_type = displays[i].display_base.type;
|
116 | 125 | if (display_type == NULL || display_type == &mp_type_NoneType) {
|
117 | 126 | continue;
|
@@ -177,7 +186,14 @@ void common_hal_displayio_release_displays(void) {
|
177 | 186 | supervisor_stop_terminal();
|
178 | 187 | }
|
179 | 188 |
|
| 189 | +void common_hal_displayio_release_displays(void) { |
| 190 | + common_hal_displayio_release_displays_impl(false); |
| 191 | +} |
| 192 | + |
180 | 193 | void reset_displays(void) {
|
| 194 | + // In CircuitPython 10, release secondary displays before doing anything else: |
| 195 | + // common_hal_displayio_release_displays_impl(true); |
| 196 | + |
181 | 197 | // The SPI buses used by FourWires may be allocated on the heap so we need to move them inline.
|
182 | 198 | for (uint8_t i = 0; i < CIRCUITPY_DISPLAY_LIMIT; i++) {
|
183 | 199 | mp_const_obj_t display_bus_type = display_buses[i].bus_base.type;
|
@@ -392,10 +408,13 @@ void displayio_gc_collect(void) {
|
392 | 408 | }
|
393 | 409 | }
|
394 | 410 |
|
| 411 | +static bool is_display_active(mp_obj_base_t *display_maybe) { |
| 412 | + return display_maybe->type != &mp_type_NoneType && display_maybe->type != NULL; |
| 413 | +} |
| 414 | + |
395 | 415 | primary_display_t *allocate_display(void) {
|
396 | 416 | for (uint8_t i = 0; i < CIRCUITPY_DISPLAY_LIMIT; i++) {
|
397 |
| - mp_const_obj_t display_type = displays[i].display_base.type; |
398 |
| - if (display_type == NULL || display_type == &mp_type_NoneType) { |
| 417 | + if (!is_display_active(&displays[i].display_base)) { |
399 | 418 | // Clear this memory so it is in a known state before init.
|
400 | 419 | memset(&displays[i], 0, sizeof(displays[i]));
|
401 | 420 | // Default to None so that it works as board.DISPLAY.
|
@@ -434,3 +453,42 @@ primary_display_bus_t *allocate_display_bus_or_raise(void) {
|
434 | 453 | }
|
435 | 454 | mp_raise_RuntimeError(MP_ERROR_TEXT("Too many display busses; forgot displayio.release_displays() ?"));
|
436 | 455 | }
|
| 456 | + |
| 457 | +mp_obj_t common_hal_displayio_get_primary_display(void) { |
| 458 | + if (primary_display_number == -1 || primary_display_number >= CIRCUITPY_DISPLAY_LIMIT) { |
| 459 | + return mp_const_none; |
| 460 | + } |
| 461 | + mp_obj_base_t *primary_display = &displays[primary_display_number].display_base; |
| 462 | + if (is_display_active(primary_display)) { |
| 463 | + return MP_OBJ_FROM_PTR(primary_display); |
| 464 | + } |
| 465 | + return mp_const_none; |
| 466 | +} |
| 467 | + |
| 468 | +void common_hal_displayio_set_primary_display(mp_obj_t new_primary_display) { |
| 469 | + if (new_primary_display == mp_const_none) { |
| 470 | + primary_display_number = -1; |
| 471 | + return; |
| 472 | + } |
| 473 | + for (uint8_t i = 0; i < CIRCUITPY_DISPLAY_LIMIT; i++) { |
| 474 | + mp_obj_t display = MP_OBJ_FROM_PTR(&displays[i]); |
| 475 | + if (new_primary_display == display && is_display_active(display)) { |
| 476 | + primary_display_number = i; |
| 477 | + return; |
| 478 | + } |
| 479 | + } |
| 480 | + // object was not a display after all... |
| 481 | + mp_raise_TypeError_varg(MP_ERROR_TEXT("%q must be of type %q or %q, not %q"), MP_QSTR_Display, MP_QSTR_AnyDisplay, MP_QSTR_None, mp_obj_get_type(new_primary_display)->name); |
| 482 | +} |
| 483 | + |
| 484 | +void common_hal_displayio_auto_primary_display(void) { |
| 485 | + if (primary_display_number != -1) { |
| 486 | + return; |
| 487 | + } |
| 488 | + for (uint8_t i = 0; i < CIRCUITPY_DISPLAY_LIMIT; i++) { |
| 489 | + if (is_display_active(&displays[i].display_base)) { |
| 490 | + primary_display_number = i; |
| 491 | + return; |
| 492 | + } |
| 493 | + } |
| 494 | +} |
0 commit comments