STM32运行FreeRtos系统出现信号量错误

本文详细分析了一起在STM32 FreeRTOS项目中遇到的信号量交互异常问题,问题源于调度器在内存分配函数pvPortMalloc执行时被挂起。该问题导致在调度器挂起期间尝试获取信号量,违反了FreeRTOS的使用规则。解决方案包括优化内存分配策略和避免在调度器挂起时进行信号量操作。同时强调了在FreeRTOS系统中使用信号量和中断服务函数的注意事项。
摘要由CSDN通过智能技术生成

最近在开发STM32项目中出现了信号量交互异常的问题,该项目搭载的是FreeRtos系统,具体内容如下:

现象:

在系统运行过程中,出现如下断言错误:

问题排查:

找到代码断言定义处如下:

#define configASSERT( x ) if( ( x ) == 0 ) { taskDISABLE_INTERRUPTS(); printf("assert: %s->%d\n", (__FUNCTION__), (__LINE__));for( ;; ); } 

由日志可以知道,是系统出现了信号量在执行获取过程中错误。
沿着错误继续追踪代码的上层调用,很快可以找到对应的configASSERT出错地方,如下:

/* Cannot block if the scheduler is suspended. */
	#if ( ( INCLUDE_xTaskGetSchedulerState == 1 ) || ( configUSE_TIMERS == 1 ) )
	{
		configASSERT( !( ( xTaskGetSchedulerState() == taskSCHEDULER_SUSPENDED ) && ( xTicksToWait != 0 ) ) );
	}
	#endif

根据configASSERT的定义可知,只有括号中的x等于0,才会报错,可得出代码运行到此处:
xTaskGetSchedulerState() == taskSCHEDULER_SUSPENDED 成立。说明调度器当前状态处于
挂起态,xTaskGetSchedulerState() 函数是用于获取当前调度器状态。
xTicksToWait != 0 成立。 说明传递的入口参数信号量等待时间不为0。
综上分析,并结合FreeRtos系统代码注释 /* Cannot block if the scheduler is suspended. 如果计
划程序已挂起,则无法阻止 */
根据我们对信号量实现功能的认知:信号量可以用来实现对共享资源的互斥访问和多任务之间的
同步。
由此结合分析可以知道:信号量的获取和释放操作运行在多任务的运行状态下,当正要执行
xQueueSemaphoreTake进行信号量获取操作的时候,任务调度器且是处于挂起状态,这是不被允
许的。
重点:当调度器处于挂起状态时,是不能调用FreeRtos API函数的
 

那到底是什么操作使得此时的调度器被挂起了呢?
可以通过结合日志出错地方的上下文查看,找到引起调度器挂起的使用地方,也就是
vTaskSuspendAll()使用。

void vTaskSuspendAll( void )
{
/* A critical section is not required as the variable is of type
BaseType_t. Please read Richard Barry's reply in the following link to a
post in the FreeRTOS support forum before reporting this as a bug! -
http://goo.gl/wu4acr */
++uxSchedulerSuspended;
}

通过一步步排查定位到是FreeRtos内存分配函数pvPortMalloc引起的调度器挂起。在内存分配过
程中,会先调用vTaskSuspendAll挂起调度器,分配完成后在调用xTaskResumeAll()开启调度器

void *pvPortMalloc( size_t xWantedSize )
{
void *pvReturn = NULL;
static uint8_t *pucAlignedHeap = NULL;

	/* Ensure that blocks are always aligned to the required number of bytes. */
	#if( portBYTE_ALIGNMENT != 1 )
	{
		if( xWantedSize & portBYTE_ALIGNMENT_MASK )
		{
			/* Byte alignment required. */
			xWantedSize += ( portBYTE_ALIGNMENT - ( xWantedSize & portBYTE_ALIGNMENT_MASK ) );
		}
	}
	#endif

	vTaskSuspendAll();
	{
		if( pucAlignedHeap == NULL )
		{
			/* Ensure the heap starts on a correctly aligned boundary. */
			pucAlignedHeap = ( uint8_t * ) ( ( ( portPOINTER_SIZE_TYPE ) &ucHeap[ portBYTE_ALIGNMENT ] ) & ( ~( ( portPOINTER_SIZE_TYPE ) portBYTE_ALIGNMENT_MASK ) ) );
		}

		/* Check there is enough room left for the allocation. */
		if( ( ( xNextFreeByte + xWantedSize ) < configADJUSTED_HEAP_SIZE ) &&
			( ( xNextFreeByte + xWantedSize ) > xNextFreeByte )	)/* Check for overflow. */
		{
			/* Return the next free byte then increment the index past this
			block. */
			pvReturn = pucAlignedHeap + xNextFreeByte;
			xNextFreeByte += xWantedSize;
		}

		traceMALLOC( pvReturn, xWantedSize );
	}
	( void ) xTaskResumeAll();

	#if( configUSE_MALLOC_FAILED_HOOK == 1 )
	{
		if( pvReturn == NULL )
		{
			extern void vApplicationMallocFailedHook( void );
			vApplicationMallocFailedHook();
		}
	}
	#endif

	return pvReturn;
}

由于挂起调度器实现的临界区只可以保护一段代码区间不被其他任务打断,而此时中断是使能
的,也就是可以被中断打断。加上频繁使用pvPortMalloc分配和释放使用不固定大小的内存空间过
程中,容易出现在挂起调度器时被中断打断后,在其他地方进行信号量操作,而导致出现以上问
题。


解决方法:
1.可以将频繁分配释放不固定内存大小的pvPortMalloc操作改为每次都分配释放同样大小的内存空
间,这样也有利于减小内存碎片发生
2.也可选择将你需要经常分配释放那块内存改为使用定义的一块全局的数组空间,而不用动态分配
的内存空间

在操作FreeRtos系统过程中应该注意事项:
1.互斥量不能在中断服务函数中使用,因为其特有的优先级继承机制只在任务起作用,在中断的上
下文环境毫无意义
2.FreeRtos系统对应的一些功能接口函数是有区分用于中断的接口和非中断中使用的接口,如果
在中断函数里调用了非中断的系统接口,就会出现系统性断言错误
 

  • 5
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 16
    评论
评论 16
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值