//与硬件有关的定义:数据口用的是P3口,控制口用的是P2口,
//如果不同,可以更改以下定义。
#include "msp430x44xh"
#define LCD_CON_DIR P2DIR
#define LCD_CON_OUT P2OUT
#define LCD_DI BIT0
#define LCD_RW BIT1
#define LCD_EN BIT2
#define LCD_CS1 BIT3
#define LCD_CS2 BIT6
#define LCD_RST BIT7
#define LCD_DATA_DIR P3DIR
#define LCD_DATA_IN P3IN
#define LCD_DATA_OUT P3OUT
#define LCD_DATA_BIT0 BIT0
#define LCD_DATA_BIT1 BIT1
#define LCD_DATA_BIT2 BIT2
#define LCD_DATA_BIT3 BIT3
#define LCD_DATA_BIT4 BIT4
#define LCD_DATA_BIT5 BIT5
#define LCD_DATA_BIT6 BIT6
#define LCD_DATA_BIT7 BIT7
//
//以下为LCD的命令字
#define DISPLAY_ON 0x3F //显示屏开命令字
#define DISPLAY_OFF 0x3E //显示屏关命令字
#define SET_SHOW 0xC0 //设定显示起始行命令字
#define SET_X 0xB8 //设定显示行命令字
#define SET_Y 0x40 //设定显示列命令字
--------------------------------------------------------------------------------------------------------
#include "LCD_DEFINEH"
void lcd_init(void)
{
LCD_DATA_OUT = 0x00;
LCD_DATA_DIR = 0xFF;
LCD_CON_OUT = LCD_CS1+LCD_CS2+LCD_RST;
LCD_CON_DIR |= LCD_DI+LCD_RW+LCD_EN+LCD_CS1+LCD_CS2+LCD_RST;
LCD_CON_OUT &= ~LCD_RST;
_NOP();
_NOP();
LCD_CON_OUT |= LCD_EN;
_NOP();
_NOP();
LCD_CON_OUT &= ~LCD_EN;
_NOP();
_NOP();
LCD_CON_OUT |= LCD_RST;
_NOP();
_NOP();
}
void write_command(unsigned char command)
{
LCD_DATA_DIR = 0xFF;
LCD_CON_OUT |= LCD_CS1 + LCD_CS2;
LCD_CON_OUT &= ~(LCD_RW + LCD_DI);
LCD_DATA_OUT = command;
LCD_CON_OUT |= LCD_EN;
LCD_CON_OUT &= ~LCD_EN;
}
void write_data( unsigned char data, unsigned char CS1, unsigned char CS2 )
{
LCD_DATA_DIR = 0xFF;
if(CS1) LCD_CON_OUT |= LCD_CS1;
else LCD_CON_OUT &=~LCD_CS1;
if(CS2) LCD_CON_OUT |= LCD_CS2;
else LCD_CON_OUT &=~LCD_CS2;
LCD_CON_OUT |= LCD_DI;
LCD_CON_OUT &=~LCD_RW;
LCD_DATA_OUT = data;
LCD_CON_OUT |= LCD_EN;
LCD_CON_OUT &=~LCD_EN;
}
void clear_lcd(void)
{
unsigned char i,j;
for(i=0;i<8;i++)
{
write_command(SET_X|i);
write_command(SET_Y);
for(j=0;j<128;j++)
{
if(j<=63)write_data(0,1,0);
else write_data(0,0,1);
_NOP();
}
}
}
void display( unsigned char chr, unsigned char nRow, unsigned char nCol,
unsigned char highth,unsigned char wideth )
//chr 显示数据的地址,nRow 在显示屏上第几行,nCol 在显示屏上第几列
//highth 字体的高度,wideth 字体的宽度
//12864有8行,128列
{
unsigned char i,tmpCol,tmpRow,h,m;
unsigned int j;
tmpRow = nRow;
m=0;
for(h=0;h<highth;h=h+8)
{
write_command(SET_X|tmpRow);
tmpCol=nCol;
for(i=0;i<wideth;i++)
{
if(tmpCol<64)
{write_command(SET_Y|tmpCol);write_data(chr[i+j],1,0);}
else
{write_command(SET_Y|(tmpCol-64));write_data(chr[i+j],0,1);}
tmpCol++;
}
m++;
j=mwideth;
tmpRow++;
}
}
/
程序名称:带汉字库的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);
}
#include
<reg51h>
#define
uchar
unsigned
char
#define
uint
unsigned
int
uchar
xpos
,ypos;
sbit
key=P2^0;
sbit
rs=P3^0;
sbit
rw=P3^1;
sbit
e=P3^2;
void
nop(){}
void
delay(uint
z)
//延时
{
uint
x,y;
for(x=z;x>0;x--)
for(y=112;y>0;y--);
}
void
check()
//判忙
{
rs=0;
rw=1;
e=0;
P1=0xff;
e=1;
nop();
nop();
while(P1&0x80){}
}
void
wrc(uchar
com)
//写指令
{
check();
rs=0;
rw=0;
e=1;
P1=com;
nop();
e=0;
nop();
}
void
wrd(uchar
date)
//写数据
{
check();
rs=1;
rw=0;
e=1;
P1=date;
nop();
e=0;
nop();
}
void
init()
//初始化
{
wrc(0x38);
wrc(0x01);
wrc(0x0c);
wrc(0x06);
}
void
lcdpos()
//内部数据地址指针定位
{
xpos&=0x01;
ypos&=0x1f;
if(xpos==0x00)
wrc(ypos|0x80);
else
wrc((ypos+0x40)|0x80);
}
void
lcdw(uchar
x,uchar
y,uchar
s)
//在指定的坐标下写字符串
{
xpos=x;
for(ypos=y;ypos<20;ypos++)
{
lcdpos();
wrd(s);
s++;
}
}
void
main()
{
init();
cgram();
while(1)
{
lcdw(0,0,"##########
");
lcdw(1,1,"!!!!!!!!!!
");
lcdw(2,2,"&&&&&&&&&&
");
lcdw(3,3,"$$$$$$$$$$
");
}
}
#include<reg51h> //包含单片机寄存器的头文件
#include<intrinsh> //包含_nop_()函数定义的头文件
sbit RS=P2^5; //寄存器选择位,将RS位定义为P25引脚
sbit RW=P2^6; //读写选择位,将RW位定义为P26引脚
sbit EN=P2^7; //使能信号位,将EN位定义为P27引脚
sbit BF=P0^7; //忙碌标志位,,将BF位定义为P07引脚
unsigned char code string[ ]={"ABCD1234"}; //字符串数组,存储待显示的字符串
/
函数功能:延时1ms
(3j+2)i=(3×33+2)×10=1010(微秒),可以认为是1毫秒
/
void delay1ms()
{
unsigned char i,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++)
;
}
/
函数功能:延时若干毫秒
入口参数:n
/
void delay(unsigned char n)
{
unsigned char i;
for(i=0;i<n;i++)
delay1ms();
}
/
函数功能:判断液晶模块的忙碌状态
返回值:result。result=1,忙碌;result=0,不忙
/
unsigned char BusyTest(void)
{
bit result;
RS=0; //根据规定,RS为低电平,RW为高电平时,可以读状态
RW=1;
EN=1; //EN=1,才允许读写
_nop_(); //空 *** 作
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
result=BF; //将忙碌标志电平赋给result
EN=0;
return result;
}
/
函数功能:将模式设置指令或显示地址写入液晶模块
入口参数:dictate
/
void WriteInstruction (unsigned char dictate)
{
while(BusyTest()==1); //如果忙就等待
RS=0; //根据规定,RS和R/W同时为低电平时,可以写入指令
RW=0;
EN=0; //EN置低电平(根据表8-6,写指令时,EN为高脉冲,
//就是让EN从0到1发生正跳变,所以应先置"0"
_nop_();
_nop_(); //空 *** 作两个机器周期,给硬件反应时间
P0=dictate; //将数据送入P0口,即写入指令或地址
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=1; //EN置高电平
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=0; //当EN由高电平跳变成低电平时,液晶模块开始执行命令
}
/
函数功能:指定字符显示的实际地址
入口参数:x
/
void WriteAddress(unsigned char x)
{
WriteInstruction(x|0x80); //显示位置的确定方法规定为"80H+地址码x"
}
/
函数功能:将数据(字符的标准ASCII码)写入液晶模块
入口参数:y(为字符常量)
/
void WriteData(unsigned char y)
{
while(BusyTest()==1);
RS=1; //RS为高电平,RW为低电平时,可以写入数据
RW=0;
EN=0; //EN置低电平(根据表8-6,写指令时,EN为高脉冲,
//就是让EN从0到1发生正跳变,所以应先置"0"
P0=y; //将数据送入P0口,即将数据写入液晶模块
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=1; //EN置高电平
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=0; //当EN由高电平跳变成低电平时,液晶模块开始执行命令
}
/
函数功能:对LCD的显示模式进行初始化设置
/
void LcdInitiate(void)
{
delay(15); //延时15ms,首次写指令时应给LCD一段较长的反应时间
WriteInstruction(0x38); //显示模式设置:16×2显示,5×7点阵,8位数据接口
delay(5); //延时5ms
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x0f); //显示模式设置:显示开,有光标,光标闪烁
delay(5);
WriteInstruction(0x06); //显示模式设置:光标右移,字符不移
delay(5);
WriteInstruction(0x01); //清屏幕指令,将以前的显示内容清除
delay(5);
}
void main(void) //主函数
{
unsigned char i;
LcdInitiate(); //调用LCD初始化函数
delay(10);
while(1)
{
WriteInstruction(0x01); //清显示:清屏幕指令
WriteAddress(0x00); //设置显示位置为第1行的第1个字。1602型LCD字符显示器在第1个地址显示完毕后,能自动指向下一地址,
//因此只要制定字符串的第1个字符的显示地址即可。
i = 0;
while(string[i] != '\0')
{ //显示字符
WriteData(string[i]);
i++;
delay(150);
}
for(i=0;i<4;i++)
delay(250);
}
}
我看你这好像是单片机的程序,汇编语言和芯片是紧密相关的,都不知道你的芯片是什么型号也就不知道都有什么指令怎么转还有怎么也得有个线路连接的吧,你这里还有个
温度传感器
,还有个LCD1,还有你的程序没写完
建议你把这个程序读懂,然后自己写吧
楼上说的不太对啊,人家是单片机的程序,
反汇编
的在那里用8086奔腾还是什么,芯片不一样指令集也不一样了,就算反汇编了来还要改很多
光看代码你是不能了解它的使用历程的,最好看看使用手册,下面的代码是用1602液晶
#include<reg52h>
#define uchar unsigned char
#define uint unsigned int
uchar code table[]="I LIKE MCU!";
uchar code table1[]=">
以上就是关于关于128X64点阵LCD液晶显示的使用-程序全部的内容,包括:关于128X64点阵LCD液晶显示的使用-程序、关于单片机液晶屏LCD12864的程序、1602液晶显示屏显示程序等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)