本文档的主要内容详细介绍的是使用AT89S51单片机制作红外遥控器的资料和源代码详细说明。
一般红外电视遥控器的输出都是用编码后串行数据对38~40kHz的方波进行脉冲幅度调制而产生的。
当发射器按键按下后,即有遥控码发出,所按的键不同遥控编码也不同。这种遥控码具有以下特征:
采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的“0”;以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的“1”。
上述“0”和“1”组成的32位二进制码经38kHz的载频进行二次调制,然后再通过红外发射二极管产生红外线向空间发射。一般电视遥控器的遥控编码是连续的32位二进制码组,其中前16位为用户识别码,能区别不同的红外遥控设备,防止不同机种遥控码互相干扰。后16位为8位的操作码和8位的操作反码,用于核对数据是否接收准确。
根据红外编码的格式,发送数据前需要先发送9ms的起始码和4.5ms的结果码。
接收方一般使用TL0038一体化红外线接收器进行接收解码,当TL0038接收到38kHz红外信号时,输出端输出低电平,否则为高电平。所以红外遥控器发送红外信号时,参考上面遥控串行数据编码波形图,在低电平处发送38kHz红外信号,高电平处则不发送红外信号。
C51程序代码:
#include 《AT89X51.h》
static bit OP; //红外发射管的亮灭
static unsigned int count; //延时计数器
static unsigned int endcount; //终止延时计数
static unsigned char flag; //红外发送标志
char iraddr1; //十六位地址的第一个字节
char iraddr2; //十六位地址的第二个字节
void SendIRdata(char p_irdata);
void delay();
void main(void)
{
count = 0;
flag = 0;
OP = 0;
P3_4 = 0;
EA = 1; //允许CPU中断
TMOD = 0x11; //设定时器0和1为16位模式1
ET0 = 1; //定时器0中断允许
TH0 = 0xFF;
TL0 = 0xE6; //设定时值0为38K 也就是每隔26us中断一次
TR0 = 1;//开始计数
iraddr1=3;
iraddr2=252;
do{
delay();
SendIRdata(12);
}while(1);
}
//定时器0中断处理
void timeint(void) interrupt 1
{
TH0=0xFF;
TL0=0xE6; //设定时值为38K 也就是每隔26us中断一次
count++;
if (flag==1)
{
OP=~OP;
}
else
{
OP = 0;
}
P3_4 = OP;
}
void SendIRdata(char p_irdata)
{
int i;
char irdata=p_irdata;
//发送9ms的起始码
endcount=223;
flag=1;
count=0;
do{}while(count《endcount);
//发送4.5ms的结果码
endcount=117
flag=0;
count=0;
do{}while(count《endcount);
//发送十六位地址的前八位
irdata=iraddr1;
for(i=0;i《8;i++)
{
//先发送0.56ms的38KHZ红外波(即编码中0.56ms的低电平)
endcount=10;
flag=1;
count=0;
do{}while(count《endcount);
//停止发送红外信号(即编码中的高电平)
if(irdata-(irdata/2)*2) //判断二进制数个位为1还是0
{
endcount=41; //1为宽的高电平
}
else
{
endcount=15; //0为窄的高电平
}
flag=0;
count=0;
do{}while(count《endcount);
irdata=irdata》》1;
}
//发送十六位地址的后八位
irdata=iraddr2;
for(i=0;i《8;i++)
{
endcount=10;
flag=1;
count=0;
do{}while(count《endcount);
if(irdata-(irdata/2)*2)
{
endcount=41;
}
else
{
endcount=15;
}
flag=0;
count=0;
do{}while(count《endcount);
irdata=irdata》》1;
}
//发送八位数据
irdata=p_irdata;
for(i=0;i《8;i++)
{
endcount=10;
flag=1;
count=0;
do{}while(count《endcount);
if(irdata-(irdata/2)*2)
{
endcount=41;
}
else
{
endcount=15;
}
flag=0;
count=0;
do{}while(count《endcount);
irdata=irdata》》1;
}
//发送八位数据的反码
irdata=~p_irdata;
for(i=0;i《8;i++)
{
endcount=10;
flag=1;
count=0;
do{}while(count《endcount);
if(irdata-(irdata/2)*2)
{
endcount=41;
}
else
{
endcount=15;
}
flag=0;
count=0;
do{}while(count《endcount);
irdata=irdata》》1;
}
endcount=10;
flag=1;
count=0;
do{}while(count《endcount);
flag=0;
}
void delay()
{
int i,j;
for(i=0;i《400;i++)
{
for(j=0;j《100;j++)
{
}
}
}
上一篇:C51编程经验汇总分享
下一篇:基于AT89C51单片机和放大器实现音频信号均幅控制放大电路的设计
推荐阅读最新更新时间:2024-11-10 08:12
推荐帖子
- MPLAB® Harmony之学习篇(五)-- 认识Harmony驱动和系统服务
- 本文转自Microchip麦田论坛——作者:炮灰哥通过本文希望大家了解MPLAB®Harmony的框架的组成模块有一个系统的概念,并对驱动和系统服务有个初步的认识。本文讨论以下内容:Harmony的主要软件模组Harmony中的驱动Harmony中的系统服务Harmony框架如下图所示,主要由六类软件模块组成:PLIB–外设库,实现直接对外设寄存器的访问;
- 橙色凯 Microchip MCU
- msgQreceive返回error,表示什么?
- 是代表队列没数据,还是什么错误啊?一般都是什么引起的?多谢啊msgQreceive返回error,表示什么?
- gtongy 嵌入式系统
- Sense和Source之间的电压限制
- 正如前面对R/2R阶梯式网络介绍的那样,HI和LO端的source和sense之间都存在5V的最大电压限制。2400并不检查实际电压是否超过了这个限制,因为每条引线上的最大电阻必须是已知的。可以通过下列计算避免测量误差:Rmax是其中一条信号源引线上的最大引线电阻。例如,如果RSourceLO=200kΩ,RSource
- Jack_ma 测试/测量
- 【安信可BW16-Kit】OLED显示
- 本篇讲述在SDK上i2c驱动,实现oled显示。一.硬件准备oled模块---开发板VCC---3V3GND---GNDSDA---PA26SCL---PA25硬件连接如下图:图1:oled模块接线二.代码准备1.驱动源码oled.c,oled.h编写准备,并分别放在hp工程src_hp和inc_hp目录下,工程添加后如下图2:工程源码添加oled.c含oled初始化函数OLED_In
- dirty RF/无线
- verilog中初值定义
- 此内容由EEWORLD论坛网友郝旭帅原创,如需转载或用于商业用途需征得作者同意并注明出处在利用verilog进行开发时,往往需要对某些寄存器进行赋初值,下面根据笔者在设计中遇到的情况进行分析。例如下面是实现流水灯(4个led),代码如下:moduleledrun(inputwireclk,inputwirerst_n,out
- 郝旭帅 模拟电子
- TMC2208测试(问题排查)
- TMC2208测试报告目录1.问题回顾2.测试平台展示3.针对性测试3.1.待机电流特性3.2.匀速过程电流特性4.总结 问题回顾 在上次测试中,主要对待机电流,微步的平滑度,电机加速/减速等进行测试,通过测试出的相电流波形不难看出电流波形圆滑,也不存在畸变的情况。但是保持电流的波动较大,匀速阶段也存在些许毛刺。初步推断是由于杜邦线的原因,为此我又进行了测试。 测试平台展示 本次测试和上次使用的平台一样,唯一的区别是将对接杜邦线改为新的单
- m9wo 电机驱动控制(Motor Control)