用串口的方法来驱动1602液晶的的长处很明显,便是串行能节省单片机有限的io口,本程序仅用四线就完成了液晶的驱动
演示程序很简单,不必多注释应该都能看懂。作为演示用处,其间有些长期延时没有没有运用守时器,在多任务体系中当然要用守时中止来替代了。
这便是电路,仔细的朋友会发现实物图中有几个贴片的阻容件,隐秘就在这儿,使用电容的回忆效应,把并行的数据转为串行。
关于本制造的更多图解请翻开:http://www.51hei.com/bbs/dpj-20365-1.html
// Drive a LCD1602 with 2 wire
//===================================================
//ICC-AVR application builder : 2010-10-3 19:30:02
// Target : M16
// Crystal: 4.0000Mhz
#include
#include
#define Set_E PORTB|=2
#define Clr_E PORTB&=~2
#define Set_D PORTB|=1
#define Clr_D PORTB&=~1
#define Set_xy(y,x) Send(0,(y<<6)|(x&15)|0x80)
//===================================================
void init_devices(void)
{
CLI(); //disable all interrupts
DDRB = 0x03;
MCUCR = 0x00;
G%&&&&&%R = 0x00;
SEI(); //re-enable interrupts
}
//===================================================
void Delay(unsigned int i)
{
while(i–);
}
//===================================================
void Send(unsigned char RS, unsigned char dat)
{
unsigned char i;
for (i = 2; i > 0; i–)
{
if (dat & 0x80) Set_D; else Clr_D;
Delay(10608);//14520us
if (RS) Set_E;
if (dat & 0x40) Set_D; else Clr_D;
Delay(462); //660us
if (dat & 0x20) Set_D; else Clr_D;
Delay(18); //30us
Set_E;
if (dat & 0x10) Set_D; else Clr_D;
_NOP(); //0.5us < t < 1.36us
Clr_E;
dat <<= 4;
}
}
//===================================================
void init_1602(void)
{
unsigned char i = 3;
Clr_D;
Clr_E;
Delay(10608);
do{
Clr_D;
Delay(462);
Set_D;
Set_E;
Delay(18);
if (i == 0) Clr_D;
_NOP();_NOP();_NOP();
Clr_E;
}while(i–);
Send(0,0×28);
Send(0,0×01);
Send(0,0x0f);
}
//===================================================
void Send_S(unsigned char *p)
{
while(*p) Send(1,*p++);
}
//===================================================
void main(void)
{
unsigned char i;
init_devices();
init_1602();
Set_xy(0,2);
Send_S(“Hello world!”);
Set_xy(1,3);
Send_S(“Im COWBOY.”);
for (i=0;i<255;i++) Delay(10000);
Send(0,0×01);
Set_xy(0,3);
Send_S(“Welcome to”);
Set_xy(1,1);
Send_S(www.51hei.com);
while(1);
}