//////////////////////////////////////////////////////////////////////////
完结功用: 按试验板所佩戴的红外遥控器上面的数字键,把红外遥控器的代码显
示在1602液晶上面,经过此试验也可测验你手上所用红外遥控器的代
码编号,必定留意要跟红外勘探头的间隔不要太远,并且
程序里只编写了遥控器上的数字键的部分
试验板类型:KBL-XYD-C52
试验称号: 红外操控1602液晶的显现
编写人: 谢应东
编写日期: 2012-5-3
//////////////////////////////////////////////////////////////////////////
#include
#include
#define uchar unsigned char
#define uint unsigned int
sbit IR = P3^3; //红外接纳器数据线
uchar code string0[] = {“IR REMOTE CONTROL”}; //界说要显现在1602上的字符串
uchar code string1[] = {“IR-CODE: 0X– H”}; //界说要显现在1602上的字符串
#define BUSY 0x80 //lcd忙检测标志
#define LCD_DATAPORT P0 //界说P0口为LCD通讯端口
sbit Duan=P2^6;//界说数码管的段选使能端
sbit Wei =P2^7; //界说数码管的位选使能端
#define Digital_tube_Wei_Enable Wei=1;//敞开操控数码管的位选使能端
#define Digital_tube_Wei_Disable Wei=0; //封闭操控数码管的位选使能端
#define Digital_tube_Duan_Enable Duan=1;//敞开操控数码管的段选使能端
#define Digital_tube_Duan_Disable Duan=0;//封闭操控数码管的段选使能端
sbit LCD_RS=P3^4; //数据/指令端
sbit LCD_RW=P3^5; //读/写挑选端
sbit LCD_EN=P3^6;
uchar cIRReceiveData[7];
//////////////////////////////////////////////////////////////////////////
函数称号:毫秒延时函数
函数功用:完结毫秒级的延时
参数介绍:Delay_MS: 界说需求延时的毫秒的数值
iNumber: 记载Delay_MS的数值,以for句子完结所要求的延时
iValue: 要延时毫秒所要进行的循环数值,本数值为实践测得
返回值: 无
留意事项:本试验是在所用晶振为12M的前提下完结的毫秒延时,本函数是经过循环的形
式完结,所以假如改动了晶振的频率,请做相应的改动
//////////////////////////////////////////////////////////////////////////
void DelayMs(uint Delay_MS)
{
uint iNumber,iValue;
for(iNumber=0;iNumber //用for句子完结单片机的延时
{
iValue=107; //107这个数值是经过测定而得
while(iValue–);
}
}
//////////////////////////////////////////////////////////////////////////
函数称号:Delay
函数功用:延时0.14毫秒,主要是用于红外解码
参数介绍:cDelay:延时0.14ms的次数
返回值: 无
留意事项:无
//////////////////////////////////////////////////////////////////////////
void Delay(unsigned char cDelay) //x*0.14MS
{
unsigned char cNumber;
while(cDelay–)
{
for (cNumber=0;cNumber<13;cNumber++) {}
}
}
///////////////////////////////////////////////////////////////////////////
函数称号:ReadStatusLCD
函数功用:检测液晶是否繁忙
参数介绍:无
返回值: LCD_DATAPORT
留意事项:无
//////////////////////////////////////////////////////////////////////////
unsigned char ReadStatusLCD()
{
LCD_DATAPORT=0xff;
LCD_RS=0;
LCD_RW=1;
LCD_EN=0;
LCD_EN=0;
LCD_EN=1;
while(LCD_DATAPORT&BUSY);
return(LCD_DATAPORT);
}
//////////////////////////////////////////////////////////////////////////
函数称号:WriteCommandLCD
函数功用:依据1602液晶的时序,经过单片机给1602液晶写一些相关的指令
参数介绍:cCommand:要写入的1602液晶指令
cBusy:1602液晶是否闲暇的标志
返回值: 无
留意事项:无
//////////////////////////////////////////////////////////////////////////
void WriteCommandLCD(uchar cCommand,uchar cBusy)
{
if(cBusy)ReadStatusLCD();//检测忙信号
LCD_DATAPORT=cCommand; //要写入的指令
LCD_RS=0; // 选中指令寄存器
LCD_RW=0; // 写形式
LCD_EN=1;// 敞开1602液晶使能端
_nop_();// 等候写入指令
_nop_();
_nop_();
LCD_EN=0; // 封闭1602液晶使能端
}
//////////////////////////////////////////////////////////////////////////
函数称号:WriteDataLCD
函数功用:依据1602液晶的时序,经过单片机给1602写入想要写入的数据
参数介绍:cData:要写入1602的数据
返回值: 无
留意事项:无
//////////////////////////////////////////////////////////////////////////
void WriteDataLCD(uchar cData)
{
ReadStatusLCD();
LCD_DATAPORT=cData;
LCD_RS=1; // 选中数据寄存器
LCD_RW=0; // 写形式
LCD_EN=1; // 翻开1602液晶使能端
_nop_(); // 等候写入数据
_nop_();
_nop_();
LCD_EN=0; // 封闭1602液晶使能端
}
//////////////////////////////////////////////////////////////////////////
函数称号:Initial_LCD
函数功用:设置单片机的外部中止0和中止1,详细的设置状况都为外部低电平触发,并
且开总中止
参数介绍:无
返回值: 无
留意事项:无
//////////////////////////////////////////////////////////////////////////
void Initial_LCD()
{
Digital_tube_Wei_Enable;
LCD_DATAPORT=0xff;
Digital_tube_Wei_Disable;
Digital_tube_Duan_Enable;
LCD_DATAPORT=0X00;
DelayMs(2);
Digital_tube_Duan_Disable;
LCD_DATAPORT=0;
DelayMs(15);
WriteCommandLCD(0x38,0); //三次显现形式设置,不检测忙信号
DelayMs(5);
WriteCommandLCD(0x38,0);
DelayMs(5);
WriteCommandLCD(0x38,0);
DelayMs(5);
WriteCommandLCD(0x38,1); //8bit数据传送,2行显现,5*7字型,检测忙信号
WriteCommandLCD(0x08,1); //封闭显现,检测忙信号
WriteCommandLCD(0x01,1); //清屏,检测忙信号
WriteCommandLCD(0x06,1); //显现光标右移设置,检测忙信号
WriteCommandLCD(0x0c,1); //显现屏翻开,光标不显现,不闪耀,检测忙信号
}
//////////////////////////////////////////////////////////////////////////
函数称号:DisplayOneCharLCD
函数功用:按指定方位显现一个字符,比如说数字,英文字母等
参数介绍:cXCoord:要显现字符在某一行傍边的方位
cYCoord:要显现字符在液晶中的行数,比如说榜首行或第二行
cData: 要显现的字符,比如说1、a等
返回值: 无
留意事项:约束cXCoord不能大于15,cYCoord不能大于1
//////////////////////////////////////////////////////////////////////////
void DisplayOneCharLCD(uchar cXCoord,uchar cYCoord,uchar cData)
{
cYCoord&=1; //取cYCoord的榜首位,其他的悉数设置成0
if(cYCoord)cXCoord|=0x40; //若y为1,显现榜首行,地址码+0X40
cXCoord|=0x80; //若不为1,则显现第二行,地址码+0X80
WriteCommandLCD(cXCoord,0); //把写入的地址送到1602液晶
WriteDataLCD(cData); //写入要写入的数据
}
//////////////////////////////////////////////////////////////////////////
函数称号:DisplayListCharLCD
函数功用:按指定方位显现一串字符,比如说显现 abcdefg,123456等
参数介绍:cXCoord:要显现字符在某一行傍边的方位
cYCoord:要显现字符在液晶中的行数,比如说榜首行或第二行
cData: 要显现的字符串,比如说123456、adcdgd等
返回值: 无
留意事项:制cXCoord不能大于15,cYCoord不能大于1
//////////////////////////////////////////////////////////////////////////
void DisplayListCharLCD(uchar cXCoord,uchar cYCoord,uchar code *Data)
{
uchar ListLength=0;//要显现字符串的长度
cYCoord&=0x01;//取cYCoord的榜首位,其他的悉数设置成0
cXCoord&=0x0f;
while(cXCoord<16) //循环写入要写入的字符串
{
DisplayOneCharLCD(cXCoord,cYCoord,Data[ListLength]);
ListLength++;
cXCoord++;
}
}
//////////////////////////////////////////////////////////////////////////
函数称号:Init_INT1
函数功用:装备外部中止1,让其为下降沿触发,一起翻开大局中止
参数介绍:无
返回值: 无
留意事项:无
//////////////////////////////////////////////////////////////////////////
void Init_INT1()
{
IT1=1; //外部中止1,为下降沿触发
EX1=1; //答应外部中止1中止
EA=1; //CPU敞开中止
}
//////////////////////////////////////////////////////////////////////////
函数称号:main
函数功用:初始化外部中止1以及1602液晶,一起把预先写好的字符串显现到1602上面,
然后进入while循环里,等候中止的产生
参数介绍:无
返回值: 无
留意事项:试验板上所用的数码管为共阴极数码管
//////////////////////////////////////////////////////////////////////////
void main()
{
Init_INT1(); //初始化外部中止1
IR=1; //I/O口初始化
Initial_LCD(); //初始化1602液晶
DisplayListCharLCD(0,0,string0); //显现字符预写好的字符串
DisplayListCharLCD(0,1,string1); //显现字符预写好的字符串
while(1);
}
//////////////////////////////////////////////////////////////////////////
函数称号:IR_Routine
函数功用:对红外接纳头接纳到的红外信号进行解码,解码后,把遥控器上的数字显现
在数码管上
参数介绍:无
返回值: 无
留意事项:详细的协议,请看光盘里所带的名为“红外操控”的数据手册
//////////////////////////////////////////////////////////////////////////
void IR_Routine() interrupt 2
{
unsigned char cNumber1,cNumber2,cCount=0;
EX1 = 0; //封闭中止
Delay(15);
if (IR==1) //然后再检测红线接纳脚是有数据接纳,有持续,没有则退出
{
EX1 =1; //翻开中止
return; //跳出去
}
while (!IR) //承认IR信号呈现
{ //等IR变为高电平,越过9ms的前导低电平信号。
Delay(1);
}
for (cNumber1=0;cNumber1<4;cNumber1++) //搜集四组数据,前两组为地址码,仅
{ //仅接着是两个指令码,这是NEC公司所界说的协议里规则的
for (cNumber2=0;cNumber2<8;cNumber2++)//每组数据有8位
{
while (IR) //等 IR 变为低电平,越过4.5ms的前导高电平信号。
{
Delay(1);
}
while (!IR) //等 IR 变为高电平
{
Delay(1);
}
while (IR) //核算IR高电平时长
{
Delay(1);
cCount++;
if (cCount>=30)
{
EX1=1;
return;
} //太长了就主动推出中止服务程序
} //高电平计数结束
cIRReceiveData[cNumber1]=cIRReceiveData[cNumber1] >> 1;//数据最高位补“0”
if(cCount>=8) //协议里界说‘1’的电平为2.25ms,由于Delay(1)延时为0.14ms
{ //加上其他指令执行时的延时,所以大于等于8时,为高电平
cIRReceiveData[cNumber1] = cIRReceiveData[cNumber1] | 0x80;
} //数据最高位补“1”
cCount=0;
}//end for k
}//end for j
if(cIRReceiveData[2]!=~cIRReceiveData[3])//判别接纳到指令是不是正确,协议里规则
{//榜首次发指令和第2次发的正好的按位取反的,以此确保接纳数据的可靠性
EX1=1;
return;
}
cIRReceiveData[5]=cIRReceiveData[2] & 0x0F; //取键码的低四位
cIRReceiveData[6]=cIRReceiveData[2] >> 4; //右移4次,高四位变为低四位
if(cIRReceiveData[5]>9)//把接纳到的数据转化成1602上面能显现的ASCII码
{ //详细请看ASCII码表,看转化的联系
cIRReceiveData[5]=cIRReceiveData[5]+0x37;
}
else
cIRReceiveData[5]=cIRReceiveData[5]+0x30;
if(cIRReceiveData[6]>9)
{
cIRReceiveData[6]=cIRReceiveData[6]+0x37;
}
else
cIRReceiveData[6]=cIRReceiveData[6]+0x30;
DisplayOneCharLCD(13,1,cIRReceiveData[6]); //显现接纳到数据的榜首位
DisplayOneCharLCD(14,1,cIRReceiveData[5]); //显现接纳到数据的第二位
EX1 = 1; //翻开中止,解码完结,以进行下次解码
}
声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/ceping/262845.html