| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576 |
- /*
- * This file is part of the Micro Python project, http://micropython.org/
- *
- * The MIT License (MIT)
- *
- * Copyright (c) 2015 Damien P. George
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- * THE SOFTWARE.
- */
- #include <string.h>
- #include "py/obj.h"
- #include "py/runtime.h"
- #include "py/gc.h"
- #include "nrf_gpio.h"
- #include "microbitimage.h"
- #include "microbitdisplay.h"
- #include "iters.h"
- #include "ticker.h"
- #define min(a,b) (((a)<(b))?(a):(b))
- void microbit_display_show(microbit_display_obj_t *display, microbit_image_obj_t *image) {
- mp_int_t w = min(imageWidth(image), 5);
- mp_int_t h = min(imageHeight(image), 5);
- mp_int_t x = 0;
- mp_int_t brightnesses = 0;
- for (; x < w; ++x) {
- mp_int_t y = 0;
- for (; y < h; ++y) {
- uint8_t pix = imageGetPixelValue(image, x, y);
- display->image_buffer[x][y] = pix;
- brightnesses |= (1 << pix);
- }
- for (; y < 5; ++y) {
- display->image_buffer[x][y] = 0;
- }
- }
- for (; x < 5; ++x) {
- for (mp_int_t y = 0; y < 5; ++y) {
- display->image_buffer[x][y] = 0;
- }
- }
- display->brightnesses = brightnesses;
- }
- #define DEFAULT_PRINT_SPEED 400
- mp_obj_t microbit_display_show_func(mp_uint_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args) {
- // Cancel any animations.
- MP_STATE_PORT(async_data)[0] = NULL;
- MP_STATE_PORT(async_data)[1] = NULL;
- static const mp_arg_t show_allowed_args[] = {
- { MP_QSTR_image, MP_ARG_REQUIRED | MP_ARG_OBJ, {.u_obj = MP_OBJ_NULL} },
- { MP_QSTR_delay, MP_ARG_INT, {.u_int = DEFAULT_PRINT_SPEED} },
- { MP_QSTR_clear, MP_ARG_KW_ONLY | MP_ARG_BOOL, {.u_bool = false} },
- { MP_QSTR_wait, MP_ARG_KW_ONLY | MP_ARG_BOOL, {.u_bool = true} },
- { MP_QSTR_loop, MP_ARG_KW_ONLY | MP_ARG_BOOL, {.u_bool = false} },
- };
- // Parse the args.
- microbit_display_obj_t *self = (microbit_display_obj_t*)pos_args[0];
- mp_arg_val_t args[MP_ARRAY_SIZE(show_allowed_args)];
- mp_arg_parse_all(n_args - 1, pos_args + 1, kw_args, MP_ARRAY_SIZE(show_allowed_args), show_allowed_args, args);
- mp_obj_t image = args[0].u_obj;
- mp_int_t delay = args[1].u_int;
- bool clear = args[2].u_bool;
- bool wait = args[3].u_bool;
- bool loop = args[4].u_bool;
- if (MP_OBJ_IS_STR(image)) {
- // arg is a string object
- mp_uint_t len;
- const char *str = mp_obj_str_get_data(image, &len);
- if (len == 0) {
- // There are no chars; do nothing.
- return mp_const_none;
- } else if (len == 1) {
- if (!clear && !loop) {
- // A single char; convert to an image and print that.
- image = microbit_image_for_char(str[0]);
- goto single_image_immediate;
- }
- }
- image = microbit_string_facade(image);
- } else if (mp_obj_get_type(image) == µbit_image_type) {
- if (!clear && !loop) {
- goto single_image_immediate;
- }
- image = mp_obj_new_tuple(1, &image);
- }
- // iterable:
- if (args[4].u_bool) { /*loop*/
- image = microbit_repeat_iterator(image);
- }
- microbit_display_animate(self, image, delay, clear, wait);
- return mp_const_none;
- single_image_immediate:
- microbit_display_show(self, (microbit_image_obj_t *)image);
- return mp_const_none;
- }
- MP_DEFINE_CONST_FUN_OBJ_KW(microbit_display_show_obj, 1, microbit_display_show_func);
- static uint8_t async_mode;
- static mp_obj_t async_iterator = NULL;
- // Record if an error occurs in async animation. Unfortunately there is no way to report this.
- static volatile bool wakeup_event = false;
- static mp_uint_t async_delay = 1000;
- static mp_uint_t async_tick = 0;
- static bool async_clear = false;
- bool microbit_display_active_animation(void) {
- return async_mode == ASYNC_MODE_ANIMATION;
- }
- STATIC void async_stop(void) {
- async_iterator = NULL;
- async_mode = ASYNC_MODE_STOPPED;
- async_tick = 0;
- async_delay = 1000;
- async_clear = false;
- MP_STATE_PORT(async_data)[0] = NULL;
- MP_STATE_PORT(async_data)[1] = NULL;
- wakeup_event = true;
- }
- STATIC void wait_for_event(void) {
- while (!wakeup_event) {
- // allow CTRL-C to stop the animation
- if (MP_STATE_VM(mp_pending_exception) != MP_OBJ_NULL) {
- async_stop();
- return;
- }
- __WFI();
- }
- wakeup_event = false;
- }
- typedef struct {
- uint8_t x;
- uint8_t y;
- } DisplayPoint;
- #define NO_CONN 0
- #define ROW_COUNT 3
- #define COLUMN_COUNT 9
- static const DisplayPoint display_map[COLUMN_COUNT][ROW_COUNT] = {
- {{0,0}, {4,2}, {2,4}},
- {{2,0}, {0,2}, {4,4}},
- {{4,0}, {2,2}, {0,4}},
- {{4,3}, {1,0}, {0,1}},
- {{3,3}, {3,0}, {1,1}},
- {{2,3}, {3,4}, {2,1}},
- {{1,3}, {1,4}, {3,1}},
- {{0,3}, {NO_CONN,NO_CONN}, {4,1}},
- {{1,2}, {NO_CONN,NO_CONN}, {3,2}}
- };
- #define MIN_COLUMN_PIN 4
- #define COLUMN_PINS_MASK 0x1ff0
- #define MIN_ROW_PIN 13
- #define MAX_ROW_PIN 15
- #define ROW_PINS_MASK 0xe000
- static inline void displaySetPinsForRow(microbit_display_obj_t * p_display, uint8_t brightness) {
- if (brightness == 0) {
- nrf_gpio_port_out_clear(NRF_GPIO, COLUMN_PINS_MASK & ~p_display->pins_for_brightness[brightness]);
- } else {
- nrf_gpio_pin_set(p_display->pins_for_brightness[brightness]);
- }
- }
- /* This is the primary PWM driver/display driver. It will operate on one row
- * (9 pins) per invocation. It will turn on LEDs with maximum brightness,
- * then let the "callback" callback turn off the LEDs as appropriate for the
- * required brightness level.
- *
- * For each row
- * Turn off all the LEDs in the previous row
- * Set the column bits high (off)
- * Set the row strobe low (off)
- * Turn on all the LEDs in the current row that have maximum brightness
- * Set the row strobe high (on)
- * Set some/all column bits low (on)
- * Register the PWM callback
- * For each callback start with brightness 0
- * If brightness 0
- * Turn off the LEDs specified at this level
- * Else
- * Turn on the LEDs specified at this level
- * If brightness max
- * Disable the PWM callback
- * Else
- * Re-queue the PWM callback after the appropriate delay
- */
- static void displayAdvanceRow(microbit_display_obj_t * p_display) {
- /* Clear all of the column bits */
- nrf_gpio_port_out_set(NRF_GPIO, COLUMN_PINS_MASK);
- /* Clear the strobe bit for this row */
- nrf_gpio_pin_clear(p_display->strobe_row + MIN_ROW_PIN);
- /* Move to the next row. Before this, "this row" refers to the row
- * manipulated by the previous invocation of this function. After this,
- * "this row" refers to the row manipulated by the current invocation of
- * this function. */
- p_display->strobe_row++;
- // Reset the row counts and bit mask when we have hit the max.
- if (p_display->strobe_row == ROW_COUNT) {
- p_display->strobe_row = 0;
- }
- // Set pin for this row.
- // Prepare row for rendering.
- for (int i = 0; i <= MAX_BRIGHTNESS; i++) {
- p_display->pins_for_brightness[i] = 0;
- }
- for (int i = 0; i < COLUMN_COUNT; i++) {
- int x = display_map[i][p_display->strobe_row].x;
- int y = display_map[i][p_display->strobe_row].y;
- int brightness = microbit_display_obj.image_buffer[x][y];
- p_display->pins_for_brightness[brightness] |= (1<<(i+MIN_COLUMN_PIN));
- (void)brightness;
- }
- /* Enable the strobe bit for this row */
- nrf_gpio_pin_set(p_display->strobe_row + MIN_ROW_PIN);
- /* Enable the column bits for all pins that need to be on. */
- nrf_gpio_port_out_clear(NRF_GPIO, p_display->pins_for_brightness[MAX_BRIGHTNESS]);
- }
- static const uint16_t render_timings[] =
- // The scale is (approximately) exponential,
- // each step is approx x1.9 greater than the previous.
- { 0, // Bright, Ticks Duration, Relative power
- 2, // 1, 2, 32µs, inf
- 2, // 2, 4, 64µs, 200%
- 4, // 3, 8, 128µs, 200%
- 7, // 4, 15, 240µs, 187%
- 13, // 5, 28, 448µs, 187%
- 25, // 6, 53, 848µs, 189%
- 49, // 7, 102, 1632µs, 192%
- 97, // 8, 199, 3184µs, 195%
- // Always on 9, 375, 6000µs, 188%
- };
- #define DISPLAY_TICKER_SLOT 1
- /* This is the PWM callback. It is registered by the animation callback and
- * will unregister itself when all of the brightness steps are complete. */
- static int32_t callback(void) {
- microbit_display_obj_t *display = µbit_display_obj;
- mp_uint_t brightness = display->previous_brightness;
- displaySetPinsForRow(display, brightness);
- brightness += 1;
- if (brightness == MAX_BRIGHTNESS) {
- clear_ticker_callback(DISPLAY_TICKER_SLOT);
- return -1;
- }
- display->previous_brightness = brightness;
- // Return interval (in 16µs ticks) until next callback
- return render_timings[brightness];
- }
- static void draw_object(mp_obj_t obj) {
- microbit_display_obj_t *display = (microbit_display_obj_t*)MP_STATE_PORT(async_data)[0];
- if (obj == MP_OBJ_STOP_ITERATION) {
- if (async_clear) {
- microbit_display_show(µbit_display_obj, BLANK_IMAGE);
- async_clear = false;
- } else {
- async_stop();
- }
- } else if (mp_obj_get_type(obj) == µbit_image_type) {
- microbit_display_show(display, (microbit_image_obj_t *)obj);
- } else if (MP_OBJ_IS_STR(obj)) {
- mp_uint_t len;
- const char *str = mp_obj_str_get_data(obj, &len);
- if (len == 1) {
- microbit_display_show(display, microbit_image_for_char(str[0]));
- } else {
- async_stop();
- }
- } else {
- MP_STATE_VM(mp_pending_exception) = mp_obj_new_exception_msg(&mp_type_TypeError, "not an image.");
- async_stop();
- }
- }
- static void microbit_display_update(void) {
- async_tick += MILLISECONDS_PER_MACRO_TICK;
- if (async_tick < async_delay) {
- return;
- }
- async_tick = 0;
- switch (async_mode) {
- case ASYNC_MODE_ANIMATION:
- {
- if (MP_STATE_PORT(async_data)[0] == NULL || MP_STATE_PORT(async_data)[1] == NULL) {
- async_stop();
- break;
- }
- /* WARNING: We are executing in an interrupt handler.
- * If an exception is raised here then we must hand it to the VM. */
- mp_obj_t obj;
- nlr_buf_t nlr;
- gc_lock();
- if (nlr_push(&nlr) == 0) {
- obj = mp_iternext_allow_raise(async_iterator);
- nlr_pop();
- gc_unlock();
- } else {
- gc_unlock();
- if (!mp_obj_is_subclass_fast(MP_OBJ_FROM_PTR(((mp_obj_base_t*)nlr.ret_val)->type),
- MP_OBJ_FROM_PTR(&mp_type_StopIteration))) {
- // An exception other than StopIteration, so set it for the VM to raise later
- // If memory error, write an appropriate message.
- if (mp_obj_get_type(nlr.ret_val) == &mp_type_MemoryError) {
- mp_printf(&mp_plat_print, "Allocation in interrupt handler");
- }
- MP_STATE_VM(mp_pending_exception) = MP_OBJ_FROM_PTR(nlr.ret_val);
- }
- obj = MP_OBJ_STOP_ITERATION;
- }
- draw_object(obj);
- break;
- }
- case ASYNC_MODE_CLEAR:
- microbit_display_show(µbit_display_obj, BLANK_IMAGE);
- async_stop();
- break;
- }
- }
- #define GREYSCALE_MASK ((1<<MAX_BRIGHTNESS)-2)
- /* This is the top-level animation/display registered callback. */
- void microbit_display_tick(void) {
- /* Do nothing if the display is not active. */
- if (!microbit_display_obj.active) {
- return;
- }
- displayAdvanceRow(µbit_display_obj);
- microbit_display_update();
- microbit_display_obj.previous_brightness = 0;
- if (microbit_display_obj.brightnesses & GREYSCALE_MASK) {
- set_ticker_callback(DISPLAY_TICKER_SLOT, callback, 1800);
- }
- }
- void microbit_display_animate(microbit_display_obj_t *self, mp_obj_t iterable, mp_int_t delay, bool clear, bool wait) {
- // Reset the repeat state.
- MP_STATE_PORT(async_data)[0] = NULL;
- MP_STATE_PORT(async_data)[1] = NULL;
- async_iterator = mp_getiter(iterable, NULL);
- async_delay = delay;
- async_clear = clear;
- MP_STATE_PORT(async_data)[0] = self; // so it doesn't get GC'd
- MP_STATE_PORT(async_data)[1] = async_iterator;
- wakeup_event = false;
- mp_obj_t obj = mp_iternext_allow_raise(async_iterator);
- draw_object(obj);
- async_tick = 0;
- async_mode = ASYNC_MODE_ANIMATION;
- if (wait) {
- wait_for_event();
- }
- }
- // Delay in ms in between moving display one column to the left.
- #define DEFAULT_SCROLL_SPEED 150
- void microbit_display_scroll(microbit_display_obj_t *self, const char* str, bool wait) {
- mp_obj_t iterable = scrolling_string_image_iterable(str, strlen(str), NULL, false, false);
- microbit_display_animate(self, iterable, DEFAULT_SCROLL_SPEED, false, wait);
- }
- mp_obj_t microbit_display_scroll_func(mp_uint_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args) {
- static const mp_arg_t scroll_allowed_args[] = {
- { MP_QSTR_text, MP_ARG_REQUIRED | MP_ARG_OBJ, {.u_obj = MP_OBJ_NULL} },
- { MP_QSTR_delay, MP_ARG_INT, {.u_int = DEFAULT_SCROLL_SPEED} },
- { MP_QSTR_wait, MP_ARG_KW_ONLY | MP_ARG_BOOL, {.u_bool = true} },
- { MP_QSTR_monospace, MP_ARG_KW_ONLY | MP_ARG_BOOL, {.u_bool = false} },
- { MP_QSTR_loop, MP_ARG_KW_ONLY | MP_ARG_BOOL, {.u_bool = false} },
- };
- // Parse the args.
- microbit_display_obj_t *self = (microbit_display_obj_t*)pos_args[0];
- mp_arg_val_t args[MP_ARRAY_SIZE(scroll_allowed_args)];
- mp_arg_parse_all(n_args - 1, pos_args + 1, kw_args, MP_ARRAY_SIZE(scroll_allowed_args), scroll_allowed_args, args);
- mp_uint_t len;
- const char* str = mp_obj_str_get_data(args[0].u_obj, &len);
- mp_obj_t iterable = scrolling_string_image_iterable(str, len, args[0].u_obj, args[3].u_bool /*monospace?*/, args[4].u_bool /*loop*/);
- microbit_display_animate(self, iterable, args[1].u_int /*delay*/, false/*clear*/, args[2].u_bool/*wait?*/);
- return mp_const_none;
- }
- MP_DEFINE_CONST_FUN_OBJ_KW(microbit_display_scroll_obj, 1, microbit_display_scroll_func);
- mp_obj_t microbit_display_on_func(mp_obj_t obj) {
- microbit_display_obj_t *self = (microbit_display_obj_t*)obj;
- /* Try to reclaim the pins we need */
- /*
- microbit_obj_pin_fail_if_cant_acquire(µbit_p3_obj);
- microbit_obj_pin_fail_if_cant_acquire(µbit_p4_obj);
- microbit_obj_pin_fail_if_cant_acquire(µbit_p6_obj);
- microbit_obj_pin_fail_if_cant_acquire(µbit_p7_obj);
- microbit_obj_pin_fail_if_cant_acquire(µbit_p9_obj);
- microbit_obj_pin_fail_if_cant_acquire(µbit_p10_obj);
- microbit_obj_pin_acquire(µbit_p3_obj, microbit_pin_mode_display);
- microbit_obj_pin_acquire(µbit_p4_obj, microbit_pin_mode_display);
- microbit_obj_pin_acquire(µbit_p6_obj, microbit_pin_mode_display);
- microbit_obj_pin_acquire(µbit_p7_obj, microbit_pin_mode_display);
- microbit_obj_pin_acquire(µbit_p9_obj, microbit_pin_mode_display);
- microbit_obj_pin_acquire(µbit_p10_obj, microbit_pin_mode_display);
- */
- /* Make sure all pins are in the correct state */
- microbit_display_init();
- /* Re-enable the display loop. This will resume any animations in
- * progress and display any static image. */
- self->active = true;
- return mp_const_none;
- }
- MP_DEFINE_CONST_FUN_OBJ_1(microbit_display_on_obj, microbit_display_on_func);
- mp_obj_t microbit_display_off_func(mp_obj_t obj) {
- microbit_display_obj_t *self = (microbit_display_obj_t*)obj;
- /* Disable the display loop. This will pause any animations in progress.
- * It will not prevent a user from attempting to modify the state, but
- * modifications will not appear to have any effect until the display loop
- * is re-enabled. */
- self->active = false;
- /* Disable the row strobes, allowing the columns to be used freely for
- * GPIO. */
- nrf_gpio_port_out_clear(0, ROW_PINS_MASK);
- /* Free pins for other uses */
- /*
- microbit_obj_pin_free(µbit_p3_obj);
- microbit_obj_pin_free(µbit_p4_obj);
- microbit_obj_pin_free(µbit_p6_obj);
- microbit_obj_pin_free(µbit_p7_obj);
- microbit_obj_pin_free(µbit_p9_obj);
- microbit_obj_pin_free(µbit_p10_obj);
- */
- return mp_const_none;
- }
- MP_DEFINE_CONST_FUN_OBJ_1(microbit_display_off_obj, microbit_display_off_func);
- mp_obj_t microbit_display_is_on_func(mp_obj_t obj) {
- microbit_display_obj_t *self = (microbit_display_obj_t*)obj;
- if (self->active) {
- return mp_const_true;
- }
- else {
- return mp_const_false;
- }
- }
- MP_DEFINE_CONST_FUN_OBJ_1(microbit_display_is_on_obj, microbit_display_is_on_func);
- void microbit_display_clear(void) {
- // Reset repeat state, cancel animation and clear screen.
- wakeup_event = false;
- async_mode = ASYNC_MODE_CLEAR;
- async_tick = async_delay - MILLISECONDS_PER_MACRO_TICK;
- wait_for_event();
- }
- mp_obj_t microbit_display_clear_func(mp_obj_t self_in) {
- microbit_display_clear();
- return mp_const_none;
- }
- MP_DEFINE_CONST_FUN_OBJ_1(microbit_display_clear_obj, microbit_display_clear_func);
- void microbit_display_set_pixel(microbit_display_obj_t *display, mp_int_t x, mp_int_t y, mp_int_t bright) {
- if (x < 0 || y < 0 || x > 4 || y > 4) {
- mp_raise_ValueError("index out of bounds.");
- }
- if (bright < 0 || bright > MAX_BRIGHTNESS) {
- mp_raise_ValueError("brightness out of bounds.");
- }
- display->image_buffer[x][y] = bright;
- display->brightnesses |= (1 << bright);
- }
- STATIC mp_obj_t microbit_display_set_pixel_func(mp_uint_t n_args, const mp_obj_t *args) {
- (void)n_args;
- microbit_display_obj_t *self = (microbit_display_obj_t*)args[0];
- microbit_display_set_pixel(self, mp_obj_get_int(args[1]), mp_obj_get_int(args[2]), mp_obj_get_int(args[3]));
- return mp_const_none;
- }
- MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(microbit_display_set_pixel_obj, 4, 4, microbit_display_set_pixel_func);
- mp_int_t microbit_display_get_pixel(microbit_display_obj_t *display, mp_int_t x, mp_int_t y) {
- if (x < 0 || y < 0 || x > 4 || y > 4) {
- mp_raise_ValueError("index out of bounds.");
- }
- return display->image_buffer[x][y];
- }
- STATIC mp_obj_t microbit_display_get_pixel_func(mp_obj_t self_in, mp_obj_t x_in, mp_obj_t y_in) {
- microbit_display_obj_t *self = (microbit_display_obj_t*)self_in;
- return MP_OBJ_NEW_SMALL_INT(microbit_display_get_pixel(self, mp_obj_get_int(x_in), mp_obj_get_int(y_in)));
- }
- MP_DEFINE_CONST_FUN_OBJ_3(microbit_display_get_pixel_obj, microbit_display_get_pixel_func);
- STATIC const mp_rom_map_elem_t microbit_display_locals_dict_table[] = {
- { MP_ROM_QSTR(MP_QSTR_get_pixel), MP_ROM_PTR(µbit_display_get_pixel_obj) },
- { MP_ROM_QSTR(MP_QSTR_set_pixel), MP_ROM_PTR(µbit_display_set_pixel_obj) },
- { MP_ROM_QSTR(MP_QSTR_show), MP_ROM_PTR(µbit_display_show_obj) },
- { MP_ROM_QSTR(MP_QSTR_scroll), MP_ROM_PTR(µbit_display_scroll_obj) },
- { MP_ROM_QSTR(MP_QSTR_clear), MP_ROM_PTR(µbit_display_clear_obj) },
- { MP_ROM_QSTR(MP_QSTR_on), MP_ROM_PTR(µbit_display_on_obj) },
- { MP_ROM_QSTR(MP_QSTR_off), MP_ROM_PTR(µbit_display_off_obj) },
- { MP_ROM_QSTR(MP_QSTR_is_on), MP_ROM_PTR(µbit_display_is_on_obj) },
- };
- STATIC MP_DEFINE_CONST_DICT(microbit_display_locals_dict, microbit_display_locals_dict_table);
- const mp_obj_type_t microbit_display_type = {
- { &mp_type_type },
- .name = MP_QSTR_MicroBitDisplay,
- .print = NULL,
- .make_new = NULL,
- .call = NULL,
- .unary_op = NULL,
- .binary_op = NULL,
- .attr = NULL,
- .subscr = NULL,
- .getiter = NULL,
- .iternext = NULL,
- .buffer_p = {NULL},
- .locals_dict = (mp_obj_dict_t*)µbit_display_locals_dict,
- };
- microbit_display_obj_t microbit_display_obj = {
- {µbit_display_type},
- {{ 0, }},
- .previous_brightness = 0,
- .active = 1,
- .strobe_row = 0,
- .brightnesses = 0,
- .pins_for_brightness = { 0 },
- };
- void microbit_display_init(void) {
- // Set pins as output.
- for (int i = MIN_COLUMN_PIN; i <= MAX_ROW_PIN; i++) {
- nrf_gpio_cfg_output(i);
- }
- }
|