You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

5.4 KiB

Project 1 Interactive Hello World

This project picks up directly from the flashing instructions in the root README.md. In this project we will be manually creating all LVGL objects within the REPL. We do this to progressively learn how an LVGL interface is built from a set of parent<->child relationships.

The code within the example assumes an ESP32-S3 DevKitC-1, pin out diagram is available here.

The code also assumes an ili9341 TFT Display with an xpt2046 touch sensor. These boards are inexpensive and available on Amazon or Aliexpress from many distributors.

Our pin mapping

The ili9341/xpt2046 uses the Serial Peripheral Interface(SPI) for communication with our board. You should review the terminology section of the documentation before continuing. The ESP32-S3 has multiple SPI buses (or in Espressif terms 'hosts') some of which are already in use for flash and PSRAM on the devkit. SPI buses 1 and 2 will be used here for our example, each of which will be connected to the ili9341 and xpt2046, with different configured speeds.

The GPIO Matrix and IO_MUX

If you are combing through the ESP-IDF documentation you will find references to SPI buses connected to specific GPIO pins. The ESP32-S3 has a flexible I/O system that allows for mapping internal peripherals (e.g. SPI) to GPIO pins. This is achieved via the I/O Multiplexer(IO_MUX) and the GPIO Matrix.

The IO_MUX is a hardware-level multiplexer that connects the chip's internal peripherals to the physical pins. Each GPIO pin has a default peripheral function, which is directly connected through the IO_MUX. Using the IO_MUX provides the fastest and most efficient connection between a peripheral and a pin. This is why you will see in the pin out diagram reference SPI functions like FSPICS0(Chip select) to GPIO pin 10.

The GPIO Matrix is a programmable routing system that sits between the IO_MUX and the chip's peripherals. It allows almost any internal signal to be routed to almost any GPIO pin. In short it allows us to map the Chip select functionality mentioned above to another pin.

This project makes generous use of the flexible pin assignment capability provided by the GPIO Matrix. Be aware that technically the IO_MUX default pins will provide better performance. Also in Micropython the SPI numbers do not directly map to the ESP-IDF SPI host numbers.

See: micropython/ports/esp32/machine_hw_spi.c

// SPI mappings by device, naming used by IDF old/new
// upython   | ESP32     | ESP32S2   | ESP32S3 | ESP32C3
// ----------+-----------+-----------+---------+---------
// SPI(id=1) | HSPI/SPI2 | FSPI/SPI2 | SPI2    | SPI2
// SPI(id=2) | VSPI/SPI3 | HSPI/SPI3 | SPI3    | err

For our ESP32-S3 the micropython hardware SPI buses (1 and 2) are mapped to the ESP-IDF SPI2 and SPI3 of our chip.

Pin Mapping

In my breadboard I have mapped the following ESP32-S3 GPIO pins to my ili9341/xpt2046. The pins chosen were simply for convenience sake, you can adjust the pin mapping as you see fit. Do note though mapping a pin to GPIO43 or GPIO44 can result in the loss of repl access. This is because UART0 TX/RX are IO_MUX'd to these pins.

On the ili9341/xpt2046 I am numbering the J2 header pins from left to right.

ESP32-S3 / Function ili9341/xpt2046 J2
3.3V 1. VCC
GND 2. GND
GPIO 6 / CS 3. CS
GPIO 5 / Reset 4. RESET
GPIO 4 / DC 5. DC
GPIO 14 / MOSI 6. SDI(MOSI)
GPIO 13 / SCK 7. SCK
3.3V 8. LED
GPIO 41 / Unused 9. SDO(MISO)
GPIO 42 10.T_CLK
GPIO 35 11.T_CS
GPIO 1 12.T_DIN
GPIO 2 13.T_DO
GPIO 42 14.T_IRQ

Running the Code

Assuming no other Micropython serial devices are connected to your PC. Plug your board in and type the command mpremote. This should automatically connect to the first device found and you will be presented with the repl input >>>.

Let's import all of the library code we will need:

import lvgl as lv
import lcd_bus
import ili9341
from machine import SPI

Now we will create our SPI bus for the ili9341 display

spi_bus = SPI.Bus(host=1, mosi=14, miso=41, sck=13)

Now we will instantiate our display object and set the data transfer rate of the bus to 40MHz.

display_bus = lcd_bus.SPIBus(
    spi_bus=spi_bus,
    dc=4,
    cs=6,
    freq=40000000,
)

display = ili9341.ILI9341(
    data_bus=display_bus,
    display_width=240,
    display_height=320,
    reset_pin=5,
    reset_state=ili9341.STATE_LOW,
    backlight_pin=40,
    color_space=lv.COLOR_FORMAT.RGB565,
    color_byte_order=ili9341.BYTE_ORDER_BGR,
    rgb565_byte_swap=True,
)
display.set_power(True)
display.init()
display.set_rotation(lv.DISPLAY_ROTATION._90)

# setting this to anything but 100 or 0 when you don't have the
# backlight state set to PWM isn't going to do anything
display.set_backlight(100)

display.invert_colors()