📄 lcd_driver.c
字号:
/*****************************************************************************
Filename: native.c
Version: 0.01
Description: calculate measure value
Returns: result
Revision History:
EQH-01, 屈宵军 09/25/04, - Original release
*****************************************************************************/
/*写汉字液晶子程 液晶屏分为4行*12列汉字,全部使用模拟接口方式。*/
#include<math.h>
#include<tms320.h>
#include "LCD.h"
/********************************/
/* 演示主程序 */
/********************************/
void Lcd_Driver(void)
{
uint x;
unsigned char temp, byteH,byteL; //word split as byte
for(x=0;x<100;x++)
{
byteH=(Hzk[x]&0x0FF00)>>8;
byteL=(Hzk[x]&0x000FF);
temp=byteH;
temp=byteH;
temp=byteH;
temp=byteH;
temp=byteL;
temp=byteL;
temp=byteL;
temp=byteL;
byteL=temp;
}
col=0;
row=0;
Delay(40); /*延时大约40Ms,等待外设准备好 */
Lcminit(); /*液晶模块初始化,包括全屏幕清屏*/
Putstr(STR2,24); /*第一行字符输出,24字节 */
col=0;
row=2;
Putstr(STR1,12); /*第二行字符输出,12字节 */
col=0;
row=4;
Putstr(STR3,25); /*第三行字符输出,24字节 */
col=0;
row=6;
Putstr(STR4,24); /*第四行字符输出,12字节 */
x=0;
col=0;
row=0;
xy = 1; /*方向标志。定为水平方向 */
Linehv(192); /*画一条横线(0,0)-(191,0) */
col=0;
row=15;
xy = 1;
Linehv(192); /*画一条横线(0,15)-(191,15) */
col=0;
row=32;
xy = 1;
Linehv(192); /*画一条横线(0,32)-(191,32) */
col=0;
row=1;
xy = 0; /*方向标志。定为垂直方向 */
Linehv(31); /*画一条竖线(0,1)-(0,31) */
col=191;
row=1;
xy = 0;
Linehv(31); /*画一条竖线(191,1)-(191,31) */
//col=0; /*设定斜线的起点坐标 */
//row=63;
//Linexy(44,31); /*画一段斜线(0,63)-(44,31) */
//col=44;
//row=31;
//Linexy(190,62); /*继续画斜线(44,31)-(191,63) */
while(0){
Rollscreen(x); /*定位新的显示起始行 */
x++;
Delay(100); /*延时,控制滚动速度 */
};
}
/************************************************/
/*画线。任意方向的斜线,不支持垂直的或水平线 */
/************************************************/
void Linexy(uint endx,uint endy)
{
register uint t;
int xerr=0,yerr=0,delta_x,delta_y,distance;
int incx,incy;
/* compute the distance in both directions */
delta_x=endx-col;
delta_y=endy-row;
/* compute the direction of the increment ,
an increment of "0" means either a vertical or horizontal lines */
if(delta_x>0) incx=1;
else if( delta_x==0 ) incx=0;
else incx=-1;
if(delta_y>0) incy=1;
else if( delta_y==0 ) incy=0;
else incy=-1;
/* determine which distance is greater */
delta_x = fabs( delta_x );
delta_y = fabs( delta_y );
if( delta_x > delta_y ) distance=delta_x;
else distance=delta_y;
/* draw the line */
for( t=0;t <= distance+1; t++ ) {
point();
xerr += delta_x ;
yerr += delta_y ;
if( xerr > distance ) {
xerr-=distance;
col+=incx;
}
if( yerr > distance ) {
yerr-=distance;
row+=incy;
}
}
}
/****************************************/
/*画线。只提供X或Y方向的,不支持斜线 */
/****************************************/
void Linehv(uint length)
{
uint xs,ys;
if (xy)
{ ys = col;
for (xs=0;xs<length;xs++)
{
col = ys + xs;
point();
}
}
else
{ xs = row;
for (ys=0;ys<length;ys++)
{
row = xs + ys;
point();
}
}
}
/****************************************/
/* 画点 */
/****************************************/
void point(void)
{
uint x1,y1,x,y;
x1=col;
y1=row;
row=y1>>3; /*取Y方向分页地址 */
Rddata();
y=y1&0x07; /*字节内位置计算 */
x=0x01;
x=x<<y; /*移入所画点 */
Wrdata(cbyte|x); /*画上屏幕 */
col=x1; /*恢复xy坐标 */
row=y1;
}
/****************************************/
/* 屏幕滚动定位 */
/****************************************/
void Rollscreen(uint x)
{
cbyte = DISPFIRST|x; /*定义显示起始行为x?*/
WrcmdL(cbyte);
WrcmdM(cbyte);
WrcmdR(cbyte);
}
/****************************************/
/* 一个字串的输出 */
/****************************************/
void Putstr(char *puts,uint i)
{
uint j,X;
for (j=0;j<i;j++)
{
X = puts[j];
if (X&0x80)
{
Putcdot(X&0x7f); /*只保留低7位*/
}
else Putedot(X-0x20); /*ascii码表从0x20开始*/
}
}
/****************************************/
/* 半角字符点阵码数据输出 */
/****************************************/
void Putedot(uint Order)
{
uint i,bakerx,bakery; /*共定义4个局部变量 */
int x; /*偏移量,字符量少的可以定义为uint */
bakerx = col; /*暂存x,y坐标,已备下半个字符使用 */
bakery = row;
x=Order * 0x10; /*半角字符,每个字符16字节 */
/*上半个字符输出,8列 */
for(i=0;i<8;i++)
{
cbyte = Ezk[x]; /*取点阵码,rom数组 */
Wrdata(cbyte); /*写输出一字节 */
x++;
col++;
if (col==LCMLIMIT){col=0;row++;row++;}; /*下一列,如果列越界换行*/
if (row>7) row=0; /*如果行越界,返回首行 */
} /*上半个字符输出结束 */
col = bakerx; /*列对齐 */
row = bakery+1; /*指向下半个字符行 */
/*下半个字符输出,8列 */
for(i=0;i<8;i++)
{
cbyte = Ezk[x]; /*取点阵码 */
Wrdata(cbyte); /*写输出一字节 */
x++;
col++;
if (col==LCMLIMIT){col=0;row=row+2;}; /*下一列,如果列越界换行*/
if (row>7) row=1; /*如果行越界,返回首行 */
} /*下半个字符输出结束 */
row=bakery;
} /*整个字符输出结束 */
/****************************************/
/* 全角字符点阵码数据输出 */
/****************************************/
void Putcdot(uint Order)
{
uint i,bakerx,bakery; /*共定义3个局部变量 */
int x; /*偏移量,字符量少的可以定义为uint */
bakerx = col; /*暂存x,y坐标,已备下半个字符使用 */
bakery = row;
x=Order * 0x20; /*每个字符32字?*/
/*上半个字符输出,16列 */
for(i=0;i<16;i++)
{
Wrdata(Hzk[x]); /*写输出一字节 */
x++;
col++;
if (col==LCMLIMIT){ col=0;row++;row++;} /*下一列,如果列越界换行*/
if (row>6) row=0; /*如果行越界,返回首行 */
} /*上半个字符输出结束 */
/*下半个字符输出,16列 */
col = bakerx;
row = bakery+1;
for(i=0;i<16;i++) /*下半部分*/
{
Wrdata(Hzk[x]);
x++;
col++;
if (col==LCMLIMIT){col=0;row++;row++;} /*下一列,如果列越界换行*/
if (row>7) row=1; /*如果行越界,返回首行 */
} /*下半个字符输出结束 */
row = bakery;
} /*整个字符输出结束 */
/****************************************/
/* 清屏,全屏幕清零 */
/****************************************/
void Lcmcls( void )
{
for(row=0;row<8;row++)
for(col=0;col<LCMLIMIT;col++) Wrdata(0);
}
/****************************************/
/* 从液晶片上读数据,保留在全局变量中 */
/****************************************/
void Rddata(void)
{
Locatexy(); /*坐标定位,返回时保留分区状态不变 */
*Datalcm_pointer=0xFF;
*Dilcm_pointer = 1; /*数据*/
*Rwlcm_pointer = 1; /*读数据*/
*Elcm_pointer = 1; /*读入到LCM*/
// _nop_();
cbyte = *Datalcm_pointer; /*虚读一次 */
*Elcm_pointer = 0;
Locatexy(); /*坐标定位,返回时保留分区状态不变 */
*Datalcm_pointer=0xFF;
// _nop_();
*Dilcm_pointer = 1; /*数据*/
*Rwlcm_pointer = 1; /*读数据*/
*Elcm_pointer = 1; /*读入到LCM*/
// _nop_();
cbyte = *Datalcm_pointer; /*从数据口读数据,真读 */
*Elcm_pointer = 0;
}
/****************************************/
/* 数据写输出 */
/****************************************/
void Wrdata(uint X)
{
Locatexy(); /*坐标定位,返回时保留分区状态不变 */
// wtcom();
*Dilcm_pointer = 1; /*数据输出*/
*Rwlcm_pointer = 0; /*写输出 */
*Datalcm_pointer = X; /*数据输出到数据口 */
*Elcm_pointer = 1; /*读入到LCM*/
// _nop_();
*Elcm_pointer = 0;
}
/****************************************/
/* 命令输出,每次输出一个分区控制口 */
/****************************************/
/*
*void Wrcmd(uint X)
{
Locatexy(); 确定分区,返回时保留分区状态不变
wtcom(); 等待LCM操作允许
*Dilcm_pointer = 0; 数据操作
*Rwlcm_pointer = 0; 写输出
*Datalcm_pointer = X; 数据输出到数据口
*Elcm_pointer = 1;_nop_();*Elcm_pointer = 0; 读入到LCM
}
*/
/********************************/
/* 命令输出到左区控制口 */
/********************************/
void WrcmdL(uint X)
{
lcdbusyL(); /*确定分区,返回时保留分区状态不变*/
*Dilcm_pointer = 0; /*命令操作 */
*Rwlcm_pointer = 0; /*写输出 */
*Datalcm_pointer = X; /*数据输出到数据口 */
*Elcm_pointer = 1;
*Elcm_pointer = 0; /*读入到LCM*/
}
/********************************/
/* 命令输出到中区控制口 */
/********************************/
void WrcmdM(uint X)
{
lcdbusyM(); /*确定分区,返回时保留分区状态不变*/
*Dilcm_pointer = 0; /*命令操作 */
*Rwlcm_pointer = 0; /*写输出 */
*Datalcm_pointer = X; /*命令输出到数据口 */
*Elcm_pointer = 1;//_nop_();
*Elcm_pointer = 0; /*读入到LCM*/
}
/********************************/
/* 命令输出到右区控制口 */
/********************************/
void WrcmdR(uint X)
{
lcdbusyR(); /*确定分区,返回时保留分区状态不变 */
*Dilcm_pointer = 0; /*命令操作 */
*Rwlcm_pointer = 0; /*写输出 */
*Datalcm_pointer = X; /*命令输出到数据口 */
*Elcm_pointer = 1;//_nop_();
*Elcm_pointer = 0; /*读入到LCM*/
}
/********************************************************/
/* 分区操作允许等待,返回时保留分区选择状态 */
/********************************************************/
void lcdbusyL(void)
{
*CS1LCM_pointer = 0; /*CLR CS1 */
*CS2LCM_pointer = 0; /*SETB CS2 */
//CS3LCM = 1; /*SETB CS3 */
wtcom(); /* waitting for enable */
}
void lcdbusyM(void)
{
*CS1LCM_pointer = 0; /*SETB CS1 */
*CS2LCM_pointer = 1; /*CLR CS2 */
//CS3LCM = 1; /*SETB CS3 */
wtcom(); /* waitting for enable */
}
void lcdbusyR(void)
{
*CS1LCM_pointer = 1; /*SETB CS1 */
*CS2LCM_pointer = 0; /*SETB CS2 */
//CS3LCM = 0; /*CLR CS3 */
wtcom(); /* waitting for enable */
}
void wtcom(void)
{
*Dilcm_pointer = 0; /*CLR DI */
*Rwlcm_pointer = 1; /*SETB RW */
*Datalcm_pointer = 0xFF; /*MOV DATA_LCM,#0FFH */
*Elcm_pointer = 1;//_nop_();
// while(*Datalcm_pointer & Lcdbusy);
*Elcm_pointer = 0;
}
/********************************************************/
/*根据设定的坐标数据,定位LCM上的下一个操作单元位置 */
/********************************************************/
void Locatexy(void)
{
unsigned char x,y;
switch (col&0xc0) /* col.and.0xC0 */
{ /*条件分支执行 */
case 0: {lcdbusyL();break;} /*左区 */
case 0x40: {lcdbusyM();break;} /*中区 */
case 0x80: {lcdbusyR();break;} /*右区 */
}
x = col&0x3F|SETX; /* col.and.0x3f.or.setx=0x40 */
y = row&0x07|SETY; /* row.and.0x07.or.sety=b8 */
wtcom(); /* waitting for enable */
*Dilcm_pointer = 0; /*CLR DI */
*Rwlcm_pointer = 0; /*CLR RW */
*Datalcm_pointer = y; /*MOV P0,Y */
*Elcm_pointer = 1;//_nop_();
*Elcm_pointer = 0;
wtcom(); /* waitting for enable */
*Dilcm_pointer = 0; /*CLR DI */
*Rwlcm_pointer = 0; /*CLR RW */
*Datalcm_pointer = x; /*MOV P0,X */
*Elcm_pointer = 1;//_nop_();
*Elcm_pointer = 0;
}
/********************************/
/*液晶屏初始化 */
/********************************/
void Lcminit(void)
{
cbyte = DISPOFF; /*关闭显示屏 */
WrcmdL(cbyte); /*左区*/
WrcmdM(cbyte); /*中区*/
WrcmdR(cbyte); /*右区*/
cbyte = DISPON; /*打开显示屏 */
WrcmdL(cbyte); /*左区*/
WrcmdM(cbyte); /*中区*/
WrcmdR(cbyte); /*右区*/
cbyte = DISPFIRST; /*定义显示起始行为零 */
WrcmdL(cbyte);
WrcmdM(cbyte);
WrcmdR(cbyte);
Lcmcls();
col=0; /*清屏 */
row=0;
Locatexy();
}
/********************************/
/* 延时 */
/********************************/
void Delay(uint MS)
{
// uint us,usn;
while(MS!=0)
{ /*usn = 4;
while(usn!=0)
{
us=0xf0;
while (us!=0){us--;};
usn--;
}*/
MS--;
}
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -