今天学习了一下外部中断检测按键引脚电平变化触发中断,并在中断服务程序中实现对LED灯的反转:

有关外部中断


EXTI ( Extern Interrupt )外部中断:


1.EXTI 可以监测指定 GPIO 口的电平信号,当其指定的 GPIO 口产生电平变化时, EXTI将立即向 NVIC 发出中断申请,经过 NVIC 裁决后即可中断 CPU 主程序,使 CPU执行 EXTI 对应的中断程序


2.支持的触发方式:上升沿 / 下降沿 / 双边沿 / 软件触发


3.支持的 GPIO 口:所有 GPIO 口,但相同的 Pin 不能同时触发中断通


4.道数: 16 个 GPIO_Pin ,外加 PVD 输出、 RTC 闹钟、 USB 唤醒、以太网唤醒


5.触发响应方式:中断响应 / 事件响应


整体代码:


exti.c


#include "exti.h" 
#include "led.h" 
#include "key.h"
#include "delay.h"
#include "stdio.h"

/* 添加新的外设步骤:
 * 第一步:首先在hardware文件夹下创建一个文件夹exti
 * 第二步:打开keil 分别创建源文件exti.c 和头文件exti.h,保存到文件夹exti中
 * 第三步:keil中,双击hardware,添加源文件exti.c
 * 第四步:魔术棒,添加头文件包含的路径
 * 第五步:源文件添加#include "exti.h" ,然后编译
 */

void EXTIX_Init(void)
{	
  GPIO_InitTypeDef  GPIO_InitStructure;
  EXTI_InitTypeDef   EXTI_InitStructure;
  NVIC_InitTypeDef   NVIC_InitStructure;

  RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOE, ENABLE);//使能GPIOF时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE);//只要使用到外部中断,就必须打开SYSCFG时钟。
	
  //GPIOE2初始化设置
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;//LED对应引脚
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;//普通输入模式
  //GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉:按键按下是低电平,则默认是高电平
  GPIO_Init(GPIOE, &GPIO_InitStructure);//初始化PE2引脚
  
  //配置 GPIO 与中断线的映射关系
  SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOE, EXTI_PinSource2);

  //配置中断源2	
  EXTI_InitStructure.EXTI_Line = EXTI_Line2;
  EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;//中断事件
  EXTI_InitStructure.EXTI_Trigger =EXTI_Trigger_Falling; //下降沿触发 
  EXTI_InitStructure.EXTI_LineCmd = ENABLE;//中断线使能
  EXTI_Init(&EXTI_InitStructure);//配置
	
  //中断源2的NVIC配置
  NVIC_InitStructure.NVIC_IRQChannel = EXTI2_IRQn;//外部中断2
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x03;//抢占优先级3
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02;//子优先级2
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能外部中断通道
  NVIC_Init(&NVIC_InitStructure);//配置
  
}

//中断服务程序
void EXTI2_IRQHandler(void)
{
	delay_ms(10);	//消抖
	if(KEY2==0)	    //检测引脚电平:上升沿 就看是不是1   下降沿就看是不是0
	{ 
		printf("KEY_2按下\n");			 
        LED0=!LED0; 
	}		 
	 EXTI_ClearITPendingBit(EXTI_Line2);//清除LINE2上的中断标志位 
}

exti.h:

#ifndef   __EXTI_H
#define   __EXTI_H

#include "stm32f4xx.h"
#include "stm32f4xx_exti.h"

void EXTIX_Init(void);
void EXTI2_IRQHandler(void); 
 
#endif

main.c

#include "sys.h"
#include "delay.h"
#include "usart.h"
#include "exti.h"
#include "led.h"
#include "key.h"
#include "beep.h" 

int main(void)
{ 
    
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
	delay_init(168);    //初始化延时函数
     
	uart_init(9600); 	//串口初始化 
	LED_Init();	     //初始化LED端口 
    KEY_Init();          
	EXTIX_Init();       //初始化外部中断输入
     
	LED0=0;		     //先点亮红灯
     
	while(1)
	{
  	    printf("OK\r\n");	//打印OK提示程序运行
		delay_ms(1000);	  //每隔1s打印一次
	}

}

注意点:




一定要开启 SYSCFG 时钟,只要我们使用到外部中断,就必须打开 SYSCFG 时钟。



另外对于中断服务函数的模板一般是:



void EXTI2_IRQHandler(void) 
 
   
{ 
 
   
         if(EXTI_GetITStatus(EXTI_Line3)!=RESET)// 
    判断某个线上的中断是否发生
 
   
         { 
 
   
               
    …中断逻辑… 
 
   
              EXTI_ClearITPendingBit(EXTI_Line3); // 
    清除  
    LINE  
    上的中断标志位
 
   
          } 
 
   
}



固件库还提供了两个函数用来判断外部中断状态以及清除外部状态 ,标志位的函数 EXTI_GetFlagStatus 和 EXTI_ClearFlag,他们的作用和前面两个函数的作用类似。



只是在 EXTI_GetITStatus 函数中会先判断这种中断是否使能,使能了才去判断中断标志位,而 EXTI_GetFlagStatus 直接用来判断状态标志位。




另外对于本例中的中断服务程序:



第一种写法:按键按下后触发中断进入中断服务程序,然后延时消抖,再if判断是不是低电平,如果是低电平,则反转led;


void EXTI2_IRQHandler(void) 
   
 { 
   
     delay_ms(10); 
   
  

        if(KEY2==0)        //检测引脚电平:上升沿 就看是不是1   下降沿就看是不是0 
   
     {              
   
         LED0=!LED0;  
   
     }          
   
      EXTI_ClearITPendingBit(EXTI_Line2);//清除LINE2上的中断标志位  
   
 }



其实在中断服务程序里面不判断,也可实现led反转:




void EXTI2_IRQHandler(void) 
    
 {           
    
      LED0=!LED0;         
    
      EXTI_ClearITPendingBit(EXTI_Line2);//清除LINE2上的中断标志位  
    
 }



第二种写法是:通过中断标志位去判断是否真的发生中断:



当外部中断线上出现选定信号沿时,便会产生中断请求,对应的挂起位也会置 1 。



在挂起寄存器的对应位写“ 1 ”,将清除该中断请求。




void EXTI2_IRQHandler(void) 
    
 { 
    
         if(EXTI_GetITStatus(EXTI_Line2)!=RESET)判断某个线上的中断是否发生 
    
        {              
    
             LED0=!LED0;  
    
   

                 EXTI_ClearITPendingBit(EXTI_Line2);//清除LINE2上的中断标志位 
    
         }          
    
           
    
 }