|  |   | 
| (71 intermediate revisions by 6 users not shown) | 
| Line 1: | Line 1: | 
|  | [[category: LPC1768 Tutorials]] |  | [[category: LPC1768 Tutorials]] | 
| − | 
 |  | 
| − | 
 |  | 
| − | =Objective=
 |  | 
|  | In this tutorial we are going to see how to interface a 2x16 LCD with LPC1768 in 4-bit mode. |  | In this tutorial we are going to see how to interface a 2x16 LCD with LPC1768 in 4-bit mode. | 
|  | As per the name the 2x16 has 2 lines with 16 chars on each lines. It supports all the ascii chars and is basically used for displaying the alpha numeric characters. Here each character is displayed in a matrix of 5x7 pixels. |  | As per the name the 2x16 has 2 lines with 16 chars on each lines. It supports all the ascii chars and is basically used for displaying the alpha numeric characters. Here each character is displayed in a matrix of 5x7 pixels. | 
| Line 11: | Line 8: | 
|  |  |  |  | 
|  | =LCD UNIT= |  | =LCD UNIT= | 
| − | {{Box|type=l_green_light|text=
 |  | 
| − | 
 |  | 
|  | Let us look at a pin diagram of a commercially available LCD like '''JHD162''' which uses a '''HD44780''' controller and then describe its operation. |  | Let us look at a pin diagram of a commercially available LCD like '''JHD162''' which uses a '''HD44780''' controller and then describe its operation. | 
|  |  |  |  | 
| − | [[File:PIN_Diagram.PNG]] | + | [[FILE:Pic16f877aLcdInterface.png]] | 
| − |   | + |  | 
| − | All the pins are identically to the lcd internal controller discussed above
 | + |  | 
| − |   | + |  | 
| − | }}
 | + |  | 
|  |  |  |  | 
| − | {|class="wikitable " | + | {| class="table table-striped table-hover table-condensed table-bordered" | 
|  | + | |-class="info" | 
|  | + | | Pin Number || Symbol || Pin Function | 
|  | |- |  | |- | 
| − | ! PIN NUMBER !! FUNCTION
 | + | |1 || VSS ||Ground | 
|  | |- |  | |- | 
| − | |1 ||Ground | + | | 2|| VCC || +5v | 
|  | |- |  | |- | 
| − | | 2||VCC  | + | | 3 || VEE || Contrast adjustment (VO)  | 
|  | |- |  | |- | 
| − | | 3 ||Contrast adjustment (VO)  | + | | 4 || RS || Register Select. 0:Command, 1: Data | 
|  | |- |  | |- | 
| − | | 4 ||Register Select (RS). RS=0:Command, RS=1:Data | + | | 5 || R/W || Read/Write, R/W=0: Write & R/W=1: Read  | 
|  | |- |  | |- | 
| − | | 5 ||Read/Write (R/W).R/W=0: Write, R/W=1: Read  | + | | 6|| EN || Enable. Falling edge triggered | 
|  | |- |  | |- | 
| − | |  6||Clock (Enable). Falling edge triggered | + | | 7 || D0 || Data Bit 0 (Not used in 4-bit operation) | 
|  | + | |-          | 
|  | + | | 8 || D1 || Data Bit 1 (Not used in 4-bit operation) | 
|  | + | |-          | 
|  | + | | 9 || D2 || Data Bit 2 (Not used in 4-bit operation) | 
|  | |- |  | |- | 
| − | | 7 || Bit0 (Not used in 4-bit operation) | + | | 10 || D3 || Data Bit 3 (Not used in 4-bit operation)   | 
|  | |- |  | |- | 
| − | | 8 || Bit1 (Not used in 4-bit operation) | + | | 11 || D4 || Data Bit 4 | 
|  | + | |-           | 
|  | + | | 12 || D5 || Data Bit 5 | 
|  | + | |-            | 
|  | + | | 13 || D6 || Data Bit 6 | 
|  | + | |-            | 
|  | + | | 14 || D7 || Data Bit 7/Busy Flag | 
|  | |- |  | |- | 
| − | | 9 || Bit 2 (Not used in 4-bit operation) | + | | 15 || A/LED+ || Back-light Anode(+) | 
|  | |- |  | |- | 
| − | | 10 ||Bit 3 (Not used in 4-bit operation)  | + | | 16 || K/LED- || Back-Light Cathode(-)   | 
| − | |-
 | + |  | 
| − | | 11 || Bit 4
 | + |  | 
| − | |-
 | + |  | 
| − | | 12 || Bit 5
 | + |  | 
| − | |-
 | + |  | 
| − | | 13 || Bit 6
 | + |  | 
| − | |-
 | + |  | 
| − | |  14|| Bit 7
 | + |  | 
| − | |-
 | + |  | 
| − | |  15|| Back-light Anode(+)
 | + |  | 
| − | |-
 | + |  | 
| − | | 16 || Back-Light Cathode(-)
 | + |  | 
|  | |} |  | |} | 
| − | <br/> | + | <br><br> | 
| − | <br/> | + |  | 
| − | <br/>
 | + |  | 
| − | <br/>
 | + |  | 
|  |  |  |  | 
|  | Apart from the voltage supply connections the important pins from the programming perspective are the data lines(8-bit Data bus), Register select, Read/Write and Enable pin.<br><br>    |  | Apart from the voltage supply connections the important pins from the programming perspective are the data lines(8-bit Data bus), Register select, Read/Write and Enable pin.<br><br>    | 
| Line 73: | Line 61: | 
|  | <b>Enable(EN):</b> This pin is used to send the enable trigger to LCD. |  | <b>Enable(EN):</b> This pin is used to send the enable trigger to LCD. | 
|  | After sending the data/cmd, Selecting the data/cmd register, Selecting the Write operation. A HIGH-to-LOW pulse has to be send on this enable pin which will latch the info into the LCD register and triggers the LCD to act accordingly. |  | After sending the data/cmd, Selecting the data/cmd register, Selecting the Write operation. A HIGH-to-LOW pulse has to be send on this enable pin which will latch the info into the LCD register and triggers the LCD to act accordingly. | 
|  | + | <br/> | 
|  | + | <br/> | 
|  |  |  |  | 
|  | + | =Hardware Connections= | 
|  | + | [[File:LCD 4Bit.jpg|680px]] | 
|  | =Schematic= |  | =Schematic= | 
|  | + | Below schematic shows the minimum connection required for interfacing the LCD with the microcontroller.<br> | 
|  | + | As we are interfacing the LCD in 4-bit mode, only the higher 4 data lines are used as data bus. | 
|  |  |  |  | 
|  | + | <html> | 
|  | + | <img src=https://www.exploreembedded.com/wiki/images/b/b8/LPC1768_4bit_LCD.PNG class="img-responsive">  | 
|  | + | </html> | 
|  | + |   | 
|  |  |  |  | 
| − |   | + | <br/> | 
|  | + | <br/> | 
|  |  |  |  | 
|  | =Port Connection= |  | =Port Connection= | 
|  | This section shows how to configure the GPIO for interfacing the LCD.<br> |  | This section shows how to configure the GPIO for interfacing the LCD.<br> | 
|  | The below configuration is as per the above schematic. You can connect the LCD to any of the PORT pins available on your boards and update this section accordingly |  | The below configuration is as per the above schematic. You can connect the LCD to any of the PORT pins available on your boards and update this section accordingly | 
|  | + | <html> | 
|  | + | <script src="https://gist.github.com/Amritach/dfd031e49d0335167eb8.js"></script> | 
|  | + | </html> | 
|  | + | <br/> | 
|  | + | <br/> | 
|  |  |  |  | 
| − | <syntaxhighlight>
 | + | =LCD Operation= | 
| − | /* Configure the databus andControl bus as per thehardware connection */ | + | In this section we are going to see how to send the data/cmd to the LCD along with the timing diagrams. | 
|  | + | First lets see the timing diagram for sending the data and the command signals(RS,RW,EN), accordingly we write the algorithm and finally the code. | 
|  |  |  |  | 
| − | #define LcdDataBusPort       LPC_GPIO1->FIOPIN
 | + | ===Timing Diagram=== | 
| − | #define LcdControlBusPort   LPC_GPIO2->FIOPIN
 | + | The below image shows the timing diagram for sending the data to the LCD.<br> | 
|  | + | As shown in the timing diagram  the data is written after sending the RS and RW signals. It is still ok to send the data before these signals.<br> | 
|  | + | The only important thing is the data should be available on the databus before generating the High-to-Low pulse on EN pin. | 
|  | + | [[File:LCD CmdWrite.jpg|figure: command write]] | 
|  | + | <br><br> | 
|  |  |  |  | 
| − | #define LcdDataBusDirnReg    LPC_GPIO1->FIODIR
 |  | 
| − | #define LcdCtrlBusDirnReg      LPC_GPIO2->FIODIR
 |  | 
|  |  |  |  | 
| − | #define LCD_D4     24 | + | ===Steps for Sending Command:===  | 
| − | #define LCD_D5     25 | + | # Send the I/P command to LCD. | 
| − | #define LCD_D6     26 | + | # Select the Control Register by making RS low. | 
| − | #define LCD_D7     27 | + | #Select Write operation making RW low. | 
|  | + | #Send a High-to-Low pulse on Enable PIN with some delay_us. | 
|  | + | <html> | 
|  | + | <script src="https://gist.github.com/Amritach/6e4b08f06568b82fcd64.js"></script> | 
|  | + | </html> | 
|  | + | <br><br> | 
|  |  |  |  | 
| − | #define LCD_RS     0 | + | ===Steps for Sending Data:===  | 
| − | #define LCD_RW     1 | + | # Send the character to LCD. | 
| − | #define LCD_EN     2 | + | # Select the Data Register by making RS high. | 
| − | </syntaxhighlight>
 | + | # Select Write operation making RW low. | 
|  | + | # Send a High-to-Low pulse on Enable PIN with some delay_us. | 
|  | + | The timings are similar as above only change is that '''RS''' is made high for selecting Data register. | 
|  |  |  |  | 
| − | =Example1=
 | + | <html> | 
| − | In this program we are going to do both INPUT and OUTPUT operation.
 | + | <script src="https://gist.github.com/Amritach/6a137d1bf6a27cefb8de.js"></script> | 
| − | The port pin to which switch is connected is configured as Input and the pin to which LED is connected is configured as OUTPUT.
 | + | </html> | 
| − | Here the switch status is read and accordingly the LED will be turned ON/OFF.
 | + | <br><br> | 
|  |  |  |  | 
| − | <syntaxhighlight> | + | = Code Examples = | 
| − | #include <lpc17xx.h>  
 | + | ===Example 1=== | 
|  | + | Here is the complete code for displaying the data on 2x16 LCD in 4-bit mode. | 
|  | + | <html> | 
|  | + | <script src="https://gist.github.com/SaheblalBagwan/96e2caabd9824981c464e68d3a11aa8b.js"></script> | 
|  | + | </html> | 
|  |  |  |  | 
| − | #define SwitchPinNumber 11
 | + | ===Using Explore Embedded Libraries :=== | 
| − | #define LedPinNumber    0
 | + | In the above tutorial we just discussed how to interface 2x16Lcd in 4-bit mode.<br> | 
|  | + | Once you know the working of lcd, you can directly use the ExploreEmbedded libraries to play around with your LCD.<br> | 
|  | + | For that you need to include the lcd.c/lcd.h and the associated files(delay/stdutils).<br>  | 
|  | + | After including these files, the only thing you got to do is to configure the PORTs in lcd.h as per your hardware connection.<br> | 
|  | + | The below sample code shows how to use the already available LCD functions.<br> | 
|  | + | Refer this link for more info on LCD libraries. | 
|  | + | <html> | 
|  | + | <script src="https://gist.github.com/Amritach/d5b38e3fc891c74a1032.js"></script> | 
|  | + | </html> | 
|  | + | <br><br> | 
|  |  |  |  | 
| − | /* start the main program */
 |  | 
| − | void main() 
 |  | 
| − | {
 |  | 
| − |     uint32_t switchStatus;
 |  | 
| − | 	SystemInit();	                //Clock and PLL configuration 
 |  | 
| − |     LPC_GPIO2->FIODIR = ((1<<LedPinNumber) | (0<<SwitchPinNumber)); //Configure the PORT pins as OUTPUT;
 |  | 
| − | 
 |  | 
| − |   while(1)
 |  | 
| − |     {
 |  | 
| − | 
 |  | 
| − | 	 /* Turn On all the leds and wait for one second */ 
 |  | 
| − |        switchStatus = (LPC_GPIO2->FIOPIN>>SwitchPinNumber) & 0x01 ;	 // Make all the Port pins as high
 |  | 
| − | 	   
 |  | 
| − | 	   if(switchStatus == 1)
 |  | 
| − | 	   {  
 |  | 
| − | 	     LPC_GPIO2->FIOPIN = (1<<LedPinNumber);
 |  | 
| − | 	   }
 |  | 
| − | 	   else
 |  | 
| − | 	   {
 |  | 
| − | 	   	 LPC_GPIO2->FIOPIN = (0<<LedPinNumber);
 |  | 
| − | 	   }	  
 |  | 
| − | 	}
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | =Example2=
 |  | 
| − | This is the alternate method using the stdutils macros. 
 |  | 
| − | <syntaxhighlight>
 |  | 
| − | #include <lpc17xx.h>   
 |  | 
| − | #include "delay.h"     //User defined library which contains the delay routines
 |  | 
| − | #include "stdutils.h"
 |  | 
| − | 
 |  | 
| − | #define SwitchPinNumber 11
 |  | 
| − | #define LedPinNumber    0
 |  | 
| − | 
 |  | 
| − | /* start the main program */
 |  | 
| − | void main() 
 |  | 
| − | {
 |  | 
| − |     uint32_t switchStatus;
 |  | 
| − | 	SystemInit();	                //Clock and PLL configuration 
 |  | 
| − |     LPC_GPIO2->FIODIR = ((1<<LedPinNumber) | (0<<SwitchPinNumber)); //Configure the PORT pins as OUTPUT;
 |  | 
| − | 
 |  | 
| − |   while(1)
 |  | 
| − |     {
 |  | 
| − | 
 |  | 
| − | 	 /* Turn On all the leds and wait for one second */ 
 |  | 
| − |        switchStatus = util_GetBitStatus(LPC_GPIO2->FIOPIN,SwitchPinNumber);	 // Make all the Port pins as high
 |  | 
| − | 	   
 |  | 
| − | 	   if(switchStatus == 1)
 |  | 
| − | 	   {  
 |  | 
| − | 	     util_BitSet(LPC_GPIO2->FIOPIN,LedPinNumber);
 |  | 
| − | 	   }
 |  | 
| − | 	   else
 |  | 
| − | 	   {
 |  | 
| − | 	   	 util_BitClear(LPC_GPIO2->FIOPIN,LedPinNumber);
 |  | 
| − | 	   }	  
 |  | 
| − | 	}
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | =Video Tutorial=
 |  | 
| − | {{Box|type=l_green_light|text=<br />
 |  | 
| − | {{#ev:youtube|YDJISiPUdA8|600}}
 |  | 
| − | {{#ev:youtube|xG7uu98qz_A|200}}
 |  | 
| − | {{#ev:youtube|jRA4gefNVhM|200}}
 |  | 
| − | {{#ev:youtube|g-3Vt7SyL9w|200}}
 |  | 
| − | =Introduction:LCD=
 |  | 
| − | 
 |  | 
| − | Liquid Crystal Display(LCDs) provide a cost effective way to put a text output unit for a microcontroller. As we have seen in the previous tutorial, LEDs or 7 Segments do no have the flexibility to display informative messages. 
 |  | 
| − | This display has 2 lines and can display 16 characters on each line. Nonetheless, when it is interfaced with the micrcontroller, we can scroll the messages with software to display information which is more than 16 characters in  length.
 |  | 
| − | 
 |  | 
| − | == LCD Internal Controller==
 |  | 
| − | [[File:HD44780U Block diagram.png|570x250px|framed|Fig 1: LCD Block diagram]]<br />
 |  | 
| − | 
 |  | 
| − | The LCD is a simple device to use but the internal details are complex. Most of the 16x2 LCDs use a '''Hitachi HD44780''' or a compatible controller. Yes, a micrcontroller is present inside a Liquid crystal display as shown in figure 1.
 |  | 
| − | [[File:LCD Char 5x7 Matrix.jpg|thumbnail]]
 |  | 
| − | The Display Controller takes commands and data from a external microcontroller and drivers the LCD panel'''(LCDP)'''.
 |  | 
| − | It takes a ASCII value as input and generate a patter for the dot matrix. E.g., to display letter 'A', it takes its value '''0X42(hex)''' or '''66(dec)''' decodes it into a dot matrix of 5x7 as shown in figure 2.
 |  | 
| − | 
 |  | 
| − | ==Basic Commands==
 |  | 
| − | The LCD controller uses RS and RW lines along with E to operate the LCD.
 |  | 
| − | *'''Resister Select (RS)''': Determines weather a command(RS = 0) is sent (to set up the display) or actual data(RS=1) is sent. 
 |  | 
| − | *'''Read/Write''' RW=0; writes to the LCD. RW=1;Reads from the LCD.
 |  | 
| − | The commonly used instructions are shown in the instruction set below.
 |  | 
| − | Observe the '''Bit names: I/D, S, D, C''' etc at the bottom of instruction set to decode the instructions completely.
 |  | 
| − | *Clear Display
 |  | 
| − | *Cursor Home
 |  | 
| − | *Set Entry Mode
 |  | 
| − | *Display on/off control 
 |  | 
| − | *Cursor/display shift 
 |  | 
| − | *Function Set
 |  | 
| − | *Read Busy Flag
 |  | 
| − | *Data Read
 |  | 
| − | *Data Write<br />
 |  | 
| − | 
 |  | 
| − | }}
 |  | 
| − | ==Instruction Set==
 |  | 
| − | {|class="wikitable" border="1" cellpadding="10" cellspacing="0" align="center"
 |  | 
| − | 
 |  | 
| − | |+ '''HD44780U based instruction set'''
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef;" rowspan="2" | Instruction
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef;" colspan="10" | Code
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef;" rowspan="2" | Description
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px;" | RS
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 0px;" | R/W
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 1px dashed;" | B7
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 0px;" | B6
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 0px;" | B5
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 0px;" | B4
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 0px;" | B3
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 0px;" | B2
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-right: 0px; border-left: 0px;" | B1
 |  | 
| − | 
 |  | 
| − | ! style="background: #efefef; border-left: 0px;" | B0
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | | Clear display
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 1px dashed;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-left: 0px;"| 1
 |  | 
| − | 
 |  | 
| − | || Clears display and returns cursor to the home position (address 0).
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | || Cursor home
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 1px dashed;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 1
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-left: 0px;"| *
 |  | 
| − | 
 |  | 
| − | || Returns cursor to home position. Also returns display being shifted to the original position. DDRAM content remains unchanged.
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | || Entry mode set
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 1px dashed;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 1
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| I/D
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-left: 0px;"| S
 |  | 
| − | 
 |  | 
| − | || Sets cursor move direction (I/D); specifies to shift the display (S). These operations are performed during data read/write.
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | || Display on/off control
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 1px dashed;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 1
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| D
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| C
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-left: 0px;"| B
 |  | 
| − | 
 |  | 
| − | || Sets on/off of all display (D), cursor on/off (C), and blink of cursor position character (B).
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | || Cursor/display shift
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 1px dashed;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 1
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| S/C
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| R/L
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| *
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-left: 0px;"| *
 |  | 
| − | 
 |  | 
| − | || Sets cursor-move or display-shift (S/C), shift direction (R/L). DDRAM content remains unchanged.
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | || Function set
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 1px dashed;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 1
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| DL
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| N
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| F
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| *
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-left: 0px;"| *
 |  | 
| − | 
 |  | 
| − | || Sets interface data length (DL), number of display line (N), and character font (F).
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | 
 |  | 
| − | || Read busy flag &<br> address counter
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px;"| 0
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 1
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 1px dashed;"| BF
 |  | 
| − | 
 |  | 
| − | |align="center" style="border-left: 0px;" colspan="7"| CGRAM/DDRAM address
 |  | 
| − | 
 |  | 
| − | || Reads busy flag (BF) indicating internal operation being performed and reads CGRAM or DDRAM address counter contents (depending on previous instruction).
 |  | 
| − | 
 |  | 
| − | |-
 |  | 
| − | || Write CGRAM or<br> DDRAM
 |  | 
| − | |align="center" style="border-right: 0px;"| 1
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | |align="center" style="border-left: 1px dashed;" colspan="8"| Write Data
 |  | 
| − | || Write data to CGRAM or DDRAM.
 |  | 
| − | |-
 |  | 
| − | || Write CGRAM or<br> DDRAM
 |  | 
| − | |align="center" style="border-right: 0px;"| 1
 |  | 
| − | |align="center" style="border-right: 0px; border-left: 0px;"| 0
 |  | 
| − | |align="center" style="border-left: 1px dashed;" colspan="8"| Write Data
 |  | 
| − | || Write data to CGRAM or DDRAM.
 |  | 
| − | |-
 |  | 
| − | |colspan="13" style="font-size: 90%;"| '''Instruction bit names —'''<br>
 |  | 
| − | '''I/D''' - 0 = decrement cursor position, 1 = increment cursor position;<br/>
 |  | 
| − | 
 |  | 
| − | '''S''' - 0 = no display shift, 1 = display shift;<br/>
 |  | 
| − | '''D''' - 0 = display off, 1 = display on;<br/>
 |  | 
| − | '''C''' - 0 = cursor off, 1 = cursor on;<br/>
 |  | 
| − | '''B''' - 0 = cursor blink off, 1 = cursor blink on ;<br/>
 |  | 
| − | '''S/C''' - 0 = move cursor, 1 = shift display;<br/>
 |  | 
| − | '''R/L''' - 0 = shift left, 1 = shift right;<br/>
 |  | 
| − | '''DL''' - 0 = 4-bit interface, 1 = 8-bit interface;<br/>
 |  | 
| − | '''N''' - 0 = 1/8 or 1/11 duty (1 line), 1 = 1/16 duty (2 lines);<br/>
 |  | 
| − | '''F''' - 0 = 5×8 dots, 1 = 5×10 dots;<br/>
 |  | 
| − | '''BF''' - 0 = can accept instruction, 1 = internal operation in progress.<br/>
 |  | 
| − | |}
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | =Interfacing LCD with 8051=
 |  | 
| − | {{Box|type=l_green_light|text=
 |  | 
| − | LCD can be interfaced with the 8051 micrcontroller in two modes, 8 bit and 4 bit.
 |  | 
| − | Let us Interface it in 8 bit mode first. 
 |  | 
| − | }}
 |  | 
| − | ==8 bit Mode==
 |  | 
| − | {{Box|type=l_green_light|text=<br />
 |  | 
| − | ===Objective===
 |  | 
| − | [[File:LCD_output.PNG|thumbnail|fig LCD display ]]
 |  | 
| − | There is lot of stuff that can be done with the LCDs, to start with we will simple display a couple of strings on the 2 lines of the LCD as shown in the image.
 |  | 
| − | ===Schematic Discription===
 |  | 
| − | *'''Data  Lines:''' In this mode, all of the 8 datalines DB0 to DB7 are connected from the micrcontroller to a LCD module as shown the schematic.
 |  | 
| − | *'''Control Lines:'''' The RS, RW and E are control lines, as discussed earlier.
 |  | 
| − | *'''Power & contrast''':Apart from that the LCD should be powered with 5V between '''PIN 2(VCC)''' and  '''PIN 1(gnd)'''. '''PIN 3''' is the contrast pin and is output of center terminal of potentiometer(voltage divider) which varies voltage between 0 to 5v to vary the contrast.
 |  | 
| − | *'''Back-light:''' The PIN 15 and 16 are used as backlight. The led backlight can be powered through a simple current limiting resistor as we do with normal leds. 
 |  | 
| − | }}
 |  | 
| − | 
 |  | 
| − | ===Schematic===
 |  | 
| − | [[File:8051 LCD8BIT Interface.PNG|Fig 3: Schematic LCD 8 bit mode]]
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | {{Box|type=l_green_light|text=<br />
 |  | 
| − | ===Code===
 |  | 
| − | As with all the interfaces to simplify thing we have separated code into two files, main.c  and  lcd_8_bit.c. You may go through the [[8051 ToolsSetup|tools setup]] tutorial on configuring the code.
 |  | 
| − | 
 |  | 
| − | The main.c is very simple it includes the standard library files.
 |  | 
| − | Then it uses several functions from the lcd_8_bit.c file to set up and display messages. As you can see, it makes things very simple when the libraries are well written. We will discuss the implementation while discussing the lcd_8_bit.c file.
 |  | 
| − | }}
 |  | 
| − | ====The main file main.c====
 |  | 
| − | <syntaxhighlight>
 |  | 
| − | /* Reg51.h contains the defnition of all ports and SFRs */
 |  | 
| − | #include <reg51.h> 
 |  | 
| − | 
 |  | 
| − | #include "lcd.h"	//Xplore labz LCD library 
 |  | 
| − | #include "delay.h" //Xplore Labz Delay library
 |  | 
| − | 
 |  | 
| − | /* start the main program */
 |  | 
| − | void main() 
 |  | 
| − | {
 |  | 
| − | 
 |  | 
| − |   /* Initilize the lcd before displaying any thing on the lcd */
 |  | 
| − |     LCD_Init();
 |  | 
| − | 
 |  | 
| − |   /* Display "hello, world" on first line*/
 |  | 
| − |   LCD_DisplayString("hello, world");
 |  | 
| − | 
 |  | 
| − |   /*Go to second line and display "good morning" */
 |  | 
| − |   LCD_GoToLineTwo();
 |  | 
| − |   LCD_DisplayString("good morning");
 |  | 
| − | 
 |  | 
| − |     while(1);
 |  | 
| − | }
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | 
 |  | 
| − | =Timing Diagram=
 |  | 
| − | [[File:LCD CmdWrite.jpg|figure: command write]]<br />
 |  | 
| − | The above image shows the timing diagram for sending the data to the LCD.
 |  | 
| − | Below are the steps for sending the command
 |  | 
| − | *step1: Send the I/P command to LCD.
 |  | 
| − | *step2: Select the Control Register by making RS low.
 |  | 
| − | *step3: Select Write operation making RW low.
 |  | 
| − | *step4: Send a High-to-Low pulse on Enable PIN with some delay_us.
 |  | 
| − | 
 |  | 
| − | Code snippet:
 |  | 
| − | <syntaxhighlight>
 |  | 
| − | /* Function to send the command to LCD. As it is 4bit mode, a byte of data is sent in two 4-bit nibbles */
 |  | 
| − | void Lcd_CmdWrite(char cmd)
 |  | 
| − | {
 |  | 
| − |     sendNibble((cmd >> 0x04) & 0x0F);   //Send higher nibble
 |  | 
| − |     LcdControlBusPort &= ~(1<<LCD_RS); // Send LOW pulse on RS pin for selecting Command register
 |  | 
| − |     LcdControlBusPort &= ~(1<<LCD_RW); // Send LOW pulse on RW pin for Write operation
 |  | 
| − |     LcdControlBusPort |= (1<<LCD_EN);  // Generate a High-to-low pulse on EN pin
 |  | 
| − |     delay(1000);
 |  | 
| − |     LcdControlBusPort &= ~(1<<LCD_EN);
 |  | 
| − | 
 |  | 
| − |     delay(10000);
 |  | 
| − | 
 |  | 
| − |     sendNibble(cmd & 0x0F);            //Send Lower nibble
 |  | 
| − |     LcdControlBusPort &= ~(1<<LCD_RS); // Send LOW pulse on RS pin for selecting Command register
 |  | 
| − |     LcdControlBusPort &= ~(1<<LCD_RW); // Send LOW pulse on RW pin for Write operation
 |  | 
| − |     LcdControlBusPort |= (1<<LCD_EN);  // Generate a High-to-low pulse on EN pin
 |  | 
| − |     delay(1000);
 |  | 
| − |     LcdControlBusPort &= ~(1<<LCD_EN); 
 |  | 
| − | 
 |  | 
| − |     delay(10000);
 |  | 
| − | }
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | '''2. void LCD_DataWrite( char dat)'''
 |  | 
| − | This function sends a character to be displayed on LCD in the following steps.
 |  | 
| − | *step1: Send the character to LCD.
 |  | 
| − | *step2: Select the Data Register by making RS high.
 |  | 
| − | *step3: Select Write operation making RW low.
 |  | 
| − | *step4: Send a High-to-Low pulse on Enable PIN with some delay_us.
 |  | 
| − | The timings are similar as above only change is that '''RS''' is made high.
 |  | 
| − | <syntaxhighlight>
 |  | 
| − | void LCD_DataWrite( char dat)
 |  | 
| − | {
 |  | 
| − |    databus=dat;	   // Send the data to LCD
 |  | 
| − |      rs=1;	   // Select the Data Register by pulling RS HIGH
 |  | 
| − |      rw=0;         // Select the Write Operation by pulling RW LOW
 |  | 
| − |      en=1;	   // Send a High-to-Low Pusle at Enable Pin
 |  | 
| − |      delay_us(10);
 |  | 
| − |      en=0;
 |  | 
| − |    delay_ms(1);
 |  | 
| − | }
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | 
 |  | 
| − | '''3. void LCD_Init()'''<br />
 |  | 
| − | Looking at the instruction set of the LCD controller, we can initialize the LCD with following steps
 |  | 
| − | * Set the display mode as 2 lines, 5 x 7 matrix
 |  | 
| − | * Turn On the dislay, and cursor.
 |  | 
| − | * Clear the LCD
 |  | 
| − | * Get the cursor to first line first position.
 |  | 
| − | The code is listed below.
 |  | 
| − | 
 |  | 
| − | <syntaxhighlight>
 |  | 
| − | void LCD_Init()
 |  | 
| − | {
 |  | 
| − |    delay_us(5000);
 |  | 
| − |    LCD_CmdWrite(0x38);   // LCD 2lines, 5*7 matrix
 |  | 
| − |    LCD_CmdWrite(0x0E);	// Display ON cursor ON  Blinking off
 |  | 
| − |    LCD_CmdWrite(0x01);	// Clear the LCD
 |  | 
| − |    LCD_CmdWrite(0x80);	// Cursor to First line First Position
 |  | 
| − | }
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | The remaining code is listed below. You could observe that the basic functions to read data and write write command are used extensively and entire library is built upon them. There are also other functions that are used to display number etc in the lcd library file. The entire listing can be found [[8051_Family_C_Library#LCD_16x2_8_bit_mode|here]]. 
 |  | 
| − | }}
 |  | 
| − | 
 |  | 
| − | {{scroll box|width=700px|height=500px|text=<br />
 |  | 
| − | <syntaxhighlight>
 |  | 
| − | #include<reg51.h>
 |  | 
| − | #include "delay.h"
 |  | 
| − | #include "lcd.h"
 |  | 
| − | 
 |  | 
| − | #define databus	P2   //	LCD databus connected to PORT2
 |  | 
| − | 
 |  | 
| − | sbit rs= P0^0;		 // Register select pin connected to P0.0
 |  | 
| − | sbit rw= P0^1;		 // Read Write pin connected to P0.1
 |  | 
| − | sbit en= P0^2;		 // Enable pin connected to P0.2
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | /* 16x2 LCD Specification */
 |  | 
| − | #define LCDMaxLines 2
 |  | 
| − | #define LCDMaxChars 16
 |  | 
| − | #define LineOne 0x80
 |  | 
| − | #define LineTwo 0xc0
 |  | 
| − | 
 |  | 
| − | #define BlankSpace ' '
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | void LCD_Init()
 |  | 
| − | {
 |  | 
| − |     delay_us(5000);
 |  | 
| − |    LCD_CmdWrite(0x38);   // LCD 2lines, 5*7 matrix
 |  | 
| − |    LCD_CmdWrite(0x0E);	// Display ON cursor ON  Blinking off
 |  | 
| − |    LCD_CmdWrite(0x01);	// Clear the LCD
 |  | 
| − |    LCD_CmdWrite(0x80);	// Cursor to First line First Position
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | void LCD_CmdWrite( char cmd)
 |  | 
| − | {
 |  | 
| − |      databus=cmd;        // Send the command to LCD
 |  | 
| − |      rs=0;             // Select Command Register by pulling RS LOW
 |  | 
| − |      rw=0;             // Select Write Operation  by pulling RW LOW
 |  | 
| − |      en=1;             // Send a High-to-Low Pusle at Enable Pin
 |  | 
| − |      delay_us(10);
 |  | 
| − |      en=0;
 |  | 
| − |      delay_ms(1);
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | void LCD_DataWrite( char dat)
 |  | 
| − | {
 |  | 
| − | 
 |  | 
| − |    databus=dat;	   // Send the data to LCD
 |  | 
| − |      rs=1;     // Select the Data Register by pulling RS HIGH
 |  | 
| − |      rw=0;   // Select the Write Operation  by pulling RW LOW
 |  | 
| − |      en=1;	// Send a High-to-Low Pusle at Enable Pin
 |  | 
| − |      delay_us(10);
 |  | 
| − |      en=0;
 |  | 
| − |     delay_ms(1);
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | void LCD_GoToXY(char row, char col)
 |  | 
| − | {
 |  | 
| − |    char pos;
 |  | 
| − | 
 |  | 
| − |     if(row<LCDMaxLines)
 |  | 
| − |       {
 |  | 
| − | 	 pos= LineOne | (row << 6); // take the line number
 |  | 
| − | 		                   //row0->pos=0x80  row1->pos=0xc0
 |  | 
| − | 
 |  | 
| − | 	 if(col<LCDMaxChars)
 |  | 
| − |             pos= pos+col;      //take the char number
 |  | 
| − | 		               //now pos points to  XY pos
 |  | 
| − | 
 |  | 
| − | 	  LCD_CmdWrite(pos) 
 |  | 
| − |        }
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | void LCD_DisplayString(char *string_ptr)
 |  | 
| − | {
 |  | 
| − |    while(*string_ptr)
 |  | 
| − |     LCD_DataWrite(*string_ptr++);
 |  | 
| − | }
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | }}
 |  | 
| − | 
 |  | 
| − | ==4 bit Mode==
 |  | 
| − | {{Box|type=l_green_light|text=<br />
 |  | 
| − | ===Schematic===
 |  | 
| − | There are following differences in 4 bit mode.
 |  | 
| − | * Only data lines D4 to D7 are used as shown in the schematic below.
 |  | 
| − | * In code, we need to send the command to select 4 bit mode as shown in the instruction set above.
 |  | 
| − | [[File:8051 LCD4BIT Interface.PNG|Fig 4: Schematic LCD 4 bit mode]]
 |  | 
| − | 
 |  | 
| − | The main program remains exactly as in 8 bit mode, we simply include the '''lcd_4_bit.c''' to work in 4 bit mode.
 |  | 
| − | The important and required code is listed below, you find other useful functions and entire library 
 |  | 
| − | [[8051_Family_C_Library#LCD_4_bit_mod|here]]
 |  | 
| − | }}
 |  | 
| − | ===Code===
 |  | 
| − | {{scroll box|width=700px|height=500px|text=<br />
 |  | 
| − | <syntaxhighlight>
 |  | 
| − | #include<reg51.h>
 |  | 
| − | #include "delay.h"
 |  | 
| − | #include "lcd.h"
 |  | 
| − | 
 |  | 
| − | #define databus	P0   //LCD databus connected to PORT0
 |  | 
| − | 
 |  | 
| − | sbit rs= databus^0;	 // Register select pin connected to P0.0
 |  | 
| − | sbit rw= databus^1;	 // Read Write pin connected to P0.1
 |  | 
| − | sbit en= databus^2;	 // Enable pin connected to P0.2
 |  | 
| − | 
 |  | 
| − | /* 16x2 LCD Specification */
 |  | 
| − | #define LCDMaxLines 2
 |  | 
| − | #define LCDMaxChars 16
 |  | 
| − | #define LineOne 0x80
 |  | 
| − | #define LineTwo 0xc0
 |  | 
| − | 
 |  | 
| − | #define BlankSpace ' '
 |  | 
| − | void LCD_Init()
 |  | 
| − | {
 |  | 
| − |     delay_us(5000);
 |  | 
| − |    LCD_CmdWrite(0x02);	//Initilize the LCD in 4bit Mode
 |  | 
| − |    LCD_CmdWrite(0x28);
 |  | 
| − |    LCD_CmdWrite(0x0E);	// Display ON cursor ON
 |  | 
| − |    LCD_CmdWrite(0x01);	// Clear the LCD
 |  | 
| − |    LCD_CmdWrite(0x80);	// Cursor to First line First Position
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | void LCD_CmdWrite( char cmd)
 |  | 
| − | {
 |  | 
| − | 
 |  | 
| − |    databus=(cmd & 0xf0); // Send the Higher Nibble 
 |  | 
| − |      rs=0;	// Select the Command Register by pulling RS LOW
 |  | 
| − |      rw=0;	// Select the Write Operation  by pulling RW LOW
 |  | 
| − |      en=1;	// Send a High-to-Low Pusle at Enable Pin
 |  | 
| − |      delay_us(1);
 |  | 
| − |      en=0;
 |  | 
| − | 
 |  | 
| − |    delay_us(10);	// wait for some time
 |  | 
| − | 
 |  | 
| − |    databus=((cmd<<4) & 0xf0);   // Send the Lower Nibble 
 |  | 
| − |      rs=0;	// Select the Command Register by pulling RS LOW
 |  | 
| − |      rw=0;	// Select the Write Operation  by pulling RW LOW
 |  | 
| − |      en=1;  // Send a High-to-Low Pusle at Enable Pin
 |  | 
| − |      delay_us(1);
 |  | 
| − |      en=0;
 |  | 
| − | 
 |  | 
| − |    delay_ms(1);
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | void LCD_DataWrite( char dat)
 |  | 
| − | {
 |  | 
| − |    databus=(dat & 0xf0); // Send the Higher Nibble
 |  | 
| − |      rs=1;	// Select the Data Register by pulling RS HIGH
 |  | 
| − |      rw=0;	 // Select the Write Operation  by pulling RW LOW
 |  | 
| − |      en=1;	 // Send a High-to-Low Pusle at Enable Pin
 |  | 
| − |      delay_us(1);
 |  | 
| − |      en=0;
 |  | 
| − | 
 |  | 
| − |    delay_us(10); // wait for some time.
 |  | 
| − | 
 |  | 
| − |    databus=((dat <<4) & 0xf0);	// Send the Lower Nibble  
 |  | 
| − |      rs=1;	// Select the Data Register by pulling RS HIGH
 |  | 
| − |      rw=0;	// Select the Write Operation  by pulling RW LOW
 |  | 
| − |      en=1;	// Send a High-to-Low Pusle at Enable Pin
 |  | 
| − | 	delay_us(1);
 |  | 
| − |      en=0;
 |  | 
| − | 
 |  | 
| − |    delay_ms(1);
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | 
 |  | 
| − | void LCD_DisplayString(char *string_ptr)
 |  | 
| − | {
 |  | 
| − |    while(*string_ptr)
 |  | 
| − |     LCD_DataWrite(*string_ptr++);
 |  | 
| − | }
 |  | 
| − | 
 |  | 
| − | void LCD_GoToLineTwo()
 |  | 
| − | {
 |  | 
| − |  LCD_CmdWrite(LineTwo);	// Move the Cursor to Second line 
 |  | 
| − | }
 |  | 
| − | </syntaxhighlight>
 |  | 
| − | }}
 |  | 
|  | = Downloads= |  | = Downloads= | 
| − | {{Box|type=l_green_light|text="The code is given for 4 and 8 bit modes with and without busy Flag"}}
 | + | Download the complete project folder from the below link:  | 
| − | [http://exploreembedded.com/wiki/images/0/02/8051_LCD_Tutorial_Code.zip '''Code''']
 | + | https://codeload.github.com/ExploreEmbedded/Explore-Cortex-M3-LPC1768-Stick-DVB-14001/zip/master<br> | 
| − |   | + |  | 
| − |   | + |  | 
| − | [[File:SwitchesLEDs.jpg|left|thumbnail|x120px|{{Box|type=l_blue_light|text=Switches and LED Interfacing with 8051 }}|link=8051_Interfacing_swithes_leds|'''''PREVIOUS TUTORIAL''''']]
 | + |  | 
| − | [[File:8051_Timers_Poster.jpg|right|thumbnail|x150px|link=8051_timers|'''''NEXT TUTORIAL''''']]
 | + |  | 
|  |  |  |  | 
| − | {{#widget:Facebook_Like_Box|profile=https://www.facebook.com/ExploreEmbedded}}
 |  | 
|  |  |  |  | 
|  | Have a opinion, suggestion , question or feedback about the article let it out here! |  | Have a opinion, suggestion , question or feedback about the article let it out here! | 
|  | {{DISQUS}} |  | {{DISQUS}} | 
In this tutorial we are going to see how to interface a 2x16 LCD with LPC1768 in 4-bit mode.
As per the name the 2x16 has 2 lines with 16 chars on each lines. It supports all the ascii chars and is basically used for displaying the alpha numeric characters. Here each character is displayed in a matrix of 5x7 pixels.
Apart from alpha numeric chars it also provides the provision to display the custom characters by creating the pattern.
Scope of this tutorial is to show how to display the alpha numeric chars on LCD, Generating and displaying the custom chars will be discussed in subsequent tutorials.
Apart from the voltage supply connections the important pins from the programming perspective are the data lines(8-bit Data bus), Register select, Read/Write and Enable pin.
  
Data Bus: As shown in the above figure and table, an alpha numeric lcd has a 8-bit data bus referenced as D0-D7.
As it is a 8-bit data bus, we can send the data/cmd to LCD in bytes. It also provides the provision to send the the data/cmd in chunks of 4-bit, which is used when there are limited number of GPIO lines on the microcontroller.
   
Register Select(RS): The LCD has two register namely a Data register and Command register. Any data that needs to be displayed on the LCD has to be written to the data register of LCD. Command can be issued to LCD by writing it to Command register of LCD.
This signal is used to differentiate the data/cmd received by the LCD.
 
If the RS signal is LOW then the LCD interprets the 8-bit info as Command and writes it Command register and performs the action as per the command.
  
If the RS signal is HIGH then the LCD interprets the 8-bit info as data and copies it to data register. After that the LCD decodes the data for generating the 5x7 pattern and finally displays on the LCD.
  
Read/Write(RW): This signal is used to write the data/cmd to LCD and reads the  busy flag of LCD.
For write operation the RW should be LOW and for read operation the R/W should be HIGH.
  
Enable(EN): This pin is used to send the enable trigger to LCD.
After sending the data/cmd, Selecting the data/cmd register, Selecting the Write operation. A HIGH-to-LOW pulse has to be send on this enable pin which will latch the info into the LCD register and triggers the LCD to act accordingly.
Below schematic shows the minimum connection required for interfacing the LCD with the microcontroller.
As we are interfacing the LCD in 4-bit mode, only the higher 4 data lines are used as data bus.
This section shows how to configure the GPIO for interfacing the LCD.
The below configuration is as per the above schematic. You can connect the LCD to any of the PORT pins available on your boards and update this section accordingly
In this section we are going to see how to send the data/cmd to the LCD along with the timing diagrams.
First lets see the timing diagram for sending the data and the command signals(RS,RW,EN), accordingly we write the algorithm and finally the code.
The below image shows the timing diagram for sending the data to the LCD.
As shown in the timing diagram  the data is written after sending the RS and RW signals. It is still ok to send the data before these signals.
The only important thing is the data should be available on the databus before generating the High-to-Low pulse on EN pin.
 
In the above tutorial we just discussed how to interface 2x16Lcd in 4-bit mode.
Once you know the working of lcd, you can directly use the ExploreEmbedded libraries to play around with your LCD.
For that you need to include the lcd.c/lcd.h and the associated files(delay/stdutils).
 
After including these files, the only thing you got to do is to configure the PORTs in lcd.h as per your hardware connection.
The below sample code shows how to use the already available LCD functions.
Refer this link for more info on LCD libraries.