728x90

728x90

Thursday, February 5, 2026

ATMega644P TWI MCP23017 Character LCD

The HD44780 based character LCD can be controlled by a micro-controller, digital circuit, an I/O expansion chip or event by hands. In this example I use an output port of the MCP23017 to control a 16x2 LCD in one direction (write only) in 4-bit data transfer mode.

ATMega644P TWI MCP23017 Character LCD

The LCD connect to GPIOB (output register OLATB). That's,

  • LCD RS (GPB0)
  • LCD R/W (GPB1)
  • LCD EN(GPB2)
  • D4:D7(GPB4:GPB7)

Source Code: "main.c"

  1. /*
  2. * 10-i2c_mcp23017_1602.c
  3. *
  4. * Created: 2/5/2026 8:03:45 PM
  5. * Author : Admin
  6. */

  7. #include<stdio.h>
  8. #include <avr/io.h>
  9. #include <util/delay.h>
  10. #define F_CPU 16000000UL

  11. //IOCON.BANK=0
  12. enum BANK0{
  13. IODIRA=0,IODIRB,IPOLA,IPOLB,GPINTENA,GPINTENB,DEFVALA,
  14. DEFVALB,INTCONA,INTCONB,IOCON1,IOCON2,GPPUA,GPPUB,
  15. INTFA,INTFB,INTCAPA,INTCAPB,GPIOA,GPIOB,OLATA,OLATB
  16. };

  17. const char RS=0,RW=1,EN=2;
  18. char BLK_ON=0;

  19. void lcd_command(uint8_t temp){
  20. uint8_t command,led;
  21. command=temp&0xF0;
  22. if(BLK_ON==1) led=0x08;
  23. else led=0;
  24. mcp23017_write(OLATB,command|led|(1<<EN));
  25. _delay_us(10);
  26. mcp23017_write(OLATB,command|led);
  27. _delay_us(25);
  28. command=temp<<4;
  29. mcp23017_write(OLATB,command|led|(1<<EN));
  30. _delay_us(10);
  31. mcp23017_write(OLATB,command|led);
  32. _delay_us(25);
  33. }

  34. void lcd_data(uint8_t temp){
  35. uint8_t data,led;
  36. data=temp&0xF0;
  37. if(BLK_ON==1) led=0x08;
  38. else led=0;
  39. mcp23017_write(OLATB,data|led|(1<<EN)|(1<<RS));
  40. _delay_us(10);
  41. mcp23017_write(OLATB,data|led|(1<<RS));
  42. _delay_us(25);
  43. data=temp<<4;
  44. mcp23017_write(OLATB,data|led|(1<<EN)|(1<<RS));
  45. _delay_us(10);
  46. mcp23017_write(OLATB,data|led|(1<<RS));
  47. _delay_us(25);
  48. }

  49. void lcd_xy(uint8_t x, uint8_t y){
  50. uint8_t cursor[]={0x80,0xC0};
  51. lcd_command(cursor[y-1]+x-1);
  52. }

  53. void lcd_text(uint8_t *text){
  54. while(*text) lcd_data(*text++);
  55. }

  56. void lcd_clear(void){
  57. lcd_command(0x01);
  58. _delay_ms(5);
  59. }

  60. void lcd_init(void){
  61. BLK_ON=1;
  62. mcp23017_write(IODIRB,0x00);
  63. lcd_command(0x33);
  64. _delay_us(10);
  65. lcd_command(0x32);
  66. _delay_us(10);
  67. lcd_command(0x28);
  68. _delay_us(10);
  69. lcd_command(0x0F);
  70. _delay_us(10);
  71. lcd_command(0x01);
  72. _delay_ms(5);
  73. lcd_command(0x06);
  74. _delay_us(10);
  75. }

  76. int main(void)
  77. {
  78. /* Replace with your application code */
  79. lcd_init();
  80. lcd_text("MCP23017 TWI");
  81. lcd_xy(1,2);
  82. lcd_text("MCP23017 LCD");

  83. _delay_ms(10000);
  84. lcd_clear();
  85. lcd_command(0x0C);
  86. lcd_text("Counter Value:");
  87. long counter=0;
  88. char msg[10];
  89. while (1)
  90. {
  91. sprintf(msg,"%ld",counter);
  92. lcd_xy(1,2); lcd_text(msg);
  93. counter++;
  94. _delay_ms(500);
  95. }
  96. }

Source Code: "mcp23017.c"

  1. /*
  2. * mcp23017.c
  3. *
  4. * Created: 2/5/2026 8:06:38 PM
  5. * Author: Admin
  6. */

  7. const char MCP23017_W=0x40;
  8. const char MCP23017_R=0x41;

  9. //IOCON.BANK=0
  10. /*
  11. enum BANK0{
  12. IODIRA=0,IODIRB,IPOLA,IPOLB,GPINTENA,GPINTENB,DEFVALA,
  13. DEFVALB,INTCONA,INTCONB,IOCON1,IOCON2,GPPUA,GPPUB,
  14. INTFA,INTFB,INTCAPA,INTCAPB,GPIOA,GPIOB,OLATA,OLATB
  15. };
  16. */


  17. void mcp23017_write(char address,char data){
  18. twi_start();
  19. /*Select the write address*/
  20. twi_write(MCP23017_W);
  21. /*Select a register address*/
  22. twi_write(address);
  23. /*Send configuration data*/
  24. twi_write(data);
  25. twi_stop();
  26. }

  27. unsigned char mcp23017_read(char address){
  28. /*Select a specific address*/
  29. twi_start();
  30. twi_write(MCP23017_W);
  31. twi_write(address);
  32. twi_stop();
  33. /*Read data from the given address*/
  34. twi_start();
  35. twi_write(MCP23017_R);
  36. unsigned char i2cData=twi_read(1);
  37. twi_stop();
  38. return i2cData;
  39. }

Source Code: "twi.c" 

  1. /*
  2. * twi.c
  3. *
  4. * Created: 2/4/2026 10:22:02 AM
  5. * Author: Admin
  6. */

  7. #include <avr/io.h>
  8. #include <util/delay.h>
  9. #define F_CPU 16000000UL


  10. void twi_init(void){
  11. TWSR|=0x00; //Prescaler Selection Bit
  12. TWBR=0x0F; //Baud Rate Generator
  13. TWCR=(1<<TWEN); //Enable The TWI Module
  14. PORTC|=(1<<0);
  15. PORTC|=(1<<1);
  16. }

  17. void twi_start(void){
  18. TWCR=(1<<TWINT)|(1<<TWSTA)|(1<<TWEN);
  19. while((TWCR&(1<<TWINT))==0);
  20. }

  21. void twi_write(unsigned char data){
  22. TWDR=data;
  23. TWCR=(1<<TWINT)|(1<<TWEN);
  24. while((TWCR&(1<<TWINT))==0);
  25. }

  26. unsigned char twi_read(char ACK){
  27. if(ACK==0)
  28. TWCR=(1<<TWINT)|(1<<TWEN)|(1<<TWEA);
  29. else
  30. TWCR=(1<<TWINT)|(1<<TWEN);
  31. while((TWCR&(1<<TWINT))==0);
  32. return TWDR;
  33. }

  34. void twi_stop(){
  35. TWCR=(1<<TWINT)|(1<<TWEN)|(1<<TWSTO);
  36. _delay_us(10);
  37. }


Schematic:

ATMega644P TWI MCP23017 Character LCD

 

AVR Prototype Board: 

ATMega644P TWI MCP23017 Character LCD
AVR Experiment Board

For PIC micro-controller using PIC16F887 is a good choice.

This PCB was offered by PCBWay (pcbway.com).

I have been using PCBWay for many years now. PCBWay fabricate PCBs at low cost, fast processing time for only 24 hours, and fast delivery time using any carrier options. This double side 10cmx10cm can be fabricate at only 5USD for 5 to 10pcs by PCBWay. It's a standard PCB with silk screen and solder mask.

A DIY dsPIC30F2010 and dsPIC30F1010 Prototype Board with Programmer
10 PCBs for only 5USD
 

For different size of PCB we can instantly quote on PCBWay website using a zip PCB Gerber file without account.

A DIY dsPIC30F2010 and dsPIC30F1010 Prototype Board with Programmer
PCBWay Instant Quote

 

There is one port left, GPIOA. So I added a 4x4 matrix keypad at GPIOA. The result of keypad scanning will show on the LCD.

Source Code:

 

  1. /*
  2. * 10-i2c_mcp23017_1602_kb.c
  3. *
  4. * Created: 2/5/2026 10:32:40 PM
  5. * Author : Admin
  6. */

  7. #include<stdio.h>
  8. #include <avr/io.h>
  9. #include <util/delay.h>
  10. #define F_CPU 16000000UL

  11. //IOCON.BANK=0
  12. enum BANK0{
  13. IODIRA=0,IODIRB,IPOLA,IPOLB,GPINTENA,GPINTENB,DEFVALA,
  14. DEFVALB,INTCONA,INTCONB,IOCON1,IOCON2,GPPUA,GPPUB,
  15. INTFA,INTFB,INTCAPA,INTCAPB,GPIOA,GPIOB,OLATA,OLATB
  16. };

  17. const char RS=0,RW=1,EN=2;
  18. char BLK_ON=0;

  19. const char key_16[4][4]={'1','2','3','A',
  20. '4','5','6','B',
  21. '7','8','9','C',
  22. '*','0','#','D'};

  23. char keyScan(void){
  24. char data=0,temp,key;
  25. for(char i=0;i<4;i++){
  26. data=0xFF;
  27. data&=~(1<<i);
  28. mcp23017_write(OLATA,data);
  29. _delay_ms(5);
  30. data=mcp23017_read(GPIOA);
  31. data&=0xF0;
  32. if((data&0x10)==0) {temp=key_16[i][0]; break;}
  33. else if((data&0x20)==0){temp=key_16[i][1]; break;}
  34. else if((data&0x40)==0){temp=key_16[i][2]; break;}
  35. else if((data&0x80)==0){temp=key_16[i][3]; break;}
  36. else temp=0;
  37. _delay_ms(10);
  38. }
  39. return temp;
  40. }
  41. void lcd_command(uint8_t temp){
  42. uint8_t command,led;
  43. command=temp&0xF0;
  44. if(BLK_ON==1) led=0x08;
  45. else led=0;
  46. mcp23017_write(OLATB,command|led|(1<<EN));
  47. _delay_us(10);
  48. mcp23017_write(OLATB,command|led);
  49. _delay_us(25);
  50. command=temp<<4;
  51. mcp23017_write(OLATB,command|led|(1<<EN));
  52. _delay_us(10);
  53. mcp23017_write(OLATB,command|led);
  54. _delay_us(25);
  55. }

  56. void lcd_data(uint8_t temp){
  57. uint8_t data,led;
  58. data=temp&0xF0;
  59. if(BLK_ON==1) led=0x08;
  60. else led=0;
  61. mcp23017_write(OLATB,data|led|(1<<EN)|(1<<RS));
  62. _delay_us(10);
  63. mcp23017_write(OLATB,data|led|(1<<RS));
  64. _delay_us(25);
  65. data=temp<<4;
  66. mcp23017_write(OLATB,data|led|(1<<EN)|(1<<RS));
  67. _delay_us(10);
  68. mcp23017_write(OLATB,data|led|(1<<RS));
  69. _delay_us(25);
  70. }

  71. void lcd_xy(uint8_t x, uint8_t y){
  72. uint8_t cursor[]={0x80,0xC0};
  73. lcd_command(cursor[y-1]+x-1);
  74. }

  75. void lcd_text(uint8_t *text){
  76. while(*text) lcd_data(*text++);
  77. }

  78. void lcd_clear(void){
  79. lcd_command(0x01);
  80. _delay_ms(5);
  81. }

  82. void lcd_init(void){
  83. BLK_ON=1;
  84. mcp23017_write(IODIRB,0x00);
  85. lcd_command(0x33);
  86. _delay_us(10);
  87. lcd_command(0x32);
  88. _delay_us(10);
  89. lcd_command(0x28);
  90. _delay_us(10);
  91. lcd_command(0x0F);
  92. _delay_us(10);
  93. lcd_command(0x01);
  94. _delay_ms(5);
  95. lcd_command(0x06);
  96. _delay_us(10);
  97. }

  98. int main(void)
  99. {
  100. /* Replace with your application code */
  101. lcd_init();
  102. lcd_text("MCP23017 TWI");
  103. lcd_xy(1,2);
  104. lcd_text("LCD and Key Pad");

  105. _delay_ms(10000);
  106. lcd_clear();
  107. //Key Pad Init
  108. mcp23017_write(IODIRA,0xF0);
  109. mcp23017_write(GPPUA,0xF0);
  110. char temp,charCount=0,newLine=0,line=1;
  111. while (1)
  112. {
  113. temp=keyScan();
  114. if(temp!=0){
  115. lcd_data(temp);
  116. charCount++;
  117. _delay_ms(500);
  118. }
  119. if(charCount>16){
  120. newLine=1;
  121. charCount=0;
  122. line+=1;
  123. }
  124. if(newLine){
  125. newLine=0;
  126. if(line==2) lcd_xy(1,2);
  127. else{
  128. lcd_xy(1,1);
  129. lcd_command(0x01);
  130. _delay_ms(5);
  131. line=1;
  132. }
  133. }
  134. }
  135. }



Program Simulation:

ATMega644P TWI MCP23017 Character LCD
Program Simulation

AVR Prototype Board: 

ATMega644P TWI MCP23017 Character LCD 

ATMega644P TWI MCP23017 Character LCD

 

No comments:

Post a Comment

320x50

Search This Blog

tyro-728x90