I/O口作为外部中断。general  purpose input and output

STM32 的每个 IO口都可以作为中断输入,要把 IO口作为外部中断输入,
有以下几个步骤:
1) 初始化 IO 口为输入。
这一步设置你要作为外部中断输入的 IO 口的状态,可以设置为上拉/下拉输入,也可以设
置为浮空输入,但浮空的时候外部一定要带上拉,或者下拉电阻。否则可能导致中断不停的触
发。在干扰较大的地方,就算使用了上拉/下拉,也建议使用外部上拉/下拉电阻,这样可以一
定程度防止外部干扰带来的影响。
2) 开启 IO 口复用时钟,设置 IO 口与中断线的映射关系。
STM32 的 IO 口与中断线的对应关系需要配置外部中断配置寄存器 EXTICR,这样我们要
先开启复用时钟,然后配置 IO 口与中断线的对应关系。才能把外部中断与中断线连接起来。
3) 开启与该 IO 口相对的线上中断/事件,设置触发条件。
这一步,我们要配置中断产生的条件, STM32 可以配置成上升沿触发,下降沿触发,或者
任意电平变化触发,但是不能配置成高电平触发和低电平触发。这里根据自己的实际情况来配
置,同时要开启中断线上的中断。这里需要注意的是:如果使用外部中断,并设置该中断的 EMR
位的话,会引起软件仿真不能跳到中断,而硬件上是可以的。而不设置 EMR,软件仿真就可以
进入中断服务函数,并且硬件上也是可以的。建议不要配置 EMR 位。
4) 配置中断分组(NVIC),并使能中断。
这一步,我们就是配置中断的分组,以及使能,对 STM32 的中断来说,只有配置了 NVIC
的设置,并开启才能被执行,否则是不会执行到中断服务函数里面去的。关于 NVIC 的详细介
绍,请参考 5.2.6 节。
5) 编写中断服务函数。
这是中断设置的最后一步,中断服务函数,是必不可少的,如果在代码里面开启了中断,
但是没编写中断服务函数,就可能引起硬件错误,从而导致程序崩溃!所以在开启了某个中断

后,一定要记得为该中断编写服务函数。在中断服务函数里面编写你要执行的中断后的操作。

与 NVIC 相关的寄存器, MDK 为其定义了如下的结构体:
typedef struct
{
__IO uint32_t ISER[8]; //中断使能寄存器组 Interrupt Set-Enable Registers
uint32_t RESERVED0[24];
__IO uint32_t ICER[8]; //中断除能寄存器组
uint32_t RSERVED1[24];
__IO uint32_t ISPR[8]; //中断挂起控制寄存器组
uint32_t RESERVED2[24];
__IO uint32_t ICPR[8]; //中断解挂控制寄存器组
uint32_t RESERVED3[24];
__IO uint32_t IABR[8]; //中断激活标志位寄存器组
uint32_t RESERVED4[56];
__IO uint8_t IP[240]; //中断优先级控制寄存器组
uint32_t RESERVED5[644];
__O uint32_t STIR; //软件触发中断寄存器组
} NVIC_Type; 

        ISER[8]: ISER 全称是: Interrupt Set-Enable Registers, 每1 bit 代表一个中断,总共有32×8=256个中断,你要使能某个中断,必须设置相应的 ISER 位为 1,使该中断被使能(这里仅仅是使能,还要配合中断分组、屏蔽、 IO 口映射等设置才算是一个完整的中断设置)。 

        ICER[8]:全称是: Interrupt Clear-Enable Registers, 该寄存器组与 ISER[8] 的作用恰好相反,是用来清除某个中断的使能的。 如果想清除一个中断,不是在对应bit写0,而应该在对应位置写1。写0是无效的。
        ISPR[8]:全称是: Interrupt Set-Pending Registers ,每个位对应的中断和 ISER 是一样的。通过置 1,可以将正在进行的中断挂起,而执行同级或更高级别的中断。写 0 是无效的。

        ICPR[8]:全称是: Interrupt Clear-Pending Registers ,通过设置 1,可以将挂起的中断取消挂起操作。写 0 无效。

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

        IP[240]:全称是: Interrupt Priority Registers, 是一个中断优先级控制的寄存器组。这个寄存器组相当重要! STM32 的中断分组与这个寄存器组密切相关。 IP 寄存器组由 240 个 8bit 的寄存器组成,每个可屏蔽中断占用 8bit,这样总共可以表示 240 个可屏蔽中断。 而 STM32 只用到了其中的 68 个。 IP[67]~IP[0]分别对应中断 67~0。 而每个可屏蔽中断占用的 8bit 并没有全部使用,而是 只用了高 4 位。这 4 位,又分为抢占优先级和子优先级。抢占优先级在前,子优先级在后。而这两个优先级各占几个位又要根据 SCB->AIRCR 中的中断分组设置来决定。抢占优先级的级别高于响应优先级。而数值越小所代表的优先级就越高。 

         STM32 的 5 个分组是通过设置 SCB->AIRCR 的 BIT[10:8]来实现的,而 SCB->AIRCR 的修改需要通过在高 16 位写入 0X05FA 这个密钥才能修改的,故在设置 AIRCR 之前,应该把密钥加入到要写入的内容的高 16 位,以保证能正常的写入 AIRCR。在修改 AIRCR 的时候,我们一般采用读->改->写的步骤,来实现不改变 AIRCR 原来的其他设置。

这里简单介绍一下 STM32 的中断分组: STM32 将中断分为 5 个组,组 0~4。该分组的设
置是由 SCB->AIRCR 寄存器的 bit10~8 来定义的。具体的分配关系如表 5.2.6.1 所示:

AIRCR[10: 8]bit[7: 4]分配情况分配结果
01110: 40 位抢占优先级, 4 位响应优先级
11101: 31 位抢占优先级, 3 位响应优先级
21012: 22 位抢占优先级, 2 位响应优先级
31003: 13 位抢占优先级, 1 位响应优先级
40114: 04 位抢占优先级, 0 位响应优先级


                                                                表 5.2.6.1 AIRCR 中断分组设置表 

   

//设置 NVIC
//NVIC_PreemptionPriority: 抢占优先级
//NVIC_SubPriority : 响应优先级
//NVIC_Channel : 中断编号
//NVIC_Group : 中断分组 0~4
//注意优先级不能超过设定的组的范围!否则会有意想不到的错误
//组划分:
//组 0: 0 位抢占优先级, 4 位响应优先级
//组 1: 1 位抢占优先级, 3 位响应优先级
//组 2: 2 位抢占优先级, 2 位响应优先级
//组 3: 3 位抢占优先级, 1 位响应优先级
//组 4: 4 位抢占优先级, 0 位响应优先级
//NVIC_SubPriority 和 NVIC_PreemptionPriority 的原则是, 数值越小, 越优先
void MY_NVIC_Init(u8 NVIC_PreemptionPriority,u8 NVIC_SubPriority,u8 NVIC_Channel,u8 NVIC_Group)
{
u32 temp;
MY_NVIC_PriorityGroupConfig(NVIC_Group);//设置分组
temp=NVIC_PreemptionPriority<<(4-NVIC_Group);
temp|=NVIC_SubPriority&(0x0f>>NVIC_Group);
temp&=0xf; //取低四位
NVIC->ISER[NVIC_Channel/32]|=(1<//使能中断位(要清除的话,相反操作就 OK)
NVIC->IP[NVIC_Channel]|=temp<<4;     //设置响应优先级和抢断优先级

 NVIC配置总结:

  1)  SCB->AIRCR 决定抢占优先级的位数,设置怎么解释IP(interrupt priority)分组。其实这个分组的设置在每个系统里面只要设置一次就够了,设置多次,则是以最后的那一次为准。整个系统的优先级分组格式都一样。

 2)   ISER 使能对应的中断管脚,

 3)设置优先级。
       IP[channel] 决定具体的抢占优先级和子优先级。

       IP和SCB->AIRCR 一起决定中断的优先级。

       IABR 自读,显示当前正在执行的中断时那个管脚的中断。

》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》

以上是对NVIC的管理和配置,下面说明一下外不中断的配置。

》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》

STM32F103 的 EXTI 控制器支持 19 个外部中断/事件请求。每个中断设有状态位,每个中断/事件都有独立的触发和屏蔽设置。 STM32F103 的 19 个外部中断为:
    线 0~15:对应外部 IO 口的输入中断。
    线 16:连接到 PVD 输出。
    线 17:连接到 RTC 闹钟事件。
    线 18:连接到 USB 唤醒事件。
对于外部中断 EXTI 控制 MDK 定义了如下结构体:
typedef struct
{
__IO uint32_t IMR;  //interrupt mask register
__IO uint32_t EMR;  //event mask register 
__IO uint32_t RTSR;  //rising trigger selection register 
__IO uint32_t FTSR;  //falling trigger selection register 
__IO uint32_t SWIER;   //software interrupt event register  
__IO uint32_t PR;     //pending register 
} EXTI_TypeDef; 
    IMR:中断屏蔽寄存器。这是一个 32 寄存器。但是只有前 19 位有效(19个外部中断)。当位 x 设置为 1 时,则开启这个线上的中断,否则关闭该线上的中断。
    EMR:事件屏蔽寄存器,同 IMR,只是该寄存器是针对事件的屏蔽和开启。
    RTSR:上升沿触发选择寄存器。该寄存器同 IMR,也是一个 32 为的寄存器,只有前 19位有效。位 x 对应线 x 上的上升沿触发,如果设置为 1,则是允许上升沿触发中断/事件。否则,不允许。
    FTSR:下降沿触发选择寄存器。同 RTSR,不过这个寄存器是设置下降沿的。下降沿和上升沿可以被同时设置,这样就变成了任意电平触发了。
    SWIER:软件中断事件寄存器。通过向该寄存器的位 x 写入 1,在未设置 IMR 和 EMR 的时候,将设置 PR 中相应位挂起。如果设置了 IMR 和 EMR 时将产生一次中断。被设置的 SWIER位,将会在 PR 中的对应位清除后清除。
    PR:挂起寄存器。当外部中断线上发生了选择的边沿事件,该寄存器的对应位会被置为 1。0:表示对应线上没有发生触发请求。通过向该寄存器的对应位写入 1 可以清除该位。在中断服务函数里面经常会要向该寄存器的对应位写 1 来清除中断请求 。

        通过以上配置就可以正常设置外部中断了,但是外部 IO 口的中断,还需要一个寄存器配置,也就是 IO 复用里的外部中断配置寄存器 EXTICR。这是因为 STM32 任何一个 IO 口都可以配置成中断输入口,但是 IO 口的数目远大于中断线数(16 个)。于是 STM32 就这样设计,GPIOA~GPIOG 的[15:0]分别对应中断线 15~0。

        这样每个中断线对应了最多 7 个 IO 口,以中断线 0为例:它对应了 GPIOA.0、 GPIOB.0、 GPIOC.0、 GPIOD.0、 GPIOE.0、 GPIOF.0、 GPIOG.0。而中断线每次只能连接到 1个 IO口上,这样就需要 EXTICR来决定对应的中断线配置到哪个 GPIO上了。

EXTICR 在 AFIO 的结构体中定义,如下:
typedef struct
{
    __IO uint32_t EVCR;
    __IO uint32_t MAPR;
    __IO uint32_t EXTICR[4];

} AFIO_TypeDef;


-------

EXTICR1


EXTICR2 对应EXTI4,EXTI5,EXTI6,EXTI7。

EXTICR3 对应EXTI8,EXTI9,EXTI10,EXTI11。

EXTICR4 对应EXTI12,EXTI13,EXTI14,EXTI15。

映射GPIOX_K 管脚。

例如,EXTI11,配置外部中断管脚11,通过对EXTIX[3:0]的设置分别设置为GPIOA  ~ GPIOG 。

//外部中断配置函数
//只针对 GPIOA~G;不包括 PVD, RTC 和 USB 唤醒这三个
//参数: GPIOx: 0~6, 代表 GPIOA~G;
//BITx: 需要使能的位;
//TRIM: 触发模式, 1, 下升沿; 2, 上降沿;3,任意电平触发
//该函数一次只能配置 1 个 IO 口, 多个 IO 口, 需多次调用
//该函数会自动开启对应中断, 以及屏蔽线
void Ex_NVIC_Config(u8 GPIOx, u8 BITx, u8 TRIM)
{
u8 EXTADDR; 
u8 EXTOFFSET;
EXTADDR=BITx/4; //得到中断寄存器组的编号
EXTOFFSET=(BITx%4)*4;//得到中断寄存器组内的偏移
RCC->APB2ENR|=0x01; //enable AFIO  clock
AFIO->EXTICR[EXTADDR]&=~(0x000F<AFIO->EXTICR[EXTADDR]|=GPIOx<//自动设置
EXTI->IMR|=1<if(TRIM&0x01)EXTI->FTSR|=1<

if(TRIM&0x02)EXTI->RTSR|=1<}

        Ex_NVIC_Config 完全是按照我们之前的分析来编写的,首先根据 GPIOx 的位得到中断寄存器组的编号,即 EXTICR 的编号,在 EXTICR 里面配置中断线应该配置到 GPIOx 的哪个位。然后使能该位的中断及事件,最后配置触发方式。这样就完成了外部中断的配置了。



 NVIC配置总结:

  1)  SCB->AIRCR 决定抢占优先级的位数,设置怎么解释IP(interrupt priority)分组。其实这个分组的设置在每个系统里面只要设置一次就够了,设置多次,则是以最后的那一次为准。整个系统的优先级分组格式都一样。

 2)   ISER 使能对应的中断bit,

 3)设置优先级。
       IP[channel] 决定具体的抢占优先级和子优先级。

       IP和SCB->AIRCR 一起决定中断的优先级。

       IABR 自读,显示当前正在执行的中断时那个管脚的中断。


外部中断配置总结:

    1) EXTI.BITx 映射到 GPIOx.BITx,通过AFIO_EXTICRn。

    2)开启外部中断线。

    3)设置触发方式。


关键字:STM32  外部中断输入 引用地址:STM32 I/O 作为外部中断输入

上一篇:用STM32F103RCT6的普通IO口模拟串口的实验
下一篇:STM32 特殊功能脚设置为普通脚的方法

推荐阅读

拿到一份UART的通信协议,上手来操作之前先做一下分析。一、帧格式说明先看一下它的帧格式说明:1、 帧头标志Head 不论是命令帧还是响应帧,帧头标志都是0x92。 2、 协议版本 协议版本号(4bit),目前值为1 加密方式(4bit),0表示采取“数据不加密+校验和”方式。 所以,当前此字段完整值为0x10 3、 控制字段中的C/R比特 用于指示该帧是命令帧还是应...
一、关于STM32F4在通过串口通信的时候乱码问题 1.刚开始弄得时候,以为和stm32一样配置完串口之后就可以用了,结果后面发现串口打印出来的东西全是乱码。后面发现是STM32F4的固件库中使用的频率是25Mhz,而板子上的外部晶振是8Mhz的。因此需要去固件库的stm32f4xx.h中把HSE_VALUE改成8Mhz就可以了。2.还有一种问题是,如果使用的是电平转换的问题。要直接...
//非常稳定可靠的单片机按键消抖程序,绝对不会产生抖动#include<reg52.h>#define uint unsigned int#define uchar unsigned charsbit K1 = P1^4;sbit K2 = P1^5;void delay1ms(uint n) { uchar a,b,c,d;for(d=n;d>0;d--) for(c=1;c>0;c--) for(b=222;b>0;b--) for(a=12;a>0;a--);}void main(){ ...
BSLAM自主导航定位系统即BITO SLAM SYSTEM,是AMR等新一代智能移动设备的眼睛,是每一台智能移动机器人不可或缺的核心模块。正是由于SLAM技术的突破和发展,才诞生了基于SLAM自然导航的智能移动机器人,它可替代传统的磁条、磁钉及二维码导航方式的AGV,不仅可以节省场地改造的费用和时间,同时也可快速完成部署,不影响工厂正常生产。BSLAM自主导航定位...

史海拾趣

问答坊 | AI 解惑

实践-----单片机制作流水灯

2008年03月13日 星期四 10:37流 水 灯   学习了一段时间,有了一定经验的朋友,肯定会觉得不过瘾,老是纸上谈兵,点个LED有什么用?能不能来点实际点的。好的,我们就安排一点实际一点的,这次是点亮真的电灯了,不过要注意安全,莫要发生触电事 ...…

查看全部问答∨

AT89S52中文资料、24c02中文资料手册

希望能帮助大家…… [ 本帖最后由 鑫海宝贝 于 2011-4-6 17:49 编辑 ]…

查看全部问答∨

酒精测试仪的设计

小弟今年大四 做的毕业设计是:酒精测试仪的设计,可是网上相关资料不多。希望大家相关课题的可以互相帮助帮助!…

查看全部问答∨

非理想Flyback CCM 模式的建模與仿真

非理想Flyback  CCM 模式的建模與仿真…

查看全部问答∨

还是Norflash的问题

我在SST39VF3201上用以下函数进行擦除: bool SST39VF320X::Erase_One_Block (U32 Dst) {     bool  ReturnStatus=TRUE;     IntOff();         *(U16 *) (BaseAddres + 0x5555 * AddrsShi ...…

查看全部问答∨

ucos2移植的困扰

   Tyep mismatch in redeclaration of \'OSTakCreate\'!!! 和他相关联的文件函数我找了,但都没找到原因。请问有谁以前遇到过这类问题,期待着您的回复!…

查看全部问答∨

经典的纯数据GSM/GPRS传输模块-M590

M590 尺寸:28mm*24mm*11mm 全球第一款单芯片GPRS模块 功耗低:待机功耗2.5MA 温宽大:-40—80 支持串口的CMUX功能 开放1Kbytes数据区供客户使用 独特AES软件加密功能 PMB7880集成了基带、射频(收发)以及电源管理四颗芯片 编码方式:CS-2 ...…

查看全部问答∨

串口工具什么都不显示!!!!

我的串口工具上面无法显示自己改的U-BOOT的选项菜单,什么都没有,做i2c和LCD实验的时候也是串口工具上面什么都没有!但是uart实验又是成功的,也就是说串口的设置是对的!。。。问题到底在什么地方阿~麻烦各位了! 我用的是mini2440的板子,USB ...…

查看全部问答∨

两段代码,上一段显示为0,,下一段显示为8.。

#include <reg52.h> #define uint unsigned int #define uchar unsigned char sbit duan=P2^6;        //申明U1锁存器的锁存端        sbit wei=P2^7;        //申U2锁存器 ...…

查看全部问答∨
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved