/
程序名称:带汉字库的12864液晶显示模块驱动
程序功能:显示字符 、汉字和
开发工具:Kile
MCU型号:AT89S52-24PU
时钟频率:110592MHZ
程序作者:yuan
版权说明:yuan
/
#include<reg52h>
#include "lcdh"
#include "utilh"
sbit E=P1^5;//脉冲使能
sbit RW=P1^6;//读写选择
sbit RS=P1^7;//数据命令选择
sbit rst=P3^6;//12864复位
// 延时ms函数:
// 12864检查状态函数:
void Check12864State(void)
{
P0=0xff;
E=0;//读状态前三控制线的状态
RS=0;
RW=1;
E=1;//拉高,读状态
while((P0&0x80)==0x80);//等待空闲
E=0;//写命令后三控制线的状态
RS=1;
RW=0;
}
// 12864写命令函数:
void Write12864Command( unsigned char com)
{
Check12864State();//检查状态
P0=com;//赋值
E=0;//写命令前三控制线的状态
RS=0;
RW=0;
E=1;//拉高,写命令
E=0;//写命令后三控制线的状态
RS=1;
RW=1;
}
//12864写数据函数:
void Write12864Data( unsigned char dat)
{
Check12864State();//检查状态
P0=dat;//赋值
E=0;//写数据前三控制线的状态
RS=1;
RW=0;
E=1;//拉高,写数据
E=0;//写数据后三控制线的状态
RS=0;
RW=1;
}
//在指定的位置显示字符串(汉字和ASCII码字符)函数:
void LCD12864DisplayString( unsigned char y,unsigned char x, unsigned char pstr)
//y-行数值0-3,x-列数值0-7,pstr-字符串指针
//12864可以显示32个汉字(四行每行8个),一个地址对应一个汉字
//可以显示64个ASCII码字符(四行每行16个),一个地址对应两个字符
//为了实现自动换行功能,这个函数比较繁琐
{
unsigned char row,n=0;
Write12864Command(0x30);//基本指令
Write12864Command(0x06);//地址计数器自动加以,光标右移
switch(y)//根据行号选择行地址
{
case 0:row=0x80;break;//第一行首地址
case 1:row=0x90;break;//第二行首地址
case 2:row=0x88;break;//第三行首地址
case 3:row=0x98;break;//第四行首地址
default:;
}
Write12864Command(row+x);//写地址
while(pstr!='\0')
{
Write12864Data(pstr);//写字符
pstr++;
n++;//计数
if((n+x2)==16)//如果一行写完 ,继续写第二行
{
if(y==0) Write12864Command(0x90);//写下一行地址
else if(y==1) Write12864Command(0x88);//写下一行地址
else if(y==2) Write12864Command(0x98);//写下一行地址
else ;
}
else if((n+x2)==32)//如果第二行写完 ,继续写第三行
{
if(y==0) Write12864Command(0x88);//写下一行地址
else if(y==1) Write12864Command(0x98);//写下一行地址
else ;
}
else if((n+x2)==48)//如果第三行写完 ,继续写第四行
{
if(y==0) Write12864Command(0x98);//写下一行地址
else ;
}
else ;
}
}
//模式清屏函数:
void Clear12864Screen()
{
unsigned char i,j;
Write12864Command(0x34);//功能设定:8位控制方式,使用扩充指令
Write12864Command(0x36);//使用扩充指令,绘图显示控制
for(i=0;i<32;i++)
//ST7920可控制25632点阵(32行256列),而12864液晶实际的行地址只有0-31行,
//12864液晶的32-63行的行是0-31行地址从第128列划分一半出来的,所以分为上下两半屏,
//也就是说第0行和第32行同属一行,行地址相同;第1行和第33行同属一行,以此类推
{
Write12864Command(0x80|i);//写行地址(垂直地址)
Write12864Command(0x80);//写列地址(水平地址)
for(j=0;j<32;j++)
Write12864Data(0x00);//清屏
}
}
//在任意位置显示任意大小的函数:
void LCD12864DisplayPictrue(unsigned char y,unsigned char x,
unsigned char px,unsigned char py, unsigned char pp)
//y-起始行(数值0-63),x-起始列(16位宽,数值0-7),
//px-宽度,py-高度,pp-指针指向数组
//因为上下屏的地址不连续,要在任意位置显示完整的图像,处理起来比较繁琐
{
unsigned char i,j,k;
Clear12864Screen();//清屏
if(y<32)//如果起始行在上半屏
{
k=32-y;//算出上半屏的行数
for(i=0;i<k;i++,y++)//上半屏行数
{
Write12864Command(0x80|y);//写行地址(垂直地址)
Write12864Command(0x80|x);//写列地址(水平地址)
for(j=0;j<px/8;j++)
Write12864Data(pp[ipx/8+j]);//写数据
}
y=0;//下半屏起始行,接上半屏继续写数据
for(;i<py;i++,y++)//下半屏剩下的行数
{
Write12864Command(0x80|y);//写行地址(垂直地址)
Write12864Command(0x80|(8+x));//写列地址(水平地址)
for(j=0;j<px/8;j++)
Write12864Data(pp[ipx/8+j]);//写数据
}
}
else //如果起始行在下半屏
{
for(i=0;i<py;i++,y++)//行数
{
Write12864Command(0x80|(y-32));//写行地址(垂直地址)
Write12864Command(0x80|(8+x));//写列地址(水平地址)
for(j=0;j<px/8;j++)
Write12864Data(pp[ipx/8+j]);//写数据
}
}
}
void Clear12864Text()
{
Write12864Command(0x34);//清屏
DelayMs(5);
Write12864Command(0x30);//清屏
DelayMs(5);
Write12864Command(0x01);//清屏
DelayMs(5);
}
//12864初始化函数:
void Initialize12864()
{
rst=0;//复位12864
DelayMs(30);
rst=1;
DelayMs(20);
Write12864Command(0x30);//功能设定:8位控制方式,使用基本指令
Write12864Command(0x08);//显示关
Write12864Command(0x01);//清屏
Write12864Command(0x06);//地址计数器加一、光标右移
Write12864Command(0x0c);//显示开
}
带字库的驱动
1602液晶的程序我有,要做数字锁的话其实用到液晶也不多。
因为你显示密码是时候不应该都是现实吗?
你只需把键盘写入的数据存在一个数组中,然后跟密码数组对比就好了。
还有就是值得注意的是数据类型问题。
输入的键值看你处理的方式而定,密码存放的格式,还有输出显示的是ascll码。注意转换。
下面附带一段51的1602LCD的C程序,自己仔细琢磨。
#include<reg52h>
#define uchar unsigned char
#define uint unsigned int
uchar table[16]="abcdefghijklmnyz";
uchar table1[16]="0123456789abcdef";
sbit lcden=P2^0;
sbit lcdrs=P2^1;
sbit dula=P2^6;
sbit wela=P2^7;
uchar num;
void delay(uint z)
{
uint x,y;
for(x=z;x>0;x--)
for(y=110;y>0;y--);
}
void write_com(uchar com)
{ wela=0;
lcdrs=0;
P0=com;
delay(5);
lcden=1;
delay(5);
lcden=0;
}
void write_data(uchar date)
{ wela=0;
lcdrs=1;
P0=date;
delay(5);
lcden=1;
delay(5);
lcden=0;
}
void init()
{
lcden=0;
write_com(0x38);
write_com(0x0e);
write_com(0x06);
write_com(0x01);
write_com(0x80);
}
void main()
{
init();
for(num=0;num<16;num++)
{
write_data(table[num]);
delay(20);
}
write_com(1);
write_com(0x80+0x40);
for(num=0;num<16;num++)
{
write_data(table1[num]);
delay(20);
}
while(1);
}
往LCD中写入一个命令字cmd
a:lcden是LCD1602的允许读写脚,高电平有效
b:lcdrs是LCD1602的命令/数据脚,相当于地址选择, 高电平是数据读写,低电平是命令读写
c:lcdrw是LCD1602的读写控制脚,高电平是读,低电平是写
P0获取CMD值的目的是查看LCD是否忙,这里应该加上遇忙循环才完美
_nop_是为了替代省略的遇忙循环,确保延时后LCD不忙
往LCD中写入一个数据
lcdrs=1是为了2的b:高电平是数据读写
因为写入了LCD控制命令:0x38,0x0c,0x06,0x01等。
2MOV LCD,#0FFH是将P2口置为输入模式。
3E端口是高电平有效。
其他的“清屏并光标复位”这段命令不是很清楚,请参考程序:>
①问,RS,RW,E 的设置为 读状态时序 的过程,读出来的数据(数据口P1) 的最高位 刚好即为 液晶的 忙碌 标志位。
②问,writecontrol(unsigned condata) 函数 是个 写指令 函数;
③问,空 *** 作 是为了让数据稳定后,才使能 液晶接收数据;
④问,写指令有时间 间隔要求,太频繁的读写会丢码,所以要加延时,三条38指令,是因为安全起见,上电后马上初始化会出现液晶电源不稳而丢码;
⑤问,液晶显示 只在 更新数据 的时候刷新一次即可,液晶会维持内容,无需重复刷新。
⑥附送,③和④问,都是驱动太恶心而造成的结果,好的驱动程序不需如此 *** 作。液晶的时序要求为ns级,单片机的指令周期普遍为us级,根本不需过多累赘。
#include"reg52h" //包含52头文件
#include"SMC1602Ah" //包含SMC1602A宏定义文件
#define BusyReadCount 10 //读忙标志等待次数
#define SMC1602_Data P0 //定义 数据接口
//sbit SMC1602_VO=P2^4; //定义 VO对比度接口
sbit SMC1602_RW=P2^5; //定义 R/W接口25
sbit SMC1602_RS=P2^6; //定义 RS接口26
sbit SMC1602_E=P2^7; //定义 E接口27
#define SMC1602_En SMC1602_E=1 //使能
#define SMC1602_Dis SMC1602_E=0 //禁止
uchar SMC1602_Read(bit read_type) //1602液晶屏读函数
{
uchar read_data;
SMC1602_Dis; //禁止使能
SMC1602_RW=ReadOperate; //读 *** 作
SMC1602_RS=read_type; //读类型:0状态,1数据
SMC1602_En; //开启使能
read_data=SMC1602_Data; //存储结果
SMC1602_Dis; //禁止使能
return read_data; //返回结果
}
void SMC1602_WriteByte(bit write_type,uchar write_data) //1602液晶屏读函数
{
uchar i=BusyReadCount;
for(;i;i--); //延时 *** 作,为写 *** 作预留回复时间
while((SMC1602_Read(CommOperate)&BusyState) &&(++i<=BusyReadCount)); //读取忙标志(BusyReadCount次),若均忙中,则不再读取忙标志,直接执行写 *** 作
//while(SMC1602_Read(CommOperate)&BusyState) if(++i>BusyReadCount) return; //读取忙标志,若BusyReadCount次均忙中,则不进行写 *** 作
//while(SMC1602_Read(CommOperate)&BusyState); //等待空闲(死等)
SMC1602_Dis; //禁止使能
SMC1602_RW=WriteOperate; //写 *** 作
SMC1602_RS=write_type; //写类型:0指令,1数据
SMC1602_Data=write_data; //写 *** 作,将 *** 作数送的数据口
SMC1602_En; //开启使能
SMC1602_Dis; //禁止使能
}
void SMC1602_WriteCGRAM(uchar write_buf,uchar start_loca,uchar word_num,uchar start_addr) //SMC1602写CGRAM函数,用于自定义字符
{
uchar i,j;
write_buf+=start_loca; //指向"需写入数据数组"的起始位置
SMC1602_WriteByte(CommOperate,CGRAMAddr|start_addr<<3); //写CGRAM *** 作,并将CGRAM起始地址设为 start_addr
for(j=0;j<word_num;j++) //自定义字符数量
for(i=0;i<8;i++) SMC1602_WriteByte(DataOperate,write_buf++); //写入一个自定义字符8个字节数据
}
void SMC1602_Init() //1602液晶屏初始化函数
{
uint i;
SMC1602_WriteByte(CommOperate,DisplayMode); //显示模式设置:16×2显示,5×7点阵,8位数据接口
SMC1602_WriteByte(CommOperate,ScreenMode|ScreenOn); //光标模式设置:开启整体显示,开启光标显示,开启光标闪烁
SMC1602_WriteByte(CommOperate,InputMode); //输入方式设置:关闭整屏移动,开启光标正移动(+1)
SMC1602_WriteByte(CommOperate,CleanLCD); //清屏,复位光标
SMC1602_WriteByte(CommOperate,FirstCol); //定位第一行
for(i=150;i;i--); //等待电源稳定,否则写CGRAM数据(自定义字符)时容易丢失,uint执行周期长,用uchar将会缩短时间,不足以稳定LCD
}
以上就是关于关于单片机液晶屏LCD12864的程序全部的内容,包括:关于单片机液晶屏LCD12864的程序、1602液晶显示屏显示程序、求帮我解答这段LCD1062程序的内容等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)