【FreeRTOS】调用vTaskStartScheduler立即进入HardFault_Handler问题解决

环境:STM32F103VCT6

问题:

先初始化了外设,然后到调用  vTaskStartScheduler  启动调度器时就直接跳转到 HardFault_Handler

启动调试,然后找到HardFault_Handler,Show Caller Code 跳转到 post.c 该函数:

__asm void prvStartFirstTask( void )
{
	PRESERVE8

	/* Use the NVIC offset register to locate the stack. */
	ldr r0, =0xE000ED08
	ldr r0, [r0]
	ldr r0, [r0]

	/* Set the msp back to the start of the stack. */
	msr msp, r0
	/* Globally enable interrupts. */
	cpsie i
	cpsie f
	dsb
	isb
	/* Call SVC to start the first task. */
	svc 0
	nop
	nop
}

 

解决:

经过逐一排查,发现是 FreeRTOS 系统可管理的最高中断优先级问题,把可管理优先级调高就不会出错了。

修改前:

#ifdef __NVIC_PRIO_BITS
	#define configPRIO_BITS       		__NVIC_PRIO_BITS
#else
	#define configPRIO_BITS       		4                  
#endif
//中断最低优先级
#define configLIBRARY_LOWEST_INTERRUPT_PRIORITY			15     

#define configKERNEL_INTERRUPT_PRIORITY 		( configLIBRARY_LOWEST_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )	/* 240 */

//系统可管理的最高中断优先级
#define configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY	5 
//设置可以在中断服务程序中安全调用FreeRTOS API函数的最高中断优先级
#define configMAX_SYSCALL_INTERRUPT_PRIORITY 	( configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )

修改后:

#ifdef __NVIC_PRIO_BITS
	#define configPRIO_BITS       		__NVIC_PRIO_BITS
#else
	#define configPRIO_BITS       		4                  
#endif
//中断最低优先级
#define configLIBRARY_LOWEST_INTERRUPT_PRIORITY			15     

#define configKERNEL_INTERRUPT_PRIORITY 		( configLIBRARY_LOWEST_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )	/* 240 */
//设置可以在中断服务程序中安全调用FreeRTOS API函数的最高中断优先级
#define configMAX_SYSCALL_INTERRUPT_PRIORITY 	9

关于 configKERNEL_INTERRUPT_PRIORITY 定义可参考基于STM32应用的FreeRTOS中断设置

 

 

解决记录:修改启动文件无效

参考 FreeRTOS简单移植到STM32F103ZET6 修改启动文件 startup_stm32f10x_hd.s 无效,并且会报错

修改前:

修改后:

__heap_limit
				IMPORT xPortPendSVHandler
				IMPORT xPortSysTickHandler
				IMPORT xPortSVCHandler	

                PRESERVE8
                THUMB


; Vector Table Mapped to Address 0 at Reset
                AREA    RESET, DATA, READONLY
                EXPORT  __Vectors
                EXPORT  __Vectors_End
                EXPORT  __Vectors_Size

__Vectors       DCD     __initial_sp               ; Top of Stack
                DCD     Reset_Handler              ; Reset Handler
                DCD     NMI_Handler                ; NMI Handler
                DCD     HardFault_Handler          ; Hard Fault Handler
                DCD     MemManage_Handler          ; MPU Fault Handler
                DCD     BusFault_Handler           ; Bus Fault Handler
                DCD     UsageFault_Handler         ; Usage Fault Handler
                DCD     0                          ; Reserved
                DCD     0                          ; Reserved
                DCD     0                          ; Reserved
                DCD     0                          ; Reserved
                ;DCD     SVC_Handler                ; SVCall Handler
                DCD     DebugMon_Handler           ; Debug Monitor Handler
                DCD     0                          ; Reserved
                ;DCD     PendSV_Handler             ; PendSV Handler
                ;DCD     SysTick_Handler            ; SysTick Handler

 后来发现已经在 FreeRTOSConfig.h 文件里面做了以下定义,所以无需以上处理:

#define xPortPendSVHandler 	PendSV_Handler
#define vPortSVCHandler 	SVC_Handler

 

 

 

  • 10
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
FreeRTOS 中,当系统发生硬件故障(如访问无效的内存地址或执行非法指令)时,会触发 HardFault_HandlerHardFault_Handler 是一个特殊的异常处理函数,用于处理硬件故障。 进入 HardFault_Handler 的过程如下: 1. 当发生硬件故障时,处理器会将当前的上下文信息(寄存器状态、堆栈等)保存到特定的存储区域。 2. 处理器会跳转到预定义的 HardFault_Handler 函数。 3. 在 HardFault_Handler 中,你可以根据需要进行一些处理,例如记录日志、重启系统等。 要进入 HardFault_Handler,你可以在 FreeRTOS 中通过以下步骤进行设置: 1. 在你的代码中包含 CMSIS(Cortex Microcontroller Software Interface Standard)头文件,通常是 "core_cm3.h"、"core_cm4.h" 或类似的。 2. 实现 HardFault_Handler 函数,并在其中添加你的处理逻辑。例如,你可以通过读取相应的寄存器状态来分析故障原因,并采取适当的措施。 3. 使用 CMSIS 提供的宏将 HardFault_Handler 函数注册为 HardFault 异常的处理函数。例如,在 ARM Cortex-M 系列处理器中,你可以使用以下代码进行注册: ```c void HardFault_Handler(void) { // 处理逻辑 } __attribute__((naked)) void HardFault_Handler(void) { __asm volatile ( "tst lr, #4\n" "ite eq\n" "mrseq r0, msp\n" "mrsne r0, psp\n" "ldr r1, [r0, #24]\n" "ldr r2, handler_address_const\n" "bx r2\n" "handler_address_const: .word HardFault_Handler_C\n" ); } ``` 这段代码将 HardFault_Handler 注册为 HardFault 异常的处理函数,并将其与 HardFault_Handler_C 函数关联起来。 请注意,以上代码是针对 ARM Cortex-M 系列处理器的示例,如果你使用的是其他类型的处理器,请参考相应的文档和头文件来进行配置。此外,根据你的需求,你可能还需要对 HardFault_Handler 函数进行适当的修改和扩展。
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值