关于FreeRTOS函数xSemaphoreGiveFromISR卡死的问题

0. 概述

关于FreeRTOS函数xSemaphoreGiveFromISR卡死的问题

1. 遇到的问题

在使用FreeRTOS调试激光雷达检测面积的项目的时候,遇到一个现象:在新加了一个线程之后,把程序下载到板子之后程序不会运行(实际上已经运行了,原因后续再进行解释),此时再下载一次程序程序才能正常运行起来。

具体现象

  1. 程序下载之后运行,发现没有正确的运行现象;此时再下载一次程序即可正常运行。
  2. 板子在正常运行状态下,如果下载程序,发现程序无法运行。
  3. 板子整体断电上电之后每次都能正常运行。
  4. 程序如果未运行到void IST8310Task(void *argument)这个线程的for循环之前重新下载程序的话(这个线程的主要任务会延时大约3s之后才启动),程序是可以正常运行的;但是如果运行到上面的线程具体任务之后再进行下载程序,程序就无法运行了。

2. 尝试的解决办法

1. 更换调试器,修改调试器的设计

通过修改调试器的设置尝试解决:尝试了多个配置发现问题依旧
将使用的H7TOOL更换为无线DAP和STLINK V3发现问题依旧
最终定位到这个问题不是仿真器以及仿真器设置的问题

2. 尝试屏蔽新加入的线程代码

void IST8310Task(void *argument)//IST8310的控制线程
{
	extern QueueHandle_t IST8310QueueHandle;//IST8310的数据
	extern SemaphoreHandle_t IST8310_DRYBinaryHandle;//IST8310转换完成的信号量
	imusensorStruct_t IST8310Data;
	BaseType_t xResult =pdPASS;//消息队列发送状态
	
	osDelay(3000);
	
	//init error handle
	if(IST8310_Init() != IST8310_NO_ERROR)
	{
		for(;;)
		{
			osDelay(100);
		}
	}
	
	osDelay(100);
	
	for(;;)
  {	
		xResult = xSemaphoreTake(IST8310_DRYBinaryHandle,0);//等待DRY完成
		if(xResult ==pdPASS)
		{		
			IST8310_Updata(&IST8310Data);
			IST8310_Data_Single_Measurement_Once();
			
			IST8310Data.yaw_RAW = atan(IST8310Data.rawMag_X/IST8310Data.rawMag_Y)/2/3.141592654*360;
			xQueueOverwrite(IST8310QueueHandle,&IST8310Data);//覆盖消息队列发送
			

		}
		osDelay(50);
  }
}

上述代码主要完成IST8310的单次数据读取,即在循环中等待IST8310的转换完成信号量(在GPIO的中断处理函数里面进行信号量的释放),然后进行数据读取,然后与IST8310通信进行下一个数据的读取,当IST8310数据完成之后会将其DRY引脚变为低电平,此时会进入到下面的GPIO中断处理函数。

void  HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) __attribute__((optnone))
{
	extern SemaphoreHandle_t IST8310_DRYBinaryHandle;//IST8310转换完成的信号量
	BaseType_t xHigherPriorityTaskWoken = pdFALSE;//定义一个变量,记录一下退出中断是否进行任务切换
	BaseType_t xResult;		
	
	if (GPIO_Pin==IST_DRDY_Pin)                     //判断是否为IST8310的外部中断
	{
		xResult = xSemaphoreGiveFromISR(IST8310_DRYBinaryHandle,&xHigherPriorityTaskWoken);//释放信号量,告诉任务已经发送完了,如果有更高优先级任务就绪就进行任务切换
		
		portYIELD_FROM_ISR(xHigherPriorityTaskWoken);//如果需要的话就进行一次任务切换
	}
	
	
}

此时经过把该线程的代码屏蔽之后,即该线程不启动,程序就可以正常运行,每次下载完程序之后都会正常运行。但此时并没有发现具体问题所在,还需要尝试其它办法进行解决。

3. 尝试加大程序的栈空间

此时怀疑是程序栈太小了,程序可能偶尔跑飞了,但是实际上把栈空间变为原来的16倍现象仍然一致,因此排除这个问题。

4. 尝试调整程序的优化等级

将程序的优化等级调整为0之后发现程序可以正常运行!!!

因此,此时怀疑是keil优化的问题(keil无端背锅0^0,因为keil的代码优化确实会出现一些问题)。

经过查阅各种网上资料,找到了在keil里面进行设置优化等级各种办法,包括不限于:

  • 整个工程的优化等级调整(最简单,直接在工程设置里面进行调整即可)
  • 文件夹或者单个文件的优化等级调整(右键文件夹或者文件进行相关设置)
  • 单个函数的优化等级(在函数的定义后面加上代码:__attribute__((optnone))太强了,也算是这次debug这个问题的一个收获吧。)

此时根据上面的内容进行了如下的尝试

  1. 更改整个工程的优化等级为0发现可以正常运行了!!!
  2. 更改工程优化等级为1,手动单独将某个.c文件的优化等级改为0。经过很久的尝试(文件太多了,汗),发现将stm32h7xx_hal_gpio.c的优化等级改为0之后程序可以运行,因此就定位到问题可能出在了GPIO上面,也给后文的调试找到了方向。(但是stm32h7xx_hal_gpio.c这个文件是STM32CubeMX自动生成的,因此不太愿意用这种修改优化等级的办法解决,因此还需要后面再排查一下问题)

5. 尝试使用DAP在线调试的功能看程序是否运行了

调试的时候发现程序没运行起来的时候实际上是进入了xSemaphoreGiveFromISR函数,卡在了下图这里:

image-20231117153249822

其实这个时候就应该发现问题是出现在这里了(在线仿真状态下程序一直卡在这里!)

3. 问题原因分析

这个问题其实是STM32CUBEMX自动生成FreeRTOS代码的坑:

  1. HAL库初始化完之后会自动打开GPIO的输入中断(当然其它中断也可能会发生类似的问题);
  2. FreeRTOS的初始化(包含任务、消息队列、信号量等OS相关内容的创建)在HAL库及其外设初始化的后面;
  3. 程序需要在GPIO的中断里面进行信号量的释放,以表示IST8310转换完成了,可以在线程里面进行读取了;
  4. IST8310的数据完成引脚可能会在FreeRTOS的初始化之前完成,从而在FreeRTOS还没初始化的时候就进入GPIO的中断进行信号量的释放了。

上述过程就会导致:

  1. 如果程序处于还未运行的状态(可能是程序卡死或者其它状态),此时GPIO没有中断,此时下载程序的话可以保证IST8310的数据完成引脚在FreeRTOS的初始化之后触发中断,因此程序会正常运行;
  2. 如果程序处于运行的状态,再下载程序的话,一瞬间IST8310的DRY引脚会变为低电平(IST8310转换比较快),此时IST8310芯片还处于上电的状态,因此改低电平状态会一直保持,此时新程序运行的时候会发生这种现象:IST8310的数据完成引脚在FreeRTOS的初始化之前触发中断,此时在GPIO中断里面IST8310_DRYBinaryHandle这个信号量还没被创建,因此会使程序整体卡死在Handle模式,不会回到正常的Thread模式,任务无法正常调度,表现为整体的卡死,表现为任务下载完程序之后程序不会自动运行。

4. 解决办法

  1. 定义一个全局变量用来记录OS是否已经初始化完成了。uint8_t IS_OS_Running = 0;
  2. 在main函数的开始地方将变量值设为0,表示OS还没运行(初始化)。IS_OS_Running = 0;
  3. void MX_FREERTOS_Init(void)函数的结尾处将变量设为1,表示OS初始化完成了。
  4. 在GPIO中断处理函数里面对变量IS_OS_Running 进行判断,参考如下代码:
void  HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) __attribute__((optnone))
{
	extern SemaphoreHandle_t IST8310_DRYBinaryHandle;//IST8310转换完成的信号量
	BaseType_t xHigherPriorityTaskWoken = pdFALSE;//定义一个变量,记录一下退出中断是否进行任务切换
	BaseType_t xResult;		
	
	if (GPIO_Pin==IST_DRDY_Pin)                     //判断是否为IST8310的外部中断
	{
		if(IS_OS_Running ==1)//cubemx的坑,freertos卡死,因为os没启动的时候进入了中断,中断里面释放信号量,造成卡死
		{
			xResult = xSemaphoreGiveFromISR(IST8310_DRYBinaryHandle,&xHigherPriorityTaskWoken);//释放信号量,告诉任务已经发送完了,如果有更高优先级任务就绪就进行任务切换
		}
		
		portYIELD_FROM_ISR(xHigherPriorityTaskWoken);//如果需要的话就进行一次任务切换
	}
}

上述代码可以实现:在OS还未初始化完成的时候不进行信号量的释放操作,只有OS被成功初始化之后才进行信号量释放,从而避免以上的问题。

5. 总结

这个问题可以简单归结为STM32CubeMX生成代码的先后逻辑顺序问题。在使用的时候要注意:在中断处理函数里面进行信号量等与OS相关的代码要先判断一下OS是否已经初始化完毕了,因为可能会出现在OS初始化完成之前会进入一些中断处理函数导致卡死。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/157355.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Spring 6 提前编译:AOT

1、AOT概述 1.1、JIT与AOT的区别 JIT和AOT 这个名词是指两种不同的编译方式,这两种编译方式的主要区别在于是否在“运行时”进行编译 (1)JIT, Just-in-time,动态(即时)编译,边运行边编译; 在程序运行时…

解决Error:java: System Java Compiler was not found in classpath

解决Error:java: System Java Compiler was not found in classpath 一、配置maven 注意:我的C盘比较大直接配置在了C盘,建议配置到其他盘,记得做maven环境变量配置 二、卸载本地JDK,换个版本安装配置环境变量 重启电脑或idea,…

开发人员面临的10个最常见的JavaScript问题

今天,JavaScript 是几乎所有现代 Web 应用的核心。这就是为什么JavaScript问题,以及找到导致这些问题的错误,是 Web 发者的首要任务。 用于单页应用程序(SPA)开发、图形和动画以及服务器端JavaScript平台的强大的基于…

C语言调用【Python3】

一、搭建编译环境 终端查询系统及软件版本dpkg -l 列出所有已安装的软件包 二、C语言中调用Python 使用 GCC编译并链接 Python 3.10 的共享库如何在C中获取和修改 sys.path 三、C语言调用无参python函数 四、C语言调用有参python函数 一、搭建编译环境 通过C语言调用Pyth…

联想Win11系统的任务栏格式调整为居中或居左

一 .目的 联想Win11系统的任务栏格式调整为居中或居左 二 .方法 2.1 鼠标任意放到电脑桌面位置,点击鼠标右键,显示后县级【显示设置】 2.2 个性化→任务栏→任务栏行为→对其方式:按需或个人习惯进行选择【靠左】 2.3 成功调整&#x…

查看libc版本

查看libc库版本 查看系统libc版本 $ ldd --version ldd (Ubuntu GLIBC 2.27-3ubuntu1.2) 2.27 Copyright (C) 2018 Free Software Foundation, Inc. This is free software; see the source for copying conditions. There is NO warranty; not even for MERCHANTABILITY or …

JDK8 TLS10 is not accepted

TLS 1.0/1.1 changes in OpenJDK and Amazon Corretto Affected OpenJDK versions: VersionRelease numberOpenJDK 88u292 and newerOpenJDK 1111.0.11 and newerOpenJDK 16 and aboveAll versions 《Amazon Corretto与OpenJDK的TLS 1.0和1.1弃用:影响与应对之策》…

Workplace Search 的演变:使用 Elasticsearch 搜索你的私人数据

作者:Dana Juratoni, Aditya Tripathi Workplace Search 功能将来将与 Elastic Search 合并。 这是你需要了解的内容。 生成式人工智能技术的最新进展为搜索带来了一系列可能性。 随着开发人员构建新的体验,用户正在采用新的搜索使用方式 —— 从用自然…

陌陌附近人打招呼脚本,可自动回复消息,按键精灵开源脚本

用按键写的一个陌陌自动打招呼发送指定话术消息的一个脚本,它还会检测对方的消息,然后自动回复指定信息,下面是UI界面和代码,你可以直接粘贴到自己的按键精灵里面运行,不会出错,已经测试过。 UI界面&#…

基于 selenium 实现网站图片采集

写在前面 有小伙伴选题,简单整理理解不足小伙伴帮忙指正 对每个人而言,真正的职责只有一个:找到自我。然后在心中坚守其一生,全心全意,永不停息。所有其它的路都是不完整的,是人的逃避方式,是对…

腾讯待办关停之后,如何提醒待办事项?

如果你之前使用腾讯待办这款小程序来记录待办事项并设置提醒,就会发现近日弹出的“业务关停通知”公告,由于业务方向调整,腾讯待办将于2023年的12月20日全面停止运营并下架,这表示以后无法继续使用了。但是腾讯待办关停之后&#…

别再吐槽大学教材了,来看看这些网友强推的数学神作!

前言 关于大学数学教材的吐槽似乎从来没停止过。有人慨叹:数学教材晦涩难懂。错!难懂,起码还可以读懂。数学教材你根本读不懂;也有人说:数学教材简直就是天书。 数学教材有好有坏,这话不假,但更…

学会使用这个平台,教你制作出色的产品画册?

产品画册是企业和用户之间的桥梁,它可以第一时间给用户传递我们企业的最新产品信息。如何制作一本精美的产品画册呢? 这个不难,给大家推荐一款免费实用的在线制作工具FLBOOK ,用这个平台可以轻松制作精美电子产品画册。 在制作产…

Python递归函数的定义和几个小例子

大家早好、午好、晚好吖 ❤ ~欢迎光临本文章 如果有什么疑惑/资料需要的可以点击文章末尾名片领取源码 递归函数 (1)什么是递归函数? 我们都知道,一个函数可以调用其他函数。如果这个函数在内部调用它自己,那么这个…

纯前端模板文件下载如何精确控制下载的文件名字

在写项目的时候,遇到了一个需要把给定的文件放到页面中,然后用户点击下载按钮将这个文件下载下来,我将其存入了云服务之中(这个云服务是不会清空的,内存又不值几个钱),但是当我下载的时候,下载的文件名是存…

「校园 Pie」 系列活动正式启航,首站走进南方科技大学!

PieCloudDB 社区校园行系列活动「校园 Pie」已正式启动。「校园 Pie」旨在促进数据库领域的学术交流,提供一个平台让学生们了解最新的数据库发展趋势和相关技术应用。 在「校园 Pie」系列活动中,PieCloudDB 社区将携拓数派技术专家,社区大咖…

LLM建模了什么,为什么需要RAG

LLM近期研究是井喷式产出,如此多的文章该处何处下手,他们到底又在介绍些什么、解决什么问题呢?“为学日增,为道日损”,我们该如何从如此多的论文中找到可以“损之又损以至于无”的更本质道或者说是这个方向的核心模型。…

轻松玩转华为MateX5分屏功能,乐趣层出不穷!

✅小窗交互,沉浸体验不打断; ✅分区截屏,花式截图,一招搞定; ✅跨屏拖拽,随心分享易如反掌; ✅悬停视频会议,沟通效率大不同。

广州华锐互动VRAR:利用VR开展刑事案件公安取证培训,沉浸式体验提升实战能力

随着科技的飞速发展,虚拟现实(VR)技术为我们的生活和工作带来了前所未有的便利。近年来,VR技术在刑事案件公安取证培训中的应用逐渐显现出其独特优势。通过模拟真实的犯罪现场,VR技术为学员提供了沉浸式的体验,使他们在安全的环境…
最新文章