关于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 handleif(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.mzph.cn/news/145576.shtml

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

相关文章

【C语法学习】26 - strcat()函数

文章目录 1 函数原型2 参数3 返回值4 使用说明5 示例5.1 示例1 1 函数原型 strcat():将src指向的字符串拼接在dest指向的字符串末尾,函数原型如下: char *strcat(char *dest, const char *src);2 参数 strcat()函数有两个参数src和dest&am…

uniapp插件开发

安装android studio:安装目录下bin下的此文件,是用来修改分配给android studio的占用内存。 Android 11足够用。 创建新项目: 目录结构介绍: UI组件介绍:在设计程序界面时可以使用可视化拖拽的方式,没有必要…

被 Next.js 的环境变量给坑了一把...

最近在使用 Next.js 时遇到了一个问题,最后原因竟是 .env 取值问题,为这个问题花费了数小时的时间,希望看到这篇文章的朋友,如果遇到类似问题,不要重蹈覆辙吧。 起初报错内容如下所示: 一开始关注点在下面…

ubuntu提高 github下载速度

Github一般用于Git的远程仓库,由于服务器位于国外,国内访问速度比较慢,为了提高访问速度,决定绕过DNS域名解析。 获取Github的IP地址 按下ctrl+alt+T打开命令终端,输入: nslookup gi…

玩具、儿童用品、儿童服装上亚马逊TEMU平台CPC认证办理

CPC认证是Childrens Product Certificate的简称,即儿童产品证书。它是美国强制性法规CPSIA要求的一部分,该法规主要针对12岁及以下儿童使用的产品,如玩具、儿童用品、儿童服装等。 一、儿童小汽车CPC测试项目可能会因产品标准和法规的不同而…

android初集成flutter,遇到的问题

环境 studio版本:2022.1.1 flutter版本:2.8.0 电脑:mac flutter项目总是报错,编译不过 以 Resources Root 加载 记得设置dart:主工程和flutter项目都需要设置,否则不出现手机链接 下面这个样子就是好了&…

窗口管理工具 Mosaic mac中文版功能特点

MosAIc mac是一种窗口管理工具,可帮助您在计算机屏幕上有效地组织和管理多个应用程序窗口。它提供了一种直观的方式来调整和排列窗口,以最大化工作效率。 MosAIc mac窗口管理软件功能和特点 窗口布局:MosAIc允许您选择不同的窗口布局&#x…

抠某区域地图方法

1.打开阿里云数据可视化平台DataV.GeoAtlas地理小工具系列 2. 选择要抠出来的区域,右侧选择要下载的json文件,如红框所示 3. 打开下载的文件,内容全部复制。 4. 打开百度地图示例Examples - Apache ECharts 5. 如下图所示,将下…

数字化转型时代,商业智能BI到底是什么?

据国际数据公司(IDC)预测,2025年时中国产生的数据量预计将达48.6ZB,在全球中的比例为27.8%。商业智能BI这一专为企业提供服务的数据类解决方案,仅2021年上半年在中国商业智能BI市场规模就达到了3.2亿美元,商…

(二)什么是Vite——Vite 和 Webpack 区别(冷启动)

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

电源线虚接,导致信号线发烫

音频板的信号是经过隔直电容接到音频板的。

【ES6标准入门】JavaScript中的模块Module的加载实现:循环加载和Node加载,非常详细,建议收藏!!!

😁 作者简介:一名大四的学生,致力学习前端开发技术 ⭐️个人主页:夜宵饽饽的主页 ❔ 系列专栏:JavaScript进阶指南 👐学习格言:成功不是终点,失败也并非末日,最重要的是继…

IDEA写mybatis程序,java.io.IOException:Could not find resource mybatis-config.xml

找不到mybatis-config.xml 尝试maven idea:module&#xff0c;不是模块构造问题 尝试检验pom.xml&#xff0c;在编译模块添加了解析resources内容依旧不行 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.or…

chrome 浏览器个别字体模糊不清

特别是在虚拟机里&#xff0c;有些字体看不清&#xff0c;但是有些就可以&#xff0c;设置办法&#xff1a; chrome://settings/fonts 这里明显可以看到有些字体就是模糊的状态&#xff1a; 把这种模糊的字体换掉即可解决一部分问题。 另外&#xff0c;经过观察&#xff0c;…

【C++】数组中出现次数超过一半的数字

代码&#xff1a; class Solution { public:/*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方法规定的值即可** * param numbers int整型vector * return int整型*/int MoreThanHalfNum_Solution(vector<int>& numbers) {int …

vue3 + ts项目(无vite)报错记录

记录项目创建后遇到的报错 1.类型“Window & typeof globalThis”上不存在属性“_CONFIG”。ts(2339) 问题描述&#xff1a; 使用全局 window 上自定义的属性&#xff0c;TypeScript 会报属性不存在 解决&#xff1a;需要将自定义变量扩展到全局 window 上&#xff0c…

word文档转换为ppt文件,怎么做?

大家是否会遇到需要将word文档转换为ppt文件的情况&#xff1f;除了反反复复粘贴复制以外&#xff0c;还有其他方法可以转换文件格式&#xff0c;今天给大家分享word转换ppt方法。 首先我们先将word文件打开大纲模式 然后我们将文中的大标题设置为1级标题&#xff0c;副标题设…

服务器集群配置LDAP统一认证高可用集群(配置tsl安全链接)-centos9stream-openldap2.6.2

写在前面 因之前集群为centos6&#xff0c;已经很久没升级了&#xff0c;所以这次配置统一用户认证也是伴随系统升级到centos9时一起做的配套升级。新版的openldap配置大致与老版本比较相似&#xff0c;但有些地方配置还是有变化&#xff0c;另外&#xff0c;铺天盖地的帮助文…

C/C++高频面经-秋招篇

自己在秋招找工作过程中遇到的一些C/C面试题&#xff0c;大中小厂都有&#xff0c;分享出来&#xff0c;希望能帮到有缘人。 C语言 snprintf()的使用 函数原型为int snprintf(char *str, size_t size, const char *format, …) 两点注意&#xff1a; (1) 如果格式化后的字符…

【广州华锐互动】消防安全宣传知识3D交互展示提升公众学习沉浸感

随着科技的快速发展&#xff0c;我们的生活与工作环境愈发复杂&#xff0c;火灾风险也随之提高。为了提高公众的消防灭火能力&#xff0c;普及消防安全知识&#xff0c;广州华锐互动开发了消防安全宣传知识3D交互展示系统。 这是一种全新的教育方式&#xff0c;它利用3D技术&am…