写出来的程序又如何验证是否正确?
其实,你只需要对LCD的打开、关闭,以及能把一个数据块显示在LCD上就行了
“常见字母、数字、符号、中文、自定义字符、图形,以及文字和图形混和显示;光标显示/隐藏(Cursor
on/off)、显示字符闪烁(Display
character
blink);画面清除(Display
clear)、光标归位(Return
home)”
这些功能,只需要在这个数据块上进行 *** 作,然后再送到LCD显示。
//与硬件有关的定义:数据口用的是P3口,控制口用的是P2口,//如果不同,可以更改以下定义。
#include "msp430x44x.h"
#define LCD_CON_DIRP2DIR
#define LCD_CON_OUTP2OUT
#define LCD_DI BIT0
#define LCD_RW BIT1
#define LCD_EN BIT2
#define LCD_CS1BIT3
#define LCD_CS2BIT6
#define LCD_RSTBIT7
#define LCD_DATA_DIR P3DIR
#define LCD_DATA_INP3IN
#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_ON0x3F //显示屏开命令字
#define DISPLAY_OFF 0x3E //显示屏关命令字
#define SET_SHOW 0xC0 //设定显示起始行命令字
#define SET_X 0xB8 //设定显示行命令字
#define SET_Y 0x40 //设定显示列命令字
--------------------------------------------------------------------------------------------------------
#include "LCD_DEFINE.H"
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=0i<8i++)
{
write_command(SET_X|i)
write_command(SET_Y)
for(j=0j<128j++)
{
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 字体的宽度
//128*64有8行,128列
{
unsigned char i,tmpCol,tmpRow,h,m
unsigned int j
tmpRow = nRow
m=0
for(h=0h<highthh=h+8)
{
write_command(SET_X|tmpRow)
tmpCol=nCol
for(i=0i<widethi++)
{
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=m*wideth
tmpRow++
}
}
①问,RS,RW,E 的设置为 读状态时序 的过程,读出来的数据(数据口P1) 的最高位 刚好即为 液晶的 忙碌 标志位。②问,writecontrol(unsigned condata) 函数 是个 写指令 函数;
③问,空 *** 作 是为了让数据稳定后,才使能 液晶接收数据;
④问,写指令有时间 间隔要求,太频繁的读写会丢码,所以要加延时,三条38指令,是因为安全起见,上电后马上初始化会出现液晶电源不稳而丢码;
⑤问,液晶显示 只在 更新数据 的时候刷新一次即可,液晶会维持内容,无需重复刷新。
⑥附送,③和④问,都是驱动太恶心而造成的结果,好的驱动程序不需如此 *** 作。液晶的时序要求为ns级,单片机的指令周期普遍为us级,根本不需过多累赘。
#include"reg52.h" //包含52头文件
#include"SMC1602A.h" //包含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(ii--) //延时 *** 作,为写 *** 作预留回复时间
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=0j<word_numj++) //自定义字符数量
for(i=0i<8i++) 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=150ii--) //等待电源稳定,否则写CGRAM数据(自定义字符)时容易丢失,uint执行周期长,用uchar将会缩短时间,不足以稳定LCD
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)