红外发射管有2个管脚,发送的是经过38KHz时钟调制过的信号。例如下图使用PWM产生一个等占空时钟信号用于调制。

接收管收下来的信号已经经过了解调,可以直接连接系统的外部中断脚。

下面通过逻辑分析仪来实际测量一下。

随便找了个红外遥控器,测量power键按下后的波形。首先是信号发送侧。

可以看到,0秒开始是一个按键动作,0.11秒后的那个波形是一个repeat,展开:

把波形重叠的部分展开,就可以看到这个38KHz的调制时钟

如果持续按下遥控器上的按键,那么就会发送连续的repeat信号,发送的间隔也基本满足协议上指出的110ms

下面是接收侧

可以看到信号被解调了,也就是说重叠的部分变成了低电平。

最后通过编写协议分析插件的方式,来描述如何通过程序来理解上面的波形。

我使用的逻辑分析软件是 Saleae Logic 1.1.15 编译环境是Microsoft Visual Studio 2008,编译时需要SaleaeAnalyzerSdk-1.1.14。完整的代码从这里下载。

分析的方法是测量两个信号下降沿之间的时间长度,这里先定义一些时间参数。

  1. /* Timing define , unit : ms*/
  2. #defineSTART_LOW_TIMING
  3. #defineSTART_HIGH_TIMING
  4. #defineREPEAT_HIGH_TIMING
  5. #defineLOGIC_ONE_TIMING (*)
  6. #defineLOGIC_ZERO_TIMING (*)
  7. #defineDATA_LOW_TIMING (*) 

从逻辑分析仪测量的结果,可以发现发射器给出的信号并不是非常的精确,所以我们需要定义误差范围。

  1. /* Timing Margin , unit : ms*/
  2.  
  3. #definedelta  

在下面代码里,通过API函数AdvanceToNextEdge来获取下一个信号发生变化的采样点,如果对应的采样点是低电平,则表示下跳沿,这时和前一个下跳沿采样点的时间做差,按照采样频率换算成时间间隔。再根据上面定义的时间常量来判断这是一个START标记、REPEAT标记、逻辑1、逻辑0还是无效的信号。对于逻辑1和0的情况,需要通过移位来整理成32bits的有效数据,这里要特别注意协议里规定,先发送的是LSB,后发送的MSB。

  1. void IRNECAnalyzer::WorkerThread()
  2. {
  3. U64 per_sample = ;
  4. U64 cur_sample = ;
  5. U64 starting_sample = ;
  6. U64 differ = ;
  7. char action = state_down;
  8. U8 fail = ;
  9. U64 data = ;
  10. U32 code = ;
  11. U8 count = ;
  12. U8 data_f = ;
  13.  
  14. mResults.reset(new IRNECAnalyzerResults( this, mSettings.get() ) );
  15. SetAnalyzerResults(mResults.get() );
  16. mResults->AddChannelBubblesWillAppearOn(mSettings->mInputChannel );
  17. mSampleRateHz= GetSampleRate();
  18. mSerial= GetAnalyzerChannelData( mSettings->mInputChannel );
  19.  
  20. if( mSerial->GetBitState() == BIT_LOW )
  21. mSerial->AdvanceToNextEdge();
  22.  
  23. for(;;){
  24. mSerial->AdvanceToNextEdge();
  25. cur_sample= mSerial->GetSampleNumber();
  26.  
  27. //只处理时钟的下跳沿
  28.  
  29. if(mSerial->GetBitState() == BIT_LOW){
  30. differ= (cur_sample - per_sample)*/mSampleRateHz;
  31.  
  32. //判断是否是REPEAT信号
  33. )<differ)&&((differ)<(START_LOW_TIMING+REPEAT_HIGH_TIMING+delta*))){
  34. action=state_repeat;
  35. fail=;
  36. })<differ)&&((differ)<(START_LOW_TIMING+START_HIGH_TIMING+delta*))){
  37. //判断是否是START信号
  38. action=state_start;
  39. fail=;
  40. }else
  41. )<differ)&&((differ)<(LOGIC_ONE_TIMING+DATA_LOW_TIMING+delta*))){
  42. //判断是否是逻辑1
  43. action=state_data;
  44. data_f=;
  45. fail=;
  46. })<differ)&&((differ)<(LOGIC_ZERO_TIMING+DATA_LOW_TIMING+delta*))){
  47. //判断是否是逻辑0
  48. action=state_data;
  49.  
  50. data_f=;
  51.  
  52. fail=;
  53.  
  54. }else{
  55. //否则为错误信号
  56. fail=;
  57. }
  58.  
  59. ){
  60. switch (action){
  61. case state_start:
  62. code= ;
  63. count= ;
  64. starting_sample= cur_sample;
  65. AddFrame(per_sample,cur_sample, , FStart);
  66. break;
  67.  
  68. case state_data:
  69. data_f= data_f << count;
  70. code= code | data_f;
  71. count++;
  72. ){
  73. count= ;
  74. AddFrame(starting_sample,cur_sample, code, FData);
  75. code= ;
  76. starting_sample= cur_sample;
  77. }
  78. break;
  79.  
  80. case state_repeat:
  81. AddFrame(per_sample,cur_sample, data, FRepeat);
  82. break;
  83.  
  84. default:
  85. break;
  86. }
  87.  
  88. }
  89. per_sample= cur_sample;
  90.  
  91. }
  92.  
  93. }
  94.  
  95. }

加载上面的插件后,可以看到分析的结果

所以如果将这份代码放在板卡上运行,首先应该将接收器的信号接到处理器的外部中断管脚,然后注册一个下跳沿触发的快速中断。然后最通常的情况你需要再注册一个标准的输入设备,映射一下遥控器码字和按键事件的对应关系就可以了。

NEC红外遥控协议理解与实现的更多相关文章

  1. 基于STM32的红外遥控重点解析

    本文有两个内容:一.红外遥控协议的的讲解:二.解码程序解析(参考正点原子的代码) 红外的介绍.优点.缺点就不给大家说了,进入正题 一.红外遥控协议的的讲解 红外遥控的编码目前广泛使用的是:NEC Pr ...

  2. 红外遥控NEC协议使用总结

    最近做了一个调试红外遥控三色灯的实习,花了一个多月的时间研究基于NEC协议的红外遥控,下面是这次实习技术方面的总结. 一.NEC协议特征: 8位地址和8位命令长度 每次传输两遍地址(用户码)和命令(按 ...

  3. 46.Linux-分析rc红外遥控平台驱动框架,修改内核的NEC解码函数BUG(1)

    内核版本          :  Linux 3.10.14 rc红外接收类型:  GPIO 类型的NEC红外编码 本章内容 1) rc体系结构分析 2) 分析红外platform_driver平台驱 ...

  4. 基于FPGA的红外遥控解码与PC串口通信

    基于FPGA的红外遥控解码与PC串口通信 zouxy09@qq.com http://blog.csdn.net/zouxy09 这是我的<电子设计EDA>的课程设计作业(呵呵,这个月都拿 ...

  5. 基于Arduino的红外遥控

    1.红外接收头介绍  一.什么是红外接收头?  红外遥控器发出的信号是一连串的二进制脉冲码.为了使其在无线传输过程中免受其他红外信号的干扰,通常都是先将其调制在特定的载波频率上,然后再经红外发射二极管 ...

  6. 基于Arduino、STM32进行红外遥控信号接收

    catalogue . 遥控器原理简介 . 红外遥控原理 . 常见红外遥控器红外线信号传输协议 . 遙控器的发展 . 实验过程 . 攻击面 . 基于STM32实现红外信号解码 1. 遥控器原理简介 0 ...

  7. 46.Linux-创建rc红外遥控平台设备,实现重复功能(2)

    上章链接:46.Linux-分析rc红外遥控平台驱动框架,修改内核的NEC解码函数BUG(1) 在上章分析了红外platform_driver后,已经修改bug后,接下来我们自己创建一个红外platf ...

  8. 玩转X-CTR100 l STM32F4 l 红外遥控接收

    我造轮子,你造车,创客一起造起来!塔克创新资讯[塔克社区 www.xtark.cn ][塔克博客 www.cnblogs.com/xtark/ ]      X-CTR100控制器具有红外接收头,例程 ...

  9. STM32之红外遥控信号自学习实现

    一.序言 很早前就想实现这个红外遥控自学习的这个实验,用于来自己控制房子里如空调等红外遥控设备的自动化,NEC的标准到具体的产品上可能就被厂家定义为不一样了,所以自学习就应该是接收到什么就发送什么,不 ...

随机推荐

  1. HttpClient post中文乱码解决

    在javase方式下使用HttpClient没有进行任何编码设置,本地从服务端获取到数据不存在中文乱码. 但是将此段代码部署到Tomcat下面出现了中文乱码,此时设置: post.getParams( ...

  2. LUN----逻辑单元号

    LUN的全称是Logical Unit Number,也就是逻辑单元号.   一.概念   LUN的全称是Logical Unit Number,也就是逻辑单元号.我们知道SCSI总线上可挂接的设备数 ...

  3. 在WPF中使用ArcGIS Engine

    原文 http://blog.csdn.net/zzahkj/article/details/9102621 1.首先,新建一个WPF项目,添加引用ESRI.ArcGIS.AxControls.ESR ...

  4. [ASP.NET] 图形验证码破解-以简单图形为例

    原文 http://www.dotblogs.com.tw/joysdw12/archive/2013/06/08/captcha-cracked.aspx 前言 这次来讲个比较有趣的主题,就是该如何 ...

  5. 安装GeoIP数据库

    1.安装GeoIP数据库 cd /usr/local/logstash/etc curl -O "http://geolite.maxmind.com/download/geoip/data ...

  6. Floyd 无向图模板

    这是无向图的 void Floyd() { memset(v, 0x3f, sizeof v); ; i <= n; i++) ; j <= n; j++) v[i][j] = map[i ...

  7. Python基础教程学习(四)类的创建与继承

    类中可以有方法,类外也可以有函数,其实类就是一种封装, Python中可以自己定义一个函数,一可以把这个函数在类中封装成一个方法, 其中的属性和方法自然就从父类中继承来了, 要想获得多个类的属性和功能 ...

  8. bash及其特性(笔记)

    bash及其特性:shell: 外壳GUI:Gnome, KDE, XfceCLI: sh, csh, ksh, bash, tcsh, zsh root, student程序:进程 进程:在每个进程 ...

  9. MapReduce程序依赖的jar包

    难得想写个mapreduce程序.发现已经不记得须要加入那些jar包了,网上找了一会也没发现准确的答案.幸好对hadoop体系结构略知一二.迅速试出了写mapreduce程序须要的五个jar包. 不多 ...

  10. MFC多线程内存泄漏问题&amp;解决方法

    在用visual studio进行界面编程时(如MFC),前台UI我们能够通过MFC的消息循环机制实现.而对于后台的数据处理.我们可能会用到多线程来处理. 那么对于大多数人(尤其是我这样的菜鸟),一个 ...