急求结果 有懂AVR单片机编写的液晶显示程序的请进来指点

急求结果 有懂AVR单片机编写的液晶显示程序的请进来指点,第1张

你这个液晶屏不是一般开发板能带有的。大致读了一下:

#include <avr/ioh> #include "boardh" #include "manageh"//调用头文件 #include "scanh" #include "timerh" #include "lcdh"

void sys_init(void);

void bus_reset(void);

int main(void) //主程序

{ uint8_t i = 0; //这里定义了一个计数器,用于指示灯闪烁计时用

sys_init(); //系统初始化,具体程序在下面

int_init(); //又初始化了一个什么东西,先不管

// timer0_init(); //没有用到time0

timer1_init(); //time1初始化,但是就现有程序而言还无法知道计时周期

// timer2_init();

// adc_init();

wdt_disable(); //先禁用看门狗

wdt_enable(WDTO_250MS); //设置看门狗250ms为一周期

wdt_reset(); //复位看门狗

sei(); //禁用中断

// 初始化LCD

lcd_init(); //调用液晶屏初始化程序

while(1) //主程序循环

{ manage(); //应该是处理液晶显示内容用的子程序

if (i & 0x10)

LED_ON(); //0x10个周期后LED闪烁

else

LED_OFF();

i++;

delay_ms(10);

wdt_reset(); //复位看门狗

}

}

// 系统初始化

void sys_init(void) //系统初始化子程序的详细内容(看来只定义了IO口还真不知道这个系统内单片机的定时器如何初始化的)

{

PORTA = 0xf2; //定义IO口

DDRA = 0; // PA7~PA4 键盘输入,PA3: LCD_A0复用电池测量时,输出1

// PA2: 输入,测电池电压, PA1:LCD_CS, PA0:输入,测总线电压?

PORTB = 0xff;

DDRB = 0x00; // PB输出,LCD_DB,PB0~PB4 KEY SCAN

PORTC = 0x80;

DDRC = 0xc0; // PC输出,PC7:LCD_RST,PC6:LCD_LED+,

PORTD = 0x03;

DDRD = 0xf3; // PD7:PWM,PD6:ZL,PD5:ZH,PD4:LED

// PD3返回,PD2开关, PD1:LCD_RD, PD0:LCD_WR

}

void lcd_init(void) //LCD的初始化

{

uint8_t i;

LCD_RST_L(); //应该是复位LCD

delay_ms(2); //延时

LCD_RST_H(); //复位完毕,该复位引脚拉高

for (i=0; i<8; i++) //复位8次看门狗?

{

delay_ms(20);

wdt_reset();

}

write_cmd(0xe2);//system reset 24 //写一个复位命令?

delay_ms(10);

write_cmd(0xeb);//set bias RATIO=1/12 27 =E8 FOR LOW POWER //设置液晶屏偏压

write_cmd(0x81);//set refence voltage ;pm 11 DOUBLE command //设置参考电压

write_cmd(0x83);//set RGB D1 21 //设置RGB方式?

write_cmd(0x2f);//set color mode,65K 22 D4 FOR 256 COLOR 3r-3g-2b//设置彩色显示深度

write_cmd(0x2b);//set line rate,358kps 15 A3//设置啥?不是很明白

write_cmd(0x24);//set n-line 5,TempCompensation 00//设置某种补偿?

write_cmd(0xa2); // 15 set line rate 296k

write_cmd(0xd0);//set partial display 21 D0 BGR//设置一个副屏幕?或许这个彩屏还附带一个小屏幕?

write_cmd(0xd6);//set cen 22 d6 64k color 5-6-5//不明白

write_cmd(0xc0); // 19 LC[2:0]

write_cmd(0xaf); //set display on 18, DC[4:2]=111//仍然不明白

delay_ms(10);

wdt_reset();//重启看门狗

}

void write_cmd(uint8_t cmd)//“写命令”子程序

{

LCD_A0_L();//应该是拉低“命令\数据”引脚的电平

LCD_CS_L(); //片选引脚

PORTB = cmd; //PORTB的8个引脚作为并行口向液晶屏输出数据

DDRB = 0xff; //PORTB的方向定义为输出

LCD_WR_L(); //写LCD引脚电平拉低,看来这程序可能是为了防止被中断或者别的情况打乱,特地发送了两遍

LCD_WR_L();

LCD_WR_H();

LCD_CS_H();

DDRB = 0; //改PORTB为高阻态

//个人觉得后面应该还需要两个语句:LCD_CS_H();和LCD_A0_H();已完成一个“写命令”周期

}

总之,你这块液晶屏应该在市场上不常见。要用这个玩意,你必须有对应这块液晶屏的详细文档、数据手册,否则光这么一个程序很难把这个液晶屏玩转了。

光看代码你是不能了解它的使用历程的,最好看看使用手册,下面的代码是用1602液晶

#include<reg52h>

#define uchar unsigned char

#define uint unsigned int

uchar code table[]="I LIKE MCU!";

uchar code table1[]=">

51单片机对lcd1602一些基础程序

#include <intrinsh>

#define dataport P1

sbit RS=P2^ 6;

sbit RW=P2^5;

sbit EN=P2^4;

//========================

//=========================

void waitfor() //检测忙信号函数

{

dataport=0xff;

RS=0;RW=1;_nop_(); //选择指令寄存器 读 *** 作

EN=1;_nop_(); //使能 *** 作

while(dataport&0x80); //如果最高位是1 表示1602正忙 原地踏步 忙完后芯片会将高位拉低

EN=0;

}

//======================

void writedata(unsigned char dataw) //写数据到lcm

{

waitfor(); //测忙

RS=1;RW=0;_nop_(); //选择数据寄存器 写 *** 作

dataport=dataw;_nop_(); //将数据送到数据口

EN=1;_nop_();_nop_();EN=0; //使能

}

//==========================

void writecmd(unsigned char cmd) //写命令到lcm

{

waitfor();

RS=0;RW=0;_nop_();

dataport=cmd;_nop_();

EN=1;_nop_();_nop_();EN=0;

}

//===========================

void init(void) // 初始化函数

{

writecmd(0x38); //功能设定 8位数据传输 双行显示

writecmd(0x0c);//显示器开关

writecmd(0x01);//清屏

writecmd(0x06);//字符进入模式 每进入一个字符光标向右移动一格 原有字符不动

//我在刚开始学的时候不知道下一个字符显示在哪 是和AC值有关还是和光标位置有关

//最后摸索出来是只和光标定位有关 现在还是不知道Ac值有什么用

}

//=========================

void location(unsigned char x,unsigned char y) //确实坐标函数

{

unsigned char temp;

temp=x&0x0f; //只要x数据的后四位

if(y){temp=temp|0x40;} //第一行为0 第二行为1 如果y=1则地址加0x40

temp|=0x80; //DDRAM地址的命令DB7为一

writecmd(temp);

}

//==============================

void displyonechar(unsigned char x,unsigned char y,unsigned char dataw) //显示一个字符函数

{

location(x,y);

writedata(dataw);

}

//=======================================

void displylistchar(unsigned char x,unsigned char y,unsigned char p) //显示字符串

{

while(p) //当一个字符型数组读完时P指的为零

{

displyonechar(x,y,(p++));

x++;

}

}

//=====================================================

void writecgram(unsigned char address,unsigned char p) //写CGRAM的数据

{

unsigned char i=8;

writecmd(address); //CGRAM里的地址 初始值0x40 每次加0x80

while(i--)

{

writedata(p);

p++;

}

}

//=====================================================

void displyonecharacter(unsigned char x,unsigned char y,unsigned char address,unsigned char p) //显示一个自定义字符

{

unsigned char i=8;

writecmd(address); //CGRAM里的地址 初始值0x40 每次加0x08

while(i--)

{

writedata(p);

p++;

}

//============================================================

location(x,y); //设定要显示的位置

writedata((address&=0x3f)/0x08); //要从CGRAM中读出数据在1602上显示 搞了半天发现CGRAM里的地址

} //和DDRAM里的地址有上面的转换关系

//========================================================

void displynumber(unsigned char x,unsigned char y,unsigned long num) //显示一个整数

{

unsigned int number[8];

int k,gh;

for(k=0;;k++)

{

(number+k)=(unsigned int)(num%10);//强制类型转换

num=num/10;

if(num==0)break;

}

for(gh=k;gh>=0;gh--)

{

displyonechar(x,y,((number+gh)+48));

x++;

}

}

//字型码

uchar code nin[]={0x08,0x0f,0x12,0x0f,0,0x1f,0x02,0x02};// "年"

uchar code yue[]={0x0f,0x09,0x0f,0x09,0x0f,0x09,0x0b,0x11};// "月"

uchar code ri[]={0x1f,0x11,0x11,0x1f,0x11,0x11,0x11,0x1f};// "日"

显示汉字

displyonecharacter(0,0,0x40,nin);

displyonecharacter(1,0,0x80,yue);

displyonecharacter(1,0,0xc0,ri);

武汉 谷鑫科技专业从事单片机驱动TFT彩屏的研发和生产。

下面是TFT6448BS-57模块的驱动程序:

sfr AUXR = 0x8e;//////////////////////////////////注意增加该语句。

#include <reg51h>

#include <absacch>

#define X_ADDR XBYTE[0x0000]

#define Y_ADDR XBYTE[0x0100]

#define CMD XBYTE[0x0200]

#define DAT XBYTE[0x0300]

unsigned char code zk[32] = {//请

0x00, 0x47, 0x20, 0x23, 0x00, 0xEF, 0x20, 0x23, //left

0x22, 0x23, 0x22, 0x23, 0x2A, 0x32, 0x22, 0x02,

0x48, 0xFC, 0x40, 0xF8, 0x40, 0xFE, 0x08, 0xFC, //right

0x08, 0xF8, 0x08, 0xF8, 0x08, 0x08, 0x28, 0x10};

unsigned char code picture[];

main()

{

unsigned int x,y,x2,y2;

unsigned int i;

unsigned char key,x1,y1,j,k,z,m,n,bright;

AUXR = 0x02;//////////////////////////////////注意增加该语句。

bright = 8;

key = 0;//////////////////////////////////////选择液晶屏型号

//0:TFT2432;

//1:TFT3224;

//2:TFT480234

//3:TFT480272

//4:TFT6448

//5:TFT8048

//6:TFT8060

//7:TFT8060

if(P13==4)

{

if(key==0){x2=240;y2=320;x1=6; y1=16; goto loop1;}//x1/32,y1/20

if(key==1){x2=320;y2=240;x1=8; y1=12; goto loop1;}

if(key==2){x2=480;y2=240;x1=12;y1=12; goto loop1;}

if(key==3){x2=480;y2=272;x1=12;y1=14; goto loop1;}

if(key==4){x2=640;y2=480;x1=16;y1=24; goto loop1;}

if(key==5){x2=800;y2=480;x1=20;y1=24; goto loop1;}

if(key==6){x2=800;y2=600;x1=20;y1=30; goto loop1;}

if(key==7){x2=800;y2=600;x1=20;y1=30; goto loop1;}//亮度测试

}

loop1:

bright--;if(bright==255)bright=7;

CMD = 0x06;

DAT = bright;

///////////////////////以下是清屏功能///////////////////////////////////////

//用红色清屏

CMD = 0x04;DAT = 0xe0;//背景色

CMD = 0x08;//启动填充 *** 作

for(y=0;y<26012;y++);//延时166毫秒

CMD = 0x00;//退出填充 *** 作

///////////////////////以上是清屏功能///////////////////////////////////////

while(key==7)

{

for(bright=7;bright!=255;bright--)

{

CMD = 0x06;

DAT = bright;

for(y=0;y<15;y++){x=1;while(x!=0)x++;}

}

CMD = 0x20;//关断显示,poweroff

for(y=0;y<15;y++){x=1;while(x!=0)x++;}

key = P1&0x07;

}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

//用红色清屏

CMD = 0x04;DAT = 0x1c;//背景色

CMD = 0x08;//启动填充 *** 作

for(y=0;y<26012;y++);//延时166毫秒

CMD = 0x00;//退出填充 *** 作

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

//用红色清屏

CMD = 0x04;DAT = 0x03;//背景色

CMD = 0x08;//启动填充 *** 作

for(y=0;y<26012;y++);//延时166毫秒

CMD = 0x00;//退出填充 *** 作

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

//用红色清屏

CMD = 0x04;DAT = 0x00;//背景色

CMD = 0x08;//启动填充 *** 作

for(y=0;y<26012;y++);//延时166毫秒

CMD = 0x00;//退出填充 *** 作

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

//用红色清屏

CMD = 0x04;DAT = 0xff;//背景色

CMD = 0x08;//启动填充 *** 作

for(y=0;y<26012;y++);//延时166毫秒

CMD = 0x00;//退出填充 *** 作

///////////////////////以上是清屏功能///////////////////////////////////////

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

///////////////////////以下是测试8点写功能///////////////////////////////////////

//8点写模式下清屏DAT=0;

CMD = 0x02;DAT=0xff;//前景色

CMD = 0x04;DAT=0x00;//背景色

CMD = 0x01;

for( y=0;y<y2;y++ )//用前景色清屏

{

X_ADDR = 0;

X_ADDR = 0;

Y_ADDR = y; ;

if( ((key==1)|(key==2))==0 )Y_ADDR = y/256;

for( x=0;x<(x2/8);x++ )DAT=0x55;

}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for( y=0;y<y2;y++ )//用背景色清屏

{

X_ADDR = 0;

X_ADDR = 0;

Y_ADDR = y;

if( ((key==1)|(key==2))==0 )Y_ADDR = y/256;

for( x=0;x<(x2/8);x++ )DAT=0x0f;

}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

//8点写模式下写单色

CMD = 0x02;DAT=0x00;//前景色

CMD = 0x04;DAT=0xff;//背景色

CMD = 0x01;

for (y=0; y<80; y++)

{

X_ADDR = 0;

X_ADDR = 0;

Y_ADDR = y;

if( ((key==1)|(key==2))==0 )Y_ADDR = y/256;

for ( n=0; n<10; n++) DAT = picture[y10+n];

}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

//8点写模式下写汉字

CMD = 0x02;DAT=0x00;//前景色

CMD = 0x04;DAT=0xff;//背景色

CMD = 0x01;

for ( n=0; n<=15; n++ )

{

X_ADDR = 0;

X_ADDR = 0;

Y_ADDR = n;

if( ((key==1)|(key==2))==0 )Y_ADDR = y/256;

DAT = zk[n];

DAT = zk[n+16];

}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

//以下是测试单点写

CMD = 0x00;

i=0;

for(j=0;j<x1;j++)//20,x/40

{for (z=0;z<y1;z++)//24,y/20

{for (k=0;k<20;k++)

{x=j40;

y=z20+k;

X_ADDR = x;

if(key!=0)X_ADDR = x/256;

Y_ADDR = y;

if( ((key==1)|(key==2))==0 )Y_ADDR = y/256;

for (m=0;m<40;m++)DAT = i;

}

i++;

}

}

for(y=0;y<1;y++){x=1;while(x!=0)x++;}

>

程序只是没有逻辑错误和语法错误,但液晶的控制貌似有些问题。给你一段1602的驱动程序做参考。

#define LCD1602_FLAG

#define LCD1602_PORT P1

#include<reg52h>

#include<stddefh>

#include"dtypeh"

sbit lcd1602_rs=P3^7;

sbit lcd1602_e=P3^5;

sbit lcd1602_rw=P3^6;

sbit lcd1602_busy=P1^7;

/

函数名称:lcd1602_CheckBusy()

函数功能:状态查询

/

void lcd1602_CheckBusy()

{

do

{

lcd1602_busy=1;

lcd1602_rs=0;

lcd1602_rw=1;

lcd1602_e=0;

lcd1602_e=1;

}

while(lcd1602_busy);

}

/

函数名称: lcd1602_WriteCmd()

函数功能:写命令

入口参数:命令字

出口参数:无

/

void lcd1602_WriteCmd(const INT8U cmd)

{

lcd1602_CheckBusy();

lcd1602_rs=0;

lcd1602_rw=0;

lcd1602_e=1;

LCD1602_PORT=cmd;

lcd1602_e=0;

}

/

函数名称:lcd1602_WriteData()

函数功能:写数据

入口参数:c--待写数据

出口参数:无

/

void lcd1602_WriteData(const INT8U c)

{

lcd1602_CheckBusy();

lcd1602_rs=1;

lcd1602_rw=0;

lcd1602_e=1;

LCD1602_PORT=c;

lcd1602_e=0;

}

/

函数名称:lcd1602_Init()

函数功能:初始化LCD

入口参数:无

出口参数:无

/

void lcd1602_Init()

{

lcd1602_WriteCmd(0x38); //显示模式为8位2行57点阵

lcd1602_WriteCmd(0x0f); //display enable,flag enable,flash enable,

lcd1602_WriteCmd(0x06); //flag move to right,screen don't move

lcd1602_WriteCmd(0x01); //clear screen

}

/

函数名称:lcd1602_Display()

函数功能: 字符显示

入口参数:ptr--字符或字符串指针

出口参数:无

说 明:用户可通过以下方式来调用:

1)lcd1602_Display("Hello,world!");

2) INT8U 存储类型 txt[]="要显示的字符串";

或者 INT8U 存储类型 txt[]={'t','x','t',,'\0'};

INT8U ptr;

ptr=&txt;

lcd1602_Display(ptr);

或 lcd1602_Display(txt);

或 lcd1602_Display(&txt);

/

void lcd1602_Display(const INT8U ptr)

{

INT8U data i=0;

INT8U data q;

q=ptr;

lcd1602_WriteCmd(0x80);

while(q!=NULL && (q!='\0') && i<16)

{

lcd1602_WriteData(q);

q++;

i++;

}

lcd1602_WriteCmd(0xc0);

while(q!=NULL && (q!='\0') && i>=16 && i<32)

{

lcd1602_WriteData(q);

q++;

i++;

}

}

#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);

}

}

以上就是关于急求结果 有懂AVR单片机编写的液晶显示程序的请进来指点全部的内容,包括:急求结果 有懂AVR单片机编写的液晶显示程序的请进来指点、最简单c51单片机液晶显示程序、给个单片机液晶1602显示汉字的程序,谢谢等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

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

原文地址:https://54852.com/zz/9311637.html

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

发表评论

登录后才能评论

评论列表(0条)

    保存