完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
|
|
相关推荐
1个回答
|
|
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 来定义的。 //设置 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《《NVIC_Channel%32); //使能中断位(要清除的话,相反操作就 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《《EXTOFFSET);//清除原来设置!!! AFIO-》EXTICR[EXTADDR]|=GPIOx《《EXTOFFSET; //EXTI.BITx 映射到 GPIOx.BITx //自动设置 EXTI-》IMR|=1《《BITx; //开启 line BITx 上的中断,写1 开启中断 if(TRIM&0x01)EXTI-》FTSR|=1《《BITx; //line BITx 上事件下降沿触发 if(TRIM&0x02)EXTI-》RTSR|=1《《BITx; //line BITx 上事件上升降沿触发 } 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)设置触发方式。 |
|
|
|
只有小组成员才能发言,加入小组>>
调试STM32H750的FMC总线读写PSRAM遇到的问题求解?
1909 浏览 1 评论
X-NUCLEO-IHM08M1板文档中输出电流为15Arms,15Arms是怎么得出来的呢?
1678 浏览 1 评论
1172 浏览 2 评论
STM32F030F4 HSI时钟温度测试过不去是怎么回事?
771 浏览 2 评论
ST25R3916能否对ISO15693的标签芯片进行分区域写密码?
1732 浏览 2 评论
1970浏览 9评论
STM32仿真器是选择ST-LINK还是选择J-LINK?各有什么优势啊?
807浏览 4评论
stm32f4下spi+dma读取数据不对是什么原因导致的?
254浏览 3评论
STM32F0_TIM2输出pwm2后OLED变暗或者系统重启是怎么回事?
624浏览 3评论
634浏览 3评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2025-1-24 01:23 , Processed in 0.814212 second(s), Total 75, Slave 59 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号