From 9efa78bf6817573cd29e6b281478bec07386b56e Mon Sep 17 00:00:00 2001 From: Scott Allen Date: Wed, 12 Apr 2017 17:36:28 -0400 Subject: [PATCH] Add alternative boot logo functions New functions bootLogoCompressed(), bootLogoSpritesSelfMasked() and bootLogoSpritesOverwrite() can be used in place of bootLogo() to reduce code size in cases where their drawing functions are shared with the same functions used by the sketch. New function bootLogoShell() added to provide common code for the above functions. Also, the Sprites class functions, and functions used for drawing the logos, were made static. --- README.md | 1 + keywords.txt | 4 +++ src/Arduboy2.cpp | 50 ++++++++++++++++++++++++--- src/Arduboy2.h | 90 +++++++++++++++++++++++++++++++++++++++++++----- src/Sprites.h | 26 +++++++------- src/ab_logo.c | 83 ++++++++++++++++++++++++++++++++++---------- 6 files changed, 210 insertions(+), 44 deletions(-) diff --git a/README.md b/README.md index 1f60579..e231157 100644 --- a/README.md +++ b/README.md @@ -211,6 +211,7 @@ This saves whatever code *blank()*, *systemButtons()* and *bootLogo()* would use There are a few functions provided that are roughly equivalent to the standard functions used by *begin()* but which use less code space. +- *bootLogoCompressed()*, *bootLogoSpritesSelfMasked()* and *bootLogoSpritesOverwrite()* will do the same as *bootLogo()* but will use *drawCompressed()*, or *Sprites* class *drawSelfMasked()* or *drawOverwrite()*, functions respectively, instead of *drawBitmask()*, to render the logo. If the sketch uses one of these functions, then using the boot logo function that also uses it may reduce code size. It's best to try each of them to see which one produces the smallest size. - *bootLogoText()* can be used in place *bootLogo()* in the case where the sketch uses text functions. It renders the logo as text instead of as a bitmap (so doesn't look as good). - *safeMode()* can be used in place of *flashlight()* for cases where it's needed to allow uploading a new sketch when the bootloader "magic key" problem is an issue. It only lights the red RGB LED, so you don't get the bright light that is the primary purpose of *flashlight()*. diff --git a/keywords.txt b/keywords.txt index 620b25e..a9d50bd 100644 --- a/keywords.txt +++ b/keywords.txt @@ -19,6 +19,10 @@ begin KEYWORD2 blank KEYWORD2 boot KEYWORD2 bootLogo KEYWORD2 +bootLogoCompressed KEYWORD2 +bootLogoShell KEYWORD2 +bootLogoSpritesOverwrite KEYWORD2 +bootLogoSpritesSelfMasked KEYWORD2 bootLogoText KEYWORD2 buttonsState KEYWORD2 clear KEYWORD2 diff --git a/src/Arduboy2.cpp b/src/Arduboy2.cpp index bd6d182..a6d72da 100644 --- a/src/Arduboy2.cpp +++ b/src/Arduboy2.cpp @@ -97,13 +97,53 @@ void Arduboy2Base::sysCtrlSound(uint8_t buttons, uint8_t led, uint8_t eeVal) } } -// bootLogoText() should be kept in sync with bootLogo() -// if changes are made to one, equivalent changes should be made to the other void Arduboy2Base::bootLogo() +{ + bootLogoShell(drawLogoBitmap); +} + +void Arduboy2Base::drawLogoBitmap(int16_t y) +{ + drawBitmap(20, y, arduboy_logo, 88, 16); +} + +void Arduboy2Base::bootLogoCompressed() +{ + bootLogoShell(drawLogoCompressed); +} + +void Arduboy2Base::drawLogoCompressed(int16_t y) +{ + drawCompressed(20, y, arduboy_logo_compressed); +} + +void Arduboy2Base::bootLogoSpritesSelfMasked() +{ + bootLogoShell(drawLogoSpritesSelfMasked); +} + +void Arduboy2Base::drawLogoSpritesSelfMasked(int16_t y) +{ + Sprites::drawSelfMasked(20, y, arduboy_logo_sprite, 0); +} + +void Arduboy2Base::bootLogoSpritesOverwrite() +{ + bootLogoShell(drawLogoSpritesOverwrite); +} + +void Arduboy2Base::drawLogoSpritesOverwrite(int16_t y) +{ + Sprites::drawOverwrite(20, y, arduboy_logo_sprite, 0); +} + +// bootLogoText() should be kept in sync with bootLogoShell() +// if changes are made to one, equivalent changes should be made to the other +void Arduboy2Base::bootLogoShell(void (*drawLogo)(int16_t)) { digitalWriteRGB(RED_LED, RGB_ON); - for (int8_t y = -18; y <= 24; y++) { + for (int16_t y = -18; y <= 24; y++) { if (pressed(RIGHT_BUTTON)) { digitalWriteRGB(RGB_OFF, RGB_OFF, RGB_OFF); // all LEDs off return; @@ -119,7 +159,7 @@ void Arduboy2Base::bootLogo() } clear(); - drawBitmap(20, y, arduboy_logo, 88, 16, WHITE); + (*drawLogo)(y); // call the function that actually draws the logo display(); delay(27); // longer delay post boot, we put it inside the loop to @@ -1010,7 +1050,7 @@ Arduboy2::Arduboy2() textWrap = 0; } -// bootLogoText() should be kept in sync with bootLogo() +// bootLogoText() should be kept in sync with bootLogoShell() // if changes are made to one, equivalent changes should be made to the other void Arduboy2::bootLogoText() { diff --git a/src/Arduboy2.h b/src/Arduboy2.h index b4f908b..e8dc6e2 100644 --- a/src/Arduboy2.h +++ b/src/Arduboy2.h @@ -240,7 +240,7 @@ class Arduboy2Base : public Arduboy2Core void systemButtons(); /** \brief - * Display the boot logo sequence. + * Display the boot logo sequence using `drawBitmap()`. * * \details * This function is called by `begin()` and can be called by a sketch @@ -258,11 +258,78 @@ class Arduboy2Base : public Arduboy2Core * which derived classes can implement to add additional information to the * logo screen. The `Arduboy2` class uses this to display the unit name. * - * \see begin() boot() Arduboy2::bootLogoExtra() Arduboy2::bootLogoText() + * \see begin() boot() Arduboy2::bootLogoExtra() bootLogoShell() + * Arduboy2::bootLogoText() */ void bootLogo(); - // Called by bootLogo() to allow derived classes to display additional + /** \brief + * Display the boot logo sequence using `drawCompressed()`. + * + * \details + * This function can be called by a sketch after `boot()` as an alternative to + * `bootLogo()`. This may reduce code size if the sketch itself uses + * `drawCompressed()`. + * + * \see bootLogo() begin() boot() + */ + void bootLogoCompressed(); + + /** \brief + * Display the boot logo sequence using the `Sprites` class + * `drawSelfMasked()` function. + * + * \details + * This function can be called by a sketch after `boot()` as an alternative to + * `bootLogo()`. This may reduce code size if the sketch itself uses + * `Sprites` class functions. + * + * \see bootLogo() begin() boot() Sprites + */ + void bootLogoSpritesSelfMasked(); + + /** \brief + * Display the boot logo sequence using the `Sprites` class + * `drawOverwrite()` function. + * + * \details + * This function can be called by a sketch after `boot()` as an alternative to + * `bootLogo()`. This may reduce code size if the sketch itself uses + * `Sprites` class functions. + * + * \see bootLogo() begin() boot() Sprites + */ + void bootLogoSpritesOverwrite(); + + /** \brief + * Display the boot logo sequence using the provided function + * + * \param drawLogo A reference to a function which will draw the boot logo + * at the given Y position. + * + * \details + * This common function executes the sequence to display the boot logo. + * It is called by `bootLogo()` and other similar functions which provide it + * with a reference to a function which will do the actual drawing of the + * logo. + * + * The prototype for the function provided to draw the logo is: + + * \code + * void drawLogo(int16_t y); + * \endcode + * + * The y parameter is the Y offset for the top of the logo. It is expected + * that the logo will be 16 pixels high and centered horizontally. This will + * result in the logo stopping in the middle of the screen at the end of the + * sequence. If the logo height is not 16 pixels, the Y value can be adjusted + * to compensate. + * + * \see bootLogo() boot() + */ + void bootLogoShell(void (*drawLogo)(int16_t)); + + // Called by bootLogoShell() to allow derived classes to display additional // information after the logo stops scrolling down. virtual void bootLogoExtra(); @@ -495,7 +562,7 @@ class Arduboy2Base : public Arduboy2Core * * The array must be located in program memory by using the PROGMEM modifier. */ - void drawBitmap(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t w, uint8_t h, uint8_t color = WHITE); + static void drawBitmap(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t w, uint8_t h, uint8_t color = WHITE); /** \brief * Draw a bitmap from a horizontally oriented array in program memory. @@ -533,6 +600,7 @@ class Arduboy2Base : public Arduboy2Core * \param sy The Y coordinate of the top left pixel affected by the bitmap. * \param bitmap A pointer to the compressed bitmap array in program memory. * \param color The color of pixels for bits set to 1 in the bitmap. + * (optional; defaults to WHITE). * * \details * Draw a bitmap starting at the given coordinates from an array that has @@ -547,7 +615,7 @@ class Arduboy2Base : public Arduboy2Core * * The array must be located in program memory by using the PROGMEM modifier. */ - void drawCompressed(int16_t sx, int16_t sy, const uint8_t *bitmap, uint8_t color = WHITE); + static void drawCompressed(int16_t sx, int16_t sy, const uint8_t *bitmap, uint8_t color = WHITE); /** \brief * Get a pointer to the display buffer in RAM. @@ -1040,6 +1108,12 @@ class Arduboy2Base : public Arduboy2Core // helper function for sound enable/disable system control void sysCtrlSound(uint8_t buttons, uint8_t led, uint8_t eeVal); + // functions passed to bootLogoShell() to draw the logo + static void drawLogoBitmap(int16_t y); + static void drawLogoCompressed(int16_t y); + static void drawLogoSpritesSelfMasked(int16_t y); + static void drawLogoSpritesOverwrite(int16_t y); + // For button handling uint8_t currentButtonState; uint8_t previousButtonState; @@ -1135,7 +1209,7 @@ class Arduboy2 : public Print, public Arduboy2Base * Show the unit name at the bottom of the boot logo screen. * * \details - * This function is called by `bootLogo()` and `bootlogoText()`. + * This function is called by `bootLogoShell()` and `bootlogoText()`. * * If a unit name has been saved in system EEPROM, it will be displayed at * the bottom of the screen. This function pauses for a short time to allow @@ -1147,8 +1221,8 @@ class Arduboy2 : public Print, public Arduboy2Base * This function would not normally be called directly from within a sketch * itself. * - * \see readUnitName() writeUnitName() bootLogo() bootLogoText() - * writeShowUnitNameFlag() begin() + * \see readUnitName() writeUnitName() bootLogo() bootLogoShell() + * bootLogoText() writeShowUnitNameFlag() begin() */ virtual void bootLogoExtra(); diff --git a/src/Sprites.h b/src/Sprites.h index 4716b20..d3a95d8 100644 --- a/src/Sprites.h +++ b/src/Sprites.h @@ -87,8 +87,8 @@ class Sprites * ..O.. OOOOO OOOOO ..O.. * ..... .OOO. OOOOO O...O */ - void drawExternalMask(int16_t x, int16_t y, const uint8_t *bitmap, - const uint8_t *mask, uint8_t frame, uint8_t mask_frame); + static void drawExternalMask(int16_t x, int16_t y, const uint8_t *bitmap, + const uint8_t *mask, uint8_t frame, uint8_t mask_frame); /** \brief * Draw a sprite using an array containing both image and mask values. @@ -124,7 +124,7 @@ class Sprites * ..O.. OOOOO OOOOO ..O.. * ..... .OOO. OOOOO O...O */ - void drawPlusMask(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); + static void drawPlusMask(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); /** \brief * Draw a sprite by replacing the existing content completely. @@ -155,7 +155,7 @@ class Sprites * ..O.. OOOOO ..O.. * ..... OOOOO ..... */ - void drawOverwrite(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); + static void drawOverwrite(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); /** \brief * "Erase" a sprite. @@ -186,7 +186,7 @@ class Sprites * ..O.. OOOOO OO.OO * ..... OOOOO OOOOO */ - void drawErase(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); + static void drawErase(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); /** \brief * Draw a sprite using only the bits set to 1. @@ -216,20 +216,20 @@ class Sprites * ..O.. OOOOO OOOOO * ..... OOOOO OOOOO */ - void drawSelfMasked(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); + static void drawSelfMasked(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t frame); // Master function. Needs to be abstracted into separate function for // every render type. // (Not officially part of the API) - void draw(int16_t x, int16_t y, - const uint8_t *bitmap, uint8_t frame, - const uint8_t *mask, uint8_t sprite_frame, - uint8_t drawMode); + static void draw(int16_t x, int16_t y, + const uint8_t *bitmap, uint8_t frame, + const uint8_t *mask, uint8_t sprite_frame, + uint8_t drawMode); // (Not officially part of the API) - void drawBitmap(int16_t x, int16_t y, - const uint8_t *bitmap, const uint8_t *mask, - uint8_t w, uint8_t h, uint8_t draw_mode); + static void drawBitmap(int16_t x, int16_t y, + const uint8_t *bitmap, const uint8_t *mask, + uint8_t w, uint8_t h, uint8_t draw_mode); }; #endif diff --git a/src/ab_logo.c b/src/ab_logo.c index 4041df2..e3f6022 100644 --- a/src/ab_logo.c +++ b/src/ab_logo.c @@ -10,27 +10,74 @@ #define ARDUBOY_LOGO_CREATED // arduboy_logo.png +// drawBitmap() format // 88x16 -PROGMEM const unsigned char arduboy_logo[] = { -0xF0, 0xF8, 0x9C, 0x8E, 0x87, 0x83, 0x87, 0x8E, 0x9C, 0xF8, -0xF0, 0x00, 0x00, 0xFE, 0xFF, 0x03, 0x03, 0x03, 0x03, 0x03, -0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0xFE, 0xFF, 0x03, 0x03, -0x03, 0x03, 0x03, 0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0xFF, -0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, -0x00, 0x00, 0xFE, 0xFF, 0x83, 0x83, 0x83, 0x83, 0x83, 0xC7, -0xEE, 0x7C, 0x38, 0x00, 0x00, 0xF8, 0xFC, 0x0E, 0x07, 0x03, -0x03, 0x03, 0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0x3F, 0x7F, -0xE0, 0xC0, 0x80, 0x80, 0xC0, 0xE0, 0x7F, 0x3F, 0xFF, 0xFF, -0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0xFF, 0xFF, 0x00, -0x00, 0xFF, 0xFF, 0x0C, 0x0C, 0x0C, 0x0C, 0x1C, 0x3E, 0x77, -0xE3, 0xC1, 0x00, 0x00, 0x7F, 0xFF, 0xC0, 0xC0, 0xC0, 0xC0, -0xC0, 0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, 0x1F, 0x3F, 0x70, -0xE0, 0xC0, 0xC0, 0xC0, 0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, -0x7F, 0xFF, 0xC1, 0xC1, 0xC1, 0xC1, 0xC1, 0xE3, 0x77, 0x3E, -0x1C, 0x00, 0x00, 0x1F, 0x3F, 0x70, 0xE0, 0xC0, 0xC0, 0xC0, -0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, +const uint8_t arduboy_logo[] PROGMEM = { +0xF0, 0xF8, 0x9C, 0x8E, 0x87, 0x83, 0x87, 0x8E, 0x9C, 0xF8, +0xF0, 0x00, 0x00, 0xFE, 0xFF, 0x03, 0x03, 0x03, 0x03, 0x03, +0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0xFE, 0xFF, 0x03, 0x03, +0x03, 0x03, 0x03, 0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0xFF, +0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, +0x00, 0x00, 0xFE, 0xFF, 0x83, 0x83, 0x83, 0x83, 0x83, 0xC7, +0xEE, 0x7C, 0x38, 0x00, 0x00, 0xF8, 0xFC, 0x0E, 0x07, 0x03, +0x03, 0x03, 0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0x3F, 0x7F, +0xE0, 0xC0, 0x80, 0x80, 0xC0, 0xE0, 0x7F, 0x3F, 0xFF, 0xFF, +0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0xFF, 0xFF, 0x00, +0x00, 0xFF, 0xFF, 0x0C, 0x0C, 0x0C, 0x0C, 0x1C, 0x3E, 0x77, +0xE3, 0xC1, 0x00, 0x00, 0x7F, 0xFF, 0xC0, 0xC0, 0xC0, 0xC0, +0xC0, 0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, 0x1F, 0x3F, 0x70, +0xE0, 0xC0, 0xC0, 0xC0, 0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, +0x7F, 0xFF, 0xC1, 0xC1, 0xC1, 0xC1, 0xC1, 0xE3, 0x77, 0x3E, +0x1C, 0x00, 0x00, 0x1F, 0x3F, 0x70, 0xE0, 0xC0, 0xC0, 0xC0, +0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF, 0xFF, 0x01, 0x00, 0x00, 0x00 +}; +// arduboy_logo.png +// drawCompressed() format +// 88x16 +const uint8_t arduboy_logo_compressed[] PROGMEM = { +0x57, 0x0F, 0x9C, 0x53, 0x72, 0x75, 0x29, 0xE5, 0x9C, 0x92, +0xCE, 0x95, 0x52, 0xAD, 0x4E, 0x49, 0xE7, 0x08, 0x09, 0xED, +0x76, 0xBB, 0xDD, 0x2A, 0xAB, 0xAC, 0x55, 0x92, 0x90, 0xD0, +0x6E, 0xB7, 0xDB, 0xAD, 0xB2, 0xCA, 0x5A, 0x25, 0xF9, 0xF8, +0xF0, 0xC6, 0x47, 0x48, 0x28, 0x95, 0x54, 0x52, 0x49, 0x25, +0x9D, 0x3A, 0x95, 0x5A, 0x3A, 0x45, 0x2A, 0xB7, 0x29, 0xA7, +0xE4, 0x76, 0xBB, 0x55, 0x56, 0x59, 0xAB, 0x24, 0x9F, 0x5D, +0x5B, 0x65, 0xD7, 0xE9, 0xEC, 0x92, 0x29, 0x3B, 0xA1, 0x4E, +0xA7, 0xD3, 0xE9, 0x74, 0x9A, 0x8F, 0x8F, 0xEF, 0xED, 0x76, +0xBB, 0x55, 0x4E, 0xAE, 0x52, 0xAD, 0x9C, 0x9C, 0x4F, 0xE7, +0xED, 0x76, 0xBB, 0xDD, 0x2E, 0x95, 0x53, 0xD9, 0x25, 0xA5, +0x54, 0xD6, 0x2A, 0xAB, 0xEC, 0x76, 0xBB, 0x54, 0x4E, 0x65, +0x97, 0x94, 0x3A, 0x22, 0xA9, 0xA4, 0x92, 0x4A, 0x2A, 0xE9, +0x94, 0x4D, 0x2D, 0x9D, 0xA2, 0x94, 0xCA, 0x5A, 0x65, 0x95, +0xDD, 0x6E, 0x97, 0xCA, 0xA9, 0xEC, 0x12, 0x55, 0x69, 0x42, +0x7A +}; + +// arduboy_logo.png +// Sprites::drawSelfMasked() format +// 88x16 +const uint8_t arduboy_logo_sprite[] PROGMEM = { +88, 16, +0xF0, 0xF8, 0x9C, 0x8E, 0x87, 0x83, 0x87, 0x8E, 0x9C, 0xF8, +0xF0, 0x00, 0x00, 0xFE, 0xFF, 0x03, 0x03, 0x03, 0x03, 0x03, +0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0xFE, 0xFF, 0x03, 0x03, +0x03, 0x03, 0x03, 0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0xFF, +0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, +0x00, 0x00, 0xFE, 0xFF, 0x83, 0x83, 0x83, 0x83, 0x83, 0xC7, +0xEE, 0x7C, 0x38, 0x00, 0x00, 0xF8, 0xFC, 0x0E, 0x07, 0x03, +0x03, 0x03, 0x07, 0x0E, 0xFC, 0xF8, 0x00, 0x00, 0x3F, 0x7F, +0xE0, 0xC0, 0x80, 0x80, 0xC0, 0xE0, 0x7F, 0x3F, 0xFF, 0xFF, +0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0xFF, 0xFF, 0x00, +0x00, 0xFF, 0xFF, 0x0C, 0x0C, 0x0C, 0x0C, 0x1C, 0x3E, 0x77, +0xE3, 0xC1, 0x00, 0x00, 0x7F, 0xFF, 0xC0, 0xC0, 0xC0, 0xC0, +0xC0, 0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, 0x1F, 0x3F, 0x70, +0xE0, 0xC0, 0xC0, 0xC0, 0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, +0x7F, 0xFF, 0xC1, 0xC1, 0xC1, 0xC1, 0xC1, 0xE3, 0x77, 0x3E, +0x1C, 0x00, 0x00, 0x1F, 0x3F, 0x70, 0xE0, 0xC0, 0xC0, 0xC0, +0xE0, 0x70, 0x3F, 0x1F, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, +0xFF, 0xFF, 0x01, 0x00, 0x00, 0x00 }; #endif