ESP32 direct port manipulation

14,066

Solution 1

There are many ways to do this. I often do it pin-by-pin.

One simple way is to make your own 'register' by defining a variable. If the register is 8-bit wide, define byte variable:

unsigned char disp_register;

Then you write to this register like it would exist in display hardware. Of course, next you must output this register to the GPIO pins of ESP32. Since the pins are all over, you must do this pin-by-pin. Define your hardware pins for readability:

/* OUTPUTS (numbers mean GPIO port) */
#define REGISTER_BIT7_ON_PIN        9
#define REGISTER_BIT6_ON_PIN        10
#define REGISTER_BIT5_ON_PIN        5
// continue with all the pins you need

Somewhere at the beginning of your program, set these pins as output, and perhaps make their default value to '0':

io_conf.intr_type = GPIO_PIN_INTR_DISABLE;
io_conf.mode = GPIO_MODE_OUTPUT;
io_conf.pull_down_en = GPIO_PULLDOWN_DISABLE;
io_conf.pull_up_en = GPIO_PULLUP_DISABLE;
io_conf.pin_bit_mask =  ((1ULL<< REGISTER_BIT7_ON_PIN) | (1ULL<< REGISTER_BIT6_ON_PIN) | (1ULL<< REGISTER_BIT5_ON_PIN)); // of course, do like this all the pins
gpio_config(&io_conf);

gpio_set_level(REGISTER_BIT7_ON_PIN, 0); // do like this all the pins you need to set the boot-up value, pin-by-pin

Next you need your function to copy your register to outside world of GPIO pins:

/*
 * wrote this simply for ease of understanding, feel free to do this in a loop
 * or shifting bit by bit
 */
void copy_register_to_GPIO_pins(unsigned char disp_register)
{
    gpio_set_level(REGISTER_BIT7_ON_PIN, (disp_register & 0x80) >> 7);
    gpio_set_level(REGISTER_BIT6_ON_PIN, (disp_register & 0x40) >> 6);
    gpio_set_level(REGISTER_BIT5_ON_PIN, (disp_register & 0x20) >> 5);
    gpio_set_level(REGISTER_BIT4_ON_PIN, (disp_register & 0x10) >> 4);
    gpio_set_level(REGISTER_BIT3_ON_PIN, (disp_register & 0x08) >> 3);
    gpio_set_level(REGISTER_BIT2_ON_PIN, (disp_register & 0x04) >> 2);
    gpio_set_level(REGISTER_BIT1_ON_PIN, (disp_register & 0x02) >> 1);
    gpio_set_level(REGISTER_BIT0_ON_PIN, (disp_register & 0x01));
}

Then, after you wrote anything into your register, call your function to output it:

disp_register = 0x2A; // example value you want to send to display
copy_register_to_GPIO_pins(disp_register);

// or, output byte WITHOUT using any register:
copy_register_to_GPIO_pins(0x2A);

Hopefully, you can do the reverse by yourself, reading the pins is done by another function, where you copy each GPIO pin value and assemble it into byte variable. Of course, pins have to be set to inputs at this point. In principle:

/*
 * wrote this simply for ease of understanding
 */
unsigned char copy_GPIO_pins_to_register(void)
{
    unsigned char retval = 0;

    retval |= gpio_get_level(REGISTER_BIT7_ON_PIN);
    retval = retval << 1;
    retval |= gpio_get_level(REGISTER_BIT6_ON_PIN);
    retval = retval << 1;
    retval |= gpio_get_level(REGISTER_BIT5_ON_PIN);
    retval = retval << 1;
    retval |= gpio_get_level(REGISTER_BIT4_ON_PIN);
    retval = retval << 1;
    retval |= gpio_get_level(REGISTER_BIT3_ON_PIN);
    retval = retval << 1;
    retval |= gpio_get_level(REGISTER_BIT2_ON_PIN);
    retval = retval << 1;
    retval |= gpio_get_level(REGISTER_BIT1_ON_PIN);
    retval = retval << 1;
    retval |= gpio_get_level(REGISTER_BIT0_ON_PIN);

    return retval;
}

Solution 2

In order to minimize computational burden when operating the 8 pins, you will want these pins to correspond to consecutive GPIO numbers (e.g. GPIO12 to GPIO19). Below is an implementation that operates multiple input/output pins in parallel and works if the above requirement (consecutive GPIO numbers) is met and if GPIO numbers are all in the range 0-31; I used GPIO12 to GPIO19 (GPIO12 corresponds to bit 0 in the input/output 8-bit values), which are handy to use if you have an ESP32 dev board with an ESP-WROOM-32 or ESP32-WROVER module. So I defined the GPIO corresponding to bit 0 as below:

#define PARALLEL_0  12

At initialization, you need to configure all 8 pins a GPIOs, e.g. by setting them all as inputs:

void setup() {
  for (int i = 0; i < 8; i++) {
    pinMode(PARALLEL_0 + i, INPUT);
  }
}

After that, you can use the following functions to set the 8 pins as inputs or outputs, and to read the input values and write the output values:

void parallel_set_inputs(void) {
  REG_WRITE(GPIO_ENABLE_W1TC_REG, 0xFF << PARALLEL_0);
}

void parallel_set_outputs(void) {
  REG_WRITE(GPIO_ENABLE_W1TS_REG, 0xFF << PARALLEL_0);
}

uint8_t parallel_read(void) {
  uint32_t input = REG_READ(GPIO_IN_REG);

  return (input >> PARALLEL_0);
}

void parallel_write(uint8_t value) {
  uint32_t output =
    (REG_READ(GPIO_OUT_REG) & ~(0xFF << PARALLEL_0)) | (((uint32_t)value) << PARALLEL_0);

  REG_WRITE(GPIO_OUT_REG, output);
}

Solution 3

For high-bandwidth parallel data output, you might want to investigate the ESP32's I2S peripheral's LCD mode.

See section 12.5.1 in the ESP32 TRM and chapter 4 on mapping the peripheral to the desired pins. The nice thing about this approach is that you can map up to 24 bits of output from the peripheral to an output pin.

Section 12.4.4 states:

The ESP32 I2S module carries out a data-transmit operation [...] Clock out data serially, or in parallel, as configured by the user

Share:
14,066
Daan van Driel
Author by

Daan van Driel

Hi, my name is Daan and I'm a young and passionate Industrial Designer. I just graduated from the Technical University of Eindhoven, the Netherlands and started working at VanBerlo. I like to come up with concepts and translate them into functional prototypes. Therefore my interest in electronics and programming grew a lot. I mainly program in c and C++ because I like to have full control over the hardware I use. However, I'm always up to learn something new! StackOverflow is for me the ideal community to learn and many people already helped me to develop my technical skills!

Updated on June 14, 2022

Comments

  • Daan van Driel
    Daan van Driel almost 2 years

    Dear StackOverflowers,

    I am trying to use HX8357D 3.5" TFT from Adafruit (link) with an esp32. The TFT driver has two interfaces: SPI and 8-bit parallel. The provided library from Adafruit (link) only supports SPI on the esp32. I need to have higher display speeds, so I decided to try and add support for the esp32 myself. I'm not experienced at all with this kind of programming, but I liked the challenge.

    I figured out how the 8-bit interface work by reverse engineering the Arduino Uno/Mega support. To add the esp32 support I need a way to directly manipulate the registers controlling the gpio ports of the esp32. I looked around on the internet, but there are very little examples of how to do this. The technical reference manual of Espressif (link) contains all the information needed, but I'm not skilled enough to figure out how to translate this into code.

    To program the esp32 I use the esp32 Arduino core. This example (link) shows how to set gpio pins as output and make them HIGH and LOW directly using registers. The problem is that I need to be able to set 8 pins as output, write data to them, make them input and then read data from them, all using registers instead of using the pinMode, digitalRead and digitalWrite functions.

    The way it works on the Arduino Uno/Mega is clear to me, there are three registers that control a port:

    • DDR* to read/write
    • PORT* to set gpio HIGH/LOW
    • PIN* to read HIGH/LOW if the gpio is INPUT.

    But how does this work on the esp32 and how can I make use of the registers to create this 8-bit parallel communication?

    If there is anybody that has more know-how than me on this topic I would super be grateful for an explanation. Thanks in advance.

  • Daan van Driel
    Daan van Driel over 5 years
    Thanks for the answer, you cleared things up for me. The thing is that I'm looking for a way to directly write the data to all gpio ports. The 8 pins that I need are free to choose and the esp32 has 32 pins on one register, so that shouldn't be a problem right? I do not really understand why you said "the pins are all over so you have to do this pin-by-pin"? I thought that if you have the right bitmask you can write to multiple gpio pins by directly writing to the register. In the datasheet I found that these registers are called W1TS & W1TC. You have any clue on how to do this?
  • EmbeddedGuy
    EmbeddedGuy over 5 years
    I am not aware if ESP32 has direct parallel port capability for use like you want (try their technical support - they are helpful).