diff --git a/Arduboy2Core.h b/Arduboy2Core.h deleted file mode 100644 index f7e03b2..0000000 --- a/Arduboy2Core.h +++ /dev/null @@ -1,735 +0,0 @@ -/** - * @file Arduboy2Core.h - * \brief - * The Arduboy2Core class for Arduboy hardware initilization and control. - */ - -#ifndef ARDUBOY2_CORE_H -#define ARDUBOY2_CORE_H - -#include -#include -#include -#include - - -// main hardware compile flags - -#if !defined(ARDUBOY_10) && !defined(AB_DEVKIT) -/* defaults to Arduboy Release 1.0 if not using a boards.txt file - * - * we default to Arduboy Release 1.0 if a compile flag has not been - * passed to us from a boards.txt file - * - * if you wish to compile for the devkit without using a boards.txt - * file simply comment out the ARDUBOY_10 define and uncomment - * the AB_DEVKIT define like this: - * - * // #define ARDUBOY_10 - * #define AB_DEVKIT - */ -#define ARDUBOY_10 //< compile for the production Arduboy v1.0 -// #define AB_DEVKIT //< compile for the official dev kit -#endif - -#define RGB_ON LOW /**< For digitially setting an RGB LED on using digitalWriteRGB() */ -#define RGB_OFF HIGH /**< For digitially setting an RGB LED off using digitalWriteRGB() */ - -// ----- Arduboy pins ----- -#ifdef ARDUBOY_10 - -#ifdef ARDUINO_AVR_PROMICRO - #define PIN_CS 3 // Pro Micro alternative display CS pin - #define CS_PORT PORTD - #define CS_BIT PORTD3 -#else - #define PIN_CS 12 // Display CS Arduino pin number - #define CS_PORT PORTD // Display CS port - #define CS_BIT PORTD6 // Display CS physical bit number -#endif - -#define PIN_DC 4 // Display D/C Arduino pin number -#define DC_PORT PORTD // Display D/C port -#define DC_BIT PORTD4 // Display D/C physical bit number - -#define PIN_RST 6 // Display reset Arduino pin number -#define RST_PORT PORTD // Display reset port -#define RST_BIT PORTD7 // Display reset physical bit number - -#define SPI_MOSI_PORT PORTB -#define SPI_MOSI_BIT PORTB2 - -#define SPI_SCK_PORT PORTB -#define SPI_SCK_BIT PORTB1 - -#define RED_LED 10 /**< The pin number for the red color in the RGB LED. */ -#ifdef ARDUINO_AVR_PROMICRO - #define GREEN_LED 3 // Pro Micro alternative green LED pin -#else - #define GREEN_LED 11 /**< The pin number for the green color in the RGB LED. */ -#endif -#define BLUE_LED 9 /**< The pin number for the blue color in the RGB LED. */ - -#define RED_LED_PORT PORTB -#define RED_LED_BIT PORTB6 - -#ifdef ARDUINO_AVR_PROMICRO - #define GREEN_LED_PORT PORTD // Pro Micro alternative green LED port - #define GREEN_LED_BIT PORTD0 -#else - #define GREEN_LED_PORT PORTB - #define GREEN_LED_BIT PORTB7 -#endif - -#define BLUE_LED_PORT PORTB -#define BLUE_LED_BIT PORTB5 - -// bit values for button states -// these are determined by the buttonsState() function -#define LEFT_BUTTON _BV(5) /**< The Left button value for functions requiring a bitmask */ -#define RIGHT_BUTTON _BV(6) /**< The Right button value for functions requiring a bitmask */ -#define UP_BUTTON _BV(7) /**< The Up button value for functions requiring a bitmask */ -#define DOWN_BUTTON _BV(4) /**< The Down button value for functions requiring a bitmask */ -#define A_BUTTON _BV(3) /**< The A button value for functions requiring a bitmask */ -#define B_BUTTON _BV(2) /**< The B button value for functions requiring a bitmask */ - -#define PIN_LEFT_BUTTON A2 -#define LEFT_BUTTON_PORT PORTF -#define LEFT_BUTTON_BIT PORTF5 - -#define PIN_RIGHT_BUTTON A1 -#define RIGHT_BUTTON_PORT PORTF -#define RIGHT_BUTTON_BIT PORTF6 - -#define PIN_UP_BUTTON A0 -#define UP_BUTTON_PORT PORTF -#define UP_BUTTON_BIT PORTF7 - -#define PIN_DOWN_BUTTON A3 -#define DOWN_BUTTON_PORT PORTF -#define DOWN_BUTTON_BIT PORTF4 - -#define PIN_A_BUTTON 7 -#define A_BUTTON_PORT PORTE -#define A_BUTTON_BIT PORTE6 - -#define PIN_B_BUTTON 8 -#define B_BUTTON_PORT PORTB -#define B_BUTTON_BIT PORTB4 - -#define PIN_SPEAKER_1 5 /**< The pin number of the first lead of the speaker */ - -#define SPEAKER_1_PORT PORTC -#define SPEAKER_1_DDR DDRC -#define SPEAKER_1_BIT PORTC6 - -#ifdef ARDUINO_AVR_PROMICRO - #define PIN_SPEAKER_2 2 //Pro Micro alternative for 2nd speaker pin - #define SPEAKER_2_PORT PORTD - #define SPEAKER_2_DDR DDRD - #define SPEAKER_2_BIT PORTD1 -#else - #define PIN_SPEAKER_2 13 /**< The pin number of the second lead of the speaker */ - #define SPEAKER_2_PORT PORTC - #define SPEAKER_2_DDR DDRC - #define SPEAKER_2_BIT PORTC7 -#endif - -#define RAND_SEED_IN A4 // Open analog input used for noise by initRandomSeed() -#define RAND_SEED_IN_PORTF -#define RAND_SEED_IN_BIT PORTF1 -// Value for ADMUX to read the random seed pin: 2.56V reference, ADC1 -#define RAND_SEED_IN_ADMUX (_BV(REFS0) | _BV(REFS1) | _BV(MUX0)) -// ----------------------- - -// ----- DevKit pins ----- -#elif defined(AB_DEVKIT) - -#define PIN_CS 6 // Display CS Arduino pin number -#define CS_PORT PORTD // Display CS port -#define CS_BIT PORTD7 // Display CS physical bit number - -#define PIN_DC 4 // Display D/C Arduino pin number -#define DC_PORT PORTD // Display D/C port -#define DC_BIT PORTD4 // Display D/C physical bit number - -#define PIN_RST 12 // Display reset Arduino pin number -#define RST_PORT PORTD // Display reset port -#define RST_BIT PORTD6 // Display reset physical bit number - -#define SPI_MOSI_PORT PORTB -#define SPI_MOSI_BIT PORTB2 - -#define SPI_SCK_PORT PORTB -#define SPI_SCK_BIT PORTB1 - -// map all LEDs to the single TX LED on DEVKIT -#define RED_LED 17 -#define GREEN_LED 17 -#define BLUE_LED 17 - -#define BLUE_LED_PORT PORTB -#define BLUE_LED_BIT PORTB0 - -// bit values for button states -// these are determined by the buttonsState() function -#define LEFT_BUTTON _BV(5) -#define RIGHT_BUTTON _BV(2) -#define UP_BUTTON _BV(4) -#define DOWN_BUTTON _BV(6) -#define A_BUTTON _BV(1) -#define B_BUTTON _BV(0) - -// pin values for buttons, probably shouldn't use these -#define PIN_LEFT_BUTTON 9 -#define LEFT_BUTTON_PORT PORTB -#define LEFT_BUTTON_BIT PORTB5 - -#define PIN_RIGHT_BUTTON 5 -#define RIGHT_BUTTON_PORT PORTC -#define RIGHT_BUTTON_BIT PORTC6 - -#define PIN_UP_BUTTON 8 -#define UP_BUTTON_PORT PORTB -#define UP_BUTTON_BIT PORTB4 - -#define PIN_DOWN_BUTTON 10 -#define DOWN_BUTTON_PORT PORTB -#define DOWN_BUTTON_BIT PORTB6 - -#define PIN_A_BUTTON A0 -#define A_BUTTON_PORT PORTF -#define A_BUTTON_BIT PORTF7 - -#define PIN_B_BUTTON A1 -#define B_BUTTON_PORT PORTF -#define B_BUTTON_BIT PORTF6 - -#define PIN_SPEAKER_1 A2 -#define SPEAKER_1_PORT PORTF -#define SPEAKER_1_DDR DDRF -#define SPEAKER_1_BIT PORTF5 -// SPEAKER_2 is purposely not defined for DEVKIT as it could potentially -// be dangerous and fry your hardware (because of the devkit wiring). -// -// Reference: https://github.com/Arduboy/Arduboy/issues/108 - -#define RAND_SEED_IN A4 // Open analog input used for noise by initRandomSeed() -#define RAND_SEED_IN_PORTF -#define RAND_SEED_IN_BIT PORTF1 -// Value for ADMUX to read the random seed pin: 2.56V reference, ADC1 -#define RAND_SEED_IN_ADMUX (_BV(REFS0) | _BV(REFS1) | _BV(MUX0)) - -#endif -// -------------------- - -// OLED hardware (SSD1306,SSD1309,SH1106,OLED_64X64) - -#define OLED_PIXELS_INVERTED 0xA7 // All pixels inverted -#define OLED_PIXELS_NORMAL 0xA6 // All pixels normal - -#define OLED_ALL_PIXELS_ON 0xA5 // all pixels on -#define OLED_PIXELS_FROM_RAM 0xA4 // pixels mapped to display RAM contents - -#define OLED_VERTICAL_FLIPPED 0xC0 // reversed COM scan direction -#define OLED_VERTICAL_NORMAL 0xC8 // normal COM scan direction - -#define OLED_HORIZ_FLIPPED 0xA0 // reversed segment re-map -#define OLED_HORIZ_NORMAL 0xA1 // normal segment re-map - -#define OLED_SET_PAGE_ADDRESS 0xB0 -#ifdef OLED_SH1106 - #define OLED_SET_COLUMN_ADDRESS_LO 0x02 //SH1106 only: 1st pixel starts on column 2 -#else - #define OLED_SET_COLUMN_ADDRESS_LO 0x00 -#endif -#define OLED_SET_COLUMN_ADDRESS_HI 0x10 -// ----- -#ifdef OLED_96X96 - #define WIDTH 96 - #define HEIGHT 96 -#elif defined(OLED_128X64) - #define WIDTH 128 - #define HEIGHT 96 -#else - #define WIDTH 128 /**< The width of the display in pixels */ - #define HEIGHT 64 /**< The height of the display in pixels */ -#endif - -#define COLUMN_ADDRESS_END (WIDTH - 1) & 127 // 128 pixels wide -#define PAGE_ADDRESS_END ((HEIGHT/8)-1) & 7 // 8 pages high - -/** \brief - * Lower level functions generally dealing directly with the hardware. - * - * \details - * This class is inherited by Arduboy2Base and thus also Arduboy2, so wouldn't - * normally be used directly by a sketch. - * - * \note - * A friend class named _Arduboy2Ex_ is declared by this class. The intention - * is to allow a sketch to create an _Arduboy2Ex_ class which would have access - * to the private and protected members of the Arduboy2Core class. It is hoped - * that this may eliminate the need to create an entire local copy of the - * library, in order to extend the functionality, in most circumstances. - */ -class Arduboy2Core -{ - friend class Arduboy2Ex; - - public: - Arduboy2Core(); - - /** \brief - * Idle the CPU to save power. - * - * \details - * This puts the CPU in _idle_ sleep mode. You should call this as often - * as you can for the best power savings. The timer 0 overflow interrupt - * will wake up the chip every 1ms, so even at 60 FPS a well written - * app should be able to sleep maybe half the time in between rendering - * it's own frames. - */ - void static idle(); - - /** \brief - * Put the display into data mode. - * - * \details - * When placed in data mode, data that is sent to the display will be - * considered as data to be displayed. - * - * \note - * This is a low level function that is not intended for general use in a - * sketch. It has been made public and documented for use by derived - * classes. - * - * \see LCDCommandMode() SPItransfer() - */ - inline void static LCDDataMode() __attribute__((always_inline)); - /** \brief - * Put the display into command mode. - * - * \details - * When placed in command mode, data that is sent to the display will be - * treated as commands. - * - * See the SSD1306 controller and OLED display documents for available - * commands and command sequences. - * - * Links: - * - * - https://www.adafruit.com/datasheets/SSD1306.pdf - * - http://www.buydisplay.com/download/manual/ER-OLED013-1_Series_Datasheet.pdf - * - * \note - * This is a low level function that is not intended for general use in a - * sketch. It has been made public and documented for use by derived - * classes. - * - * \see LCDDataMode() sendLCDCommand() SPItransfer() - */ - inline void static LCDCommandMode() __attribute__((always_inline)); - /** \brief - * Transfer a byte to the display. - * - * \param data The byte to be sent to the display. - * - * \details - * Transfer one byte to the display over the SPI port and wait for the - * transfer to complete. The byte will either be interpreted as a command - * or as data to be placed on the screen, depending on the command/data - * mode. - * - * \see LCDDataMode() LCDCommandMode() sendLCDCommand() - */ - void static SPItransfer(uint8_t data); - - /** \brief - * Turn the display off. - * - * \details - * The display will clear and be put into a low power mode. This can be - * used to extend battery life when a game is paused or when a sketch - * doesn't require anything to be displayed for a relatively long period - * of time. - * - * \see displayOn() - */ - void static displayOff(); - - /** \brief - * Turn the display on. - * - * \details - * Used to power up and reinitialize the display after calling - * `displayOff()`. - * - * \note - * The previous call to `displayOff()` will have caused the display's - * buffer contents to be lost. The display will have to be re-painted, - * which is usually done by calling `display()`. - * - * \see displayOff() - */ - void static displayOn(); - - /** \brief - * Get the width of the display in pixels. - * - * \return The width of the display in pixels. - * - * \note - * In most cases, the defined value `WIDTH` would be better to use instead - * of this function. - */ - uint8_t static width(); - - /** \brief - * Get the height of the display in pixels. - * - * \return The height of the display in pixels. - * - * \note - * In most cases, the defined value `HEIGHT` would be better to use instead - * of this function. - */ - uint8_t static height(); - - /** \brief - * Get the current state of all buttons as a bitmask. - * - * \return A bitmask of the state of all the buttons. - * - * \details - * The returned mask contains a bit for each button. For any pressed button, - * its bit will be 1. For released buttons their associated bits will be 0. - * - * The following defined mask values should be used for the buttons: - * - * LEFT_BUTTON, RIGHT_BUTTON, UP_BUTTON, DOWN_BUTTON, A_BUTTON, B_BUTTON - */ - uint8_t static buttonsState(); - - /** \brief - * Paint 8 pixels vertically to the display. - * - * \param pixels A byte whose bits specify a vertical column of 8 pixels. - * - * \details - * A byte representing a vertical column of 8 pixels is written to the - * display at the current page and column address. The address is then - * incremented. The page/column address will wrap to the start of the - * display (the top left) when it increments past the end (lower right). - * - * The least significant bit represents the top pixel in the column. - * A bit set to 1 is lit, 0 is unlit. - * - * Example: - * - * X = lit pixels, . = unlit pixels - * - * blank() paint8Pixels() 0xFF, 0, 0xF0, 0, 0x0F - * v TOP LEFT corner (8x9) v TOP LEFT corner - * . . . . . . . . (page 1) X . . . X . . . (page 1) - * . . . . . . . . X . . . X . . . - * . . . . . . . . X . . . X . . . - * . . . . . . . . X . . . X . . . - * . . . . . . . . X . X . . . . . - * . . . . . . . . X . X . . . . . - * . . . . . . . . X . X . . . . . - * . . . . . . . . (end of page 1) X . X . . . . . (end of page 1) - * . . . . . . . . (page 2) . . . . . . . . (page 2) - */ - void static paint8Pixels(uint8_t pixels); - - /** \brief - * Paints an entire image directly to the display from program memory. - * - * \param image A byte array in program memory representing the entire - * contents of the display. - * - * \details - * The contents of the specified array in program memory is written to the - * display. Each byte in the array represents a vertical column of 8 pixels - * with the least significant bit at the top. The bytes are written starting - * at the top left, progressing horizontally and wrapping at the end of each - * row, to the bottom right. The size of the array must exactly match the - * number of pixels in the entire display. - * - * \see paint8Pixels() - */ - void static paintScreen(const uint8_t *image); - - /** \brief - * Paints an entire image directly to the display from an array in RAM. - * - * \param image A byte array in RAM representing the entire contents of - * the display. - * \param clear If `true` the array in RAM will be cleared to zeros upon - * return from this function. If `false` the RAM buffer will remain - * unchanged. (optional; defaults to `false`) - * - * \details - * The contents of the specified array in RAM is written to the display. - * Each byte in the array represents a vertical column of 8 pixels with - * the least significant bit at the top. The bytes are written starting - * at the top left, progressing horizontally and wrapping at the end of - * each row, to the bottom right. The size of the array must exactly - * match the number of pixels in the entire display. - * - * If parameter `clear` is set to `true` the RAM array will be cleared to - * zeros after its contents are written to the display. - * - * \see paint8Pixels() - */ - void static paintScreen(uint8_t image[], bool clear = false); - - /** \brief - * Blank the display screen by setting all pixels off. - * - * \details - * All pixels on the screen will be written with a value of 0 to turn - * them off. - */ - void static blank(); - - /** \brief - * Invert the entire display or set it back to normal. - * - * \param inverse `true` will invert the display. `false` will set the - * display to no-inverted. - * - * \details - * Calling this function with a value of `true` will set the display to - * inverted mode. A pixel with a value of 0 will be on and a pixel set to 1 - * will be off. - * - * Once in inverted mode, the display will remain this way - * until it is set back to non-inverted mode by calling this function with - * `false`. - */ - void static invert(bool inverse); - - /** \brief - * Turn all display pixels on or display the buffer contents. - * - * \param on `true` turns all pixels on. `false` displays the contents - * of the hardware display buffer. - * - * \details - * Calling this function with a value of `true` will override the contents - * of the hardware display buffer and turn all pixels on. The contents of - * the hardware buffer will remain unchanged. - * - * Calling this function with a value of `false` will set the normal state - * of displaying the contents of the hardware display buffer. - * - * \note - * All pixels will be lit even if the display is in inverted mode. - * - * \see invert() - */ - void static allPixelsOn(bool on); - - /** \brief - * Flip the display vertically or set it back to normal. - * - * \param flipped `true` will set vertical flip mode. `false` will set - * normal vertical orientation. - * - * \details - * Calling this function with a value of `true` will cause the Y coordinate - * to start at the bottom edge of the display instead of the top, - * effectively flipping the display vertically. - * - * Once in vertical flip mode, it will remain this way until normal - * vertical mode is set by calling this function with a value of `false`. - * - * \see flipHorizontal() - */ - void static flipVertical(bool flipped); - - /** \brief - * Flip the display horizontally or set it back to normal. - * - * \param flipped `true` will set horizontal flip mode. `false` will set - * normal horizontal orientation. - * - * \details - * Calling this function with a value of `true` will cause the X coordinate - * to start at the left edge of the display instead of the right, - * effectively flipping the display horizontally. - * - * Once in horizontal flip mode, it will remain this way until normal - * horizontal mode is set by calling this function with a value of `false`. - * - * \see flipVertical() - */ - void static flipHorizontal(bool flipped); - - /** \brief - * Send a single command byte to the display. - * - * \param command The command byte to send to the display. - * - * \details - * The display will be set to command mode then the specified command - * byte will be sent. The display will then be set to data mode. - * Multi-byte commands can be sent by calling this function multiple times. - * - * \note - * Sending improper commands to the display can place it into invalid or - * unexpected states, possibly even causing physical damage. - */ - void static sendLCDCommand(uint8_t command); - - /** \brief - * Set the light output of the RGB LED. - * - * \param red,green,blue The brightness value for each LED. - * - * \details - * The RGB LED is actually individual red, green and blue LEDs placed - * very close together in a single package. By setting the brightness of - * each LED, the RGB LED can show various colors and intensities. - * The brightness of each LED can be set to a value from 0 (fully off) - * to 255 (fully on). - * - * \note - * \parblock - * Certain libraries that take control of the hardware timers may interfere - * with the ability of this function to properly control the RGB LED. - *_ArduboyPlaytune_ is one such library known to do this. - * The digitalWriteRGB() function will still work properly in this case. - * \endparblock - * - * \note - * \parblock - * Many of the Kickstarter Arduboys were accidentally shipped with the - * RGB LED installed incorrectly. For these units, the green LED cannot be - * lit. As long as the green led is set to off, setting the red LED will - * actually control the blue LED and setting the blue LED will actually - * control the red LED. If the green LED is turned fully on, none of the - * LEDs will light. - * \endparblock - * - * \see digitalWriteRGB() - */ - void static setRGBled(uint8_t red, uint8_t green, uint8_t blue); - - /** \brief - * Set the RGB LEDs digitally, to either fully on or fully off. - * - * \param red,green,blue Use value RGB_ON or RGB_OFF to set each LED. - * - * \details - * The RGB LED is actually individual red, green and blue LEDs placed - * very close together in a single package. This 3 parameter version of the - * function will set each LED either on or off, to set the RGB LED to - * 7 different colors at their highest brightness or turn it off. - * - * The colors are as follows: - * - * RED LED GREEN_LED BLUE_LED COLOR - * ------- --------- -------- ----- - * RGB_OFF RGB_OFF RGB_OFF OFF - * RGB_OFF RGB_OFF RGB_ON Blue - * RGB_OFF RGB_ON RGB_OFF Green - * RGB_OFF RGB_ON RGB_ON Cyan - * RGB_ON RGB_OFF RGB_OFF Red - * RGB_ON RGB_OFF RGB_ON Magenta - * RGB_ON RGB_ON RGB_OFF Yellow - * RGB_ON RGB_ON RGB_ON White - * - * \note - * Many of the Kickstarter Arduboys were accidentally shipped with the - * RGB LED installed incorrectly. For these units, the green LED cannot be - * lit. As long as the green led is set to off, turning on the red LED will - * actually light the blue LED and turning on the blue LED will actually - * light the red LED. If the green LED is turned on, none of the LEDs - * will light. - * - * \see digitalWriteRGB(uint8_t, uint8_t) setRGBled() - */ - void static digitalWriteRGB(uint8_t red, uint8_t green, uint8_t blue); - - /** \brief - * Set one of the RGB LEDs digitally, to either fully on or fully off. - * - * \param color The name of the LED to set. The value given should be one - * of RED_LED, GREEN_LED or BLUE_LED. - * - * \param val Indicates whether to turn the specified LED on or off. - * The value given should be RGB_ON or RGB_OFF. - * - * \details - * This 2 parameter version of the function will set a single LED within - * the RGB LED either fully on or fully off. See the description of the - * 3 parameter version of this function for more details on the RGB LED. - * - * \see digitalWriteRGB(uint8_t, uint8_t, uint8_t) setRGBled() - */ - void static digitalWriteRGB(uint8_t color, uint8_t val); - - /** \brief - * Initialize the Arduboy's hardware. - * - * \details - * This function initializes the display, buttons, etc. - * - * This function is called by begin() so isn't normally called within a - * sketch. However, in order to free up some code space, by eliminating - * some of the start up features, it can be called in place of begin(). - * The functions that begin() would call after boot() can then be called - * to add back in some of the start up features, if desired. - * See the README file or documentation on the main page for more details. - * - * \see Arduboy2Base::begin() - */ - void static boot(); - - /** \brief - * Allow upload when the bootloader "magic number" could be corrupted. - * - * \details - * If the UP button is held when this function is entered, the RGB LED - * will be lit and timer 0 will be disabled, then the sketch will remain - * in a tight loop. This is to address a problem with uploading a new - * sketch, for sketches that interfere with the bootloader "magic number". - * The problem occurs with certain sketches that use large amounts of RAM. - * - * This function should be called after `boot()` in sketches that - * potentially could cause the problem. - * - * It is intended to replace the `flashlight()` function when more - * program space is required. If possible, it is more desirable to use - * `flashlight()`, so that the actual flashlight feature isn't lost. - * - * \see Arduboy2Base::flashlight() boot() - */ - void static safeMode(); - - /** \brief - * Delay for the number of milliseconds, specified as a 16 bit value. - * - * \param ms The delay in milliseconds. - * - * \details - * This function works the same as the Arduino `delay()` function except - * the provided value is 16 bits long, so the maximum delay allowed is - * 65535 milliseconds (about 65.5 seconds). Using this function instead - * of Arduino `delay()` will save a few bytes of code. - */ - void static delayShort(uint16_t ms) __attribute__ ((noinline)); - - protected: - // internals - void static setCPUSpeed8MHz(); - void static bootSPI(); - void static bootOLED(); - void static bootPins(); - void static bootPowerSaving(); -}; - -#endif