STM32单片机(12) 红外信号接收解码(外部中断)

发布者:RadiantBreeze最新更新时间:2018-05-19 来源: eefocus关键字:STM32  单片机  红外信号  接收解码  外部中断 手机看文章 扫描二维码
随时随地手机看文章

本程序主要利用外部中断,实现红外遥控器信号接收解码,并利用串口通信把编码传至计算机显示

注:请用质量好点的遥控器实验,用了劣质遥控器浪费了一天时间,数据位接收总是不完整,后来用宿舍空调遥控器就解码成功了

相关资料

STM32单片机学习(2) 外部中断 http://blog.csdn.net/leytton/article/details/38063335 

STM32单片机学习(3) 串口中断通信  http://blog.csdn.net/leytton/article/details/38393553 

STM32单片机学习(7) 串口通信printf重定向 http://blog.csdn.net/leytton/article/details/38393967

STM32-外设篇 视频教程(Cortex-M3)-主讲人:刘洋   http://yun.baidu.com/pcloud/album/info?uk=2853967793&album_id=5492137931588632574

效果如图:格力空调遥控器红外信号接收解码,很多复合按键编码,不同编码实现不同空调功能


贴代码:

/******************************************************************************* 

*    

* 软件功能:  红外线接收 

*  

*******************************************************************************/  

#include "stm32f10x.h"  

#include  

#include "delay.h"  

    

void RCC_Configuration(void);  

void GPIO_Configuration(void);  

void USART1_Configuration(void);  

void Uart1_PutChar(u8 ch);  

void Uart1_PutString(u8* buf , u8 len);  

int fputc(int ch, FILE *f);  

  

void NVIC_Configuration(void);  

void EXTI_Configuration(void);  

  

u8 HW_ReceiveTime(void);  

                              

/************************************************* 

函数: int main(void) 

功能: main主函数 

参数: 无 

返回: 无 

**************************************************/  

int main(void)  

{  

  RCC_Configuration();  

  GPIO_Configuration();  

    

  delay_init(72);  

  USART1_Configuration();  

  NVIC_Configuration();  

  EXTI_Configuration();  

  printf("start\n");  

  GPIO_ResetBits(GPIOA,GPIO_Pin_0);//灭     

  while(1);  

}  

  

/************************************************* 

函数: void RCC_Configuration(void) 

功能: 复位和时钟控制 配置 

参数: 无 

返回: 无 

**************************************************/  

void RCC_Configuration(void)  

{  

  ErrorStatus HSEStartUpStatus;                    //定义外部高速晶体启动状态枚举变量  

  RCC_DeInit();                                    //复位RCC外部设备寄存器到默认值  

  RCC_HSEConfig(RCC_HSE_ON);                       //打开外部高速晶振  

  HSEStartUpStatus = RCC_WaitForHSEStartUp();      //等待外部高速时钟准备好  

  if(HSEStartUpStatus == SUCCESS)                  //外部高速时钟已经准别好  

  {  

    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //开启FLASH预读缓冲功能,加速FLASH的读取。所有程序中必须的用法.位置:RCC初始化子函数里面,时钟起振之后  

    FLASH_SetLatency(FLASH_Latency_2);                    //flash操作的延时  

          

    RCC_HCLKConfig(RCC_SYSCLK_Div1);               //配置AHB(HCLK)时钟等于==SYSCLK  

    RCC_PCLK2Config(RCC_HCLK_Div1);                //配置APB2(PCLK2)钟==AHB时钟  

    RCC_PCLK1Config(RCC_HCLK_Div2);                //配置APB1(PCLK1)钟==AHB1/2时钟  

           

    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);  //配置PLL时钟 == 外部高速晶体时钟 * 9 = 72MHz  

    RCC_PLLCmd(ENABLE);                                   //使能PLL时钟  

     

    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)    //等待PLL时钟就绪  

    {  

    }  

    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);            //配置系统时钟 = PLL时钟  

    while(RCC_GetSYSCLKSource() != 0x08)                  //检查PLL时钟是否作为系统时钟  

    {  

    }  

  }  

    

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_USART1 | RCC_APB2Periph_AFIO, ENABLE);  //允许 GPIOA、USART1、AFIO时钟  

}  

  

/************************************************* 

函数: void GPIO_Configuration(void) 

功能: GPIO配置 

参数: 无 

返回: 无 

**************************************************/  

void GPIO_Configuration(void)  

{  

    

  GPIO_InitTypeDef GPIO_InitStructure;        //定义GPIO初始化结构体  

  

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;   

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复合推挽输出     

  GPIO_Init(GPIOA, &GPIO_InitStructure);       //PA9串口输出  

  

  

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;   //红外接收  

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;  //配置成上拉输入;   

  GPIO_Init(GPIOA, &GPIO_InitStructure);  

  

    

}  

  

  

/******************************************************************************* 

    函数名:USART1_Configuration 

    输  入: 

    输  出: 

    功能说明: 

    初始化串口硬件设备,启用中断 

    配置步骤: 

    (1)打开GPIO和USART1的时钟 

    (2)设置USART1两个管脚GPIO模式 

    (3)配置USART1数据格式、波特率等参数 

    (4)使能USART1接收中断功能 

    (5)最后使能USART1功能 

*/  

void USART1_Configuration(void)   //串口配置   详见《STM32的函数说明(中文).pdf》P346  

{  

    USART_InitTypeDef USART_InitStructure;  

    USART_InitStructure.USART_BaudRate=9600;   //波特率为9600  

    USART_InitStructure.USART_WordLength=USART_WordLength_8b;  //数据位为8  

    USART_InitStructure.USART_StopBits=USART_StopBits_1; //在帧结尾传输 1 个停止位  

    USART_InitStructure.USART_Parity=USART_Parity_No; //校验模式:奇偶失能  

    USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None; //硬件流控制失能  

    USART_InitStructure.USART_Mode=USART_Mode_Tx | USART_Mode_Rx; //USART_Mode 指定了使能或者失能发送和接收模式:发送使能|接收失能  

    USART_Init(USART1, &USART_InitStructure);     //初始化配置  

  

    USART_Cmd(USART1,ENABLE);   //使能或者失能 USART 外设  

    USART_ClearFlag(USART1, USART_FLAG_TC);//清除传输完成标志位,否则可能会丢失第1个字节的数据.USART_FLAG_TC为发送完成标志位  

}  

  

  

//发送一个字符  

void Uart1_PutChar(u8 ch)  

{  

    USART_SendData(USART1, (u8) ch);  

    while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);//等待发送完成  

}  

  

//发送一个字符串 Input : buf为发送数据的地址 , len为发送字符的个数  

void Uart1_PutString(u8* buf , u8 len)  

{     

    u8 i;  

    for(i=0;i

    {  

        Uart1_PutChar(*(buf++));  

    }  

}  

  

int fputc(int ch, FILE *f)  

{  

Uart1_PutChar((u8)ch);  //此处为自定义函数,参见串口中断通信,请勿盲目复制  

return (ch);  

}  

  

void NVIC_Configuration(void)    //中断分组和优先级配置    详见《STM32的函数说明(中文).pdf》P165  

{  

    NVIC_InitTypeDef NVIC_InitStructure;  

    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);  //0组,先占优先级0位,从优先级4位  

    NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQn; //外部中断线 9_5 中断(5-9号引脚是统一使用中断线EXTI9_5_IRQn)  

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;    //0组无抢占优先级,所以只配置从优先级,即响应式优先级  

    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能  

    NVIC_Init(& NVIC_InitStructure); //初始化配置  

}  

  

void EXTI_Configuration(void)  //中断配置  //详见《STM32的函数说明(中文).pdf》   P99  

{  

    EXTI_InitTypeDef EXTI_InitStructure;          

  

    EXTI_ClearITPendingBit(EXTI_Line8);//清除 EXTI 线路挂起位  

  

    EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //设置 EXTI 线路为中断请求  

    EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; //设置输入线路下降沿为中断请求  

    EXTI_InitStructure.EXTI_Line = EXTI_Line8; //外部中断线 8  

    EXTI_InitStructure.EXTI_LineCmd = ENABLE;//使能  

    EXTI_Init(& EXTI_InitStructure); //初始化配置  

      

  GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource8);    ///*将EXTI线8连接到PA8*/  

}  

  

  

  

u8 HW_ReceiveTime(void)  

{  

    u8 t=0;  

  

    while(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_8)==1)//高电平  

    {  

        t++;  

        delay_us(20);  

  

        if(t>=250) return t;//超时溢出  

    }  

  

    return t;  

}  

  

/********************************************************* 

                中断服务程序 

********************************************************/  

void EXTI9_5_IRQHandler()       

{  

     u8 time=0,startSignal=0,Data=0,Count=0;  

     u32 HW_ReceivedData=0;  

     //printf("1\n");  

     if(EXTI_GetITStatus(EXTI_Line8) == SET)//检查指定的EXTI线路触发请求发生与否,返回一个EXTI_Line新状态  

      {  

            //printf("5\n");   

            //if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0))  GPIO_SetBits(GPIOA,GPIO_Pin_0);//点亮LED  

            //else  GPIO_ResetBits(GPIOA,GPIO_Pin_0);//灭  

            while(1)  

            {  

                    if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_8)==1)  

                    {  

                         time = HW_ReceiveTime();  

                            

                         if(time>=250)  

                         {  

                            //printf("VoverTime:%x\n",time);   

                            startSignal=0;  

                            HW_ReceivedData=0;  

                            Count=0;  

                            break;  

                         }   

                         else if(time>=200 && time<250)  

                         {  

                              startSignal=1;//收到起始信号  

                              HW_ReceivedData=0;  

                              Count=0;  

                              continue;  

                         }   

                         else if(time>=60 && time<90)  Data=1;//收到数据 1  

                         else if(time>=10 && time<50)  Data=0;//收到数据 0  

                         //printf("%x ",Data);  

                         if(startSignal==1)  

                          {  

                                HW_ReceivedData<<=1;  

                                HW_ReceivedData+=Data;  

                                Count++;  

                                if(Count>=32)  

                                {  

                                    //printf("received:%x\n",HW_ReceivedData);  

                                    if(HW_ReceivedData==0x1090000a)  printf("红外编码:0x%8X,指令:打开空调\n",HW_ReceivedData);  

                                    else if(HW_ReceivedData==0x90000a)  printf("红外编码:0x%08X,指令:关闭空调\n",HW_ReceivedData);  

                                    else   printf("红外编码:0x%08X,指令:未知\n",HW_ReceivedData);  

                                    HW_ReceivedData=0;  

                                    Count=0;  

                                    startSignal=0;  

                                    break;  

                                }   

                                  

                          }     

                    }  

            }  

              

            EXTI_ClearITPendingBit(EXTI_Line8);//清除线路挂起位  

            EXTI_ClearFlag(EXTI_Line8);//清除中断挂起位   

     }   

}  


关键字:STM32  单片机  红外信号  接收解码  外部中断 引用地址:STM32单片机(12) 红外信号接收解码(外部中断)

上一篇:STM32单片机(13) I2C读写AT24Cxx存储器实验
下一篇:STM32单片机(11) DS18B20温度传感器实验

推荐阅读最新更新时间:2024-03-16 16:03

STM32的瞬态运动参数存储测试系统设计
在瞬态运动参数测试中,对存储测试系统的实时性和功耗提出了更高的要求。提出了一种基于STM32的嵌入式存储测试系统的设计方案,介绍了该系统关键部分的软硬件设计,主要包括模拟信号调理、数据采集存储和USB数据回读。该系统具有实时性好、体积小、功耗低的特点,适合于恶劣环境下加速度信号的采集存储。试验结果表明,该系统工作稳定,实现了设计目标。 存储测试技术是在特殊环境下记录运动物体参数行之有效的方法,先将测试数据存入存储器,待装置回收后通过特定接口与上位机进行通信,还原数据信息。在诸多领域的测试中,对数据采集存储系统的实时性和功耗提出了更高的要求,随着半导体技术的发展,各种技术的进步使得高速度、低功耗的存储测试系统能够实现。 本系
[单片机]
<font color='red'>STM32</font>的瞬态运动参数存储测试系统设计
PIC系列单片机选型及应用
当今单片机厂商琳琅满目,产品性能各异。针对具体情况,我们应选何种型号呢?首先,我们来弄清两个概念:集中指令集(CISC)和精简指令集(RISC)。采用CISC结构的单片机数据线和指令线分时复用,即所谓冯.诺伊曼结构。它的指令丰富,功能较强,但取指令和取数据不能同时进行,速度受限,价格亦高。采用RISC结构的单片机数据线和指令线分离,即所谓哈佛结构。这使得取指令和取数据可同时进行,且由于一般指令线宽于数据线,使其指令较同类CISC单片机指令包含更多的处理信息,执行效率更高,速度亦更快。同时,这种单片机指令多为单字节,程序存储器的空间利用率大大提高,有利于实现超小型化。属于CISC结构的单片机有Intel8051系列、Motorola
[单片机]
基于AT89C52单片机的日历时钟模块设计
在实时监控系统的设计中,要实时监测各个控制信号,更重要的是在发生故障时能准确记录故障数据,以便准确分析排除错误。监控系统中一般都要定时采集现场数据,对某些重要的信息不仅要记录其内容,还要记录下该信息发生的准确时间,所记录的实时时间信息应长期保存,因此需要实时时钟来实现。常用的单片机没有实时时钟,若需采用定时器实现,一旦系统掉电,时钟就不能运行,这是实时监控系统 不允许的,而采用独立运行的实时时钟便可实现。但一般的时钟芯片在系统掉电时,其数据也会丢失,需提供备用电池。而时钟芯片DS12887在系统掉电时数据不丢失,广泛应用于测量和控制系统。因此,这里给出了实时时钟DS12887在单片机应用系统中的应用。 1 主要器件造型 1
[单片机]
基于AT89C52<font color='red'>单片机</font>的日历时钟模块设计
51单片机总线时序概述
一、总线概述 计算机系统是以微处理器为核心的,各器件要与微处理器相连,且必须协调工作,所以在微处理机中引入了总线的概念,各器件共同享用总线,任何时候只能有一个器件发送数据(可以有多个器件同时接收数据) 。 计算机的总线分为控制总线、地址总线和数据总线等三种。而数据总线用于传送数据,控制总线用于传送控制信号, 地址总线则用于选择存储单元或外设。 二、单片机的三总线结构 51系列单片机具有完善的总线接口时序,可以扩展控制对象,其直接寻址能力达到64k( 2的16次方) 。在总线模式下,不同的对象共享总线,独立编址、分时复用总线,CPU 通过地址选择访问的对象,完成与各对象之间的信息传递。 单片机三总线扩展示意如图1 所示。
[单片机]
51<font color='red'>单片机</font>总线时序概述
STM32之DMA+ADC
简介:有时候找BUG真是一件痛苦的事、我找了整整一个晚上、第二天上完通信原理回来再找了一会、偶然发现我在主函数里的ADC_DMA_Init()没有写、、这种无奈、、这种、、已经无法用普通话能表达的了、找各种BUG的时候、相信大家都有体会、、在这里就不多说了哈、、希望能帮到你们、、在这里为什么这么直接的讲DMA、、因为DMA很直接、、所以我也很直接、、所以你懂的、、 学习到了STM32的DMA模块、、琢磨了一下中文参考手册,官方是这样描述的: 直接存储器存取(DMA)用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。无须CPU干预,数据可以通过DMA快速地移动,这就节省了CPU的资源来做其他操作。 是的,无
[单片机]
51单片机实验——输出周期为200ms的方波
1.实验题目: 利用定时器T0的16位不可重加载模式,编制一段程序,只有当INT0(P3.2)引脚输出高电平时,在P1.6输出一个方波控制一个指示灯。设单片机的时钟为12MHz,输出周期为200ms的方波,观察指示灯的变化。 2.KEIL代码 #include reg51.h sbit OUT=P1^6; sbit P32=P3^2; void Int0_Init(); void main() { Int0_Init(); TMOD &=0X0F; TMOD |=0x09;//选择定时器T0的工作模式1 TH0 = (65536 - 50000)/256;//高四位初值 TL0 = (65536 - 5
[单片机]
51<font color='red'>单片机</font>实验——输出周期为200ms的方波
基于单片机PID恒温温度控制系统设计
#include reg52.h #include intrins.h #include math.h #include string.h struct PID { unsigned int SetPoint; // 设定目标 Desired Value unsigned int Proportion; // 比例常数 Proportional Const unsigned int Integral; // 积分常数 Integral Const unsigned int Derivative; // 微分常数 Derivative Const unsigned int LastError; // Error
[单片机]
基于<font color='red'>单片机</font>PID恒温温度控制系统设计
STM32虚拟串口usb_printf函数及接收函数
环境:STM32CubeMX STM32F429IGT6 STlink 首先要确保硬件电路USB部分没问题;USB相关的概念知识大概需要了解一下,网上挺多这类文章的,自行百度。 点击USB_OTG_FS,模式选择Device_Only,其他保持默认。 点击USB_DEVICE,选择IP 为VPC(虚拟串口),其他保持默认。 我使用的芯片是F429IGT6,最大时钟180MHz,但是USB时钟必须为48MHz(详情看STM32中文参考手册930页),180MHz是分频不出来48MHz的USB时钟,所以把系统配置成168MHz就能分频出48MHz的USB时钟。 堆空间需要改大一点,不然在USB插入电脑的时候,
[单片机]
<font color='red'>STM32</font>虚拟串口usb_printf函数及<font color='red'>接收</font>函数
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved