OLED显示红外遥控键码

2024-02-16 09:28
文章标签 显示 oled 红外 遥控 键码

本文主要是介绍OLED显示红外遥控键码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

基本原理

本遥控器的编码是NEC编码,为PWM(脉冲宽度调制)。
发射红外载波的时间固定,通过改变不发射载波的时间来改变占空比。

  • 逻辑“0”是由0.56ms的38KHZ载波和0.560ms的无载波间隔组成;
  • 逻辑“1”是由0.56ms的38KHZ载波和1.685m 的无载波间隔组成;
  • 结束位是0.56ms的38K载波。
  • 重复码由9ms红外脉冲和2.25ms的无红外脉冲以及560us的红外脉冲组成。

若发了一次命令码之后,一直按住遥控器按键并未松手,遥控器发射端将不会再发送命令码,而是每隔 110ms 时间,发送一段重复码。
image.png

过程分析

初始化部分

通用配置过程:

  1. 将相关的gpio和硬件挂载到RCC
  2. 初始化GPIO
  3. 初始化TIM时基单元
  4. 初始化IC输入捕获
  5. 初始化NVIC中断管理
  6. 实现中断处理函数
  7. 启用中断
  8. 启动定时器

image.png
TIM3的CH1通道位于PA6引脚。

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IPU;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
GPIO_InitStructure.GPIO_Speed =GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);

配置时基单元。arr和psc作为函数参数传入,支持动态调整。
在本文中,分频系数为72,这样标准周期就是1us,方便后续计算。

TIM_InternalClockConfig(TIM3);
TIM_TimeBaseInitTypeDef	TIM_TimeBaseStructure;
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseStructure.TIM_Period = arr - 1;
TIM_TimeBaseStructure.TIM_Prescaler = psc - 1;
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;
TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);

配置输入捕获。将TIM_ICInitStructure.TIM_ICFilter设置为0xf是为了对输入捕获信号进行滤波,滤波时间为8个时钟周期。

  • 对应16个采样点的平均,意味着最终捕获值是16个相邻采样值的平均值。
  • 这相当于对输入信号进行8个时钟周期的滤波,因为每个采样周期占用一个时钟周期。
TIM_ICInitTypeDef	TIM_ICInitStructure;
TIM_ICInitStructure.TIM_Channel = TIM_Channel_1;
TIM_ICInitStructure.TIM_ICFilter = 0xf;
TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Falling;
TIM_ICInitStructure.TIM_ICPrescaler = TIM_CKD_DIV1;
TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
TIM_ICInit(TIM3,&TIM_ICInitStructure);

配置中断优先级。中断优先级的配置方案需要在主函数中设置:NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

NVIC_InitTypeDef	NVIC_InitStructure;
NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority =1;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
NVIC_Init(&NVIC_InitStructure);

在之前的《PWM输入/输出》中,输入捕获采用的是事件方式:

TIM_SelectInputTrigger(TIM3,TIM_TS_TI1FP1);
TIM_SelectSlaveMode(TIM3,TIM_SlaveMode_Reset);

当时没有设置中断,设置了事件。事件由硬件自动完成,不需要我们实现中断处理函数,只需要指定处理方式即可。
我们这次用不到事件。
涉及各种码的判断,以及信号处理阶段的判断,硬件事件难以处理这种复杂任务,需要通过中断实现。
启用中断标志位,在TIM3的CC1通道出现下降沿时,自动触发TIM_IT_CC1中断:

TIM_ITConfig(TIM3,TIM_IT_CC1,ENABLE);

配置完毕,启动TIM定时器:

TIM_Cmd(TIM3,ENABLE);

中断处理函数

通过输入捕获,判断两个下降沿之间的时间间隔:

  • 引导码:13.5ms
  • 重复码:11.25ms
  • 逻辑0:1.12ms
  • 逻辑1:2.245ms

将输入捕获通道的GPIO设置为上拉输入。在没有外界信号时会自动上拉到高电平。
时间间隔通过TIM_GetCounter(TIM3)获取。TIM定时器设置的标准周期为1us,返回值即为下降沿之间的时间间隔,单位为us。
获取之后通过TIM_SetCounter(TIM3,0);设置寄存器值为0,相当于重新开始计数。由于一个周期的开始和结束都是下降沿,所以上一个周期的结束接着就是下一个周期的开始,计数值误差可以接受。
操作类似于一个状态机:

  • 初始处于空闲状态,TIM定时器的值是无意义的。在第一个下降沿到来时,设置TIM寄存器值为0,开始计数,进入准备状态。
  • 准备状态下,第二个下降沿到来时,根据时长判断是引导码还是重复码。重复码则回到空闲状态。引导码还需要读取具体的指令,进入接受状态。
  • 接收状态下,逐位接收,共32位。接收结束后回到空闲状态。等待下一个响应。

为什么引导码和重复码在同一个状态下判断?

对于上面四种码的时间间隔,很明显:引导码与重复码相近,逻辑0和逻辑1相近。
如果同一个阶段去判断时间间隔差距过大的码,可能造成数据遗漏。

如何按位接收?

需要用到指针,通过模运算、位运算实现。
指令一共有32位,需要一个char[4]类型的数组。
创建一个指针,其实是整型,

  • 对8做除法,用于对指定下标的char赋值
  • 对8取模之后,值介于[0,7],为指定位赋值

这种做法在存储器扩展、cache映射中都有应用,映射的常用方式就是取模。
如果学过,应该不会陌生。

数据校验

NEC编码的数据包括:
8位地址码+8位地址码的反码+8位命令码+8位命令码的反码
验证操作就是将反码部分按位取反,判断是否相等。

Addr_Temp = ~IR_DATA[1];
Cmd_Temp  = ~IR_DATA[3];
if (IR_DATA[0] == Addr_Temp && IR_DATA[2] == Cmd_Temp)  //数据验证
{IR_Address = IR_DATA[0];IR_Command = IR_DATA[2];IR_DataFlag = 1;
}

代码实现

到目前,需要创建如下变量:

  • uint8_t IR_State:记录当前状态:空闲/准备/接收
  • uint16_t IR_Time:下降沿间隔时间:引导/重复/0/1
  • uint8_t IR_RepeatFlag,IR_DataFlag:数据状态:重复/有数据
  • uint8_t IR_DATA[4]:数据:地址码+地址码反码+命令码+命令码反码
  • uint8_t IR_pData:指针:对IR_DATA进行段选和位选
void TIM3_IRQHandler()
{uint8_t Addr_Temp,Cmd_Temp;if (TIM_GetITStatus(TIM3,TIM_FLAG_CC1) != RESET){if (IR_State == 0)     //空闲态{TIM_SetCounter(TIM3,0);IR_State = 1;}else if (IR_State == 1) //准备态{IR_Time = TIM_GetCounter(TIM3);if (IR_Time > 13500 - 500 && IR_Time < 13500+500)  //Start信号{IR_State = 2;}else if (IR_Time > 11250 - 500 && IR_Time < 11250+500)  //Repeat信号{IR_State = 0;IR_RepeatFlag = 1;}else{}TIM_SetCounter(TIM3,0);}else if (IR_State == 2) //接收态{IR_Time = TIM_GetCounter(TIM3);if (IR_Time > 1120 -500 && IR_Time < 1120+500)   //逻辑0{IR_DATA[IR_pData/8] &= ~(0x01 << (IR_pData %8));IR_pData++;}else if (IR_Time > 2250 -500 && IR_Time < 2250 + 500) //逻辑1{IR_DATA[IR_pData/8] |= (0x01 << (IR_pData %8));IR_pData++;}else {IR_pData = 0;IR_State = 1;}if (IR_pData>=32)   //如果接收完32位数据{IR_pData = 0;Addr_Temp = ~IR_DATA[1];Cmd_Temp  = ~IR_DATA[3];if (IR_DATA[0] == Addr_Temp && IR_DATA[2] == Cmd_Temp)  //数据验证{IR_Address = IR_DATA[0];IR_Command = IR_DATA[2];IR_DataFlag = 1;}IR_State = 0;}TIM_SetCounter(TIM3,0);}else{}TIM_ClearITPendingBit(TIM3,TIM_IT_CC1);}
}

实验效果

飞线

红外部分并不直接与PA6相连,需要飞线。
用杜邦线连接IF和TIM3的CH1所在的PA6即可。
image.png

在主函数中调用

uint8_t Address;
uint8_t Command;
uint8_t Num;
int main(void)
{NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);Remote_Init(30000,72);OLED_Init();OLED_ShowString(1,1,"ADDR CMD NUM");OLED_ShowString(2,1,"00   00  000");while(1){if (IR_GetDataFlag() || IR_GetRepeatFlag()){Address = IR_GetAddress();Command = IR_GetCommand();OLED_ShowHexNum(2,1,Address,2);OLED_ShowHexNum(2,6,Command,2);if (Command == IR_VOL_ADD){Num++;}if(Command == IR_VOL_MINUS){Num--;}OLED_ShowNum(2,10,Num,3);}}
}

VID_20240215_014048

参考

  • STM32F103C8T6引脚定义.xlsx
  • stm32 使用说明+笔记(必读).pdf
  • 32版开发板原理图.pdf

这篇关于OLED显示红外遥控键码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/714145

相关文章

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

安卓链接正常显示,ios#符被转义%23导致链接访问404

原因分析: url中含有特殊字符 中文未编码 都有可能导致URL转换失败,所以需要对url编码处理  如下: guard let allowUrl = webUrl.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {return} 后面发现当url中有#号时,会被误伤转义为%23,导致链接无法访问

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

lvgl8.3.6 控件垂直布局 label控件在image控件的下方显示

在使用 LVGL 8.3.6 创建一个垂直布局,其中 label 控件位于 image 控件下方,你可以使用 lv_obj_set_flex_flow 来设置布局为垂直,并确保 label 控件在 image 控件后添加。这里是如何步骤性地实现它的一个基本示例: 创建父容器:首先创建一个容器对象,该对象将作为布局的基础。设置容器为垂直布局:使用 lv_obj_set_flex_flow 设置容器

C# dateTimePicker 显示年月日,时分秒

dateTimePicker默认只显示日期,如果需要显示年月日,时分秒,只需要以下两步: 1.dateTimePicker1.Format = DateTimePickerFormat.Time 2.dateTimePicker1.CustomFormat = yyyy-MM-dd HH:mm:ss Tips:  a. dateTimePicker1.ShowUpDown = t

小程序button控件上下边框的显示和隐藏

问题 想使用button自带的loading图标功能,但又不需要button显示边框线 button控件有一条淡灰色的边框,在控件上了样式 border:none; 无法让button边框隐藏 代码如下: <button class="btn">.btn{border:none; /*一般使用这个就是可以去掉边框了*/} 解决方案 发现button控件有一个伪元素(::after

MFC中Spin Control控件使用,同时数据在Edit Control中显示

实现mfc spin control 上下滚动,只需捕捉spin control 的 UDN_DELTAPOD 消息,如下:  OnDeltaposSpin1(NMHDR *pNMHDR, LRESULT *pResult) {  LPNMUPDOWN pNMUpDown = reinterpret_cast(pNMHDR);  // TODO: 在此添加控件通知处理程序代码    if

微信小程序uniappvue3版本-控制tabbar某一个的显示与隐藏

1. 首先在pages.json中配置tabbar信息 2. 在代码根目录下添加 tabBar 代码文件 直接把微信小程序文档里面的四个文件复制到自己项目中就可以了   3. 根据自己的需求更改index.js文件 首先我这里需要判断什么时候隐藏某一个元素,需要引入接口 然后在切换tabbar时,改变tabbar当前点击的元素 import getList from '../

gazebo 已加载模型但无法显示

目录 写在前面的话问题一:robot_state_publisher 发布机器人信息失败报错一 Error: Error document empty.报错二 .xcaro 文件中有多行注释成功启动 问题二:通过 ros2 启动 gazebo 失败成功启动 问题三:gazebo 崩溃和无法显示模型问题四: 缺少 robot_description 等话题正确的输出 写在前面的话

JSP 简单表单显示例子

<html><!--http://localhost:8080/test_jsp/input.html --><head><meta http-equiv="Content-Type" content="text/HTML; charset=utf-8"><title>input页面</title></head><body><form action="input.jsp" method