1.器件简介

本次测试采用R903V1红外接收头与NEC协议的红外遥控器,接收头原理图如下:

器件的供电电压VCC在2.7V~5.5V之间,输出电压VOUT正常在0.2v ~(VCC-0.3±0.2)v,注意高低电平输出脉冲宽度最小都在400us~800us之间。

NEC 码的位定义:一个脉冲对应 560us 的连续载波,一个逻辑 1 传输需要 2.25ms(560us脉冲+1680us 低电平),一个逻辑 0 的传输需要 1.125ms(560us 脉冲+560us 低电平)。而遥控接收头在收到脉冲的时候为低电平,在没有脉冲的时候为高电平,这样,我们在接收头端收到的信号为:逻辑 1 应该是 560us 低+1680us 高,逻辑 0 应该是 560us 低+560us 高。同时NEC码还规定了连发码由 9ms 低电平+2.5m 高电平+0.56ms 低电平+97.94ms 高电平组成。

NEC协议的数据格式:同步码头、地址码、地址反码、控制码、控制反码。同步码由一个 9ms 的低电平和一个 4.5ms 的高电平组成,地址码、地址反码、控制码、控制反码均是8 位数据格式。正常是按照低位在前,高位在后的顺序发送,但是我测试是按照高位在前,低位在后的。采用反码是为了增加传输的可靠性(可用于校验)。

2.硬件连接

简单说一下,红外接收器电压接3.3V,VOUT连接单片机的输入捕获引脚即可。

3.软件设计

使用stm32单片机的输入捕获功能,用stm32cubemx进行设置:

这里解释主要参数部分:

prescaler:预分频器,设置为72

counter mode:计数模式,设为向上计数

counter period:计数周期,设为65535

polarity selection:边沿检测方式,选择下降沿检测,程序里还是改成了上升沿检测

input filter:输入滤波器,设为8,是指连续采集到8个一样的高/低电平才计作高/低电平

在MAIN中开启输入捕获:

  while (1)
{
switch(cap_state)
{
case 0:
__HAL_TIM_SET_CAPTUREPOLARITY(&htim5, TIM_CHANNEL_2, TIM_INPUTCHANNELPOLARITY_RISING);//设置为上升沿捕获
HAL_TIM_IC_Start_IT(&htim5, TIM_CHANNEL_2); //启动输入捕获
cap_state++;
break;
} /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */
}

main循环体

/定时器输入捕获中断回调函数
void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)//捕获中断发生时执行
{
if(htim->Instance==TIM5)
{
switch(cap_state)
{
case 1:
rise_value[rise_i] = HAL_TIM_ReadCapturedValue(&htim5,TIM_CHANNEL_2);//取上升沿时刻
__HAL_TIM_SET_CAPTUREPOLARITY(&htim5,TIM_CHANNEL_2,TIM_ICPOLARITY_FALLING); //设置为下降沿捕获
cap_state++;
break;
case 2:
fall_value[fall_i] = HAL_TIM_ReadCapturedValue(&htim5,TIM_CHANNEL_2);//取下降沿时刻
if(fall_value[fall_i] > rise_value[rise_i])
{
rising_time[t_i] = fall_value[fall_i] - rise_value[rise_i];//高电平时间获取
}else
{
rising_time[t_i] = 65535 + fall_value[fall_i] - rise_value[rise_i];//高电平时间获取
}
//控制码
if(same_i == 1)
{
cnt+=1;
//地址码1~8
if(t_i<8)
{
if(rising_time[t_i]>500&&rising_time[t_i]<600)
{
address <<= 1;
address+=0;
}else if(rising_time[t_i]>1500&&rising_time[t_i]<1800)
{
address <<= 1;
address+=1;
}
}else if(t_i>16&&t_i<25)//控制码17~24
{
if(rising_time[t_i]>500&&rising_time[t_i]<600)
{
rec <<= 1;
rec+=0;
}else if(rising_time[t_i]>1500&&rising_time[t_i]<1800)
{
rec <<= 1;
rec+=1;
}
}
t_i++;
rise_i++;
fall_i++;
}
//同步码
if(rising_time[t_i]>4300&&rising_time[t_i]<4700)
{
//同步码正确
same_i=1;
t_i++;
rise_i++;
fall_i++;
}
HAL_TIM_IC_Stop_IT(&htim5,TIM_CHANNEL_2); //停止捕获
cap_state=0;
break;
}
}
}

输入捕获中断回调

//按键中断回调
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
//KEY0
if (GPIO_Pin==GPIO_PIN_5)
{
HAL_Delay(20);/* 延时一小段时间,消除抖动 */
if (HAL_GPIO_ReadPin(GPIOC,GPIO_PIN_5)==0)
{
HAL_GPIO_TogglePin(GPIOA,GPIO_PIN_8); for(int i=0;i<=fall_i;i++)
{
if(fall_value[i] > rise_value[i])
{
rising_time[i] = fall_value[i] - rise_value[i];//高电平时间获取
}else
{
rising_time[i] = 65535 + fall_value[i] - rise_value[i];//高电平时间获取
printf("hear:%d\r\n",i);
}
printf("高电平[%d]:%d us 下降沿时刻[%d]:%d us 上升沿时刻[%d]:%d us\r\n",i,rising_time[i],i,fall_value[i],i,rise_value[i]);
} printf("地址:%d 控制码:%lld\r\n",address,rec);
rise_i=fall_i=same_i=t_i=cnt=address=rec=0;
}
__HAL_GPIO_EXTI_CLEAR_IT(GPIO_PIN_5);
}
}

按键中断回调

大致说一下程序流程,在main中循环开启上升沿检测输入捕获功能(case 0),接着在输入捕获中断回调里记下上升沿触发时刻(rise_value),并改为下降沿检测捕获,再记下下降沿触发时刻(fall_value)并关闭捕获功能。

高电平持续时间(rising_time) = 下降沿触发时刻(fall_value) - 上升沿触发时刻(rise_value)

获取到一系列高电平以后(看示波器图),根据高电平持续时间进行数值移位换算,可以得到对应的地址码和控制码,串口打印如下:

打印结果中的hear是下降沿时刻溢出的情况,计算时需要在下降沿时刻基础上再加65535才可以。

感兴趣的可以试一试,需要原工程的留言我发。

stm32与红外遥控器(NEC协议)的更多相关文章

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

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

  2. 手把手教你DIY尼康ML-L3红外遥控器

    项目介绍 ML-L3是用于尼康部分型号相机的无线红外遥控器,可以通过红外方式来控制快门的释放,支持B门拍摄.官方售价100RMB左右,山寨版售价10RMB左右.虽然也能实现基本的遥控功能,但是功能还是 ...

  3. 红外NEC协议

    注意: 用示波器在接收头抓的电平看起来和NEC协议刚好相反, 那是因为:HS0038B 这个红外一体化接收头,当收到有载波的信号的时候,会输出一个低电平,空闲的时候会输出高电平. 具体情况,具体分析. ...

  4. 红外协议之NEC协议

    NEC协议载波:38khz 其逻辑1与逻辑0的表示如图所示: 逻辑1为2.25ms,脉冲时间560us:逻辑0为1.12ms,脉冲时间560us.所以我们根据脉冲时间长短来解码.推荐载波占空比为1/3 ...

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

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

  6. NEC协议

    注意: 用示波器在接收头抓的电平看起来和NEC协议刚好相反, 那是因为:HS0038B 这个红外一体化接收头,当收到有载波的信号的时候,会输出一个低电平,空闲的时候会输出高电平. 具体情况,具体分析. ...

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

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

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

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

  9. IRM3800 红外遥控器解码 linux驱动

    这一次还是接在 Cemera 上.用 中断引脚 EINT20 也就是 GPG12. 之前焊的 51 板子上有一个红外接收器. 请注意了,是 标准的 NEC 码规范:首次发送的是9ms的高电平脉冲,其后 ...

随机推荐

  1. zookeeper(5) 客户端

    zookeeper客户端主要负责与用户进行交互,将命令发送到服务器,接收服务器的响应,反馈给用户.主要分为一下三层: 用户命令处理层 用户命令处理层的功能是读取用户输入的命令,解析用户命令和输入参数, ...

  2. 容器云平台No.2~kubeadm创建高可用集群v1.19.1

    通过kubernetes构建容器云平台第二篇,最近刚好官方发布了V1.19.0,本文就以最新版来介绍通过kubeadm安装高可用的kubernetes集群. 市面上安装k8s的工具很多,但是用于学习的 ...

  3. Oracle学习(十七)数据库锁在分布式系统里的应用(老板,乐观锁了解一下?~)

    一.引言 随着业务量的增加,单机部署已经无法满足日常需求了,我们可能会把代码部署到多台服务器上去来进行服务的扩容,也就是负载均衡,那在这种场景下,怎么能实现锁的概念呢? 那么我们知道如果是一台主机部署 ...

  4. 分享一些好用的 Chrome 插件!

    使用浏览器扩展程序可以使你的工作效率提高数倍不止,那么下面我就向大家分享一下我日常使用的扩展,可能大多数扩展大家都已经在使用了,不过也难免有一两个是你不知道的. 以下排名并不分先后,请坚持看到最后,或 ...

  5. SpringMVC执行流程和原理

    SpringMVC流程: 01.用户发送出请求到前端控制器DispatcherServlet. 02.DispatcherServlet收到请求调用HandlerMapping(处理器映射器). 03 ...

  6. Oracle添加键值对盲注

    前言 遇到一种注入点,存在于POST参数中,却不能用sqlmap扫出: 分析 request参数格式: %24Q_value1=test1&orderCol=&order=+ASC+& ...

  7. 《Duubo系列》-Dubbo服务暴露过程

    我今天来就带大家看看 Dubbo 服务暴露过程,这个过程在 Dubbo 中其实是很核心的过程之一,关乎到你的 Provider 如何能被 Consumer 得知并调用. 今天还是会进行源码解析,毕竟我 ...

  8. MySQL手注之报错注入

    报错注入: 指在页面中没有一个合适的数据返回点的情况下,利用mysql函数的报错来创造一个显位的注入.先来了解一下报错注入常用的函数 XML:指可扩展标记语言被设计用来传输和存储数据. concat: ...

  9. tomcat源码--springboot整合tomcat源码分析

    1.测试代码,一个简单的springboot web项目:地址:https://gitee.com/yangxioahui/demo_mybatis.git 一:tomcat的主要架构:1.如果我们下 ...

  10. 使用JWT登录生成token

    package com.example.demo.util; import com.auth0.jwt.JWT; import com.auth0.jwt.JWTVerifier; import co ...