华大HC32L136 单定时器红外解码
背景
近期有个工业衡器需要用到红外解码,在网上也看了其他人写的程序,包括正点原子的STM32F103的例程。感觉他们的解码要么复杂,要么采用了高大上的输入捕获功能,多次编译移植,有些实在修改地方多而繁琐,而且都解码失败。最终,还是不得不静下心对照时序图,编写了一个基于定时循环执行的解码程序,就是只要定期执行就能解码的程序。 解码程序主要使用了两个外部变量,一个是ir_decode_ok_flag 解码成功标志位,一个是ir_code[4]码值保存变量。 使用方法,就是定时100us执行一次就可以了。
红外遥控时序
因为串行解码,对时序和定时准确性要求较高,100us定时必须卡准,建议用示波器实测确认。 单片机采用华大HC32L136,内部时钟倍频到24Mhz, TIM2, 8分频。
void App_Timer2Cfg100us(uint16_t us100)
{
uint32_t u16ArrValue;
uint32_t u16CntValue;
stc_bt_mode0_cfg_t stcBtBaseCfg;
uint32_t convertPeriod;
DDL_ZERO_STRUCT(stcBtBaseCfg);
Sysctrl_SetPeripheralGate(SysctrlPeripheralBaseTim, TRUE);
stcBtBaseCfg.enWorkMode = BtWorkMode0;
stcBtBaseCfg.enCT = BtTimer;
stcBtBaseCfg.enPRS = BtPCLKDiv8;
stcBtBaseCfg.enCntMode = Bt16bitArrMode;
stcBtBaseCfg.bEnTog = FALSE;
stcBtBaseCfg.bEnGate = FALSE;
stcBtBaseCfg.enGateP = BtGatePositive;
Bt_Mode0_Init(TIM2, &stcBtBaseCfg);
convertPeriod = (us100 * Sysctrl_GetHClkFreq())/(8*10000);
u16ArrValue = 0x10000 - convertPeriod;
Bt_M0_ARRSet(TIM2, u16ArrValue);
u16CntValue = 0x10000 - convertPeriod;
Bt_M0_Cnt16Set(TIM2, u16CntValue);
Bt_ClearIntFlag(TIM2,BtUevIrq);
Bt_Mode0_EnableIrq(TIM2);
EnableNvic(TIM2_IRQn, IrqLevel2, TRUE);
Bt_M0_Run(TIM2);
}
TIM2中断函数调用解码子函数Ir_Decode(), 以100us间隔定时调用即可
```c
void Tim2_IRQHandler(void)
{
#if 1
if(TRUE == Bt_GetIntFlag(TIM2, BtUevIrq))
{
Bt_ClearIntFlag(TIM2,BtUevIrq);
Ir_Decode();
}
#endif
}
主程序检测标志位ir_decode_ok_flag,如果为1,表示收到有效遥控发射码流,打印出键值
```c
void appIrLoop(void)
{
if( ir_decode_ok_flag == 1 )
{
ir_decode_ok_flag = 0;
printf("IrKey:%02X %02X %02X %02X\r\n",ir_code[0],ir_code[1],ir_code[2],ir_code[3]);
}
}
完整工程代码可通过https://download.csdn.net/download/igreeniot/23359224下载测试
|