12864使用串口通信,驱动程序怎么写

12864使用串口通信,驱动程序怎么写,第1张

这个问题有两种回答:

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给单片烧程序时候很容易蓝屏等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zz/9336192.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-27
下一篇 2023-04-27

发表评论

登录后才能评论

评论列表(0条)

保存