历史上的今天
返回首页

历史上的今天

今天是:2024年09月18日(星期三)

正在发生

2019年09月18日 | 第17章 EXTI—外部中断/事件控制器—零死角玩转STM32-F429系列

2019-09-18 来源:eefocus

本章参考资料:《STM32F4xx中文参考手册》系统配置控制器以及中断和事件章节。


上一章节我们已经详细介绍了NVIC,对STM32F4xx中断管理系统有个全局的了解,我们这章的内容是NVIC的实例应用,也是STM32F4xx控制器非常重要的一个资源。学习本章时,配合《STM32F4xx中文参考手册》系统配置控制器以及中断和事件章节一起阅读,效果会更佳,特别是涉及到寄存器说明的部分。


特别说明,本书内容是以STM32F42xxx系列控制器资源讲解。


17.1 EXTI简介

外部中断/事件控制器(EXTI)管理了控制器的23个中断/事件线。每个中断/事件线都对应有一个边沿检测器,可以实现输入信号的上升沿检测和下降沿的检测。EXTI可以实现对每个中断/事件线进行单独配置,可以单独配置为中断或者事件,以及触发事件的属性。


17.2 EXTI功能框图

EXTI的功能框图包含了EXTI最核心内容,掌握了功能框图,对EXTI就有一个整体的把握,在编程时就思路就非常清晰。EXTI功能框图见图 171。


在图 171可以看到很多在信号线上打一个斜杠并标注"23"字样,这个表示在控制器内部类似的信号线路有23个,这与EXTI总共有23个中断/事件线是吻合的。所以我们只要明白其中一个的原理,那其他22个线路原理也就知道了。

图 171 EXTI功能框图


EXTI可分为两大部分功能,一个是产生中断,另一个是产生事件,这两个功能从硬件上就有所不同。


首先我们来看图 171中红色虚线指示的电路流程。它是一个产生中断的线路,最终信号流入到NVIC控制器内。


编号1是输入线,EXTI控制器有23个中断/事件输入线,这些输入线可以通过寄存器设置为任意一个GPIO,也可以是一些外设的事件,这部分内容我们将在后面专门讲解。输入线一般是存在电平变化的信号。


编号2是一个边沿检测电路,它会根据上升沿触发选择寄存器(EXTI_RTSR)和下降沿触发选择寄存器(EXTI_FTSR)对应位的设置来控制信号触发。边沿检测电路以输入线作为信号输入端,如果检测到有边沿跳变就输出有效信号1给编号3电路,否则输出无效信号0。而EXTI_RTSR和EXTI_FTSR两个寄存器可以控制器需要检测哪些类型的电平跳变过程,可以是只有上升沿触发、只有下降沿触发或者上升沿和下降沿都触发。


编号3电路实际就是一个或门电路,它一个输入来自编号2电路,另外一输入来自软件中断事件寄存器(EXTI_SWIER)。EXTI_SWIER允许我们通过程序控制就可以启动中断/事件线,这在某些地方非常有用。我们知道或门的作用就是有"就为1,所以这两个输入随便一个有有效信号1就可以输出1给编号4和编号6电路。


编号4电路是一个与门电路,它一个输入编号3电路,另外一个输入来自中断屏蔽寄存器(EXTI_IMR)。与门电路要求输入都为1才输出1,导致的结果如果EXTI_IMR设置为0时,那不管编号3电路的输出信号是1还是0,最终编号4电路输出的信号都为0;如果EXTI_IMR设置为1时,最终编号4电路输出的信号才由编号3电路的输出信号决定,这样我们可以简单的控制EXTI_IMR来实现是否产生中断的目的。编号4电路输出的信号会被保存到挂起寄存器(EXTI_PR)内,如果确定编号4电路输出为1就会把EXTI_PR对应位置1。


编号5是将EXTI_PR寄存器内容输出到NVIC内,从而实现系统中断事件控制。


接下来我们来看看绿色虚线指示的电路流程。它是一个产生事件的线路,最终输出一个脉冲信号。


产生事件线路是在编号3电路之后与中断线路有所不同,之前电路都是共用的。编号6电路是一个与门,它一个输入编号3电路,另外一个输入来自事件屏蔽寄存器(EXTI_EMR)。如果EXTI_EMR设置为0时,那不管编号3电路的输出信号是1还是0,最终编号6电路输出的信号都为0;如果EXTI_EMR设置为1时,最终编号6电路输出的信号才由编号3电路的输出信号决定,这样我们可以简单的控制EXTI_EMR来实现是否产生事件的目的。


编号7是一个脉冲发生器电路,当它的输入端,即编号6电路的输出端,是一个有效信号1时就会产生一个脉冲;如果输入端是无效信号就不会输出脉冲。


编号8是一个脉冲信号,就是产生事件的线路最终的产物,这个脉冲信号可以给其他外设电路使用,比如定时器TIM、模拟数字转换器ADC等等。


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


另外,EXTI是在APB2总线上的,在编程时候需要注意到这点。


17.3 中断/事件线

EXTI有23个中断/事件线,每个GPIO都可以被设置为输入线,占用EXTI0至EXTI15,还有另外七根用于特定的外设事件,见表 171。


七根特定外设中断/事件线由外设触发,具体用法参考《STM32F4xx中文参考手册》中对外设的具体说明。


表 171 EXTI中断/事件线

image.png?imageView2/2/w/550

EXTI0至EXTI15用于GPIO,通过编程控制可以实现任意一个GPIO作为EXTI的输入源。由表 171可知,EXTI0可以通过SYSCFG外部中断配置寄存器1(SYSCFG_EXTICR1)的EXTI0[3:0]位选择配置为PA0、PB0、PC0、PD0、PE0、PF0、PG0、PH0或者PI0,见图 172。其他EXTI线(EXTI中断/事件线)使用配置都是类似的。


图 172 EXTI0输入源选择


17.4 EXTI初始化结构体详解

标准库函数对每个外设都建立了一个初始化结构体,比如EXTI_InitTypeDef,结构体成员用于设置外设工作参数,并由外设初始化配置函数,比如EXTI_Init()调用,这些设定参数将会设置外设相应的寄存器,达到配置外设工作环境的目的。


初始化结构体和初始化库函数配合使用是标准库精髓所在,理解了初始化结构体每个成员意义基本上就可以对该外设运用自如了。初始化结构体定义在stm32f4xx_exti.h文件中,初始化库函数定义在stm32f4xx_exti.c文件中,编程时我们可以结合这两个文件内注释使用。


代码清单 171 EXTI初始化结构体


1 typedef struct {


2 uint32_t EXTI_Line; // 中断/事件线


3 EXTIMode_TypeDef EXTI_Mode; // EXTI模式


4 EXTITrigger_TypeDef EXTI_Trigger; // 触发事件


5 FunctionalState EXTI_LineCmd; // EXTI控制


6 } EXTI_InitTypeDef;


1)    EXTI_Line:EXTI中断/事件线选择,可选EXTI0至EXTI22,可参考表 171选择。


2)    EXTI_Mode:EXTI模式选择,可选为产生中断(EXTI_Mode_Interrupt)或者产生事件(EXTI_Mode_Event)。


3)    EXTI_Trigger:EXTI边沿触发事件,可选上升沿触发(EXTI_Trigger_Rising)、下降沿触发( EXTI_Trigger_Falling)或者上升沿和下降沿都触发( EXTI_Trigger_Rising_Falling)。


4)    EXTI_LineCmd:控制是否使能EXTI线,可选使能EXTI线(ENABLE)或禁用(DISABLE)。


17.5 外部中断控制实验

中断在嵌入式应用中占有非常重要的地位,几乎每个控制器都有中断功能。中断对保证紧急事件得到第一时间处理是非常重要的


我们设计使用外接的按键来作为触发源,使得控制器产生中断,并在中断服务函数中实现控制RGB彩灯的任务。


17.5.1 硬件设计

轻触按键在按下时会使得引脚接通,通过电路设计可以使得按下时产生电平变化,见图 171。

图 173 按键电路设计


17.5.2 软件设计

这里只讲解核心的部分代码,有些变量的设置,头文件的包含等并没有涉及到,完整的代码请参考本章配套的工程。我们创建了两个文件:bsp_exti.c和bsp_exti.h文件用来存放EXTI驱动程序及相关宏定义,中断服务函数放在stm32f4xx_it.h文件中。


1.    编程要点

1)    初始化RGB彩灯的GPIO;


2)    开启按键GPIO时钟和SYSCFG时钟;


3)    配置NVIC;


4)    配置按键GPIO为输入模式;


5)    将按键GPIO连接到EXTI源输入;


6)    配置按键EXTI中断/事件线;


7)    编写EXTI中断服务函数。


2.    软件分析

按键和EXTI宏定义

代码清单 172 按键和EXTI 宏定义


1 //引脚定义


2 /*******************************************************/


3 #define KEY1_INT_GPIO_PORT GPIOA


4 #define KEY1_INT_GPIO_CLK RCC_AHB1Periph_GPIOA


5 #define KEY1_INT_GPIO_PIN GPIO_Pin_0


6 #define KEY1_INT_EXTI_PORTSOURCE EXTI_PortSourceGPIOA


7 #define KEY1_INT_EXTI_PINSOURCE EXTI_PinSource0


8 #define KEY1_INT_EXTI_LINE EXTI_Line0


9 #define KEY1_INT_EXTI_IRQ EXTI0_IRQn


10


11 #define KEY1_IRQHandler EXTI0_IRQHandler


12


13 #define KEY2_INT_GPIO_PORT GPIOC


14 #define KEY2_INT_GPIO_CLK RCC_AHB1Periph_GPIOC


15 #define KEY2_INT_GPIO_PIN GPIO_Pin_13


16 #define KEY2_INT_EXTI_PORTSOURCE EXTI_PortSourceGPIOC


17 #define KEY2_INT_EXTI_PINSOURCE EXTI_PinSource13


18 #define KEY2_INT_EXTI_LINE EXTI_Line13


19 #define KEY2_INT_EXTI_IRQ EXTI15_10_IRQn


20


21 #define KEY2_IRQHandler EXTI15_10_IRQHandler


使用宏定义方法指定与电路设计相关配置,这对于程序移植或升级非常有用的。


嵌套向量中断控制器NVIC配置

代码清单 173 NVIC配置


1 static void NVIC_Configuration(void)


2 {


3 NVIC_InitTypeDef NVIC_InitStructure;


4


5 /* 配置NVIC为优先级组1 */


6 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);


7


8 /* 配置中断源:按键1 */


9 NVIC_InitStructure.NVIC_IRQChannel = KEY1_INT_EXTI_IRQ;


10 /* 配置抢占优先级:1 */


11 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;


12 /* 配置子优先级:1 */


13 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;


14 /* 使能中断通道 */


15 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;


16 NVIC_Init(&NVIC_InitStructure);


17


18 /* 配置中断源:按键2,其他使用上面相关配置 */


19 NVIC_InitStructure.NVIC_IRQChannel = KEY2_INT_EXTI_IRQ;


20 NVIC_Init(&NVIC_InitStructure);


21 }


有关NVIC配置问题可参考上一章节内容,这里不做过多解释。


EXTI中断配置

代码清单 174 EXTI中断配置


1 void EXTI_Key_Config(void)


2 {


3 GPIO_InitTypeDef GPIO_InitStructure;


4 EXTI_InitTypeDef EXTI_InitStructure;


5


6 /*开启按键GPIO口的时钟*/


7 RCC_AHB1PeriphClockCmd(KEY1_INT_GPIO_CLK|KEY2_INT_GPIO_CLK ,ENABLE);


8


9 /* 使能 SYSCFG 时钟,使用GPIO外部中断时必须使能SYSCFG时钟*/


10 RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE);


11


12 /* 配置 NVIC */


13 NVIC_Configuration();


14


15 /* 选择按键1的引脚 */


16 GPIO_InitStructure.GPIO_Pin = KEY1_INT_GPIO_PIN;


17 /* 设置引脚为输入模式 */


18 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;


19 /* 设置引脚不上拉也不下拉 */


20 GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;


21 /* 使用上面的结构体初始化按键 */


22 GPIO_Init(KEY1_INT_GPIO_PORT, &GPIO_InitStructure);


23


24 /* 连接 EXTI 中断源到key1引脚 */


25 SYSCFG_EXTILineConfig(KEY1_INT_EXTI_PORTSOURCE,


26 KEY1_INT_EXTI_PINSOURCE);


27


28 /* 选择 EXTI 中断源 */


29 EXTI_InitStructure.EXTI_Line = KEY1_INT_EXTI_LINE;


30 /* 中断模式 */


31 EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;


32 /* 下降沿触发 */


33 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;


34 /* 使能中断/事件线 */


35 EXTI_InitStructure.EXTI_LineCmd = ENABLE;


36 EXTI_Init(&EXTI_InitStructure);


37


38 /* 选择按键2的引脚 */


39 GPIO_InitStructure.GPIO_Pin = KEY2_INT_GPIO_PIN;


40 /* 其他配置与上面相同 */


41 GPIO_Init(KEY2_INT_GPIO_PORT, &GPIO_InitStructure);


42


43 /* 连接 EXTI 中断源到key2 引脚 */


44 SYSCFG_EXTILineConfig(KEY2_INT_EXTI_PORTSOURCE,


45 KEY2_INT_EXTI_PINSOURCE);


46


47 /* 选择 EXTI 中断源 */


48 EXTI_InitStructure.EXTI_Line = KEY2_INT_EXTI_LINE;


49 EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;


50 /* 上升沿触发 */


51 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;


52 EXTI_InitStructure.EXTI_LineCmd = ENABLE;


53 EXTI_Init(&EXTI_InitStructure);


54 }


首先,使用GPIO_InitTypeDef和EXTI_InitTypeDef结构体定义两个用于GPIO和EXTI初始化配置的变量,关于这两个结构体前面都已经做了详细的讲解。


使用GPIO之前必须开启GPIO端口的时钟;用到EXTI必须开启SYSCFG时钟。


调用NVIC_Configuration函数完成对按键1、按键2优先级配置并使能中断通道。


作为中断/时间输入线把GPIO配置为输入模式,这里不使用上拉或下拉,有外部电路完全决定引脚的状态。


SYSCFG_EXTILineConfig函数用来指定中断/事件线的输入源,它实际是设定SYSCFG外部中断配置寄存器的值,该函数接收两个参数,第一个参数指定GPIO端口源,第二个参数为选择对应GPIO引脚源编号。


我们的目的是产生中断,执行中断服务函数,EXTI选择中断模式,按键1使用下降沿触发方式,并使能EXTI线。


按键2基本上采用与按键1相关参数配置,只是改为上升沿触发方式。


EXTI中断服务函数

代码清单 175 EXTI中断服务函数


1 void KEY1_IRQHandler(void)


2 {


3 //确保是否产生了EXTI Line中断


4 if (EXTI_GetITStatus(KEY1_INT_EXTI_LINE) != RESET) {


5 // LED1 取反


6 LED1_TOGGLE;


7 //清除中断标志位


8 EXTI_ClearITPendingBit(KEY1_INT_EXTI_LINE);


9 }


10 }


11


12 void KEY2_IRQHandler(void)


13 {


14 //确保是否产生了EXTI Line中断


15 if (EXTI_GetITStatus(KEY2_INT_EXTI_LINE) != RESET) {


16 // LED2 取反


17 LED2_TOGGLE;


18 //清除中断标志位


19 EXTI_ClearITPendingBit(KEY2_INT_EXTI_LINE);


20 }


21 }


当中断发生时,对应的中断服务函数就会被执行,我们可以在中断服务函数实现一些控制。


一般为确保中断确实发生,我们会在中断服务函数调用中断标志位状态读取函数读取外设中断标志位并判断标志位状态。


EXTI_GetITStatus函数用来获取EXTI的中断标志位状态,如果EXTI线有中断发生函数返回"SET"否则返回"RESET"。实际上,EXTI_GetITStatus函数是通过读取EXTI_PR寄存器值来判断EXTI线状态的。


按键1的中断服务函数我们让LED1翻转其状态,按键2的中断服务函数我们让LED2翻转其状态。执行任务后需要调用EXTI_ClearITPendingBit函数清除EXTI线的中断标志位。


主函数

代码清单 176 主函数


1 int main(void)


2 {


3 /* LED 端口初始化 */


4 LED_GPIO_Config();


5


6 /* 初始化EXTI中断,按下按键会触发中断,


7 * 触发中断会进入stm32f4xx_it.c文件中的函数


8 * KEY1_IRQHandler和KEY2_IRQHandler,处理中断,反转LED灯。


9 */


10 EXTI_Key_Config();


11


12 /* 等待中断,由于使用中断方式,CPU不用轮询按键 */


13 while (1) {


14 }


15 }


主函数非常简单,只有两个任务函数。LED_GPIO_Config函数定义在bsp_led.c文件内,完成RGB彩灯的GPIO初始化配置。EXTI_Key_Config函数完成两个按键的GPIO和EXTI配置。


17.5.3 下载验证

保证开发板相关硬件连接正确,把编译好的程序下载到开发板。此时RGB彩色灯是暗的,如果我们按下开发板上的按键1,RGB彩灯变亮,再按下按键1,RGB彩灯又变暗;如果我们按下开发板上的按键2并弹开,RGB彩灯变亮,再按下开发板上的KEY2并弹开,RGB彩灯又变暗。

推荐阅读

史海拾趣

德国ACAM公司的发展小趣事

ACAM公司在超声波计量领域也取得了革命性的突破。2012年,ACAM公司与世强电讯合作,在第二届供热计量技术与管理国际研讨会上推出了业内领先的超声波热量表解决方案。这一解决方案利用ACAM公司的先进测量技术,为热量表厂商和流量计量专家提供了全新的解决方案,受到了广泛关注。

北京人民电器厂公司的发展小趣事

近年来,随着新能源产业的快速发展,北京人民电器紧跟时代步伐,在新能源领域取得了重要突破。公司推出的真安型零飞弧直流断路器,解决了直流低压系统中长期存在的外喷电弧问题,为新能源行业的发展提供了重要的技术支持。这一创新成果不仅得到了行业的广泛认可,也为公司在新能源领域的发展打开了新的局面。

这五个故事只是北京人民电器在电子行业发展历程中的一部分,但它们足以展现出公司在技术创新、产品应用、生产自动化、人才队伍建设以及新能源领域发展等方面的努力和成就。这些故事不仅记录了北京人民电器的成长轨迹,也见证了中国电子行业的蓬勃发展。

ADDtek公司的发展小趣事

为了加强技术研发和创新能力,北京人民电器建立了北京市级技术研究中心,并吸引了教授级高工、博士后、博士、硕士等多层次的专业技术人才。这些人才为公司的新产品研发、技术创新提供了强大的智力支持,使得北京人民电器在激烈的市场竞争中始终保持领先地位。

Golledge Electronics公司的发展小趣事

为了加强技术研发和创新能力,北京人民电器建立了北京市级技术研究中心,并吸引了教授级高工、博士后、博士、硕士等多层次的专业技术人才。这些人才为公司的新产品研发、技术创新提供了强大的智力支持,使得北京人民电器在激烈的市场竞争中始终保持领先地位。

福斯特半导体(Foster)公司的发展小趣事

为了提高生产效率和产品质量,北京人民电器投入大量资金建设了自动化装配检测流水线。这一举措使得公司的生产能力得到了大幅提升,同时也确保了产品的一致性和可靠性。自动化流水线的建设,是北京人民电器在现代化生产道路上迈出的重要一步。

Goldentech Discrete Semiconductor Inc公司的发展小趣事

进入21世纪后,随着全球电子市场的竞争加剧,Goldentech意识到单靠技术创新已不足以维持其市场地位。因此,公司开始积极寻求与行业领先企业的战略合作。通过与一家知名芯片设计公司的深度合作,Goldentech成功将其高性能离散半导体器件集成到对方的芯片设计中,从而实现了产品的广泛应用。这一合作不仅扩大了Goldentech的市场份额,还提升了其品牌知名度和行业影响力。

问答坊 | AI 解惑

发个我用的元件库

这是我平常所用到的元件库…

查看全部问答>

wince 的cab 安装包问题(vs2005)

想做一个cab安装包,安装今日插件,按照网上的说明写了个安装程序setupdll.dll, 但是在模拟器上(pocket pc se 2003 Emulator)安装发现setupdll没有被调用(在函数Install_Exit中加了MessageBox,没有弹出,注册表也没写) cab安装包使用vs2005做的 ...…

查看全部问答>

GPRS连接问题

我用OPEN AT 已经建立完成了GPRS激活部分,但是为什么数据流中什么都收不到,在超级终端里使用ATD*99***1#,可以收到PPP包,如何使用OPEN AT ADL也能收到这些PPP包?请指教一二!我在软件里已经加入了AT命令,但是没有任何反应,到底是什么地方出了 ...…

查看全部问答>

如何用WinDbg或Waston Dump Viewer分析WinCE机台上抓取的Dump File问题

    基于ARM+WinCE 5.0的机台上有概率性的发生Data Abort,直接采用加入Debug信息逐步缩小范围的方法太费时间,所以在Image中加入了ErrorReporting的功能。     机器发生了Data Abort后,将dump file拷贝出来,就是那个后缀名 ...…

查看全部问答>

dshow CreateMediaType FreeMediaType 无法解析的外部符号

我在wince6.0上做dshow开发,已经包含的头文件和库 #include #include #include #include #include                                     &n ...…

查看全部问答>

请教中断问题

我在做关于MPC8260的工作.目前,我想为DMA加入中断处理程序.MPC8260的参考手册中 说,IDMA1的中断号是6,我使用如下函数: intConnect(INUM_TO_IVEC(6),dma_isr,0); 连接中断处理程序与中断源.但是一旦DMA结束,BC中断到来之后,整个EP8260板子就死掉 ...…

查看全部问答>

怎样取得Windows的启动分区?

我的机器上有两个硬盘 我的boot.ini: [boot loader] timeout=6 default=multi(0)disk(0)rdisk(0)partition(1)\\WINDOWS [operating systems] multi(0)disk(0)rdisk(0)partition(1)\\WINDOWS=\"Microsoft Windows XP Professional\" /noexecut ...…

查看全部问答>

MCS-51单片机定时器问题

MCS-51单片机中,采用12Mhz时钟,定时器T0采用模式1(16位计数器),请问在下面程序中,p1.0的输出频率 ? MOV TMOD,#01H SETB TR0 LOOP:MOV TH0,#0B1H MOV TL0,#0E0H LOOP1:JNB TF0,LOOP1 CLR TR0 CPL P1.0 SJMP LOOP…

查看全部问答>

ADS1.2 调用strtoul(str,NULL,0,NULL),地址0的内容会被更改

strtoul的定义: unsigned long strtoul(const char *str, char **endptr, int requestedbase, int *ret) 可见ADS1.2会把NULL指针指向地址0,但是地址0放着复位向量,怎么样才能不改变0地址的内容呢?(当然在调用的时候可以不用NULL,而定义一个 ...…

查看全部问答>