关于FreeRTOS函数xSemaphoreGiveFromISR卡死的问题

news2024/11/18 10:33:08

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函数,卡在了下图这里:

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

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.coloradmin.cn/o/1222149.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C语言二进制数(ZZULIOJ1068:二进制数)

题目描述 将一个二进制数,转换为对应的十进制数。 输入:输入一个只含有’0’和’1’的字符串,以回车结束,表示一个二进制数。该二进制数无符号位,长度不超过31。 输出:输出一个整数,为该二进制数…

【C语言期末不挂科——指针篇1】

C语言指针初阶 文章目录 C语言指针初阶**什么是指针?**   **1)初识指针**  **2)地址的大小**  **3)指针变量** **指针的类型**   **1)指针对整数加减运算**  **2)指针的解引用** **野指针**  **1&#xff…

linux使用spi读取icm20608传感器数值一直显示0

1.这个问题困扰了我两天&#xff0c;把下面这些uart2相关引脚屏蔽掉也不行&#xff0c;ICM20608 ID 读出的数值一直为 0X0 pinctrl_flexcan2: flexcan2grp{ fsl,pins < /* MX6UL_PAD_UART2_RTS_B__FLEXCAN2_RX 0x1b020*/ /* MX6UL_PAD_UART2_CTS_B__FLEXCAN2_TX 0x1b020*/…

关于Ultra HDR Image的那些事

一、什么是Ultra HDR Image 2023年10月初&#xff0c;google正式发布了Android 14。该版本中引入了一个新的功能Ultra HDR Image&#xff0c;被誉为”图像技术的未来”。之前Android版本各手机厂商或许有自己的HDR图片技术&#xff0c;本文这里重点分析下Android14上google的实…

JavaScript 浮点数运算的精度问题及解决

JavaScript 浮点数运算的精度问题及解决 在 JavaScript 中整数和浮点数都属于 Number 数据类型&#xff0c;当浮点数做数学运算的时候&#xff0c;你经常会发现一些问题&#xff0c;举几个例子&#xff1a; 0.1 0.2 0.30000000000000004 console.log(0.1 0.2) 0.3000000…

springcloud新闻发布系统源码

开发技术&#xff1a; jdk1.8&#xff0c;mysql5.7&#xff0c;nodejs&#xff0c;idea&#xff0c;vscode springcloud springboot mybatis vue elementui 功能介绍&#xff1a; 用户端&#xff1a; 登录注册 首页显示搜索新闻&#xff0c;新闻分类&#xff0c;新闻列表…

VS+Qt+C++ Yolov8物体识别窗体程序onnx模型

程序示例精选 VSQtC Yolov8物体识别窗体程序onnx模型 如需安装运行环境或远程调试&#xff0c;见文章底部个人QQ名片&#xff0c;由专业技术人员远程协助&#xff01; 前言 这篇博客针对《VSQtC Yolov8物体识别窗体程序onnx模型》编写代码&#xff0c;代码整洁&#xff0c;规…

【Hello Go】Go语言工程管理

工程管理 工作区工作区介绍GOPATH设置 包自定义包main包main函数和init函数导入包点操作别名操作_操作 测试案例GOPATH配置go install使用 在我们实际的工作中 直接运用到编译器进行编译链接的场景少之又少 这是因为 在工程中不会只有一个源文件 并且源文件之间也有着相互依赖…

java学习part05

43-流程控制-使用Scanner类从键盘获取数据_哔哩哔哩_bilibili 1.接收输入 步骤 例子 2.生成随机数 3.switch-case 4.for 5.while

IDEA插件下载到本地

IDEA插件下载到本地 官网下载【点击跳转】

基于DE10-Standard Cyclone V SoC FPGA学习---开发板简介

基于DE10-Standard Cyclone V SoC FPGA学习---开发板简介 简介产品规格基于 ARM 的 HPS配置与调试存储器件通讯连接头显示器音频视频输入模数转换器开关、按钮、指示器传感器电源 DE10-Standard 开发板系统框图Connect HTG 组件配置设计资源其他资源 简介 开发板资料 见 DE10-…

rv1126-rv1109-openssh

这是一个工具&#xff0c;可以通过ssh远程登录来操作&#xff0c;非常逆天&#xff01; 于是rv1109代码自身自带有openssh 所以只需要打开config即可 diff --git a/buildroot/configs/rockchip_rv1126_rv1109_spi_nand_defconfig b/buildroot/configs/rockchip_rv1126_rv1109…

Pytorch plt.scatter()函数用法

一.scatter&#xff08;&#xff09;函数的定义 matplotlib.pyplot.scatter(x, y, sNone, cNone, markerNone, cmapNone, normNone, vminNone, vmaxNone, alphaNone, linewidthsNone, vertsNone, edgecolorsNone, *, dataNone, **kwargs) 特征值作用x&#xff0c;y绘制散点图…

MAC电脑连接外接显示屏,颜色显示有问题,又粉、紫色蒙版,问题处理(1)

问题描述 买了一个显示器&#xff0c;想给mac做分屏使用&#xff0c;结果连上之后发现&#xff0c;整个屏幕像是被蒙上了一层紫色的蒙版。 就像下面展示的一样&#xff1a; 解决 将显示器颜色空间改为RGB颜色空间即可。 打开显示器菜单&#xff0c;找到颜色空间选项&#…

(六)什么是Vite——热更新时vite、webpack做了什么

vite分享ppt&#xff0c;感兴趣的可以下载&#xff1a; ​​​​​​​Vite分享、原理介绍ppt 什么是vite系列目录&#xff1a; &#xff08;一&#xff09;什么是Vite——vite介绍与使用-CSDN博客 &#xff08;二&#xff09;什么是Vite——Vite 和 Webpack 区别&#xff0…

2023数维杯数学建模C题完整版本

已经完成全部版本&#xff0c;获取请查看文末下方名片 摘要 随着人工智能在多个领域的快速发展&#xff0c;其在文本生成上的应用引起了广泛关注。本研究聚焦于辨识人工智能&#xff08;AI&#xff09;生成文本的基本规则&#xff0c;并探究AI文本的检测及其与人类文本的区分…

Boolean源码解剖学

原创/朱季谦 有天突发其想&#xff0c;想看一下Boolean底层都做了些什么&#xff0c;故而去看了一番Boolean的源码&#xff0c;基于一些思考的基础上&#xff0c;输出了这篇文章。 一.类继承 Boolean的源码类定义部分如下&#xff1a; 1 public final class Boolean implemen…

Cascade-MVSNet论文笔记

Cascade-MVSNet论文笔记 摘要1 立体匹配&#xff08;Stereo Matching&#xff09;2 多视图立体视觉&#xff08;Multi-View Stereo&#xff09;3 立体视觉和立体视觉的高分辨率输出4 代价体表达方式&#xff08;Cost volume Formulation&#xff09;4.1 多视图立体视觉的3D代价…

农户建档管理系统的设计与实现-计算机毕业设计源码20835

摘 要 随着互联网趋势的到来&#xff0c;各行各业都在考虑利用互联网将自己推广出去&#xff0c;最好方式就是建立自己的互联网系统&#xff0c;并对其进行维护和管理。在现实运用中&#xff0c;应用软件的工作规则和开发步骤&#xff0c;采用Java技术建设农户建档管理系统。 本…

keepalived离线安装

上传离线安装包 将离线安装包拖动到服务器上 进入到离线安装包路径&#xff0c;执行下面脚本进行安装 rpm -Uvh --force --nodeps *.rpm