基于stm32 ev1527学习型射频遥控的解码

2019-07-21 02:50发布

      第一次发帖  部分资料来自网上  侵权删  还有许多需要改进的地方  望指点        eV1527是一片由CMOS设计制造的可预烧内码的学习码编码IC,由软件解码;内码共有20个位元可预烧1048576组(220)内码组合,降低使用上编码重复的机率.它由百万组不同的编码组成,接收器只有通过对码学习,才能识别遥控器。它大大增加了产品的安全性。一般通过单片机进行解码。(附1527中文手册) ev1527.pdf (143.56 KB, 下载次数: 551) 2016-11-4 16:23 上传 点击文件名下载附件
ev1527
项目需要用到射遥控模块 在网上找了好多  没找到好用的程序  就在原子哥的红外遥控实验进行修改 .硬件 :        用到的有一个遥控器和一个接收模块和战舰开发板. QQ截图20161104164155.png QQ截图20161104164210.png 为了省事 我将接收模块的数据端口接到了开发板的PB8 也就是红外的口 .省的改IO口了.EV1527的编码方式和PT2262/2272那种固定码是不一样的 1527是20位地址码+四位键值码. 2262是8位地址码+四位键值码 QQ截图20161104163930.png
上程序吧初始化 : 和红外接收一样  没什么修改   就把溢出时间加长了点void Remote_Init(void)                              {          GPIO_InitTypeDef GPIO_InitStructure;        NVIC_InitTypeDef NVIC_InitStructure;        TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;        TIM_ICInitTypeDef  TIM_ICInitStructure;  
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE); //使能PORTB时钟         RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4,ENABLE);        //TIM4 时钟使能
        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;                                 //PB9 输入         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;                 //上拉输入         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;         GPIO_Init(GPIOB, &GPIO_InitStructure);         GPIO_SetBits(GPIOB,GPIO_Pin_9);        //初始化GPIOB.9                                                                   TIM_TimeBaseStructure.TIM_Period = 50000; //设定计数器自动重装值 最大50ms溢出          TIM_TimeBaseStructure.TIM_Prescaler =(72-1);         //预分频器,1M的计数频率,1us加1.                           TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim        TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
        TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure); //根据指定的参数初始化TIMx
  TIM_ICInitStructure.TIM_Channel = TIM_Channel_4;  // 选择输入端 IC4映射到TI4上  TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;        //上升沿捕获  TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;  TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;         //配置输入分频,不分频   TIM_ICInitStructure.TIM_ICFilter = 0x03;//IC4F=0011 配置输入滤波器 8个定时器时钟周期滤波  TIM_ICInit(TIM4, &TIM_ICInitStructure);//初始化定时器输入捕获通道
  TIM_Cmd(TIM4,ENABLE );         //使能定时器4
        NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;  //TIM3中断        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;  //先占优先级0级        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;  //从优先级3级        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能        NVIC_Init(&NVIC_InitStructure);  //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器       
        TIM_ITConfig( TIM4,TIM_IT_Update|TIM_IT_CC4,ENABLE);//允许更新中断 ,允许CC4IE捕获中断                                                                 }
中断函数:
//遥控器接收状态
//[7]:收到了引导码标志

//[5]:保留       
//[4]:标记上升沿是否已经被捕获                                                                  
//[3:0]:溢出计时器
u8         RmtSta=0;                    
u32 Dval;                //下降沿时计数器的值
u32 Dval2;                //上升沿时计数器的值
u32 RmtRec=0;        //接收到的数据                               

u8  num = 0;
u8  key_flag = 0;  //学习按键 按下学习键后置1
//定时器4中断服务程序         
void TIM4_IRQHandler(void)
{                              


        if(TIM_GetITStatus(TIM4,TIM_IT_CC4)!=RESET)
        {          
                if(RDATA)//上升沿捕获
                {
                          TIM_OC4PolarityConfig(TIM4,TIM_ICPolarity_Falling);                                                //CC4P=1        设置为下降沿捕获
                        Dval2=TIM_GetCapture4(TIM4);                                        //读取CCR4也可以清CC4IF标志位  //读取低电平的时间
                        TIM_SetCounter(TIM4,0);                                                        //清空定时器值
                        RmtSta|=0X10;                                                        //标记上升沿已经被捕获
                }else //下降沿捕获
                {
                        Dval=TIM_GetCapture4(TIM4);                                        //读取CCR4也可以清CC4IF标志位   //高电平的时间
                        TIM_SetCounter(TIM4,0);                                                        //清空定时器值
                        TIM_OC4PolarityConfig(TIM4,TIM_ICPolarity_Rising);                                //CC4P=0        设置为上升沿捕获
                        RmtSta&=~(1<<4);
                }       
                        if(RmtSta&0X10)                                                        //完成一次高低电平捕获
                        {
                                if(RmtSta&0X80)//接收到了引导码
                                {
                                       
                                        if(Dval2<600&&Dval>600&&Dval<2000&&Dval2<2000)                        //   收到1
                                        {
                                                RmtRec<<=1;                                        //左移一位.
                                                RmtRec|=1;                                                        //接收到1       
                                                num ++;
                                        }else if(Dval<600&&Dval2>600&&Dval<2000&&Dval2<2000)        //   收到0
                                        {
                                                RmtRec<<=1;                                        //左移一位.
                                                RmtRec|=0;                                        //接收到0
                                                num ++;
                                        }
                                        else
                                                if(Dval>2000||Dval2>2000)        //数据错误
                                        {
                                                num = 0;
                                                RmtRec = 0;
                                                RmtSta&=0XF0;                                //清空计时器
                                                RmtSta|=0<<7;                                        //取消标记接收到引导码                                               
                                        }
                                }else
                                {
                                if(Dval2>6000&&Dval2<18000)               
                                {
                                        RmtSta|=1<<7;                                        //标记成功接收到了引导码
                                        RmtRec = 0;
                                        num = 0;
                                }        
               
                        }
                       
                        if(num == 24)
                        {
                                key_rmote();
                        }
                }       
                       
        }
        TIM_ClearITPendingBit(TIM4,TIM_IT_Update|TIM_IT_CC4);                     
}


感觉判断高低电平时间那里不严谨  希望大神指点
//////////按键处理函数///////////
u32 daaress= 0;
u32  key_num = 0;
u32   key_daaress = 0;  //存储地址编码信息
u8 key_rmote(void)
{
        daaress =        RmtRec>>4;
                if(key_flag == 1)
        {
                key_flag =0;
                key_daaress = daaress;
        }
        else
                if(daaress ==key_daaress)
                {
                key_num = RmtRec;
                key_num &= 0x0000000F;
                }
        RmtRec = 0;

        printf("num = %d ",key_num);
        printf("RmtRec = %x ",daaress);


return 0;
}



主函数就没什么了 很简单  
int main(void)
{         
        u8 key;
        delay_init();                     //延时函数初始化          
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级
        uart_init(115200);                 //串口初始化为115200
        LED_Init();                             //LED端口初始化
        LCD_Init();       
        KEY_Init();                
        Remote_Init();                        //无线遥控接收初始化        

        POINT_COLOR=RED;                //设置字体为红 {MOD}
        LCD_ShowString(30,50,200,16,16,"WarShip STM32");
        LCD_ShowString(30,70,200,16,16,"REMOTE TEST");       
        LCD_ShowString(30,90,200,16,16,"ATOM@ALIENTEK");
        LCD_ShowString(30,110,200,16,16,"2015/1/15");

           LCD_ShowString(30,130,200,16,16,"KEY:");       
                                                                                                            
        while(1)
        {       
                key = KEY_Scan(0);
                if(key_num)
                {         
                        LCD_ShowNum(86,130,key_num,1,16);                //显示键值          
                       
                }
                 
               
                if(key==WKUP_PRES)
                {
                        LED0=!LED0;
                    key_flag = 1;
                }
        }
       
}



只写了储存一个遥控信息   
按开发板的上键后 按遥控进行学习  学习一后才能识别遥控  按键的值会在LCD上面显示


附上工程吧

无线射频遥控接收.rar (334.55 KB, 下载次数: 588) 2016-11-4 16:50 上传 点击文件名下载附件


友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。