你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

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

[复制链接]
STMCU小助手 发布时间:2022-9-18 18:00
1 序言
    很早前就想实现这个红外遥控自学习的这个实验,用于来自己控制房子里如空调等红外遥控设备的自动化,NEC的标准到具体的产品上可能就被厂家定义为不一样了,所以自学习就应该是接收到什么就发送什么,不用管内容是什么!
2 硬件实现原理
640 (5).png
    由上述原理图可知,当IE为高电平时发送红外光,为低电平时不发送红外光。
    在NEC协议中,信息传输是基于38K载波,也就是说红外线是以载波的方式传递。
    发送波形如下图所示:
640 (4).png
    NEC协议规定:
  • 发送协议数据“0” = 发送载波560us + 不发送载波560us
  • 发送协议数据“1” = 发送载波560us+ 不发送载波1680us
  • 发送引导码 = 发送载波9000us + 不发送载波4500us


     在红外接收端,如果接收到红外38K载波,则IR输出为低电平,如果不是载波包括固定低电平和固定高电平则输出高电平。在IR端接收的信号如下所示:
640 (3).png
640 (2).png
3 软件实现自学习
    设计原理:
640 (1).png
    1、 根据接收波形记录电平和电平持续时间,以便于发送。
    2、电平记录采用定时器捕获功能,从下降沿接收引导信号开始,每触发一次改变触发方式,从而使每个电平变化都能捕获到。
    源码实现如下:
    定时器捕获初始化设置(CubeMax软件自动配置生成):
  1. void MX_TIM4_Init(void)
  2. {
  3. TIM_ClockConfigTypeDef sClockSourceConfig = {0};
  4. TIM_MasterConfigTypeDef sMasterConfig = {0};
  5. TIM_IC_InitTypeDef sConfigIC = {0};


  6. htim4.Instance = TIM4;
  7. htim4.Init.Prescaler = 71;
  8. htim4.Init.CounterMode = TIM_COUNTERMODE_UP;
  9. htim4.Init.Period = 10000;
  10. htim4.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
  11. htim4.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
  12. if (HAL_TIM_Base_Init(&htim4) != HAL_OK)
  13. {
  14. Error_Handler();
  15. }
  16. sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
  17. if (HAL_TIM_ConfigClockSource(&htim4, &sClockSourceConfig) != HAL_OK)
  18. {
  19. Error_Handler();
  20. }
  21. if (HAL_TIM_IC_Init(&htim4) != HAL_OK)
  22. {
  23. Error_Handler();
  24. }
  25. sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  26. sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  27. if (HAL_TIMEx_MasterConfigSynchronization(&htim4, &sMasterConfig) != HAL_OK)
  28. {
  29. Error_Handler();
  30. }
  31. sConfigIC.ICPolarity = TIM_INPUTCHANNELPOLARITY_RISING;
  32. sConfigIC.ICSelection = TIM_ICSELECTION_DIRECTTI;
  33. sConfigIC.ICPrescaler = TIM_ICPSC_DIV1;
  34. sConfigIC.ICFilter = 0;
  35. if (HAL_TIM_IC_ConfigChannel(&htim4, &sConfigIC, TIM_CHANNEL_4) != HAL_OK)
  36. {
  37. Error_Handler();
  38. }


  39. }
复制代码
    定时器捕获中断回调处理:
  1. void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)
  2. {
  3. if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_4)
  4.     {
  5. if(TIM4->CCER & (TIM_CCER_CC4P))   //下降沿触发
  6.         {
  7.             TIM4->CCER &= ~(TIM_CCER_CC4P); //切换
  8.             gu8BitVal = 1;
  9.         }
  10. else                               //上升沿触发
  11.         {
  12.             TIM4->CCER |= TIM_CCER_CC4P;    //切换
  13.             gu8BitVal = 0;
  14.         }




  15. if(gsInfrared.State == NONE_STATE)
  16.         {
  17.             gsInfrared.State = RECV_STATE;
  18.         }
  19. else if(gsInfrared.State == RECV_STATE)
  20.         {
  21.             NowTimCnt = HAL_TIM_ReadCapturedValue(&htim4, TIM_CHANNEL_4);
  22.             gsInfrared.KeepTime[gsInfrared.SampleCount] = Round(NowTimCnt);
  23.             gsInfrared.BitValue[gsInfrared.SampleCount ++] = gu8BitVal;
  24.         }


  25.         TIM4->CNT = 0;
  26.     }
  27. }
复制代码
    3、设置的定时器溢出时间为10ms,如果10毫秒内不再接收电平变化则默认接收结束,设置结束标志。
  1. void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
  2. {
  3. if(htim == &htim4)
  4.     {
  5. if(gsInfrared.State == RECV_STATE)
  6.         {
  7.             gsInfrared.State = END_STATE;
  8.         }
  9.     }
  10. }
复制代码
    至此,实现了红外遥控的学习功能,获得的记录数据为记录长度和电平信号数组与电平信号维持的时间数组。
640.png
    4、发送实现
    设置定时器输出38KPWM信号,在记录电平为0是输出记录时间的38K载波信号,如果为1则不输出载波,实现如下:
    PWM生成设置(CubeMax自动配置生成):
  1. void MX_TIM5_Init(void)
  2. {
  3.   TIM_MasterConfigTypeDef sMasterConfig = {0};
  4.   TIM_OC_InitTypeDef sConfigOC = {0};


  5.   htim5.Instance = TIM5;
  6.   htim5.Init.Prescaler = 0;
  7.   htim5.Init.CounterMode = TIM_COUNTERMODE_UP;
  8.   htim5.Init.Period = 1896;
  9.   htim5.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
  10.   htim5.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
  11. if (HAL_TIM_PWM_Init(&htim5) != HAL_OK)
  12.   {
  13.     Error_Handler();
  14.   }
  15.   sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  16.   sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  17. if (HAL_TIMEx_MasterConfigSynchronization(&htim5, &sMasterConfig) != HAL_OK)
  18.   {
  19.     Error_Handler();
  20.   }
  21.   sConfigOC.OCMode = TIM_OCMODE_PWM1;
  22.   sConfigOC.Pulse = 0;
  23.   sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;
  24.   sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
  25. if (HAL_TIM_PWM_ConfigChannel(&htim5, &sConfigOC, TIM_CHANNEL_2) != HAL_OK)
  26.   {
  27.     Error_Handler();
  28.   }
  29.   HAL_TIM_MspPostInit(&htim5);


  30. }
复制代码
    发送实现,注意点就是记录为0时发载波,记录为1时不发载波:
  1. void InfraredSend(void)
  2. {
  3. uint16_t Count = 0;

  4. while(Count < gsInfrared.SampleCount &&  gsInfrared.State == END_STATE)
  5.     {
  6. if(gsInfrared.BitValue[Count] == 0)
  7.         {
  8.             TIM5->CCR2 = 948;
  9.             delay_us(gsInfrared.KeepTime[Count]);
  10.             TIM5->CCR2 = 0;
  11.         }
  12. else
  13.         {
  14.             TIM5->CCR2 = 0;
  15.             delay_us(gsInfrared.KeepTime[Count]);
  16.             TIM5->CCR2 = 0;
  17.         }


  18.         Count ++;
  19.     }


  20.     delay_us(20000);
  21. }
复制代码
    往PWM比较寄存器设置948即为设置38KPWM波,也可在初始化时固定948,在此函数内启停定时器即可;
    至此,自学习功能的全部思路已实现,通过对各个不同类型的红外遥控进行功能测试,均成功。
     PS:查看很多资料发现很多红外解码未判断低电平时间,个人感觉不是很好,应该是不仅高电平时间得符合,低电平时间也应该符合。
    自己写了一个小函数验证了一下,这个函数只是验证,未经仔细推敲,还可优化,仅供参考这一思想。
    误差设计:±200us(拍脑袋值)
  1. void InFraredDataDeal(void)
  2. {
  3. uint32_t DataBuff = 0;
  4. uint16_t Count = 0;


  5. if(gsInfrared.State == END_STATE)
  6.     {
  7.         gsInfraredData.State = 0;


  8. do
  9.         {
  10. switch(gsInfraredData.State)
  11.             {
  12. case 0:   //引导码识别
  13.             {


  14. if(gsInfrared.KeepTime[0] >= 8800 && gsInfrared.KeepTime[0] <= 9200 && gsInfrared.BitValue[0] == 0)
  15.                 {
  16. if(gsInfrared.KeepTime[1] >= 4300 && gsInfrared.KeepTime[1] <= 4700 && gsInfrared.BitValue[1] == 1)
  17.                     {
  18. if(gsInfrared.KeepTime[2] >= 360 && gsInfrared.KeepTime[2] <= 760 && gsInfrared.BitValue[2] == 0)
  19.                         {
  20.                             Count = 3;
  21.                             gsInfraredData.State = 1;
  22.                         }
  23.                     }
  24. else if(gsInfrared.KeepTime[1] >= 2300 && gsInfrared.KeepTime[1] <= 2700 && gsInfrared.BitValue[1] == 1)
  25.                     {
  26. if(gsInfrared.KeepTime[2] >= 360 && gsInfrared.KeepTime[2] <= 760 && gsInfrared.BitValue[2] == 0)
  27.                         {
  28.                             gsInfraredData.ReDataCount ++;
  29.                             gsInfraredData.State = 3;
  30.                         }
  31.                     }
  32. else
  33.                     {
  34.                         gsInfraredData.State = 3;
  35.                     }


  36.                 }
  37. else
  38.                 {
  39.                     gsInfraredData.State = 3;
  40.                 }




  41.             }
  42. break;




  43. case 1:   //数据解析
  44.             {


  45. if(gsInfrared.KeepTime[Count + 1] >= 360 && gsInfrared.KeepTime[Count + 1] <= 760 && gsInfrared.BitValue[Count + 1] == 0)
  46.                 {


  47. if(gsInfrared.BitValue[Count] == 1)
  48.                     {
  49. if(gsInfrared.KeepTime[Count] >= 1480 && gsInfrared.KeepTime[Count] <= 1880)
  50.                         {
  51.                             DataBuff <<= 1;
  52.                             DataBuff |= 1;
  53.                         }
  54. else if(gsInfrared.KeepTime[Count] >= 360 && gsInfrared.KeepTime[Count] <= 760 && gsInfrared.BitValue[Count] == 1)
  55.                         {
  56.                             DataBuff <<= 1;
  57.                             DataBuff |= 0;
  58.                         }
  59. else
  60.                         {
  61.                             gsInfraredData.State = 3;
  62.                         }
  63.                     }
  64.                 }


  65. if(Count < gsInfrared.SampleCount)
  66.                 {
  67.                     Count += 2;
  68.                 }
  69. else
  70.                 {
  71.                     gsInfraredData.State = 2;
  72.                 }


  73.             }
  74. break;


  75. case 2:   //成功解析
  76.             {
  77.                 gsInfraredData.Data = DataBuff;
  78.                 gsInfraredData.State = 3;


  79.             }
  80. break;


  81. default:
  82.             {
  83.                 gsInfraredData.State = 3;   //解析结束


  84.             }
  85. break;
  86.             }


  87.         }
  88. while(gsInfraredData.State != 3);


  89.         gsInfrared.State = NONE_STATE;
  90.         gsInfrared.SampleCount = 0;
  91.     }


  92. }
复制代码
    解析的话一般高位在前,所以左移,经测试帧格式为:引导码+用户码+用户码反码+命令码+命令码反码,能成功解析数据!解析的话根据具体协议,具体分析。

转载自:树荫下的阳光
收藏 评论0 发布时间:2022-9-18 18:00

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版