遥控红外通信原理

在实际的通信领域,发出来的信号一般有较宽的频谱,而且都是在比较低的频率段分布大量的能量,所以称之为基带信号,这种信号是不适合直接在信道中传输的。为便于传输、提高抗干扰能力和有效的利用带宽,通常需要将信号调制到适合信道和噪声特性的频率范围内进行传输,这就叫做信号调制。在通信系统的接收端要对接收到的信号进行解调,恢复出原来的基带信号。
我们平时用到的红外遥控器里的红外通信,通常是使用38K左右的载波进行调制的
调制:就是用待传送信号去控制某个高频信号的幅度、相位、频率等参量变化的过程,即用一个信号去装载另一个信号。比如我们的红外遥控信号要发送的时候,先经过38K调制,如图1所示。

图1 红外信号调制

原始信号就是我们要发送的一个数据“0”位或者一位数据“1”位,而所谓38K载波就是频率为38K的方波信号,调制后信号就是最终我们发射出去的波形。我们使用原始信号来控制38K载波,当信号是数据“0”的时候,38K载波毫无保留的全部发送出去,当信号是数据“1”的时候,不发送任何载波信号。

图2 红外发射原理图

38K载波,我们可以用455K晶振,经过12分频得到37.91K,也可以由时基电路NE555来产生,或者使用单片机的PWM来产生。当信号输出引脚输出高电平时,Q2截止,不管38K载波信号如何控制Q1,右侧的竖向支路都不会导通,红外管L1不会发送任何信息。当信号输出是低电平的时候,那么38K载波就会通过Q1释放出来,在L1上产生38K的载波信号。这里要说明的是,大多数家电遥控器的38K的占空比是1/3,也有1/2的,但是相对少一些。
正常的通信来讲,接收端要首先对信号通过监测、放大、滤波、解调等等一系列电路处理,然后输出基带信号。但是红外通信的一体化接收头HS0038B,已经把这些电路全部集成到一起了,我们只需要把这个电路接上去,就可以直接输出我们所要的基带信号了

 图3 红外接收原理图

由于红外接收头内部放大器的增益很大,很容易引起干扰,因此在接收头供电引脚上必须加上滤波电容,官方手册给的值是4.7uF,我们这里直接用的10uF,手册里还要求在供电引脚和电源之间串联100欧的电阻,进一步降低干扰。

图3所示的电路,用来接收图16-5电路发送出来的波形,当HS0038监测到有38K的红外信号时,就会在OUT引脚输出低电平,当没有38K的时候,OUT引脚就会输出高电平。那我们把OUT引脚接到单片机的IO口上,通过编程,就可以获取红外通信发过来的数据了。

大家想想,OUT引脚输出的数据是不是又恢复成为基带信号数据了呢?那我们单片机在接收这个基带信号数据的时候,如何判断接收到的是什么数据,应该遵循什么协议呢?像我们前边学到的UART、I2C、SPI等通信协议都是基带通信的通信协议,而红外的38K仅仅是对基带信号进行调制解调,让信号更适合在信号中传输。

由于我们的红外调制信号是半双工的,而且同时空间只能允许一个信号源,所以我们红外的基带信号不适合在I2C或者SPI通信协议中进行的,我们前边提到过UART虽然是2条线,但是通信的时候,实际上一条线即可,所以红外可以在UART中进行通信。当然,这个通信也不是没有限制的,比如在HS0038B的数据手册中标明,要想让HS0038B识别到38K的红外信号,那么这个38K的载波必须要大于10个周期,这就限定了我们红外通信的基带信号的比特率必须不能高于3800,那如果把串口输出的信号直接用38K调制的话,波特率也就不能高于3800。

常用红外遥控器协议

一、 NEC 协议

特征:

8 位地址和 8 位命令长度为提高可靠性每次传输两遍地址(用户码)和命令(按键值)通过脉冲串之间的时间间隔来实现信号的调制 38Khz 载波每位的周期为 1.12ms 或者 2.25ms

Note:对于测试红外接收头的信号来说,有脉冲信号的地方就是高电平。即逻辑“1” 为 0.56ms 高电平+1.69ms 低电平,逻辑“0”为 0.56ms 高电平+0.56ms 低电平。

上图为典型的 NEC 协议传输格式,起始位(引导码)为 9ms 高+4.5ms 低组成,有效数据为地址+地址反码+命令+命令反码。反码的作用是用来校准前面的地址和命令,如果对可靠性不感兴趣,也可以去掉取反的数据,或者将地址和命令扩展到 16 位

上图传输的地址数据为 10011010,需要注意的是先发低位地址再发高位地址,因此该波形的地址为 01011001=0X59,同理,命令为 00010110=0X16。
长按键时,如下图所示,每隔 110ms 重复发送一次,但是命令只发送一次,重复发送的是 9ms 高电平+2.25ms 低电平+0.56ms 高电平+低电平

  扩展协议:
  
  扩展协议只是将地址改为 16 位,其他不变。

实测波形:

下面的波形是从红外接收头上得到的波形:(调制脉冲信号转变成高低电平了)
  
   由于红外接收头在接收信号时(或者是发送的时候)将波形反向了,因此在读数据时可以将示波器的反向功能打开,就能读到有效数据了。

下面实例是已知 NEC 类型遥控器所截获的波形:

遥控器的识别码是 Address=0xDD20;其中一个键值是 Command=0x0E
  

  rk3308:
MODULE_PARM_DESC 对模块的描述信息
module_param_named  加载模块可修改的参数查看模块信息:
modinfo  *.ko
parm:(MODULE_PARM_DESC中的信息)insmod *.ko module_param_named设置的变量=xxx
insmod hello.ko  watchdog=1000IR发射
compatible = "pwm-ir-tx"pwm_ir_probedevm_pwm_getdevm_rc_register_device(rc1)device_create_file(&rcdev->dev, &ir_attrs[i])debugfsduty_ratio_store duty_ratio_show pwm_ir->duty_cyclefrequency_store frequency_showpwm_ir->carriertransmit_store transmit_showpwm_ir_tx(rcd, patterns, index);echo 25000 > /sys/class/rc/rc1/frequency 频率
echo 90 > /sys/class/rc/rc1/duty_ratio 占空比
cat /tmp/ir_recv > /sys/class/rc/rc1/transmit 发送IR接收
compatible = "gpio-ir-receiver"gpio_ir_recv_probedevm_gpiod_getgpiod_to_irqdevm_rc_register_device(rc0)device_create_file(&rcdev->dev, &ir_attrs[i])gpio_ir_recv_irqgpiod_get_valueir_raw_event_store_edge波形   1T=560us 载波频率为:38kHz
1.头码由9ms高电平加4.5ms低电平表示
2.数据 “1” 由1T高电平加3T低电平表示数据“1”
3.数据 “0” 由1T高电平加1T低电平表示数据“0”

红外发射与调制信号

NE555

使用NE555产生38kHz, 占空比为1/3的方波信号。

产生方波的频率=0.693((RA+2RB)*C)
占空比=RB/(RA+2RB)
因为红外发射管最佳的占空比是1/3,C一般为0.01uf,所以计算之后RA=RB=1.2k

使用三极管来增强红外发射管的发射功率。


直径3mm 5mm为小功率红外发射管,正向电压: 1.1~1.5V,电流20mA
直径8mm为中功率红外发射管,正向电压: 1.4~1.65V,电流50~100mA
直径10mm为大功率红外发射管,正向电压: 1.51.9V,电流200350mA

以为中功率红外发射管为例:基极电流IB=(5v-0.7v-1.4v)/R1=2.9v/R1
三极管β为30,则发射极流过红外发射管的电流为(1+β)*IB=89.9mA

正常情况下IR一体化接收头未收到38kHz的调制信号时为管脚信号为高电平,收到38kHz的调制信号时管脚信号为低电平。

TIM3 PWM输出 4路

一、设置TIM3的GPIO为推挽输出

void TIM3_IOConfig(void)
{GPIO_InitTypeDef  GPIO_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB,ENABLE);//这里TIM3的通道1是GPIOA_Pin_6,通道2是GPIOA_Pin_7;通道3是GPIOB_Pin_0;//这里TIM3的通道4是GPIOB_Pin_1;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//使用PWM的功能需要设置成AF_PP模式GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7;GPIO_Init(GPIOA,&GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;GPIO_Init(GPIOB,&GPIO_InitStructure);
}二、设置TIM3的PWM1方式4路输出
//arr:自动重装值 psc:时钟预分频数
//定时器溢出时间计算:Tout=((arr+1)*(psc+1))/Ft
void TIM3_PWM_Init(void)
{TIM_TimeBaseInitTypeDef  TIM_TimeBaseInitStructure;//TIM定时器的配置TIM_OCInitTypeDef        TIM_OCInitStructure;//TIMPWM的RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);//使能TIM3的时钟线;//复位定时器TIM_DeInit(TIM3);//TIM_Prescaler:72,TIM_Period:20000周期为20msTIM_TimeBaseInitStructure.TIM_Prescaler = 72-1;                        //定时器周期TIM_TimeBaseInitStructure.TIM_Period = 20000-1;                        //定时器更新的重装载值0-65535TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);            //初始化TIM3      TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;TIM_OCInitStructure.TIM_Pulse = 1000;                                //预设值TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;    TIM_OC1Init(TIM3,&TIM_OCInitStructure);TIM_OC2Init(TIM3,&TIM_OCInitStructure);TIM_OC3Init(TIM3,&TIM_OCInitStructure);TIM_OC4Init(TIM3,&TIM_OCInitStructure);TIM_OC1PreloadConfig(TIM3,TIM_OCPreload_Enable);TIM_OC2PreloadConfig(TIM3,TIM_OCPreload_Enable);TIM_OC3PreloadConfig(TIM3,TIM_OCPreload_Enable);TIM_OC4PreloadConfig(TIM3,TIM_OCPreload_Enable);TIM_CtrlPWMOutputs(TIM3, ENABLE);
//        TIM_ITConfig(TIM3,TIM_IT_CC1|TIM_IT_CC2|TIM_IT_CC3|TIM_IT_CC4,ENABLE);//使能标志位CC1,CC2,CC3,CC4TIM_Cmd(TIM3,ENABLE);//使能TIM;
}
上面的TIM周期是20ms

三、修改PWM的方法

void TIM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare1);
void TIM_SetCompare2(TIM_TypeDef* TIMx, uint16_t Compare2);
void TIM_SetCompare3(TIM_TypeDef* TIMx, uint16_t Compare3);
void TIM_SetCompare4(TIM_TypeDef* TIMx, uint16_t Compare4);

STM32定时器初始化完成后修改频率问题

问题代码

void timer_PWM_init(TIM_TypeDef *TIMx, uint32_t tim_ch)
{LL_TIM_InitTypeDef TIM_InitStruct = {0};timer_clk_irq_config(TIMx, 0);TIM_InitStruct.Prescaler = 71;TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;TIM_InitStruct.Autoreload = 999;TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV1;LL_TIM_Init(TIMx, &TIM_InitStruct);LL_TIM_DisableARRPreload(TIMx);LL_TIM_OC_EnablePreload(TIMx, tim_ch);TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_PWM1;TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_DISABLE;TIM_OC_InitStruct.OCNState = LL_TIM_OCSTATE_DISABLE;TIM_OC_InitStruct.CompareValue = 500;TIM_OC_InitStruct.OCPolarity = LL_TIM_OCPOLARITY_HIGH;LL_TIM_OC_Init(TIMx, tim_ch, &TIM_OC_InitStruct);LL_TIM_OC_DisableFast(TIMx, tim_ch);LL_TIM_SetTriggerOutput(TIMx, LL_TIM_TRGO_RESET);LL_TIM_DisableMasterSlaveMode(TIMx);timer_io_init();
}
void timer_PWM_freq_set(TIM_TypeDef *TIMx, uint32_t tim_ch, uint16_t freq)
{LL_TIM_DisableCounter(TIMx);LL_TIM_SetPrescaler(TIMx, 7200-1); // 10KHzLL_TIM_SetAutoReload(TIMx, (uint16_t)((10000/freq))-1);
}

问题现象:
调用:timer_PWM_freq_set(TIM3, LL_TIM_CHANNEL_CH4, 1000)
PWM没问题,1KHz,占空比50%
但是,调用:timer_PWM_freq_set(TIM3,LL_TIM_CHANNEL_CH4,100)
出问题了,PWM一直是高定平。

问题解决

因为初始化PWM的频率为1K,计数周期为500,占空比为50%
但是在后面修改频率后并未修改计数周期,也就是频率为100HZ,计数周期还是500,计数周期大于频率了,于是就出现一直高电平的现象。(个人分析的,不知正确与否,应该是这样的。。。。)

解决方法:
修改频率的同时也要修改计数周期,要保证计数周期小于频率,不然占空比比周期还长,自然就一直高电平或者低电平了
代码修改

*** @brief  设置PWM输出频率* @param  fre取值范围(1-10000)Hz* @retval 设置完成频率之后不启动定时器*/
void timer_PWM_freq_set(TIM_TypeDef *TIMx, uint32_t tim_ch, uint16_t freq)
{LL_TIM_DisableCounter(TIMx);LL_TIM_SetPrescaler(TIMx, 7200-1); // 10KHzLL_TIM_SetAutoReload(TIMx, (uint16_t)((10000/freq))-1);LL_TIM_OC_SetCompareCH4(TIMx, (uint16_t)((10000/freq)/2))
}

抄写与 https://blog.csdn.net/yangyang_1024/article/details/82999694?ops_request_misc=&request_id=&biz_id=102&utm_term=红外信号&utm_medium=distribute.pc_search_result.none-task-blog-2allsobaiduweb~default-1-82999694

红外遥控实现回充原理红外发射与调制信号相关推荐

  1. 扫地机器人自动回充原理

    扫地机器人自动回充,有多种技术原理,有采用雷达定位的.采用蓝牙定位的.红外线定位的.在扫地机器人领域主要是利用这3种定位技术. 一.采用超声波定位 一种扫地机器人是采用超声波定位来寻找充电基座的,超声 ...

  2. 红外遥控C语言程序设计,光电红外遥控开关设计(光电系统课程设计)【PCB图仿真图单片机C语言分工心得】..doc...

    光电红外遥控开关设计(光电系统课程设计)[PCB图仿真图单片机C语言分工心得]. 本科生课程论文 论文题目光电红外遥控开关设计课程名称光电系统设计学生姓名学号所在学院所在班级指导教师 目 录 摘要3 ...

  3. c语言红外解码程序,[转载]红外遥控和C语言51红外遥控解码程序设计实例

    什么是红外线?人的眼睛能看到的可见光按波长从长到短排列,依次为红.橙.黄.绿.青.蓝.紫.其中红光的波长范围为0.62-0.76μm:比红光波长还长的光叫红外线.红外遥控在生产和生活中应用越来越广泛, ...

  4. 步进电机红外遥控C语言程序,单片机红外遥控+步进电机+1602液晶显示c语言源程序...

    这是我做的红外遥控控制的1602显示转速和转向的步进电机c语言原程序,已经调试成功!有些模块是以前做其他实验时做的,现在是直接调用的,包括1602液晶显示技术,TC9012红外解码,然后就是四相步进电 ...

  5. 红外遥控c语言,NEC协议红外遥控器

    家电遥控器通信距离往往要求不高,而红外的成本比其它无线设备要低的多,所以家电遥控器应用中红外始终占据着一席之地.遥控器的基带通信协议很多,大概有几十种,常用的就有 ITT 协议.NEC 协议.Shar ...

  6. android如何编程红外遥控,全志A20[android教程]-红外遥控器调试

    第1章 前言 要在树莓派3 (A20)android系统上实现红外遥控器,因为super3上有红外接收头,所以可以直接支持红外遥控器,可以把树莓派3应用到家庭控制当中.当作网络机顶盒用 树莓派3的 ...

  7. STM32F103红外遥控密码锁

    红外接收,在学习MCU中比较常见的一个功能,生活中使用的地方也比较多 相关方面的程序 野火官网.正点原子官网都有,这两个网站的资料也比较丰富(开源) 正点原子官网:http://www.aliente ...

  8. 【STC单片机学习】高级外设和项目篇二:红外遥控

    [朱老师课程总结 侵删] 第一部分.章节目录 2.2.1.红外遥控背景知识 2.2.2.原理图电路分析 2.2.3.NEC协议讲解 2.2.4.官方示例代码解析 2.2.5.红外接收程序的移植和调试1 ...

  9. 基于STM32F103ZET6红外遥控制步进电机

    基于STM32F103ZET6红外遥控制步进电机# 基于STM32F103ZET6红外遥控制步进电机 希望在学习的过程中留下一点点记录,以便以后自己方便查找,特意再次留下记录. 第一次发博文写的不好请 ...

最新文章

  1. java timer和timertask_Timer和TimerTask与Java中的线程+睡眠
  2. 网络请求可以返回数据的网站_实例解析|Python加解密VIP网站反爬请求头实现数据爬取...
  3. Python Django设置中文语言及时区
  4. python turtle库setpos_Python内置海龟(turtle)库绘图命令详解(二)
  5. 梅森增益matlab求解,梅森公式互不接触回路及其增益
  6. CreateThread函数
  7. linux通过I2C地址查看设备名称
  8. JAVA只要掌握内部类,多继承和单继承都不是问题
  9. linux 虚拟文件系统 源码,Linux内核源代码情状分析-虚拟文件系统
  10. gps l1带宽_民用GPS接收机可达到的最高更新速率是多少?
  11. 外部中断器微型计算机课程设计,课程设计-电子时钟参考.doc
  12. 配置vscode作为STM32代码的编辑器(替代keil5)。实现:代码自动补全, 编译,下载。nRF52也可以编译。
  13. d3开发Svg编辑器
  14. 抖音培训教程,抖音培训班,抖音培训课程(2019实时更新中) -
  15. URAL1389. Roadworks(dp)
  16. 移动端touch触屏滑动事件、滑动触屏事件监听!
  17. 颜色的搭配适用,摘自某论坛
  18. 评:日本的“泡沫”代价
  19. 他儿子就这样娶到了比尔·盖茨的女儿
  20. 【数据结构】无向图与有向图的连通性及相关算法

热门文章

  1. 计算机维护bios设置u盘启动不了,U盘装系统一直显示正在启动大白菜pe维护系统,就是进不去...
  2. 在领英(LinkedIn)开发客户,及时撤销没通过的添加好友邀请竟这么重要...
  3. vuepress 热更新失效研究
  4. 读书:《人生的底气》
  5. 组图:2004雅典奥运会
  6. LeetCode137.只出现一次的数字
  7. Java 包装类型的缓存机制
  8. Seetaface 02 Seetaface AndroidStudio Demo
  9. 危害女性健康的四种常见职业
  10. CSS_03_盒子模型