| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352 |
- /* SPI Master example
- This example code is in the Public Domain (or CC0 licensed, at your option.)
- Unless required by applicable law or agreed to in writing, this
- software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
- CONDITIONS OF ANY KIND, either express or implied.
- */
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include "freertos/FreeRTOS.h"
- #include "freertos/task.h"
- #include "esp_system.h"
- #include "driver/spi_master.h"
- #include "soc/gpio_struct.h"
- #include "driver/gpio.h"
- /*
- This code displays some fancy graphics on the 320x240 LCD on an ESP-WROVER_KIT board.
- It is not very fast, even when the SPI transfer itself happens at 8MHz and with DMA, because
- the rest of the code is not very optimized. Especially calculating the image line-by-line
- is inefficient; it would be quicker to send an entire screenful at once. This example does, however,
- demonstrate the use of both spi_device_transmit as well as spi_device_queue_trans/spi_device_get_trans_result
- as well as pre-transmit callbacks.
- Some info about the ILI9341/ST7789V: It has an C/D line, which is connected to a GPIO here. It expects this
- line to be low for a command and high for data. We use a pre-transmit callback here to control that
- line: every transaction has as the user-definable argument the needed state of the D/C line and just
- before the transaction is sent, the callback will set this line to the correct state.
- */
- #define PIN_NUM_MISO 25
- #define PIN_NUM_MOSI 23
- #define PIN_NUM_CLK 19
- #define PIN_NUM_CS 22
- #define PIN_NUM_DC 21
- #define PIN_NUM_RST 18
- #define PIN_NUM_BCKL 5
- /*
- The LCD needs a bunch of command/argument values to be initialized. They are stored in this struct.
- */
- typedef struct {
- uint8_t cmd;
- uint8_t data[16];
- uint8_t databytes; //No of data in data; bit 7 = delay after set; 0xFF = end of cmds.
- } lcd_init_cmd_t;
- typedef enum {
- LCD_TYPE_ILI = 1,
- LCD_TYPE_ST,
- LCD_TYPE_MAX,
- } type_lcd_t;
- //Place data into DRAM. Constant data gets placed into DROM by default, which is not accessible by DMA.
- DRAM_ATTR static const lcd_init_cmd_t st_init_cmds[]={
- {0x36, {(1<<5)|(1<<6)}, 1},
- {0x3A, {0x55}, 1},
- {0xB2, {0x0c, 0x0c, 0x00, 0x33, 0x33}, 5},
- {0xB7, {0x45}, 1},
- {0xBB, {0x2B}, 1},
- {0xC0, {0x2C}, 1},
- {0xC2, {0x01, 0xff}, 2},
- {0xC3, {0x11}, 1},
- {0xC4, {0x20}, 1},
- {0xC6, {0x0f}, 1},
- {0xD0, {0xA4, 0xA1}, 1},
- {0xE0, {0xD0, 0x00, 0x05, 0x0E, 0x15, 0x0D, 0x37, 0x43, 0x47, 0x09, 0x15, 0x12, 0x16, 0x19}, 14},
- {0xE1, {0xD0, 0x00, 0x05, 0x0D, 0x0C, 0x06, 0x2D, 0x44, 0x40, 0x0E, 0x1C, 0x18, 0x16, 0x19}, 14},
- {0x11, {0}, 0x80},
- {0x29, {0}, 0x80},
- {0, {0}, 0xff}
- };
- DRAM_ATTR static const lcd_init_cmd_t ili_init_cmds[]={
- {0xCF, {0x00, 0x83, 0X30}, 3},
- {0xED, {0x64, 0x03, 0X12, 0X81}, 4},
- {0xE8, {0x85, 0x01, 0x79}, 3},
- {0xCB, {0x39, 0x2C, 0x00, 0x34, 0x02}, 5},
- {0xF7, {0x20}, 1},
- {0xEA, {0x00, 0x00}, 2},
- {0xC0, {0x26}, 1},
- {0xC1, {0x11}, 1},
- {0xC5, {0x35, 0x3E}, 2},
- {0xC7, {0xBE}, 1},
- {0x36, {0x28}, 1},
- {0x3A, {0x55}, 1},
- {0xB1, {0x00, 0x1B}, 2},
- {0xF2, {0x08}, 1},
- {0x26, {0x01}, 1},
- {0xE0, {0x1F, 0x1A, 0x18, 0x0A, 0x0F, 0x06, 0x45, 0X87, 0x32, 0x0A, 0x07, 0x02, 0x07, 0x05, 0x00}, 15},
- {0XE1, {0x00, 0x25, 0x27, 0x05, 0x10, 0x09, 0x3A, 0x78, 0x4D, 0x05, 0x18, 0x0D, 0x38, 0x3A, 0x1F}, 15},
- {0x2A, {0x00, 0x00, 0x00, 0xEF}, 4},
- {0x2B, {0x00, 0x00, 0x01, 0x3f}, 4},
- {0x2C, {0}, 0},
- {0xB7, {0x07}, 1},
- {0xB6, {0x0A, 0x82, 0x27, 0x00}, 4},
- {0x11, {0}, 0x80},
- {0x29, {0}, 0x80},
- {0, {0}, 0xff},
- };
- //Send a command to the LCD. Uses spi_device_transmit, which waits until the transfer is complete.
- void lcd_cmd(spi_device_handle_t spi, const uint8_t cmd)
- {
- esp_err_t ret;
- spi_transaction_t t;
- memset(&t, 0, sizeof(t)); //Zero out the transaction
- t.length=8; //Command is 8 bits
- t.tx_buffer=&cmd; //The data is the cmd itself
- t.user=(void*)0; //D/C needs to be set to 0
- ret=spi_device_transmit(spi, &t); //Transmit!
- assert(ret==ESP_OK); //Should have had no issues.
- }
- //Send data to the LCD. Uses spi_device_transmit, which waits until the transfer is complete.
- void lcd_data(spi_device_handle_t spi, const uint8_t *data, int len)
- {
- esp_err_t ret;
- spi_transaction_t t;
- if (len==0) return; //no need to send anything
- memset(&t, 0, sizeof(t)); //Zero out the transaction
- t.length=len*8; //Len is in bytes, transaction length is in bits.
- t.tx_buffer=data; //Data
- t.user=(void*)1; //D/C needs to be set to 1
- ret=spi_device_transmit(spi, &t); //Transmit!
- assert(ret==ESP_OK); //Should have had no issues.
- }
- //This function is called (in irq context!) just before a transmission starts. It will
- //set the D/C line to the value indicated in the user field.
- void lcd_spi_pre_transfer_callback(spi_transaction_t *t)
- {
- int dc=(int)t->user;
- gpio_set_level(PIN_NUM_DC, dc);
- }
- uint32_t lcd_get_id(spi_device_handle_t spi)
- {
- //get_id cmd
- lcd_cmd( spi, 0x04);
- spi_transaction_t t;
- memset(&t, 0, sizeof(t));
- t.length=8*3;
- t.flags = SPI_TRANS_USE_RXDATA;
- t.user = (void*)1;
- esp_err_t ret = spi_device_transmit(spi, &t);
- assert( ret == ESP_OK );
- return *(uint32_t*)t.rx_data;
- }
- //Initialize the display
- void lcd_init(spi_device_handle_t spi)
- {
- int cmd=0;
- const lcd_init_cmd_t* lcd_init_cmds;
- //Initialize non-SPI GPIOs
- gpio_set_direction(PIN_NUM_DC, GPIO_MODE_OUTPUT);
- gpio_set_direction(PIN_NUM_RST, GPIO_MODE_OUTPUT);
- gpio_set_direction(PIN_NUM_BCKL, GPIO_MODE_OUTPUT);
- //Reset the display
- gpio_set_level(PIN_NUM_RST, 0);
- vTaskDelay(100 / portTICK_RATE_MS);
- gpio_set_level(PIN_NUM_RST, 1);
- vTaskDelay(100 / portTICK_RATE_MS);
- //detect LCD type
- uint32_t lcd_id = lcd_get_id(spi);
- int lcd_detected_type = 0;
- int lcd_type;
- printf("LCD ID: %08X\n", lcd_id);
- if ( lcd_id == 0 ) {
- //zero, ili
- lcd_detected_type = LCD_TYPE_ILI;
- printf("ILI9341 detected...\n");
- } else {
- // none-zero, ST
- lcd_detected_type = LCD_TYPE_ST;
- printf("ST7789V detected...\n");
- }
- #ifdef CONFIG_LCD_TYPE_AUTO
- lcd_type = lcd_detected_type;
- #elif defined( CONFIG_LCD_TYPE_ST7789V )
- printf("kconfig: force CONFIG_LCD_TYPE_ST7789V.\n");
- lcd_type = LCD_TYPE_ST;
- #elif defined( CONFIG_LCD_TYPE_ILI9341 )
- printf("kconfig: force CONFIG_LCD_TYPE_ILI9341.\n");
- lcd_type = LCD_TYPE_ILI;
- #endif
- if ( lcd_type == LCD_TYPE_ST ) {
- printf("LCD ST7789V initialization.\n");
- lcd_init_cmds = st_init_cmds;
- } else {
- printf("LCD ILI9341 initialization.\n");
- lcd_init_cmds = ili_init_cmds;
- }
- //Send all the commands
- while (lcd_init_cmds[cmd].databytes!=0xff) {
- lcd_cmd(spi, lcd_init_cmds[cmd].cmd);
- lcd_data(spi, lcd_init_cmds[cmd].data, lcd_init_cmds[cmd].databytes&0x1F);
- if (lcd_init_cmds[cmd].databytes&0x80) {
- vTaskDelay(100 / portTICK_RATE_MS);
- }
- cmd++;
- }
- ///Enable backlight
- gpio_set_level(PIN_NUM_BCKL, 0);
- }
- //To send a line we have to send a command, 2 data bytes, another command, 2 more data bytes and another command
- //before sending the line data itself; a total of 6 transactions. (We can't put all of this in just one transaction
- //because the D/C line needs to be toggled in the middle.)
- //This routine queues these commands up so they get sent as quickly as possible.
- static void send_line(spi_device_handle_t spi, int ypos, uint16_t *line)
- {
- esp_err_t ret;
- int x;
- //Transaction descriptors. Declared static so they're not allocated on the stack; we need this memory even when this
- //function is finished because the SPI driver needs access to it even while we're already calculating the next line.
- static spi_transaction_t trans[6];
- //In theory, it's better to initialize trans and data only once and hang on to the initialized
- //variables. We allocate them on the stack, so we need to re-init them each call.
- for (x=0; x<6; x++) {
- memset(&trans[x], 0, sizeof(spi_transaction_t));
- if ((x&1)==0) {
- //Even transfers are commands
- trans[x].length=8;
- trans[x].user=(void*)0;
- } else {
- //Odd transfers are data
- trans[x].length=8*4;
- trans[x].user=(void*)1;
- }
- trans[x].flags=SPI_TRANS_USE_TXDATA;
- }
- trans[0].tx_data[0]=0x2A; //Column Address Set
- trans[1].tx_data[0]=0; //Start Col High
- trans[1].tx_data[1]=0; //Start Col Low
- trans[1].tx_data[2]=(320)>>8; //End Col High
- trans[1].tx_data[3]=(320)&0xff; //End Col Low
- trans[2].tx_data[0]=0x2B; //Page address set
- trans[3].tx_data[0]=ypos>>8; //Start page high
- trans[3].tx_data[1]=ypos&0xff; //start page low
- trans[3].tx_data[2]=(ypos+1)>>8; //end page high
- trans[3].tx_data[3]=(ypos+1)&0xff; //end page low
- trans[4].tx_data[0]=0x2C; //memory write
- trans[5].tx_buffer=line; //finally send the line data
- trans[5].length=320*2*8; //Data length, in bits
- trans[5].flags=0; //undo SPI_TRANS_USE_TXDATA flag
- //Queue all transactions.
- for (x=0; x<6; x++) {
- ret=spi_device_queue_trans(spi, &trans[x], portMAX_DELAY);
- assert(ret==ESP_OK);
- }
- //When we are here, the SPI driver is busy (in the background) getting the transactions sent. That happens
- //mostly using DMA, so the CPU doesn't have much to do here. We're not going to wait for the transaction to
- //finish because we may as well spend the time calculating the next line. When that is done, we can call
- //send_line_finish, which will wait for the transfers to be done and check their status.
- }
- static void send_line_finish(spi_device_handle_t spi)
- {
- spi_transaction_t *rtrans;
- esp_err_t ret;
- //Wait for all 6 transactions to be done and get back the results.
- for (int x=0; x<6; x++) {
- ret=spi_device_get_trans_result(spi, &rtrans, portMAX_DELAY);
- assert(ret==ESP_OK);
- //We could inspect rtrans now if we received any info back. The LCD is treated as write-only, though.
- }
- }
- //Simple routine to generate some patterns and send them to the LCD. Don't expect anything too
- //impressive. Because the SPI driver handles transactions in the background, we can calculate the next line
- //while the previous one is being sent.
- static void display_pretty_colors(spi_device_handle_t spi)
- {
- uint16_t line[2][320];
- int x, y, frame=0;
- //Indexes of the line currently being sent to the LCD and the line we're calculating.
- int sending_line=-1;
- int calc_line=0;
-
- while(1) {
- frame++;
- for (y=0; y<240; y++) {
- //Calculate a line.
- for (x=0; x<320; x++) {
- line[calc_line][x]=((x<<3)^(y<<3)^(frame+x*y));
- }
- //Finish up the sending process of the previous line, if any
- if (sending_line!=-1) send_line_finish(spi);
- //Swap sending_line and calc_line
- sending_line=calc_line;
- calc_line=(calc_line==1)?0:1;
- //Send the line we currently calculated.
- send_line(spi, y, line[sending_line]);
- //The line is queued up for sending now; the actual sending happens in the
- //background. We can go on to calculate the next line as long as we do not
- //touch line[sending_line]; the SPI sending process is still reading from that.
- }
- }
- }
- void app_main()
- {
- esp_err_t ret;
- spi_device_handle_t spi;
- spi_bus_config_t buscfg={
- .miso_io_num=PIN_NUM_MISO,
- .mosi_io_num=PIN_NUM_MOSI,
- .sclk_io_num=PIN_NUM_CLK,
- .quadwp_io_num=-1,
- .quadhd_io_num=-1
- };
- spi_device_interface_config_t devcfg={
- .clock_speed_hz=10*1000*1000, //Clock out at 10 MHz
- .mode=0, //SPI mode 0
- .spics_io_num=PIN_NUM_CS, //CS pin
- .queue_size=7, //We want to be able to queue 7 transactions at a time
- .pre_cb=lcd_spi_pre_transfer_callback, //Specify pre-transfer callback to handle D/C line
- };
- //Initialize the SPI bus
- ret=spi_bus_initialize(HSPI_HOST, &buscfg, 1);
- assert(ret==ESP_OK);
- //Attach the LCD to the SPI bus
- ret=spi_bus_add_device(HSPI_HOST, &devcfg, &spi);
- assert(ret==ESP_OK);
- //Initialize the LCD
- lcd_init(spi);
- //Go do nice stuff.
- display_pretty_colors(spi);
- }
|