#include "st7789.hpp" #include #include #include "hardware/dma.h" #include "hardware/pwm.h" namespace pimoroni { uint8_t madctl; uint16_t caset[2] = {0, 0}; uint16_t raset[2] = {0, 0}; void ST7789::init(bool auto_init_sequence, bool round) { // configure spi interface and pins spi_init(spi, spi_baud); gpio_set_function(dc, GPIO_FUNC_SIO); gpio_set_dir(dc, GPIO_OUT); gpio_set_function(cs, GPIO_FUNC_SIO); gpio_set_dir(cs, GPIO_OUT); gpio_set_function(sck, GPIO_FUNC_SPI); gpio_set_function(mosi, GPIO_FUNC_SPI); if(miso != -1) { gpio_set_function(miso, GPIO_FUNC_SPI); } // if supported by the display then the vsync pin is // toggled high during vertical blanking period if(vsync != -1) { gpio_set_function(vsync, GPIO_FUNC_SIO); gpio_set_dir(vsync, GPIO_IN); gpio_set_pulls(vsync, false, true); } // if a backlight pin is provided then set it up for // pwm control if(bl != -1) { pwm_config cfg = pwm_get_default_config(); pwm_set_wrap(pwm_gpio_to_slice_num(bl), 65535); pwm_init(pwm_gpio_to_slice_num(bl), &cfg, true); gpio_set_function(bl, GPIO_FUNC_PWM); set_backlight(255); // Turn backlight on by default to avoid nasty surprises } // if auto_init_sequence then send initialisation sequence // for our standard displays based on the width and height if(auto_init_sequence) { command(reg::SWRESET); sleep_ms(150); command(reg::TEON, 1, "\x00"); // enable frame sync signal if used command(reg::COLMOD, 1, "\x05"); // 16 bits per pixel command(reg::INVON); // set inversion mode command(reg::SLPOUT); // leave sleep mode command(reg::DISPON); // turn display on sleep_ms(100); // setup correct addressing window if(width == 240 && height == 240) { caset[0] = 0; caset[1] = 239; raset[0] = round ? 40 : 0; raset[1] = round ? 279 : 239; madctl = MADCTL::HORIZ_ORDER; } if(width == 240 && height == 135) { caset[0] = 40; // 240 cols caset[1] = 279; raset[0] = 53; // 135 rows raset[1] = 187; madctl = MADCTL::COL_ORDER | MADCTL::SWAP_XY | MADCTL::SCAN_ORDER; } if(width == 135 && height == 240) { caset[0] = 52; // 135 cols caset[1] = 186; raset[0] = 40; // 240 rows raset[1] = 279; madctl = 0; } // Byte swap the 16bit rows/cols values caset[0] = __builtin_bswap16(caset[0]); caset[1] = __builtin_bswap16(caset[1]); raset[0] = __builtin_bswap16(raset[0]); raset[1] = __builtin_bswap16(raset[1]); command(reg::CASET, 4, (char *)caset); command(reg::RASET, 4, (char *)raset); command(reg::MADCTL, 1, (char *)&madctl); } // the dma transfer works but without vsync it's not that useful as you could // be updating the framebuffer during transfer... // // this could be avoided by creating another buffer to draw into and flip // buffers (but costs another ~100kb of ram) // // it's probably not worth it for this particular usecase but will consider it // some more... // setup spi for 16-bit transfers // spi_set_format(spi, 16, SPI_CPOL_0, SPI_CPHA_0, SPI_MSB_FIRST); // initialise dma channel for transmitting pixel data to screen // dma_channel = dma_claim_unused_channel(true); // dma_channel_config config = dma_channel_get_default_config(dma_channel); // channel_config_set_transfer_data_size(&config, DMA_SIZE_16); // channel_config_set_dreq(&config, spi_get_index(spi) ? DREQ_SPI1_TX : DREQ_SPI0_TX); // dma_channel_configure( // dma_channel, &config, &spi_get_hw(spi)->dr, frame_buffer, width * height, false); } spi_inst_t* ST7789::get_spi() const { return spi; } int ST7789::get_cs() const { return cs; } int ST7789::get_dc() const { return dc; } int ST7789::get_sck() const { return sck; } int ST7789::get_mosi() const { return mosi; } int ST7789::get_bl() const { return bl; } void ST7789::command(uint8_t command, size_t len, const char *data) { //dma_channel_wait_for_finish_blocking(dma_channel); gpio_put(cs, 0); gpio_put(dc, 0); // command mode spi_write_blocking(spi, &command, 1); if(data) { gpio_put(dc, 1); // data mode spi_write_blocking(spi, (const uint8_t*)data, len); } gpio_put(cs, 1); } void ST7789::update(bool dont_block) { ST7789::command(reg::RAMWR, width * height * sizeof(uint16_t), (const char*)frame_buffer); /*if(dma_channel_is_busy(dma_channel) && dont_block) { return; } dma_channel_wait_for_finish_blocking(dma_channel); uint8_t r = reg::RAMWR; gpio_put(cs, 0); gpio_put(dc, 0); // command mode spi_write_blocking(spi, &r, 1); gpio_put(dc, 1); // data mode dma_channel_set_read_addr(dma_channel, frame_buffer, true);*/ } void ST7789::set_backlight(uint8_t brightness) { // gamma correct the provided 0-255 brightness value onto a // 0-65535 range for the pwm counter float gamma = 2.8; uint16_t value = (uint16_t)(pow((float)(brightness) / 255.0f, gamma) * 65535.0f + 0.5f); pwm_set_gpio_level(bl, value); } void ST7789::vsync_callback(gpio_irq_callback_t callback) { gpio_set_irq_enabled_with_callback(vsync, GPIO_IRQ_EDGE_RISE, true, callback); } void ST7789::flip(){ madctl ^= MADCTL::ROW_ORDER | MADCTL::COL_ORDER; command(reg::MADCTL, 1, (char *)&madctl); } }