STM32-调用 vTaskStartScheduler API 后出现 HardFault

STM32 移植 FreeRTOS 后调用 vTaskStartScheduler() 后出现 HardFault 异常。

原因分析:

FreeRTOS 配置头文件 FreeRTOSConfig.h 中与中断有关的配置和通过系统接口 void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup) 设置的中断分组冲突。

c 复制代码
/* The lowest interrupt priority that can be used in a call to a "set priority"
function. */
#define configLIBRARY_LOWEST_INTERRUPT_PRIORITY   15

/* The highest interrupt priority that can be used by any interrupt service
routine that makes calls to interrupt safe FreeRTOS API functions.  DO NOT CALL
INTERRUPT SAFE FREERTOS API FUNCTIONS FROM ANY INTERRUPT THAT HAS A HIGHER
PRIORITY THAN THIS! (higher priorities are lower numeric values. */
#define configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY 5

/* Interrupt priorities used by the kernel port layer itself.  These are generic
to all Cortex-M ports, and do not rely on any particular library functions. */
#define configKERNEL_INTERRUPT_PRIORITY 		( configLIBRARY_LOWEST_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )
/* !!!! configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to zero !!!!
See http://www.FreeRTOS.org/RTOS-Cortex-M3-M4.html. */
#define configMAX_SYSCALL_INTERRUPT_PRIORITY 	( configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )
c 复制代码
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);

FreeRTOSConfig.h 中设置的中断最低优先级为 15 ,说明 可编程中断优先级的范围为 0 ~ 15 ,也即需要 4 bits 来表示抢占优先级。而通过 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0) API 设置为 0 bit 用于抢占优先级3 bits 用于子优先级 ,与 FreeRTOSConfig.h 中的中断设置冲突导致。

c 复制代码
/** @defgroup Preemption_Priority_Group 
  * @{
  */

#define NVIC_PriorityGroup_0         ((uint32_t)0x700) /*!< 0 bits for pre-emption priority
                                                            4 bits for subpriority */
#define NVIC_PriorityGroup_1         ((uint32_t)0x600) /*!< 1 bits for pre-emption priority
                                                            3 bits for subpriority */
#define NVIC_PriorityGroup_2         ((uint32_t)0x500) /*!< 2 bits for pre-emption priority
                                                            2 bits for subpriority */
#define NVIC_PriorityGroup_3         ((uint32_t)0x400) /*!< 3 bits for pre-emption priority
                                                            1 bits for subpriority */
#define NVIC_PriorityGroup_4         ((uint32_t)0x300) /*!< 4 bits for pre-emption priority
                                                            0 bits for subpriority */

解决方案:

将中断分组设置为 NVIC_PriorityGroup_4 即可。FreeRTOS 官方文档 也建议对于 STM32 (Cortex-M3),建议设置为 NVIC_PriorityGroup_4

c 复制代码
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);

题外:

这里顺便也记录下 STM32 (Cortex-M3) 结合 FreeRTOS 对中断的设置。

STM32 (Cortex-M3)SCB_AIRCR 寄存器中的 Bits 10:9 设置中断优先级分组。

FreeRTOS 在配置文件 FreeRTOSConfig.h 中通过宏 configKERNEL_INTERRUPT_PRIORITYconfigMAX_SYSCALL_INTERRUPT_PRIORITY 来设置 FreeRTOS 中的中断设置。

  • configKERNEL_INTERRUPT_PRIORITY 用于设置中断最低优先级。这个宏必须结合 void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup) 来设置。
  • configMAX_SYSCALL_INTERRUPT_PRIORITY 用于 FreeRTOS 中的一些中断功能。典型应用就是 FreeRTOS 中以 FromISR 结尾的 API 可以安全的在 configMAX_SYSCALL_INTERRUPT_PRIORITY ~ configKERNEL_INTERRUPT_PRIORITY 之间的中断回调中使用。对于这部分描述,可以参考下图或者 FreeRTOS 官方文档中对 FreeRTOSConfig.h 中宏的说明
相关推荐
Stanford_sun5 小时前
基于Zigbee的无线火灾报警系统(云平台版)
网络·嵌入式硬件·物联网·zigbee
minglie15 小时前
ESP32 波特律动oled
mcu
CFZPL5 小时前
espidf用CMake文件构建项目
单片机·esp32
猪八戒1.06 小时前
9.5 【定时器】输入捕获
单片机·嵌入式硬件
xiaohai@Linux6 小时前
STM32之移植原生的infoNES nes游戏模拟器源码实现游戏自由!!!(原生纯C版,非汇编版)
stm32·游戏·模拟器·infones·nes游戏机
无人装备硬件开发爱好者6 小时前
初级菜鸟快速学习无人机电调教程:第2节
嵌入式硬件·无人机·电调开发
Wnq100727 小时前
当无人机 “飞” 入生活,安全隐患如何破解?
嵌入式硬件·物联网·网络安全·信息与通信·信号处理
无垠的广袤7 小时前
【启明云端 WT9932S3-Nano 开发板】介绍、环境搭建、工程测试
python·单片机·嵌入式硬件
小李做物联网7 小时前
【单片机毕业设计】143.1基于单片机stm32塔吊控制反馈物联网嵌入式项目程序开发系统
stm32·单片机·嵌入式硬件·物联网
源代码•宸15 小时前
分布式缓存-GO(分布式算法之一致性哈希、缓存对外服务化)
开发语言·经验分享·分布式·后端·算法·缓存·golang