这个问题有两种回答:
1,你问的不对。你其实想问的是,怎么使用串口,修改12864屏幕上的内容。这个网上有很多程序。随便看一下就好。
2,你沉迷于技术了。模拟本身提供的是八位的输出,你偏偏要自己用UART串口去驱动。这就好比:是要用到PWM的时候,明明单片机内部已经有了集成的模块。配置定时器就可以输出完美的PWM。可你觉得,这样太简单了。于是,你用两个定时器去实现:一个定时,一个控制输出频率。
效果不好还不错,而且浪费资源。
再举个极端的例子,你想实现一个很简单的for循环,使用高级的语言(就C语言来说,一下就好),
但是,你偏偏要用二进制0,1来写。
你觉得,这样才牛B,你才是比较厉害的人。
--------------------------------
喜欢请点赞,不喜欢,另说
阁下说的pic16f877a单片机的12864液晶串行显示驱动程序!
你用的屏应该是12864带中文字库,控制芯片是ST7920的屏,那你就可以用模块内置自带的中文字库,只要根据ST7920中文字型码表里提出你想的内容就可以了,你可以在网下一个"ST7920中文字型码表",网上有很多,里面都是12864液晶串行接口的汉字库,你从里面调出来用就可以!
参考资料(不包含自调字库)
/-------------------------------------
ST7920串行驱动程序
--------------------------------------/
#ifndef ST7920_H
#define ST7920_H
/--------------------------------/
//#include<reg52h>
/--------------------------------/
#define uchar unsigned char
#define uint unsigned int
/--------------------------------
下面是ST7920的内部指令宏定义
--------------------------------/
#define bascmd 0x30 /使用基本指令集/
#define clear 0x01 /清屏/
#define turn 0x02 /地址归位/
#define entset 0x06 /游标右移,DDRAM地址计数器AC+1/
#define gor 0x1c /整屏右移/
#define gol 0x18 /整屏左移/
#define dison 0x0c /显示开,关光标/
#define cgrama 0x40 /cgram基地止/
#define ddrama 0x80 / 设定DDRAM 地址,第一行80H~87H,第二行90H~97H /
#define page0 0x80
#define page1 0x90
#define page2 0x88
#define page3 0x98
#define extcmd 0x34 /使用扩展指令集/
#define reverse 0x04 /反白显示/
#define mapdison 0x36/扩充指令图形显示开/
/--------------------------------
硬件接口
--------------------------------/
#define LCM_DATA P0 / P0口,作为数据总线 /
sbit LCM_RS = P1^7; / 数据(H)/指令(L) /
sbit LCM_RW = P1^6; / 读(H)/写(L) /
sbit LCM_EN = P1^5; / 使能-写(H->L)/读(H) /
sbit LCM_PSB = P1^4; / 8位或4位串口 /
sbit LCM_BF = LCM_DATA^7; / LCD模块内部忙标志 /
/------------------------------/
void delay(uint t);
void Ldelay(uint t1);
void LCM_busy();
void LCM_wrcmd(uchar cmd);
void LCM_wrdat(uchar dat);
void LCM_add(uchar x,uchar y);
void LCM_basinit();
void LCM_wrpagestr(uchar add,uchar dis,uchar num);
void LCM_mapclr();
void LCM_extinit();
void LCM_12864map(uchar code dis);
void LCM_6464map(uchar x,uchar p);
void LCM_3232word(uchar x,uchar p);
#endif
以下是并行方式的驱动:
#include"ST7920h"
/------------------------------
函数名:void delay(uint t)
延时t个5倍指令周期,1个指令周期约为1us
--------------------------------/
void delay(uint t)
{
while(t--)
{
_nop_();_nop_();_nop_();_nop_();_nop_();
}
}
void Ldelay(uint t1)
{
while(t1--)
delay(1000);
}
/------------------------------
读忙函数,等待液晶处于就绪态
LCM_busy=1时为忙
-------------------------------/
void LCM_busy()
{
LCM_RS = 0;
LCM_RW = 1;
LCM_EN = 1;
while(LCM_BF);
LCM_EN = 0;
}
/------------------------------
向LCM写指令
-------------------------------/
void LCM_wrcmd(uchar cmd)
{
LCM_busy();
LCM_RS = 0;
LCM_RW = 0;
LCM_DATA = cmd;
LCM_EN = 1;
delay(1);
LCM_EN = 0;
}
/------------------------------
向LCM写数据
-------------------------------/
void LCM_wrdat(uchar dat)
{
LCM_busy();
LCM_RS = 1;
LCM_RW = 0;
LCM_DATA = dat;
LCM_EN = 1;
delay(1);
LCM_EN = 0;
}
/------------------------------
写地址函数,基本指令集,文字
-------------------------------/
void LCM_add(uchar x,uchar y)
{
switch(x)
{
case 0:x=page0;break;
case 1:x=page1;break;
case 2:x=page2;break;
case 3:x=page3;break;
}
LCM_wrcmd(x+y);
// delay(8);
}
/------------------------------
LCM初始化函数,基本指令集
-------------------------------/
void LCM_basinit()
{
LCM_PSB = 1;
LCM_wrcmd(bascmd);
delay(80);
LCM_wrcmd(entset);
delay(80);
LCM_wrcmd(dison);
delay(80);
LCM_wrcmd(clear);
delay(1000);
}
void LCM_extinit()
{
LCM_wrcmd(0x34);
delay(80);
LCM_wrcmd(0x02);
delay(80);
}
/------------------------------
向某一行写字符串
-------------------------------/
void LCM_wrpagestr(uchar add,uchar dis,uchar num)
{
LCM_wrcmd(add);
while(num--)
{
LCM_wrdat(dis++);
}
}
/------------------------------
图形清屏函数
-------------------------------/
void LCM_mapclr()
{
uint i,j;
LCM_extinit();
for(i=0;i<512;i+=16)
{
LCM_wrcmd(0x80+i/16);
LCM_wrcmd(0x80);
for(j=0;j<16;j++)
{
LCM_wrdat(0x00);
}
}
for(i=0;i<512;i+=16)
{
LCM_wrcmd(0x80+i/16);
LCM_wrcmd(0x88);
for(j=0;j<16;j++)
{
LCM_wrdat(0x00);
}
}
}
/----------------------------
输入一幅12864的图像
----------------------------/
void LCM_12864map(uchar p)
{
uint i,j;
LCM_extinit();
LCM_mapclr();
for(i=0;i<512;i+=16)
{
LCM_wrcmd(0x80+i/16);
LCM_wrcmd(0x80);
delay(1);
for(j=0;j<16;j++)
{
LCM_wrdat(p[j+i]);
}
LCM_wrcmd(0x80+i/16);
LCM_wrcmd(0x88);
delay(1);
for(j=0;j<16;j++)
{
LCM_wrdat(p[512+j+i]);
}
}
LCM_wrcmd(mapdison);
delay(16);
}
void LCM_6464map(uchar x,uchar p)
{
uint i,j;
LCM_extinit();
LCM_mapclr();
for(i=0;i<256;i+=8)
{
LCM_wrcmd(0x80+i/8);
LCM_wrcmd(0x80+x);
delay(1);
for(j=0;j<8;j++)
{
LCM_wrdat(p[j+i]);
}
LCM_wrcmd(0x80+i/8);
LCM_wrcmd(0x88+x);
delay(1);
for(j=0;j<8;j++)
{
LCM_wrdat(p[256+j+i]);
}
}
LCM_wrcmd(mapdison);
delay(16);
}
/----------------------------
自定义图形文字3232
----------------------------/
void LCM_3232word(uchar x,uchar p)
{
uint i,j;
for(i=0;i<128;i+=4)
{
LCM_wrcmd(0x80+i/4);
LCM_wrcmd(0x80+x);
delay(1);
for(j=0;j<4;j++)
{
LCM_wrdat(p[j+i]);
}
}
LCM_wrcmd(mapdison);
delay(16);
}
以下是串行方式的驱动
#include"ST7920_sh"
/------------------------------
函数名:void delay(uint t)
延时t个5倍指令周期,1个指令周期约为1us
--------------------------------/
void delay(uint t)
{
while(t--)
{
_nop_();_nop_();_nop_();_nop_();_nop_();
}
}
void Ldelay(uint t1)
{
while(t1--)
delay(1000);
}
/------------------------------
读忙函数,等待液晶处于就绪态
LCM_busy=1时为忙
-------------------------------/
/void LCM_busy()
{
LCM_RS = 0;
LCM_RW = 1;
LCM_EN = 1;
while(LCM_BF);
LCM_EN = 0;
}
/------------------------------
向LCM写1byte
-------------------------------/
void LCM_wrbyte(uchar DATA)
{
uchar i;
for(i=0;i<8;i++)
{
LCM_SID=DATA&0x80; //取出最高位
LCM_CLK=1;
// _nop_();
LCM_CLK=0;
// _nop_();
DATA<<=1; //左移
}
}
/------------------------------
start=0向LCM写指令
start=1向LCM写数据
-------------------------------/
void LCM_wrcord(bit start, uchar DATA) //写指令或数据
{
uchar start_data,Hdata,Ldata;
if(start==0) start_data=0xf8; //写指令
else start_data=0xfa; //写数据
Hdata=DATA&0xf0; //取高四位
Ldata=(DATA<<4)&0xf0; //取低四位
LCM_wrbyte(start_data); //发送起始信号
delay(3); //延时是必须的
LCM_wrbyte(Hdata); //发送高四位
delay(1); //延时是必须的
LCM_wrbyte(Ldata); //发送低四位
delay(1); //延时是必须的
}
/------------------------------
写地址函数,基本指令集,文字
-------------------------------/
void LCM_add(uchar x,uchar y)
{
switch(x)
{
case 0:x=page0;break;
case 1:x=page1;break;
case 2:x=page2;break;
case 3:x=page3;break;
}
LCM_wrcord(0,x+y);
}
/------------------------------
LCM初始化函数,基本指令集
-------------------------------/
void LCM_basinit()
{
LCM_PSB = 0; //串口驱动模式
LCM_CS=1;
LCM_wrcord(0,bascmd); //8 位介面,基本指令集
LCM_wrcord(0,dison); //显示打开,光标关,反白关
LCM_wrcord(0,clear); //清屏,将DDRAM的地址计数器归零
}
void LCM_extinit()
{
LCM_wrcord(0,0x34);
LCM_wrcord(0,0x02);
}
/------------------------------
向某一行写字符串
-------------------------------/
void LCM_wrpagestr(uchar add,uchar dis,uchar num)
{
LCM_wrcord(0,add);
while(num--)
{
LCM_wrcord(1,dis++);
}
}
希望能对你有帮助,祝你开发成功!
是串口驱动的问题,开发程序,尤其是硬件这边经常跟串口打交道,一般的ghost系统由于是通用驱动,经常会在底层出问题。所以你最好重新装系统,别用ghost的,用安装的,这样就不会出驱动问题了,简单的解决方法是先用个USB转串口线试试吧。
要有一系列的功能函数将单片机与液晶屏联系起来,根据LCD驱动的时序要编写bit数据函数,写字节函数,写地址函数等等,单片机通过这些函数将要显示的内容写入12864自身的RAM存储器,而后LCD会自己将这些东西映射到屏上,我们就看到显示的东西了。
首先,你要用软件允许中断,即C语言中EA=1;//允许总中断ES=1;//允许串口中断汇编中可用SETBEA;允许总中断SETBES;允许串口中断当单片机接收到一帧数据后,RI会置1,向CPU申请中断,若之前有中断允许,则产生了中断,进入中断服务程序。
当然,单片机发送完一帧数据,TI也会置1,同样会产生中断!一般我们在发送数据时要关中断,因为一般你不用在发送时不用处理数据;接收数据时要开中断,以便你在中断服务程序中将接收到的数据进行存储并处理。补充:其实,不管你有没有允许中断,上位机(此时即给单片机发送信息的机器)只要给单片机发送数据,单片机就会自动接收数据,并把它放在数据缓冲器SBUF中,如果你之前有允许串行口中断,RI就会置1,向单片机CPU申请中断,并进入中断服务程序,即你问题中的serial()函数,做完这个函数后就会自动返回断点。
如果你没有允许中断,便不会产生串行中断。
其实,你的问题有个错误:别的中断都是某个I/O口电平变化产生。
这只是外部中断产生条件,不过,你之前也需要用软件允许外部中断。
另外,常见的51系列单片机有5个中断源三种中断5:
1、外部中断0和1;
2、定时器/计数器溢出中断0和1;
3、串行口中断。
另外,STC51系列还有定时器/计数器T2中断,A/D转换中断,PWM中断,串行中断2等等。你还有什么不懂的可以自己多翻一翻书,学到后面去了你就明白了。
你的问题中那个serial()函数中P1=SBUF;是把接收的数据送到并行P1口,然后那条RI=0;是不可少的,这是用软件清零RI,准备再次接受一帧数据产生中断,如果你没有这条语句,就会进入死循环,出不来了。有什么问题还可以Q我(1445090023),希望能给我评个最佳答案。
以上就是关于12864使用串口通信,驱动程序怎么写全部的内容,包括:12864使用串口通信,驱动程序怎么写、求pic16f877a单片机的12864液晶串行显示驱动程序!、为什么我用keil软件写的LCM12864程序,用STC-ISP给单片烧程序时候很容易蓝屏等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)