Skip to Content

Interfacing ESP32 And 16×2 LCD Parallel Data (Without I2C)

Interfacing ESP32 And 16×2 LCD Parallel Data (Without I2C)

In this article, I will show you how to connect an ESP32 microcontroller to a 16×2 LCD display with a parallel data interface. 

(In a separate article, I have already taught how to connect the ESP32 to the I2C 16×2 LCD – go check it out if you are looking for an I2C LCD with ESP32). 

The parallel LCD accepts data from ESP32 over an eight-bit data line. We will only use 4 data bits. We cannot sometimes afford 8 data lines; the LCD also supports a 4-bit data interface, which is what I will show you in this tutorial.

I will start by taking you through the LCD controller Integrated Circuit (IC) specifications, share a few tips and tricks, and show you how to display random characters and look at possible applications. 

I’ll then go into more detail and share a connection guide, ESP32 code for the 16×2 LCD parallel data interface, and answer some frequently asked questions about using a 16×2 LCD with an ESP32. 

Grab your ESP32, and let’s get started!

Components Needed To Build ESP32 And 16×2 LCD Project

Hardware Components

Software

Guide

Makerguides.com is a participant in the Amazon Services LLC Associates Program, an affiliate advertising program designed to provide a means for sites to earn advertising fees by advertising and linking to products on Amazon.com.


Fundamentals Of The 16×2 LCD Displays

Let us understand the features of the 16×2 LCD and the built-in LCD controller.

The HD44780U 16×2 LCD is a dot matrix liquid crystal display driver, with the following key features:

ParameterRange
Matrix supported5 x 8, and 5 x 10
Operating voltage2.7 V to 5.5 V
High-speed bus interfaceUp to 2 MHz
Number of data bits4-bit and 8-bit data interface
Display RAM80 x 8-bit display RAM
Character fonts208 ( 5 x 8 dot)
32 ( 5 x 10 dot)
Duty cycleProgrammable (⅛, 1/11, 1/16)

A typical connection between a Microcontroller Unit (MCU) and the display controller is shown below:

MCU to Display Controller connection
MCU to Display Controller connection

I want to begin by teaching you about CGROM and CGRAM of the LCD. 

CGROM and CGRAM

CGROM (Character Generator ROM)

Based on 8-bit character codes, the CGROM generates character patterns in either a 5×8 dot or a 5×10 dot format.

 In the 5×8 dot format, this ROM can produce 208 character patterns, and in the 5×10 dot format, 32 character patterns.

 A mask-programmed ROM can also be used to construct user-defined character patterns. These come from factory predefined. You cannot change the during run time. 

CGROM (Character Generator ROM)

You can rewrite character patterns using a program in the character generator RAM. 

Eight character patterns may be created for 5×8 dots, while four can be produced for 5×10 dots.

To display the character patterns recorded in CGRAM, enter the character codes at the addresses listed in the left column into DDRAM.

Character Patterns
Character Patterns

The correlation between CGRAM addresses, data, and display patterns is below.

CGRAM addresses, data, and display patterns
CGRAM addresses

General data RAM can be employed in areas not used for displays.

Refer to the I2C LCD Article, where you can see more examples of how to program your characters!

Interfacing the LCD controller to the MCU

4-bit interface

Used only four data lines (DB to DB7). DB0 to DB3 bus lines are disabled. The data transfer between the HD44780U and the MPU is completed after the 4-bit data has been transferred twice.

4-bit interface
4-bit interface

8-bit interface

For 8-bit interface data, all eight bus lines (DB0 to DB7) are used.

Pin definition of 16×2 LCD Module

Here is the pin description of the LCD module. Let’s begin.

Pinout of 16x2 LCD Module
Pinout of 16×2 LCD Module
Pin NumberPin NameLCD Pin Description
1VSSGround connections
2VDD5 V supply connection
3V0Contrast adjust pin (connect pot’s center pin here)
4RSRegister Select

0: Instruction register (for write) busy flag: address counter (for read)

1: Data register (for write and read)
5RWGround (always WRITE mode only)

0: Write1: Read
6EEnable the LCD controller
7D0Data bit 0 (Not used during 4-bit operation) 
8D1Data bit 1  (Not used during 4-bit operation) 
9D2Data bit 2  (Not used during 4-bit operation) 
10D3Data bit 3  (Not used during 4-bit operation) 
11D4Data bit 4
12D5Data bit 5
13D6Data bit 6
14D7Data bit 7
15ALED backlight Positive 
16KLED backlight Negative

Applications of 16×2 LCD Modules

If you want the cheapest low-power and easy-to-program display, 16×2 LCDs are the go-to solution. Here’s a list of common applications you can use them in:

  1. To display time, day of the week, and timers or counters – I have personally used LCD for this purpose on an intelligent metering system. I used to display kW consumption, and with a button, the user could change the display content to show day, time, units consumed, and more. 
  1. Sensor data display – Weather information, air quality, and other sensor data can readily be displayed on the LCDs. It helps to remove dependency on a laptop (for serial terminal) or mobile (Bluetooth). 

The simplest solution is to plug in an LCD module. You can have a switch to toggle the power to the LCD when you don’t need it. So power saving plus comfort both. 

  1. LCDs in the home automation system help you understand the status of various sensors, lights, appliances, etc. It also helps to have an LCD to see what we are entering. 
  1. The LCD acts as a rugged long-term solution in industries, especially in harsh environments. The LCDs can read the machine status and configure a few settings faster. 
  1. During the development of projects, too, you can print a few variables and the status of GPIOs on the LCD. It will help you to understand what is happening with the system. 

You don’t have to connect the debugger or multimeters. In some cases, you will not have those luxuries during testing. LCDs are your friend during those situations. 

  1. An exciting application is a dedicated message display board displaying random motivational quotes – Consider a Geek birthday gift.

Instructions To Connect The 16×2 LCD To ESP32

I will show you how to build a project using ESP32 and the 16×2 LCD. Let’s get started with the hardware connections. We will use a 4-bit data interface to talk to the LCD.

Step 1: Complete the ESP32 and LCD 16×2 hardware connections

Connect ESP32 and LCD 16x2
Connect ESP32 and LCD 16×2

Here is the table summarizing the LCD and ESP32 connections. Please use the table below as a guide and connect the LCD pins to the ESP32 pins one by one. 

LCD PinLCD LabelESP32 Pins
PIN 01VSSGND
PIN 02VDD5 V
PIN 03V010 k Pot ( center pin)
PIN 04RSGPIO19
PIN 05RWGND
PIN 06EGPIO23
PIN 07D0Not used
PIN 08D1Not used
PIN 09D2Not used
PIN 10D3Not used
PIN 11D4GPIO18
PIN 12D5GPIO17
PIN 13D6GPIO16
PIN 14D7GPIO15
PIN 15A5 V
PIN 16KGND

LCD interface using the following pins on ESP32. The potentiometer is used to control the contrast of the display. If you skip the potentiometer, you might see nothing on the display. 

A potentiometer is an excellent solution to set the contrast and avoid unnecessary troubles.

You should remember that contrast is not set and try debugging the connections. Unlike the I2C interface, the parallel interface uses several wires.

Step 2: Program the ESP32 with the code below

Follow the next step to understand the code implementation. You can use the code below to test the 16×2 LCD interface with a parallel data interface module and the connected ESP32.

// include the library code:
#include "LiquidCrystal.h"


// initialize the library with the numbers of the interface pins
LiquidCrystal lcd(19, 23, 18, 17, 16, 15);


void setup() {
  // set up the LCD's number of columns and rows:
  lcd.begin(16, 2);
  // Print a message to the LCD.
  lcd.print("MakerGuides.");
}


void loop() {
  // set the cursor to column 0, line 1
  // (note: line 1 is the second row, since counting begins with 0):
  lcd.setCursor(0, 1);
  // print the number of seconds since reset:
  lcd.print(millis() / 1000);
}

Step 3: Code Walkthrough

Let’s walk through the code. This example tests a connected 16×2 LCD. The code is used to display custom messages on the LCD.

#include "LiquidCrystal.h"

This line includes the LiquidCrystal library in the code. It allows us to use functions that communicate with an LCD screen.

LiquidCrystal lcd(19, 23, 18, 17, 16, 15);

This line initializes the LCD object with the specified interface pins. The six arguments in the parentheses represent the pins connected to the RS, E, D4, D5, D6, and D7 pins of the LCD, respectively.

void setup() {
  // set up the LCD's number of columns and rows:
  lcd.begin(16, 2);
  // Print a message to the LCD.
  lcd.print("circuitschools.");
}

The setup() function is called only once when the ESP32 board is powered up or reset. Here, we are initializing the LCD with the begin() function.

The function sets the number of columns and rows for the LCD screen. 

In the example, we set the LCD to be 16 columns and two rows. Next, we are printing the string “MakerGuides.” on the first row of the LCD screen.

void loop() {
  // set the cursor to column 0, line 1
  // (note: line 1 is the second row, since counting begins with 0):
  lcd.setCursor(0, 1);
  // print the number of seconds since reset:
  lcd.print(millis() / 1000);
}

The loop() function is called repeatedly after the `setup()` function. Here, we are setting the cursor to the first column of the second row (i.e., line 1), and printing the number of seconds that have passed since the ESP32 board was powered up or reset. 

The millis() function returns the number of milliseconds that have elapsed since the board was last reset, and we divide it by 1000 to convert it to seconds before printing it on the LCD.

I hope the code explanation helped. I hope you will use a 16×2 LCD in the near future in your upcoming projects.

Example 2: Create a Heart Pattern 

The code below creates a heart pattern.

// https://wokwi.com/projects/294395602645549578
#include "LiquidCrystal.h"


LiquidCrystal lcd(12, 11, 10, 9, 8, 7);


uint8_t heart[8] = {
  0b00000,
  0b01010,
  0b11111,
  0b11111,
  0b11111,
  0b01110,
  0b00100,
  0b00000,
};


void setup() {
  lcd.createChar(3, heart);
  lcd.begin(16, 2);
  lcd.print("  I \x03 Arduino");
}


void loop() { }
I love Adruino

Example 3: Create an Animation

This example provides you with an example of how creative you can get using the CGRAM feature.

Create an Animation

Source code: 

/*
    butterfly metamorphosis animation
    https://projecthub.arduino.cc/tusindfryd/57d77608-cd7e-4d6a-b52f-7a604cc121c5

    2021 ~ by   tusindfryd
    this code is in public domain
*/


#include <LiquidCrystal.h>
LiquidCrystal   lcd(12, 11, 5, 4, 3, 2); // RS, E, D4, D5, D6, D7
void setup()
{
  lcd.begin(16,   2);
}
void loop()
{
  image00();
  delay(250);
  image01();
  delay(250);
  image02();
  delay(250);
  image03();
  delay(700);
  image04();
  delay(250);
  image05();
  delay(250);
  image06();
  delay(700);
  image07();
  delay(1250);
}


void image00()
{
  lcd.clear();


  byte image22[8] = {B00110, B01101, B11011, B10011, B00111,   B01111, B01111, B11111};
  byte image23[8] = {B01111, B11110, B11100, B11000,   B11000, B10000, B10000, B00000};
  byte image07[8] = {B00000, B00000, B00000,   B00000, B00000, B00000, B00001, B00111};
  byte image08[8] = {B00000, B01000,   B10000, B10000, B10000, B11111, B11111, B11000};
  byte image09[8] = {B00000,   B00000, B00000, B00000, B00000, B11000, B11000, B00100};


  lcd.createChar(0,   image22);
  lcd.createChar(1, image23);
  lcd.createChar(2, image07);
  lcd.createChar(3, image08);
  lcd.createChar(4, image09);


  lcd.setCursor(5,   1);
  lcd.write(byte(0));
  lcd.setCursor(6, 1);
  lcd.write(byte(1));
  lcd.setCursor(6, 0);
  lcd.write(byte(2));
  lcd.setCursor(7, 0);
  lcd.write(byte(3));
  lcd.setCursor(8, 0);
  lcd.write(byte(4));
}


void   image01()
{
  lcd.clear();


  byte image22[8] = {B00110, B00101,   B00011, B00011, B00111, B01111, B01111, B11111};
  byte image23[8] = {B01111,   B11110, B11100, B11000, B11000, B10000, B10000, B00000};
  byte image07[8]   = {B00000, B00000, B00000, B00000, B00000, B00000, B11001, B10111};
  byte   image08[8] = {B00000, B01000, B10000, B10000, B10000, B11111, B11111, B11000};
  byte image09[8] = {B00000, B00000, B00000, B00000, B00000, B11000, B11000, B00100};
  byte image06[8] = {B00000, B00000, B00000, B00000, B00000, B00000, B00000, B00011};


  lcd.createChar(0, image22);
  lcd.createChar(1, image23);
  lcd.createChar(2,   image07);
  lcd.createChar(3, image08);
  lcd.createChar(4, image09);
  lcd.createChar(5, image06);


  lcd.setCursor(5, 1);
  lcd.write(byte(0));
  lcd.setCursor(6, 1);
  lcd.write(byte(1));
  lcd.setCursor(6, 0);
  lcd.write(byte(2));
  lcd.setCursor(7, 0);
  lcd.write(byte(3));
  lcd.setCursor(8, 0);
  lcd.write(byte(4));
  lcd.setCursor(5, 0);
  lcd.write(byte(5));
}


void image02()
{
  lcd.clear();


  byte image22[8] = {B00000, B00001, B00011, B00011, B00111, B01111, B01111, B11111};
  byte image23[8] = {B01111, B11110, B11100, B11000, B11000, B10000, B10000, B00000};
  byte image07[8] = {B00000, B00000, B00000, B00001, B00111, B00100, B11001, B10111};
  byte image08[8] = {B00000, B01000, B10000, B10000, B10000, B11111, B11111, B11000};
  byte image09[8] = {B00000, B00000, B00000, B00000, B00000, B11000, B11000, B00100};


  lcd.createChar(0, image22);
  lcd.createChar(1, image23);
  lcd.createChar(2,   image07);
  lcd.createChar(3, image08);
  lcd.createChar(4, image09);


  lcd.setCursor(5, 1);
  lcd.write(byte(0));
  lcd.setCursor(6, 1);
  lcd.write(byte(1));
  lcd.setCursor(6, 0);
  lcd.write(byte(2));
  lcd.setCursor(7, 0);
  lcd.write(byte(3));
  lcd.setCursor(8, 0);
  lcd.write(byte(4));
}


void image03()
{
  lcd.clear();


  byte image22[8] = {B00000, B00001, B00011, B00011, B00111, B01111, B01111, B11111};
  byte image23[8] = {B01111, B11110, B11100, B11000, B11000, B10000, B10000, B00000};
  byte image07[8] = {B00000, B00000, B00000, B00000, B00000, B00000, B00001, B00111};
  byte image08[8] = {B00000, B01000, B10000, B10000, B10000, B11111, B11111, B11010};
  byte image09[8] = {B00000, B00000, B00000, B00000, B00000, B11000, B11000, B00100};
  byte image24[8] = {B00010, B00111, B00111, B00111, B00111, B00111, B00010, B00000};


  lcd.createChar(0, image22);
  lcd.createChar(1, image23);
  lcd.createChar(2,   image07);
  lcd.createChar(3, image08);
  lcd.createChar(4, image09);
  lcd.createChar(5, image24);


  lcd.setCursor(5, 1);
  lcd.write(byte(0));
  lcd.setCursor(6, 1);
  lcd.write(byte(1));
  lcd.setCursor(6, 0);
  lcd.write(byte(2));
  lcd.setCursor(7, 0);
  lcd.write(byte(3));
  lcd.setCursor(8, 0);
  lcd.write(byte(4));
  lcd.setCursor(7, 1);
  lcd.write(byte(5));
}


void image04()
{
  lcd.clear();


  byte image22[8] = {B00000, B00001, B00011, B00011, B00111, B01111, B01111, B11111};
  byte image23[8] = {B01111, B11110, B11100, B11000, B11000, B10001, B10000, B00000};
  byte image07[8] = {B00000, B00000, B00000, B00000, B00000, B00000, B00001, B00111};
  byte image08[8] = {B00000, B01000, B10000, B10000, B10000, B11111, B11111, B11010};
  byte image09[8] = {B00000, B00000, B00000, B00000, B00000, B11000, B11000, B00100};
  byte image24[8] = {B00010, B00100, B01011, B10101, B11010, B10101, B11010, B01110};
  byte image25[8] = {B00000, B00000, B00000, B10000, B10000, B00000, B00000, B00000};


  lcd.createChar(0, image22);
  lcd.createChar(1, image23);
  lcd.createChar(2,   image07);
  lcd.createChar(3, image08);
  lcd.createChar(4, image09);
  lcd.createChar(5, image24);
  lcd.createChar(6, image25);


  lcd.setCursor(5,   1);
  lcd.write(byte(0));
  lcd.setCursor(6, 1);
  lcd.write(byte(1));
  lcd.setCursor(6, 0);
  lcd.write(byte(2));
  lcd.setCursor(7, 0);
  lcd.write(byte(3));
  lcd.setCursor(8, 0);
  lcd.write(byte(4));
  lcd.setCursor(7, 1);
  lcd.write(byte(5));
  lcd.setCursor(8, 1);
  lcd.write(byte(6));
}


void image05()
{
  lcd.clear();


  byte image24[8] = {B01010, B10100, B01011, B10101, B11010, B10101, B11010, B01110};
  byte image25[8] = {B00000, B00000, B00000, B10000, B10000, B00000, B00000, B00000};
  byte image23[8] = {B01101, B01010, B01101, B00111, B00000, B00000, B00000, B00000};
  byte image07[8] = {B00000, B00000, B00000, B00000, B00000, B00000, B00001, B00011};
  byte image08[8] = {B00000, B00000, B00000, B00000, B00000, B00000, B00000, B10000};


  lcd.createChar(0, image24);
  lcd.createChar(1, image25);
  lcd.createChar(2,   image23);
  lcd.createChar(3, image07);
  lcd.createChar(4, image08);


  lcd.setCursor(7, 1);
  lcd.write(byte(0));
  lcd.setCursor(8, 1);
  lcd.write(byte(1));
  lcd.setCursor(6, 1);
  lcd.write(byte(2));
  lcd.setCursor(6, 0);
  lcd.write(byte(3));
  lcd.setCursor(7, 0);
  lcd.write(byte(4));
}


void image06()
{
  lcd.clear();


  byte image08[8] = {B00000, B00100, B01010, B01010, B10001, B00011, B00110, B01100};
  byte image07[8] = {B00000, B00000, B00000, B00000, B00001, B00010, B00010, B00001};
  byte image09[8] = {B00000, B00000, B10000, B11000, B00000, B00000, B11000, B00100};
  byte image24[8] = {B00100, B00100, B00011, B00000, B00000, B00000, B00000, B00000};
  byte image25[8] = {B10000, B00000, B00000, B00000, B00000, B00000, B00000, B00000};


  lcd.createChar(0, image08);
  lcd.createChar(1, image07);
  lcd.createChar(2,   image09);
  lcd.createChar(3, image24);
  lcd.createChar(4, image25);


  lcd.setCursor(7, 0);
  lcd.write(byte(0));
  lcd.setCursor(6, 0);
  lcd.write(byte(1));
  lcd.setCursor(8, 0);
  lcd.write(byte(2));
  lcd.setCursor(7, 1);
  lcd.write(byte(3));
  lcd.setCursor(8, 1);
  lcd.write(byte(4));
}


void image07()
{
  lcd.clear();


  byte image24[8] = {B10101, B01110, B01110, B00100, B10101, B01110, B00100, B11111};
  byte image08[8] = {B00000, B00100, B01010, B01010, B10001, B00011, B00110, B01100};
  byte image07[8] = {B00000, B00000, B00000, B00000, B00001, B00010, B00010, B00001};
  byte image09[8] = {B00000, B00000, B10000, B11000, B00000, B00000, B00000, B00000};


  lcd.createChar(0, image24);
  lcd.createChar(1, image08);
  lcd.createChar(2,   image07);
  lcd.createChar(3, image09);


  lcd.setCursor(7, 1);
  lcd.write(byte(0));
  lcd.setCursor(7, 0);
  lcd.write(byte(1));
  lcd.setCursor(6, 0);
  lcd.write(byte(2));
  lcd.setCursor(8, 0);
  lcd.write(byte(3));
}

FAQs About The 16×2 LCD And ESP32 Projects

I have included a list of the most frequently asked questions about projects built using the ESP32 and the 16×2 LCD module.

What is a 16×2 parallel data LCD?

A 16×2 parallel data LCD with 16 characters in each row for 32 characters. It is controlled by a microcontroller such as Arduino or an ESP32 using a parallel interface. It involves sending data to the LCD one byte at a time through data pins.

What interface pins are required to connect a 16×2 parallel data LCD to a microcontroller?

A 16×2 parallel data LCD typically requires six interface pins connected to a microcontroller. These pins include RS (Register Select), E (Enable), D4, D5, D6, and D7.

How do I initialize and display text on a 16×2 LCD?

To initialize a 16×2 parallel data LCD, include the LiquidCrystal library in your code, create an instance of the LiquidCrystal class with the appropriate interface pins, and call the begin() function to set the number of rows and columns for the LCD. 

To display text on the LCD, you can use the print() function to send the text to the LCD one character at a time. Please refer to the connection instructions section of the article.

How do you set the cursor position on a 16×2 LCD?

To set the cursor position on a 16×2 parallel data LCD, you can use the setCursor() function, which takes two arguments: the column number (0-15) and the row number (0-1).

Can you use a 16×2 LCD with a microcontroller that doesn’t have enough pins?

You can use an I2C interface module to reduce the interface pins required to control a 16×2 parallel data LCD. The module acts as an intermediary between the LCD and the microcontroller, converting the parallel data to serial data that can be transmitted using only two interface pins.

Conclusion

This article has covered all the essential information about using the 16×2 LCD with an ESP32 microcontroller.

I have given you complete information on working the 16×2 LCD module.

Knowing the complete information about how commands work, how data is framed, and the sequences needed to initiate LCD and write to the LCD can help you get creative with your displays and also debug your code if it goes wrong.

I also shared an ESP32 connection guide and example code with explanations explaining how the code works.

Was the article easy to follow? If you have suggestions to improve the article, you are always welcome to share feedback. 

I’d love to hear from you! Let us know if there’s anything else you’d like me to cover in future articles.