解决未进入中断服务程序的一般思路

解决未进入中断服务程序的一般思路

  1. GPIO 是否连接

  2. GPIO 引脚是否配置,支持引脚复用功能的则需要判断是否选定使用的功能

  3. 检查中断向量表

  4. 对于支持重映射中断向量表的硬件,检查存储中断向量表起始地址的寄存器的值是否正确

  5. 当你在程序中对向量表重新进行了映射时,检查起始地址的合法性与复制区域大小,可以读取复制完成后的新中断向量表的首尾中,检查获取到的中断服务程序地址与旧中断向量表中的内容是否对应

  6. 中断向量表确认不存在问题后,开始检查中断配置状态

  7. 总中断是否打开

  8. 子中断是否打开

  9. 中断号与 ISR 连接是否正确?(不支持动态修改中断向量表的硬件可以跳过)

  10. 中断是否触发

  11. 当前设定的中断优先级是否会造成影响

  12. debug 正常 release 异常的情况,可能是 release 优化后程序乱序执行造成的问题。考虑是否需要在代码中添加必要的编译屏障(阻止编译器优化造成的乱序执行问题)或运行时的内存屏障。着重分析读写关键变量的情况。

能够使用编译屏障来解决问题,尽可能的使用编译屏障。能够使用不太严格的内存屏障来解决问题,就尽可能使用。在保证正常执行的情况下,考虑最优的性能。越严格的屏障带来更安全的保障的同时也增强了性能开销。

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在STM32中,中断是一种事件驱动的机制,它能够及时响应外部事件并处理。中断可以分为内部中断和外部中断,内部中断是由芯片内部的某些硬件模块触发的,如定时器溢出中断、ADC转换完成中断等;而外部中断是由外部设备触发的,如按键、外部输入信号等。 在使用STM32的中断机制时,需要编写中断服务程序(ISR),中断服务程序中断触发后自动执行的程序,用于处理中断事件。下面是一个简单的中断服务程序的例子: ```c void EXTI0_IRQHandler(void) //中断服务程序 { if(EXTI_GetITStatus(EXTI_Line0) != RESET) //检查是否触发中断 { //执行中断处理操作 //... EXTI_ClearITPendingBit(EXTI_Line0); //清除中断标志位 } } ``` 在上面的例子中,EXTI0_IRQHandler()就是一个中断服务程序,用于处理外部中断0事件。当外部中断0被触发时,程序会自动跳转到该中断服务程序中执行相关的中断处理操作。在中断服务程序中,需要检查中断标志位,判断中断是否被触发,然后进行相应的中断处理操作。最后,需要清除中断标志位,以便下一次中断触发时能够再次进入中断服务程序。 下面是一个使用外部中断的例子: ```c #include "stm32f10x.h" void EXTI0_IRQHandler(void); //中断服务程序声明 int main(void) { GPIO_InitTypeDef GPIO_InitStructure; EXTI_InitTypeDef EXTI_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; //使能外部中断0所在的GPIO口 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //配置GPIOA.0为输入口 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOA, &GPIO_InitStructure); //使能外部中断线0 RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE); GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0); //配置外部中断线0为下降沿触发 EXTI_InitStructure.EXTI_Line = EXTI_Line0; EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; EXTI_InitStructure.EXTI_LineCmd = ENABLE; EXTI_Init(&EXTI_InitStructure); //使能外部中断0所在的中断向量 NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); while(1) { //主程序循环 //... } } void EXTI0_IRQHandler(void) //中断服务程序实现 { if(EXTI_GetITStatus(EXTI_Line0) != RESET) //检查是否触发中断 { //执行中断处理操作 //... EXTI_ClearITPendingBit(EXTI_Line0); //清除中断标志位 } } ``` 在上面的例子中,使用了外部中断0来触发中断,对应的GPIO口为GPIOA.0。首先,需要配置GPIOA.0为输入口,然后使能外部中断线0,并将其配置为下降沿触发。接着,使能外部中断0所在的中断向量,并编写对应的中断服务程序。在中断服务程序中,可以编写相应的中断处理操作。最后,需要清除中断标志位。 需要注意的是,在使用中断时,一定要注意中断服务程序的执行时间,避免中断服务程序执行时间过长,导致主程序无法及时执行,影响系统的响应速度。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值