STM32HAL库--NVIC和EXTI

1. 外部中断实验

1.1 NVIC和EXTI简介

1.1.1 NVIC简介

        NVIC 即嵌套向量中断控制器,全称 Nested vectored interrupt controller。是ARM Cortex-M处理器中用于管理中断的重要组件。负责处理中断请求,分配优先级,并协调中断的触发和响应。
它是内核的器件,所以它的更多描述可以看内核有关的资料。

        M3/M4/M7 内核都是支持 256 个中断,其中包含了 16 个系统中断240 个外部中断,并且具有 256 级的可编程中断设置。芯片厂商一般不会把内核的这些资源全部用完,如 STM32F429 的系统中断有 10 个,外部中断有 86 个。

中断的宏定义可以在stm32fxxx.h里面找,中断向量表在startup_stm32fxxxx.s启动文件里面找

 1.1.2 NVIC寄存器

        NVIC 相关的寄存器定义了可以在 core_cm4.h 文件中找到。

typedef struct
{
     __IO uint32_t ISER[8U]; /* 中断使能寄存器 */
     uint32_t RESERVED0[24U];
     __IO uint32_t ICER[8U]; /* 中断除能寄存器 */
     uint32_t RSERVED1[24U];
     __IO uint32_t ISPR[8U]; /* 中断使能挂起寄存器 */
     uint32_t RESERVED2[24U];
     __IO uint32_t ICPR[8U]; /* 中断解挂寄存器 */
     uint32_t RESERVED3[24U];
     __IO uint32_t IABR[8U]; /* 中断有效位寄存器 */
     uint32_t RESERVED4[56U];
     __IOM uint8_t IP[240U]; /* 中断优先级寄存器(8Bit 位宽) */
     uint32_t RESERVED5[644U];
     __O uint32_t STIR; /* 中断触发中断寄存器 */
} NVIC_Type;

         STM32F429 的中断在这些寄存器的控制下有序的执行的。只有了解这些中断寄存器,才
能方便的使用 STM32F429 的中断。下面重点介绍这几个寄存器:

        ISER[8]:ISER 全称是:Interrupt Set Enable Registers,这是一个中断使能寄存器组,包含8个32位寄存器,每个位控制一个中断。但是STM32F429 的可屏蔽中断最多只有 86 个,所以只用到前4个(ISER[0~3]),总共可以表示 128 个中断。要使能某个中断,必须设置相应的ISER位为1,使该中断被使能(这里仅仅是使能,还要配合中断分组、屏蔽、IO 口映射等设置才算是一个完整的中断设置)。具体每一位对应哪个中断,请参考 STM32F429xx.h文件。

        ICER[8]:全称是:Interrupt Clear Enable Registers,是一个中断除能寄存器组。该寄存器组与 ISER 的作用恰好相反,是用来清除某个中断的使能的。其对应位的功能,也和 ICER 一样。这里要专门设置一个 ICER 来清除中断位,而不是向 ISER 写 0 来清除,是因为 NVIC 的这些寄存器都是写 1 有效的,写 0 是无效的。

        ISPR[8]:全称是:Interrupt Set Pending Registers,是一个中断挂起控制寄存器组。每个位对应的中断和 ISER 是一样的。通过置 1,可以将正在进行的中断挂起,而执行同级或更高级别的中断。写 0 是无效的。

        ICPR[8]:全称是:Interrupt Clear Pending Registers,是一个中断解挂控制寄存器组。其作
用与 ISPR 相反,对应位也和 ISER 是一样的。通过设置 1,可以将挂起的中断解挂。写 0 无效。

         IABR[8]:全称是:Interrupt Active Bit Registers,是一个中断激活标志位寄存器组。对应位所代表的中断和 ISER 一样,如果为 1,则表示该位所对应的中断正在被执行。这是一个只读寄存器,通过它可以知道当前在执行的中断是哪一个。在中断执行完了由硬件自动清零。

        IPR[240]:全称是:Interrupt Priority Registers,是一个中断优先级控制的寄存器组IP 寄存器组由 240 个 8bit 的寄存器组成,每个可屏蔽中断占用 8bit,这样总共可以表示 240 个可屏蔽中断。而STM32F429只用到了其中的 86个。IP[85]~IP[0]分别对应中断 85~0。而每个可屏蔽中断占用的8bit 并没有全部使用,而是只用了高 4 位。这 4 位,又分为抢占优先级和子优先级。抢占优先级在前,子优先级在后。而这两个优先级各占几个位又要根据 SCB->AIRCR 中的中断分组设置来决定。

NVIC寄存器总结

        NVIC,嵌套向量中断控制器,ARM Cortex-M内核的组件,

用来中断使能挂起记录中断激活标志位配置中断优先级

        分别对应寄存器组有

        中断使能和除能  ISER[8]  ICER[8]                   8个*32位寄存器

        中断挂起和解挂  ISPR[8]  ICPR[8]

        记录中断激活标志位 IABR[8]                                   Interrupt Active Bit Registers

        配置中断优先级        IP[240]               240*8位寄存器,仅用高4位,放抢占优先级和子优先级

        SCB->AIRCR的中断分组设置决定抢占优先级和子优先级占几位      

1.1.3 中断优先级

        STM32 中的中断优先级可以分为:抢占式优先级和响应优先级,响应优先级也称子优先
级,每个中断源都需要被指定这两种优先级。

        抢占优先级:抢占优先级高的中断 可以 打断正在执行的抢占优先级低的中断。
        
响应优先级:抢占优先级相同,响应优先级高的中断 不能 打断响应优先级低的中断。但可以按响应优先级顺序先后执行。

        当多个中断的抢占式优先级和响应优先级相同时,那么就遵循自然优先级,看中断向量表的中断排序,数值越小,优先级越高。

        在NVIC中有 IPR[60] 控制中断优先级。每个 IPR 寄存器每8位又分为一组,可以分4组,所以就有了 240 组宽度为 8bit 的中断优先级控制寄存器,原则上每个外部中断可配置的优先级为 0~255,数值越小,优先级越高。但是实际上 M3 /M4 /M7 芯片为了精简设计,只使用了高四位[7:4],低四位取零,这样以至于最多只有 16 级中断嵌套,即 2^4=16。

正点原子开发指南V1.0该章节所说的 IP[240]和 IPR0~IPR59 其实是一个东西,指的是中断优先级控制的寄存器组,60个32位 IPR 寄存器,8位一组,只使用高4位,由SCB->AIRCR设置抢占优先级和响应优先级分别占几位

1.1.4 NVIC相关函数

        ST 公司把 core_cm4.h 文件的 NVIC 相关函数封装到 stm32f4xx_hal_cortex.c 文件中。

1.1.4.1 中断优先级分组

        首先要讲解的是中断优先级分组函数 NVIC_SetPriorityGrouping,其声明如下:

    void HAL_NVIC_SetPriorityGrouping(uint32_t PriorityGroup);
    函数形参:
    NVIC_PRIORITYGROUP_0 到 NVIC_PRIORITYGROUP_4(共5组)

        这个函数唯一目的就是通过设置 SCB->AIRCR 寄存器来设置中断优先级分组。

1.1.4.2 中断优先级设置

        中断优先级设置函数 NVIC_SetPriority,其声明如下:

void HAL_NVIC_SetPriority(IRQn_Type IRQn, uint32_t PreemptPriority, 
                          uint32_t SubPriority);
    函数形参:
    形参1  中断号, IRQn_Type 定义的枚举类型,定义在 stm32f429xx.h
    形参2  抢占优先级
    形参3  子优先级

        用于设置中断的抢占优先级和子优先级

1.1.4.3 中断使能

        中断使能函数 NVIC_EnableIRQ,其声明如下:

    void HAL_NVIC_EnableIRQ(IRQn_Type IRQn);
1.1.4.4 中断除能

        中断除能函数 NVIC_DisableIRQ,其声明如下:

    void HAL_NVIC_disableIRQ(IRQn_Type IRQn);
1.1.4.5 系统复位

        系统复位函数 HAL_NVIC_SystemReset 函数,其声明如下:

    void HAL_NVIC_SystemReset(void);

//SCB(System Control Block)寄存器中的特定位,例如SCB->AIRCR寄存器的位[2]:SYSRESETREQ。
//通过将这个位设置为1,可以触发系统的软件复位(Software System Reset)。

        用于软件复位系统。

1.1.5 EXTI简介

        EXTI 即是外部中断和事件控制器,它由三个部分组成:APB 接口访问的寄存器模块事件输入触发模块屏蔽模块。其中寄存器块包含了所有 EXTI 寄存器,事件输入触发模块提供事件输入边沿触发逻辑,屏蔽模块为不同的唤醒·、中断和事件输出及其屏蔽功能提供事件分配。EXTI 的功能框图如下图所示:

        从 EXTI 功能框图可以看到有两条主线,一条是由输入线到 NVIC 中断控制器,一条是由输入线到脉冲发生器。这就恰恰是 EXTI 的两大部分功能,产生中断与产生事件,两者从硬件上就存在不同。 

        EXTI 功能框图的产生中断的线路,最终信号是流入 NVIC 中断控制器中。

        下面让我们看一下 EXTI 功能框图的产生中断的线路,最终信号是流入 NVIC 控制器中。输入线是线路的信息输入端,它可以通过配置寄存器设置为任何一个 GPIO 口,或者是一些外设事件。

        标号①是一个边沿检测电路,包括边沿检测电路,上升沿触发选择寄存器(EXTI_RTSR)和
下降沿触发选择寄存器(EXTI_FTSR)。

        标号②是一个或门电路,它的两个信号输入端分别是软件中断事件寄存器(EXTI_SWIER)和边沿检测电路的输入信号。或门电路只要输入端有信号‘1’,就会输出‘1’,所以就会输出‘1’到标号③电路和标号④电路。

        标号③是一个与门电路,它的两个信号输入端分别是中断屏蔽寄存器(EXTI_IMR)和标号②电路输出信号。与门电路要求输入都为‘1’才输出‘1’,这样子的情况下,如果中断屏蔽寄存器(EXTI_IMR)设置为 0 时,不管从标号②电路输出的信号特性如何,最终标号③电路输出的信号都是 0;假如中断屏蔽寄存器(EXTI_IMR)设置为 1 时,最终标号③电路输出的信号才由标号②电路输出信号决定,这样子就可以简单控制 EXTI_IMR来实现中断的目的。标号④电路输出‘1’就会把请求挂起寄存器(EXTI_PR)对应位置 1。

        最后,请求挂起寄存器(EXTI_PR)的内容就输出到 NVIC 内,实现系统中断事件的控制。

        接下来我们看看 EXTI 功能框图的产生事件的线路。

        产生事件线路是从标号 2 (软件中断事件寄存器)之后与中断线路有所不用,之前的线路都是共用的。标号④是一个与门,输入端来自标号 2 电路以及来自于事件屏蔽寄存器(EXTI_EMR)。

        标号④电路输出有效信号 1 就会使脉冲发生器电路产生一个脉冲,而无效信号就不会使其产生脉冲信号。脉冲信号产生可以给其他外设电路使用,例如定时器,模拟数字转换器等,这样的脉冲信号一般用来触发 TIM 或者 ADC 开始转换。

        产生中断线路目的使把输入信号输入到 NVIC,进一步运行中断服务函数,实现功能。而产生事件线路目的是传输一个脉冲信号给其他外设使用,属于硬件级功能。

EXTI功能框图总结

        EXTI功能框图有两条线路,

        一条是中断发生线路:

                GPIO 输入脉冲信号,经过

                1. 边沿检测电路

                2. 上升沿、下降沿触发选择器

                3. 软件中断事件寄存器

                4. 中断屏蔽寄存器

                5. 中断挂起请求寄存器

                最后输入 NVIC,进行中断控制

        一条是事件发生线路:

                GPIO 输入脉冲信号,经过

                1. 边沿检测电路

                2. 上升沿、下降沿触发选择器

                3. 软件中断事件寄存器

                4. 事件屏蔽寄存器

                5. 脉冲发生器

                最后产生脉冲信号,给其他外设使用

        EXTI支持 22个外部中断/事件请求,这些都是信息输入端,也就是上面提及到了输入线,具体如下:

        EXTI 线 0~15:对应外部 IO 口的输入中断
        EXTI 线 16:连接到 PVD 输出
        EXTI 线 17:连接到 RTC 闹钟事件
        EXTI 线 18:连接到 USB OTG FS 唤醒事件
        EXTI 线 19:连接到以太网唤醒事件
        EXTI 线 20:连接到 USB OTG HS(在 FS 中配置)唤醒事件
        EXTI 线 21:连接到 RTC 入侵和时间戳事件
        EXTI 线 22:连接到 RTC 唤醒事件

STM32F4 供给 IO 口使用的中断线只有 16 个,每个中断线对应了最多 9 个 IO 口,以线 0 为例:它对应了 GPIOA.0、GPIOB.0、GPIOC.0、GPIOD.0、GPIOE.0、GPIOF.0、GPIOG.0、GPIOH.0、GPIOI.0。而中断线每次只能连接到 1 个 IO 口上,这样就需要通过配置决定对应的中断线配置到哪个 GPIO 上了

EXTI 程序设计

        例程功能:

        通过外部中断的方式,让开发板上的独立按键控制LED灯。

        思路:

        一个按键按下,信号输入GPIO,触发外部中断点灯。

        EXTI 外部中断配置步骤:

        1)使能按键的 IO 口时钟

    __HAL_RCC_GPIOH_CLK_ENABLE()

        2)设置按键的 IO 口模式,触发条件,开启 SYSCFG 时钟,设置 IO 口与中断线的映射关系

        3) 配置中断优先级并使能中断

    //HAL_INIT函数中设置中断优先级分组,配置Systic时钟
    //GPIO_INIT函数配置了SYSCFG时钟

    GPIO_InitTypeDef gpio_init_struct; 

    gpio_init_struct.Pin = GPIO_PIN_3;                       /* KEY0引脚 */
    gpio_init_struct.Mode = GPIO_MODE_INPUT;                    /* 输入 */
    gpio_init_struct.Pull = GPIO_PULLUP;                        /* 上拉 */
    gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;              /* 高速 */
    gpio_init_struct.Mode = GPIO_MODE_IT_FALLING;               /* 下降沿触发 */
    HAL_GPIO_Init(KEY0_GPIO_PORT, &gpio_init_struct);           /* KEY0引脚模式设置,上拉输入 

    HAL_NVIC_SetPriority(KEY0_INT_IRQn, 0, 2);               /* 抢占0,子优先级2 */
    HAL_NVIC_EnableIRQ(EXTI3_IRQn);                       /* 使能中断线3 */

        4)编写中断服务函数

        每开启一个中断,就必须编写其对应的中断服务函数,否则将导致死机(CPU 将找不到中断服务函数)。中断服务函数接口厂家已经在 startup_stm32f429xx.s 中做好了。STM32F4 的 IO 口外部中断函数只有 7 个,分别为:

        void EXTI0_IRQHandler();
        void EXTI1_IRQHandler();
        void EXTI2_IRQHandler();
        void EXTI3_IRQHandler();
        void EXTI4_IRQHandler();
        void EXTI9_5_IRQHandler();
        void EXTI15_10_IRQHandler();

        中断线 0-4,每个中断线对应一个中断函数,中断线 5-9 用中断函数 EXTI9_5_IRQHandler,中断线10-15共用中断函数 EXTI15_10_IRQHandler。一般情况下,我们可以把中断控制逻辑直接编写在中断服务函数中,但是 HAL 库把中断处理过程进行了简单封装,请看下面步骤 5 讲解。

        5) 编写中断处理回调函数 GPIO_EXTI_Callback

        HAL 库为了用户使用方便,提供了一个中断通用入口函数 HAL_GPIO_EXTI_IRQHandler,
在该函数内部直接调用回调函数 GPIO_EXTI_Callback。

        我们先来看一下 HAL_GPIO_EXTI_IRQHandler 函数定义:

void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
{
     if(__HAL_GPIO_EXTI_GET_IT(GPIO_Pin) != 0x00U)
     {
     __HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin); /* 清中断标志位 */
     HAL_GPIO_EXTI_Callback(GPIO_Pin); /* 外部中断回调函数 */
     }
}

        该函数实现的作用非常简单,通过入口参数 GPIO_Pin 判断中断来自哪个 IO 口,然后清除相应的中断标志位,最后调用回调函数 HAL_GPIO_EXTI_Callback()实现控制逻辑。在所有的外部中断服务函数中直接调用外部中断共用处理函数 HAL_GPIO_EXTI_IRQHandler,然后在回调函数 HAL_GPIO_EXTI_Callback 中通过判断中断是来自哪个 IO 口编写相应的中断服务控制逻辑。

void KEY0_INT_IRQHandler(void)
{
     HAL_GPIO_EXTI_IRQHandler(KEY0_INT_GPIO_PIN); 
    /* 调用中断处理公用函数 清除 KEY0 所在中断线 的中断标志位 */
     __HAL_GPIO_EXTI_CLEAR_IT(KEY0_INT_GPIO_PIN); 
    /* HAL 库默认先清中断再处理回调,退出时再清一次中断,避免按键抖动误触发 */
}
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
     delay_ms(20); /* 消抖 */
     switch(GPIO_Pin)
     {
         case KEY0_INT_GPIO_PIN:
         if (KEY0 == 0)
         {
         LED1_TOGGLE(); /* LED1 状态取反 */
         LED0_TOGGLE(); /* LED0 状态取反 */
         }
         break;
         default : break;
     }
}

  • 13
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值