STM32-07-外部中断

【STM32系列文章】

STM32-01-认识单片机
STM32-02-基础知识
STM32-03-HAL库
STM32-04-时钟树
STM32-05-SYSTEM文件夹
STM32-06-GPIO
STM32-07-外部中断
STM32-08-串口
STM32-09-IWDG和WWDG
STM32-10-定时器
STM32-11-电容触摸按键
STM32-12-OLED模块
STM32-13-MPU
STM32-14-FSMC_LCD
STM32-15-DMA
STM32-16-ADC
STM32-17-DAC

STM32 中断系统

1. 中断

  • 目的:中断系统是为使CPU具有对外界紧急事件的实时处理能力而设置的。
  • 中断过程:当CPU正在处理某事件的时候外界发生了紧急事件请求 中断发生,要求CPU暂停当前的工作,转而去处理这个紧急事件 中断响应和中断服务,处理完以后,再回到原来被中断的地方,继续原来的工作 中断返回
  • 中断系统:实现中断功能的部件称为中断系统,请求CPU中断的请求源称为中断源。微型机的中断系统一般允许有多个中断源,当几个中断源同时向CPU请求中断,要求它服务的时候,这就存在CPU优先响应哪个中断源请求的问题。因此设置中断优先级,CPU总是先响应优先级别最高的中断请求。
  • 作用与意义
    1. 实时控制
    2. 故障处理
    3. 数据传输

2. NVIC

  • 什么是NVIC?

    Nested vectored interrupt controller 即嵌套向量中断控制器,属于内核。

    支持256个中断,其中包含了16个系统中断和240个外部中断,并且具有256级的可编程中断设置。

    对于STM32F103ZET6的系统中断10个,外部中断60个,16中断优先级

    特性:

    • 68个可屏蔽中断通道
    • 16个可编程的优先等级
    • 低延时的异常和中断处理
    • 电源管理的控制
    • 系统控制寄存器的实现
  • NVIC相关寄存器介绍

    NVIC相关寄存器位数寄存器个数备注
    中断使能寄存器(ISER)328每个位控制一个中断
    中断失能寄存器(ICER)328每个位控制一个中断
    应用程序中断及复位控制寄存器(AIRCR)321位[10:8]控制优先级分组
    中断优先级寄存器(IPR)82408个位对应一个中断,而STM32只使用高4位
  • 工作原理

    内核中断是不可屏蔽中断,直接由SHPR决定优先级,外部中断过来以后,会首先经过ISERICER寄存器进行使能或者失能,之后由IPR寄存器确定优先级,然后去抢占CPU执行中断。
    在这里插入图片描述

  • 中断优先级

    • 抢占优先级:高抢占优先级可以打断正在执行的低抢占优先级中断
    • 响应优先级:当抢占优先级相同时,响应优先级高的先执行,但是不能互相打断
    • 自然优先级:中断向量表的优先级
    • 抢占和响应优先级相同时,自然优先级高的先执行,数值越小,表示优先级越高
  • 优先级分组

    优先级分组AIRCR[10:8]IPRx bit[7:4]分配分配结果
    0111None :[7:4]0位抢占优先级,4位响应优先级
    1110[7] :[6:4]1位抢占优先级,3位响应优先级
    2101[7:6] :[5:4]2位抢占优先级,2位响应优先级
    3100[7:5] :[4]3位抢占优先级,1位响应优先级
    4011[7:4] :None4位抢占优先级,0位响应优先级
  • NVIC的使用

    1. 设置中断分组 SCB_AIRCR
    2. 设置中断优先级 NVIC_IPRx
    3. 使能中断 NVIC_ISER

3. EXTI

  • 什么是EXTI?

    External interrupt/event Controller 即外部中断/事件控制器,由20个产生事件/中断请求的边沿检测器组成,非互联型产品有19个边沿检测器。每个输入线可以独立地配置输入类型和对应的触发事件。每个输入线都可以独立地被屏蔽。挂起寄存器保持着状态线的中断请求。

  • 特性:

    • 每个中断/事件都有独立地触发和屏蔽
    • 每个中断线都有专用的状态位
    • 支持多达20个软件的中断/事件请求
    • 检测脉冲宽度低于APB2时钟宽度的外部信号
  • 框图
    在这里插入图片描述

    功能框图包含两条主线:一是由输入线到NVIC中断控制器,二是由输入线到脉冲发生器。这就是EXTI的两大功能,产生中断与产生事件,两者从硬件上就存在不同。

    输入线是线路的信息输入端,它可以通过配置寄存器设置为任何一个GPIO口,或者是一些外设的事件。输入线一般都存在电平变化的信号。

    ①:边缘检测上升沿触发选择寄存器(EXTI_RTSR)和下降沿触发选择寄存器(EXTI_FTSR)。边沿检测电路以输入线作为信号输入端,如果检测到边缘跳变,就输出有效信号1,有效信号1会传输到②电路。

    ②:软件触发,两个信号输入端分别为软件中断事件寄存器(EXTI_SWIER)和边沿检测电路的输入信号。或门电路只要输入有效信号1,就会把有效信号输入到③和④电路。通过对软件中断事件寄存器的读写操作就可以启动中断/事件线。

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

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

4. AFIO

  • Alternate Function IO,即复用功能IO,主要用于重映射和外部中断映射配置

    寄存器AFIO_EXTICR1~4,配置EXTI中断线0~15具体对应哪个IO口

  • EXTI与IO对应关系
    在这里插入图片描述

5. 中断配置步骤

在这里插入图片描述

  • EXTI配置步骤:
    1. 使能GPIO时钟
    2. 设置GPIO工作模式
    3. 使能AFIO
    4. 设置EXTI和IO对应关系
    5. 设置EXTI屏蔽,上下沿
    6. 设置NVIC
    7. 设计中断服务函数

  • HAL库配置步骤:

    1. 使能GPIO时钟

      __HAL_RCC_GPIOx_CLK_ENABLE();
      
    2. GPIO/AFIO/EXTI配置

      HAL_GPIO_Init();
      
    3. 设置中断分组

      HAL_NVIC_SetPriorityGrouping();
      
    4. 设置中断优先级

      HAL_NVIC_SetPriority();
      
    5. 使能中断

      HAL_NVIC_EnableIRQ();
      
    6. 设计中断服务函数

      EXTIx_IRQHandler();
      
    7. 编写中断处理回调函数

      HAL_GPIO_EXTI_Callback();
      
  • 中断回调处理机制
    在这里插入图片描述

6. 外部中断代码

Exti.h

#ifndef _EXTI_H
#define _EXTI_H


#include "./SYSTEM/sys/sys.h"

void exti_init(void);

#endif

EXTI.c

#include "./BSP/EXTI/exti.h"
#include "./SYSTEM/delay/delay.h"
#include "./BSP/LED/led.h"
#include "./BSP/KEY/key.h"


//外部中断初始化
void exti_init(void)
{
    GPIO_InitTypeDef gpio_init_struct;
    
    //1.打开GPIO的时钟
    __HAL_RCC_GPIOE_CLK_ENABLE();
    
    //2.配置GPIO
    gpio_init_struct.Pin = GPIO_PIN_4;
    gpio_init_struct.Mode = GPIO_MODE_IT_FALLING;
    gpio_init_struct.Pull = GPIO_PULLUP;
    HAL_GPIO_Init(GPIOE, &gpio_init_struct);
    
    gpio_init_struct.Pin = GPIO_PIN_3;
    gpio_init_struct.Mode = GPIO_MODE_IT_FALLING;
    gpio_init_struct.Pull = GPIO_PULLUP;
    HAL_GPIO_Init(GPIOE, &gpio_init_struct);
    
    gpio_init_struct.Pin = GPIO_PIN_2;
    gpio_init_struct.Mode = GPIO_MODE_IT_FALLING;
    gpio_init_struct.Pull = GPIO_PULLUP;
    HAL_GPIO_Init(GPIOE, &gpio_init_struct);
    
    gpio_init_struct.Pin = GPIO_PIN_0;
    gpio_init_struct.Mode = GPIO_MODE_IT_RISING;
    gpio_init_struct.Pull = GPIO_PULLDOWN;
    HAL_GPIO_Init(GPIOA, &gpio_init_struct);
    
    //4.设置中断优先级
    HAL_NVIC_SetPriority(EXTI4_IRQn, 2, 0);
    HAL_NVIC_SetPriority(EXTI3_IRQn, 3, 0);
    HAL_NVIC_SetPriority(EXTI2_IRQn, 4, 0);
    HAL_NVIC_SetPriority(EXTI0_IRQn, 5, 0);
    
    //5.使能中断
    HAL_NVIC_EnableIRQ(EXTI4_IRQn);
    HAL_NVIC_EnableIRQ(EXTI3_IRQn);
    HAL_NVIC_EnableIRQ(EXTI2_IRQn);
    HAL_NVIC_EnableIRQ(EXTI0_IRQn);
}


void EXTI4_IRQHandler(void)
{
    HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_4);
    __HAL_GPIO_EXTI_CLEAR_FLAG(GPIO_PIN_4);
}

void EXTI3_IRQHandler(void)
{
    HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_3);
    __HAL_GPIO_EXTI_CLEAR_FLAG(GPIO_PIN_3);
}

void EXTI2_IRQHandler(void)
{
    HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_2);
    __HAL_GPIO_EXTI_CLEAR_FLAG(GPIO_PIN_2);
}

void EXTI0_IRQHandler(void)
{
    HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_0);
    __HAL_GPIO_EXTI_CLEAR_FLAG(GPIO_PIN_0);
}

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{ 
    delay_ms(20);
    
    switch(GPIO_Pin)
    {
        case GPIO_PIN_0: 
            if(KEY_UP == 1)
                LED0(0); 
            break;
        case GPIO_PIN_2:
            if(KEY2 == 0)            
                LED0(1); 
            break;
        case GPIO_PIN_3: 
            if(KEY1 == 0)
                LED1(0); 
            break;
        case GPIO_PIN_4: 
            if(KEY0 == 0)
                LED1(1); 
            break;
        default : 
            break;
    }
}

main.c

#include "./SYSTEM/sys/sys.h"
#include "./SYSTEM/usart/usart.h"
#include "./SYSTEM/delay/delay.h"
#include "./BSP/LED/led.h"
#include "./BSP/KEY/key.h"
#include "./BSP/BEEP/beep.h"
#include "./BSP/EXTI/exti.h"


int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    sys_stm32_clock_init(RCC_PLL_MUL9); /* 设置时钟, 72Mhz */
    delay_init(72);                     /* 延时初始化 */
    led_init();                         /* LED初始化 */
    key_init();     //按键初始化
    beep_init();    //蜂鸣器初始化
    exti_init();    //外部中断初始化
    
    while(1)
    {     

    }
}

声明:资料来源(战舰STM32F103ZET6开发板资源包)

  1. Cortex-M3权威指南(中文).pdf
  2. STM32F10xxx参考手册_V10(中文版).pdf
  3. STM32F103 战舰开发指南V1.3.pdf
  4. STM32F103ZET6(中文版).pdf
  5. 战舰V4 硬件参考手册_V1.0.pdf
  • 31
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
外部中断STM32单片机中的一种事件触发机制,它可以让外部设备向单片机发送一个中断信号,从而使单片机在正常执行的过程中立即转移到中断服务程序中执行相应的代码。中断可以用于实现实时性要求较高的功能,例如按键检测、传感器数据采集等。 在STM32单片机中配置外部中断需要经过以下几个步骤: 1. 设置中断系统的优先级分组。这是一个系统级的设置,可以通过寄存器配置来实现。该设置决定了不同中断源之间的优先级关系。 2. 配置外部中断引脚。首先需要将对应的GPIO口配置为输入模式,然后将外部中断引脚与GPIO口连接起来。具体的配置过程包括设置引脚模式、引脚速度、上下拉电阻等。 3. 配置外部中断触发方式。根据具体的需求,可以选择边沿触发或电平触发。边沿触发可以进一步分为上升沿触发、下降沿触发和双边沿触发。 4. 编写中断服务程序。中断服务程序是在中断发生时执行的代码段,用于处理中断事件。在中断服务程序中可以编写相应的逻辑,例如读取按键状态、处理传感器数据等。 5. 配置中断优先级和使能中断。通过中断控制器的设置,可以决定中断的优先级和是否允许中断。优先级高的中断会在多个中断同时发生时优先得到处理。 6. 清除中断挂起位。在中断服务程序中,需要手动清除中断挂起位,表示该中断已经被处理。 总的来说,配置外部中断需要进行引脚配置、触发方式设置、中断服务程序编写和中断优先级设置等步骤。这样,当外部中断事件发生时,单片机会立即转移到中断服务程序中执行相应的代码,从而实现相应的功能。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [STM32单片机--外部中断详解](https://blog.csdn.net/WYFDMW/article/details/107847097)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [【STM32】HAL库-睡眠模式-外部中断唤醒demo](https://download.csdn.net/download/qq_45607873/85319880)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值