STM32_HAL库_外部中断

一、设置分组 

stm32f1xx_hal_cortex.c

查看分组

五个形参,分组0~4

stm32f1xx_hal.c

设置了分组为2, 此工程就不需要再设置了

再回到stm32f1xx_hal_cortex.c

查看NVIC_SetPriorityGrouping的定义,若无法跳转,先编译一下,

继续找定义

这段代码是用于设置 NVIC 的优先级分组的内联函数。关键部分:

  • SCB->AIRCR = reg_value;: 最后,将更新后的寄存器值写入 SCB_AIRCR 寄存器,完成对 NVIC 优先级分组的设置。

总的来说,这个函数的目的是安全地设置 NVIC 的优先级分组,确保在进行任何更改之前正确地处理了旧的配置值,并且将写入操作限制在允许的范围内。

二、设置优先级

stm32f1xx_hal_cortex.c

没有返回值,三个形参

这段代码是用于设置特定中断的优先级的函数。

  1. void HAL_NVIC_SetPriority(IRQn_Type IRQn, uint32_t PreemptPriority, uint32_t SubPriority): 这是一个函数声明,用于设置特定中断的优先级。它接受三个参数,分别是中断号 IRQn,抢占优先级 PreemptPriority,以及响应优先级 SubPriority

  2. uint32_t prioritygroup = 0x00U;: 这一行定义了一个变量 prioritygroup 并初始化为 0x00U,该变量用于存储当前 NVIC 的优先级分组。

  3. assert_param(IS_NVIC_SUB_PRIORITY(SubPriority));assert_param(IS_NVIC_PREEMPTION_PRIORITY(PreemptPriority));: 这两行是参数检查,确保传递给函数的 SubPriorityPreemptPriority 在有效范围内。如果参数不在有效范围内,这些断言将会触发。

  4. prioritygroup = NVIC_GetPriorityGrouping();: 这一行调用了 NVIC_GetPriorityGrouping() 函数,获取当前 NVIC 的优先级分组。

  5. NVIC_SetPriority(IRQn, NVIC_EncodePriority(prioritygroup, PreemptPriority, SubPriority));: 这一行调用了 NVIC_SetPriority() 函数,设置指定中断的优先级。它使用了 NVIC_EncodePriority() 函数将抢占优先级和子优先级编码为一个优先级值,并将其传递给 NVIC_SetPriority() 函数来设置中断的优先级。

这个函数的目的是设置特定中断的优先级,确保传递给函数的优先级参数在有效范围内,并将优先级值编码后传递给 NVIC。

查看IRQn_Type定义

跳转到stm32f103xe.h

举个例子

RTC_IRQn

中断号是3

NVIC_SetPriority

看定义 

跳转到这里,再去看定义

 跳转到:

用于设置特定中断的优先级的内联函数。每个部分的作用:

  1. __STATIC_INLINE void __NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority): 这是一个内联函数的声明,用于设置特定中断的优先级。它接受两个参数,一个是中断号 IRQn,另一个是优先级 priority

  2. if ((int32_t)(IRQn) >= 0): 这个条件语句检查中断号是否为非负值,如果是非负值,则说明是可编程中断,需要设置 NVIC->IP 寄存器的相应位置;如果是负值,则说明是固定优先级中断,需要设置 SCB->SHP 寄存器的相应位置。

  3. NVIC->IP[((uint32_t)IRQn)] = (uint8_t)((priority << (8U - __NVIC_PRIO_BITS)) & (uint32_t)0xFFUL);: 如果是可编程中断,就将优先级值左移以便正确放置到 NVIC->IP 寄存器相应位置。这里通过移位来确保优先级值的正确性,同时保留了 __NVIC_PRIO_BITS 指定的位数,这是由硬件定义的。

  4. SCB->SHP[(((uint32_t)IRQn) & 0xFUL)-4UL] = (uint8_t)((priority << (8U - __NVIC_PRIO_BITS)) & (uint32_t)0xFFUL);: 如果是固定优先级中断,则将优先级值左移并设置到 SCB->SHP 寄存器的相应位置。由于固定优先级中断的索引不是从零开始的,因此需要进行一些偏移和掩码操作。

这个函数的目的是根据中断类型设置对应中断的优先级,确保优先级值在正确的范围内,并将其正确地写入到相应的寄存器中。

stm32f1xx_hal_cortex.c对cortex_cm3.h文件里的函数进行了封装

这段代码是一个内联函数,用于检查特定中断是否已经被使能。让我们逐步解释每个部分的作用:

  1. __STATIC_INLINE uint32_t __NVIC_GetEnableIRQ(IRQn_Type IRQn): 这是一个内联函数的声明,用于获取特定中断是否已经被使能。它接受一个参数,即中断号 IRQn

  2. if ((int32_t)(IRQn) >= 0): 这个条件语句检查中断号是否为非负值,如果是非负值,则说明是可编程中断,需要检查 NVIC->ISER 寄存器的相应位;如果是负值,则说明是固定优先级中断,这些中断无法被使能,直接返回 0。

  3. return((uint32_t)(((NVIC->ISER[(((uint32_t)IRQn) >> 5UL)] & (1UL << (((uint32_t)IRQn) & 0x1FUL))) != 0UL) ? 1UL : 0UL));: 如果是可编程中断,则检查 NVIC->ISER 寄存器中相应位是否被置位,如果被置位则返回 1,否则返回 0。这里通过对中断号进行移位和掩码操作来确定具体的位位置。

  4. return(0U);: 如果是固定优先级中断,则直接返回 0,表示该中断无法被使能。

这个函数的目的是根据中断类型检查特定中断是否已经被使能,并返回相应的结果。

三、EXTI

p60

P61

两种中断

四、GPIO外部中断

复制跑马灯实验,BSP新建EXTI文件夹,keil5中新建exti.c与exti.h并添加到EXTI文件夹,添加到工程里

复制拿过来用

 若无法跳转定义,参考这份博客

http://t.csdnimg.cn/RCJyL

重新编译程序

再次执行Go To Definition of ''的操作,可以了

参考

需选用外部中断下降沿触发,且输入上拉

#define  GPIO_MODE_IT_RISING                    0x10110000u   /*!< External Interrupt Mode with Rising edge trigger detection          */
#define  GPIO_MODE_IT_FALLING                   0x10210000u   /*!< External Interrupt Mode with Falling edge trigger detection         */
#define  GPIO_MODE_IT_RISING_FALLING            0x10310000u   /*!< External Interrupt Mode with Rising/Falling edge trigger detection  */

#define  GPIO_MODE_EVT_RISING                   0x10120000u   /*!< External Event Mode with Rising edge trigger detection               */
#define  GPIO_MODE_EVT_FALLING                  0x10220000u   /*!< External Event Mode with Falling edge trigger detection              */
#define  GPIO_MODE_EVT_RISING_FALLING           0x10320000u   /*!< External Event Mode with Rising/Falling edge trigger detection       */

这些宏定义了用于 GPIO(通用输入/输出)引脚的不同模式,用于处理外部中断和事件,根据触发边沿的不同来触发。具体而言:

  • GPIO_MODE_IT_RISING:表示外部中断模式,检测上升沿触发。
  • GPIO_MODE_IT_FALLING:表示外部中断模式,检测下降沿触发。
  • GPIO_MODE_IT_RISING_FALLING:表示外部中断模式,同时检测上升沿和下降沿触发。

类似地:

  • GPIO_MODE_EVT_RISING:表示外部事件模式,检测上升沿触发。
  • GPIO_MODE_EVT_FALLING:表示外部事件模式,检测下降沿触发。
  • GPIO_MODE_EVT_RISING_FALLING:表示外部事件模式,同时检测上升沿和下降沿触发。

这些宏通常与 GPIO 配置函数一起使用,以设置 GPIO 引脚的所需模式,以便根据特定边沿触发来处理外部中断或事件。

上下拉配置

#define  GPIO_NOPULL        0x00000000u   /*!< No Pull-up or Pull-down activation  */
#define  GPIO_PULLUP        0x00000001u   /*!< Pull-up activation                  */
#define  GPIO_PULLDOWN      0x00000002u   /*!< Pull-down activation                */

这些宏定义了用于 GPIO 引脚的不同上拉和下拉配置选项。具体来说:

  • GPIO_NOPULL:表示不启用上拉或下拉。
  • GPIO_PULLUP:表示启用上拉。
  • GPIO_PULLDOWN:表示启用下拉。

这些选项通常用于配置 GPIO 引脚的电气特性,以确保引脚在特定条件下的正确操作,如减少干扰、提高信号质量等。

选择输入上拉

中断优先级:HAL_NVIC_SetPriority()

go to definition

选择中断号

笔者选用的是B1,选择EXTI1_IRQn

设置抢断优先级与响应优先级

笔者设置的是2和0

使能中断:HAL_NVIC_EnableIRQ()

笔者的是EXTI1_IRQn

设计中断服务函数:

去到配置文件查找

选择EXTI1_IRQHandler

 

void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
{
  /* EXTI line interrupt detected */
  if (__HAL_GPIO_EXTI_GET_IT(GPIO_Pin) != 0x00u)
  {
    __HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin);
    HAL_GPIO_EXTI_Callback(GPIO_Pin);
  }
}

这段代码是一个 HAL(Hardware Abstraction Layer,硬件抽象层)的 GPIO(通用输入/输出)外部中断处理函数。让我来解释它:

  • HAL_GPIO_EXTI_IRQHandler 是一个函数,用于处理外部中断事件。它接收一个参数 GPIO_Pin,表示触发了外部中断的 GPIO 引脚。

  • 在函数内部,首先检测是否检测到了 EXTI(外部中断)线的中断事件。

  • 如果 GPIO_Pin 对应的 EXTI 中断事件标志被置位(不为 0),则说明该 GPIO 引脚触发了外部中断。

  • 接着,通过 __HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin) 清除相应的中断标志。

  • 最后,调用 HAL_GPIO_EXTI_Callback(GPIO_Pin),这是一个回调函数,用于在外部中断触发时执行用户定义的操作。这个函数的实现应该由用户在程序的其他地方提供。

用于在外部中断事件发生时执行相应的操作,如清除中断标志并调用用户定义的回调函数。

回调函数

__weak void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
  /* Prevent unused argument(s) compilation warning */
  UNUSED(GPIO_Pin);
  /* NOTE: This function Should not be modified, when the callback is needed,
           the HAL_GPIO_EXTI_Callback could be implemented in the user file
   */
}

用于处理 GPIO 外部中断回调的弱定义函数 HAL_GPIO_EXTI_Callback

  • __weak 关键字表示这是一个弱定义的函数,它可以被用户在程序的其他地方重新定义以满足特定需求。

  • 这个函数的目的是处理 GPIO 外部中断事件。它接收一个参数 GPIO_Pin,表示触发了外部中断的 GPIO 引脚。

  • 函数内部通过调用 UNUSED(GPIO_Pin) 来避免编译器产生未使用参数的警告。这是因为在某些情况下,用户可能不需要使用该参数。

  • 注释部分指出,如果需要在外部中断发生时执行特定的操作,用户应该在自己的代码文件中实现 HAL_GPIO_EXTI_Callback 函数,并在其中编写所需的处理逻辑。

这个函数提供了一个默认的外部中断处理回调函数,并允许用户根据需要进行修改或扩展。

中断信号到来,去翻转led引脚的信号,在led.h中查看引脚是B5

4.1使能GPIO时钟

__HAL_RCC_GPIOB_CLK_ENABLE();

4.2 HAL_GPIO_init

	GPIO_InitTypeDef gpio_init_struct;
	__HAL_RCC_GPIOB_CLK_ENABLE();

    gpio_init_struct.Pin = LED0_GPI1_PIN;                   /* 引脚B1 */
    gpio_init_struct.Mode = GPIO_MODE_IT_FALLING;            /* 推挽输出 */
    gpio_init_struct.Pull = GPIO_PULLUP;                    /* 上拉 */
    gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;          /* 高速 */
    HAL_GPIO_Init(GPIOB, &gpio_init_struct);       /* 初始化 */
		

4.3 设置中断分组

配置文件已设置分组为2 

4.4 设置中断优先级

HAL_NVIC_SetPriority(EXTI1_IRQn,2,0);

4.5 使能中断

HAL_NVIC_EnableIRQ(EXTI1_IRQn);

4.6 设计中断服务函数

void EXTI1_IRQHandler(void)
{
	HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_1);

}

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
	delay_ms(20);
	if(GPIO_Pin=GPIO_Pin_1)
	{
		if(HAL_GPIO_ReadPin(GPIOB,GPIO_Pin_1) ==0)
		{
			HAL_GPIO_TogglePin(GPIOB,GPIO_Pin_5);
		
		}
	}

}

exti.h

函数声明

void exti_init(void);

EXTI1_IRQHandler与HAL_GPIO_EXTI_Callback已经在配置文件里声明过

startup_stm32f103xe.s

stm32f1xx_hal_gpio.h

五、源码

exti.h

#ifndef _EXTI_H
#define _EXTI_H
#include "./SYSTEM/sys/sys.h"


void exti_init(void);



#endif

exit.c

#include "./BSP/EXTI/exti.h"


void exti_init(void)
{

	GPIO_InitTypeDef gpio_init_struct;
	__HAL_RCC_GPIOB_CLK_ENABLE();

    gpio_init_struct.Pin = GPIO_PIN_1;                   /* 引脚B1 */
    gpio_init_struct.Mode = GPIO_MODE_IT_FALLING;            /* 推挽输出 */
    gpio_init_struct.Pull = GPIO_PULLUP;                    /* 上拉 */
    gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;          /* 高速 */
    HAL_GPIO_Init(GPIOB, &gpio_init_struct);       /* 初始化 */
		

	HAL_NVIC_SetPriority(EXTI1_IRQn,2,0);
	HAL_NVIC_EnableIRQ(EXTI1_IRQn);
}

void EXTI1_IRQHandler(void)
{
	HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_1);

}

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
	delay_ms(20);
	if(GPIO_Pin == GPIO_PIN_1)
	{
		if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_1) ==0)
		{
			HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_5);
		
		}
	
	}

}

main.c


#include "./SYSTEM/sys/sys.h"
#include "./SYSTEM/delay/delay.h"
#include "./SYSTEM/usart/usart.h"
#include "./BSP/LED/led.h"
#include "./BSP/EXTI/exti.h"


int main(void)
{
    HAL_Init();                                 /* 初始化HAL库 */
    sys_stm32_clock_init(RCC_PLL_MUL9);         /* 设置时钟,72M */
    delay_init(72);                             /* 初始化延时函数 */
    led_init();                                 /* 初始化LED */
    exti_init();
	
	
    while(1)
    {
		HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_5);
		delay_ms(5000);
		
    }
}

参考

【【正点原子】手把手教你学STM32 HAL库开发全集【真人出镜】STM32入门教学视频教程 单片机 嵌入式】https://www.bilibili.com/video/BV1bv4y1R7dp?p=56&vd_source=be33b1553b08cc7b94afdd6c8a50dc5a

相关推荐

最近更新

  1. TCP协议是安全的吗?

    2024-06-08 09:20:06       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-06-08 09:20:06       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-06-08 09:20:06       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-06-08 09:20:06       20 阅读

热门阅读

  1. 部件库(Widget Factory)

    2024-06-08 09:20:06       10 阅读
  2. 图像处理 -- 自适应色调映射(ATM)整理

    2024-06-08 09:20:06       9 阅读
  3. python项目中到底使用什么解释器更合适?

    2024-06-08 09:20:06       9 阅读
  4. python安装Django

    2024-06-08 09:20:06       7 阅读
  5. 设计模式-责任链模式

    2024-06-08 09:20:06       5 阅读
  6. react组件中的this

    2024-06-08 09:20:06       8 阅读
  7. 换热器设计参数的选用

    2024-06-08 09:20:06       7 阅读
  8. Stream对List进行排序

    2024-06-08 09:20:06       8 阅读