FreeRTOS【4】线程挂起和恢复
1.开发背景
基于上一篇指引,成功创建并启动线程后,线程已经开始运行了,但是有时我们需要线程暂停运行,例如某个线程是控制 LED 闪灯的,如果现在需要让 LED 停止工作,单纯的关闭 LED 是没用的,因为下一时刻线程可能就会重新打开 LED 导致程序没有达到预期,所以线程引入了一个挂起的状态,如下图,FreeRTOS 引入了线程的多个状态。

其中包含 Running、Ready、Suspended、Blocked 等状态。
Running:即运行态,是成功获取了 CPU 使用权的线程。
Ready:即准备态,单核 CPU 在同一时刻只能做一件事情,所以如果当前有其他线程获取了 CPU 的使用权(一般是高优先级线程),这个时候等待运行的线程就是准备态。
Suspended:即挂起态,正如上文说到的,如果我们想让某个线程暂时停止工作,就可以挂起对应线程,被挂起的线程就是挂起态。
Blocked:即阻塞态,因为线程存在高低优先级,如果高优先级线程一直运行会导致低优先级线程一直抢占不到 CPU 的使用权,如果使用挂起的方式去挂起高优先级线程,那么高优先级线程的实时性就会大打折扣,所以就引入了阻塞的概念,高优先级线程可以一直阻塞在某个事件,在阻塞期间会让出 CPU 的使用权,但是一旦高优先级线程满足指定事件就会立刻抢占低优先级线程的 CPU 使用权,这样就保证了高优先级线程的实时性。
上述纯粹个人理解,如有误请见谅,可以参考链接:FreeRTOS task states and state transitions described
2.开发需求
挂起、恢复和删除已有线程
3.开发环境
window10 + MDK + STM32F429 + FreeRTOS10.3.1
4.实现步骤
4.1 线程挂起其他线程
1)创建控制线程和 2 个测试线程
/* 测试初始化 */
void aTest_Init(void)
{/* 创建动态任务 */xTaskCreate(TaskCtrl, "TaskCtrl", 500, NULL, 5, &p->taskCtrl);/* 共用一个任务函数 创建多个任务 */static char whichTask[TASK_LIST_SIZE][3] = {0};for (int i = 0; i < TASK_LIST_SIZE; i++){snprintf(whichTask[i], 2, "%d", i);xTaskCreate(TaskList, "TaskList", 500, (void*)whichTask[i], 5, &p->taskList[i]);}
}
2)测试线程循环打印
/* 动态任务组 */
static void TaskList(void *pvParameters)
{int count = 0;int whichTask = atoi(pvParameters);Log_Debug("%s [%d]\r\n", __func__, whichTask);for ( ; ; ){vTaskDelay(1000);Log_Debug("%s [%d] count = %d\r\n", __func__, whichTask, count++);}
}
3)控制线程间断挂起和恢复,使用 vTaskSuspend 挂起线程,vTaskResume 恢复线程。
/* 动态任务 */
static void TaskCtrl(void *pvParameters)
{Log_Debug("%s\r\n", __func__);/* 挂起线程 0 */vTaskDelay(3000);vTaskSuspend(p->taskList[0]);vTaskDelay(3000);vTaskResume(p->taskList[0]);for ( ; ; ){vTaskDelay(1000);}
}
4)测试结果

如图所示,可以看出线程0 中间有 3 个周期是停止工作的。
4.2 线程挂起线程本身
1)基于上面试验的基础上,修改测试线程为打印日志后挂起自身,这里做这个试验是为了验证 vTaskSuspend 如果传入参数为 NULL,即指向线程本身。
/* 动态任务组 */
static void TaskList(void *pvParameters)
{int count = 0;int whichTask = atoi(pvParameters);Log_Debug("%s [%d]\r\n", __func__, whichTask);for ( ; ; ){vTaskDelay(1000);Log_Debug("%s [%d] count = %d\r\n", __func__, whichTask, count++);vTaskSuspend(NULL);}
}
2)控制线程只需定期恢复线程即可
/* 动态任务 */
static void TaskCtrl(void *pvParameters)
{Log_Debug("%s\r\n", __func__);/* 挂起线程 0 */vTaskDelay(3000);vTaskResume(p->taskList[0]);for ( ; ; ){vTaskDelay(1000);}
}
3)测试结果

如图所示,测试线程执行了一次就挂起了本身,控制线程间隔 3s 之后唤醒了一次测试线程0
4)源码解析
vTaskSuspend
-> prvGetTCBFromHandle
/** Several functions take an TaskHandle_t parameter that can optionally be NULL,* where NULL is used to indicate that the handle of the currently executing* task should be used in place of the parameter. This macro simply checks to* see if the parameter is NULL and returns a pointer to the appropriate TCB.*/
#define prvGetTCBFromHandle( pxHandle ) ( ( ( pxHandle ) == NULL ) ? pxCurrentTCB : ( pxHandle ) )
如源码所示,如果传入指针为空,即获取当前控制块 pxCurrentTCB
4.3 中断中恢复线程
1)基于上面的实验使用中断来代替控制线程来恢复已挂起的测试线程
/* Key2 PC13 Key0 PH3 Key1 PH2 */
void Exti13_TriggerInterrupt(void)
{if (mspGpio_GetInput("PC13") == 0){Log_Debug("%s 按键执行测试线程0 恢复\r\n", __func__);BaseType_t xYieldRequired;xYieldRequired = xTaskResumeFromISR(p->taskList[0]);portYIELD_FROM_ISR(xYieldRequired);mspExti_Close(13);}
}
在中断中使用 FreeRTOS 接口需要带 FromISR 后缀的,如 xTaskResumeFromISR,需要portYIELD_FROM_ISR 切换上下文,否则实时性会收到一定的影响,为了调试和演示方便在中断中打印了数据,在实际项目中切记不要在中断中停留,特别是打印等高延时操作。
2)测试结果

如图所示,在中断中恢复已经挂起的线程也是可以的。
4.4 线程删除
1)线程删除后会释放内存,由于现在的线程都是在系统堆栈动态开辟的,所以线程删除后内存会回归系统内存堆栈。
/* 动态任务 */
static void TaskCtrl(void *pvParameters)
{Log_Debug("%s\r\n", __func__);/* 挂起线程 0 */vTaskDelay(3000);Log_Info("FreeRTOS Remain Space = %d Bytes\r\n", xPortGetFreeHeapSize());vTaskDelete(p->taskList[0]);Log_Info("Delete Task0 And FreeRTOS Remain Space = %d Bytes\r\n", xPortGetFreeHeapSize());vTaskDelete(p->taskList[1]);Log_Info("Delete Task1 And FreeRTOS Remain Space = %d Bytes\r\n", xPortGetFreeHeapSize());for ( ; ; ){vTaskDelay(1000);
// Log_Debug("%s\r\n", __func__);}
}
2)测试结果

如图所示,删除2个任务后,系统内存由 49472Bytes -> 51584Bytes -> 53696Bytes
相关文章:
FreeRTOS【4】线程挂起和恢复
1.开发背景 基于上一篇指引,成功创建并启动线程后,线程已经开始运行了,但是有时我们需要线程暂停运行,例如某个线程是控制 LED 闪灯的,如果现在需要让 LED 停止工作,单纯的关闭 LED 是没用的,因…...
CPU占用率过高排查
CPU占用率高是设备本身的一种现象,直观表现为display cpu-usage命令查询结果中整机CPU占用率“CPU usage”偏高,如超过70%。在网络运行中CPU高常常会导致其他业务异常,如BGP震荡、VRRP频繁切换、甚至设备无法登录。 通常,整机CPU占…...
关于 vs2019 c++20 规范里的 STL 库里模板 decay_t<T>
(1) 这个模板,在库代码里非常常见。 decay 英文是“衰弱,消减” 的意思,大概能感觉到就是要简化模板参数 T 的类型,去掉其上的修饰符。因为常用且复杂,故单独列出其源码和注释。先举例其应用场景…...
android C++打印堆栈
Android在Java层打印堆栈比较方便,代码如下: try {throw new Exception("Debug xxx call stack"); }catch(Exception e) {e.printStackTrace(); }但是在C模块中能打印调用堆栈吗?怎么打印调用栈呢? 答案是肯定的&…...
MySQL Undo Log、Redo Log、bin Log
Undo Log 回滚日志,用于将数据回滚到之前的状态。 MySQL在进行数据的增、删、改时,会将数据写入到Undo Log日志中。 对于Undo Log存在着insert和update两种类型的数据。插入语句对应的是insert类型,修改、删除语句对应的是update类型。 U…...
vld.ini配置文件说明
vld.ini配置文件说明 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; Visual Leak Detector - 初始化/配置文件 ;; 版权所有 (c) 2005-2017 VLD团队 ;; ;; 本库是自由软件;你可以在自由软件基金会发布的GNU宽通用公共…...
NSS【web】刷题
[SWPUCTF 2021 新生赛]jicao 类型:PHP、代码审计、RCE 主要知识点:json_decode()函数 json_decode():对JSON字符串解码,转换为php变量 用法: <?php $json {"ctf":"web","question"…...
将TailwindCSS默认单位rem转换为px
前言: 我这里需要将 默认的rem 转换为 px 原因是要使用 postcss-px-to-viewport 插件做移动端适配。 在tailwind.config.js文件中进行配置: 注意:这里 padding(内边距)、spacing(外边距)、width…...
命令模式(命令)
命令模式 文章目录 命令模式什么时命令模式通过示例了解命令模式 什么时命令模式 命令模式(Command),将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化:对请求排队或记录请求日志,以及支持可撤销的操作。 通过示例了解命令模…...
Android ashmem 原理分析
源码基于:Andoird U Kernel-5.10 0. 简介 ashmem 称为匿名共享内存(Anonymous Shared Memory),它以驱动程序的形式实现在内核空间中。它有两个特点: 能否辅助内存管理系统来有效地管理不再使用的内存块(pin / unpin); 通过Bind…...
redis报错500
之前自己举一反三把value也给序列化了: 然后报错了: 原因是这里传入的是Integer类型,序列化的话就变为string类型了...
GPT-3
论文:Language Models are Few-Shot Learners(巨无霸OpenAI GPT3 2020) 摘要 最近的工作表明,通过对大量文本进行预训练,然后对特定任务进行微调,在许多NLP任务和基准方面取得了实质性进展。虽然这种方法…...
MATLAB数组
文章目录 数组创建通过冒号创建一维数组通过logspace函数创建一维数组通过linspace函数创建一维数组 通过randperm生成随机整数排列运算算术运算关系运算逻辑运算优先顺序 矩阵创建矩阵操作下标引用矩阵信息提取删除与扩展合并矩阵元素的运算矩阵运算 数组 在MATLAB中一般使用…...
JAVA实验项目(二): 抽象类、接口的定义与使用
实验项目二 抽象类、接口的定义与使用 Tips:"分享是快乐的源泉💧,在我的博客里,不仅有知识的海洋🌊,还有满满的正能量加持💪,快来和我一起分享这份快乐吧😊&…...
JVM内存模型最新面试题(持续更新)
问题:java中创建的对象一般放在哪里?(全流程包含从创建到回收) 回答 大部分对象在堆中,这个基本都知道; 少部分对象是会在栈中的,比如作用域不局限于方法内的方法内部变量,这类对象的特征一般就是生命周期…...
Nginx wss to ws 折腾记
jssip 或 sipml5 <----wss--->nginx<---ws---->fs(5066) fs_cli -x sofia loglevel all 9 日志如下: REGISTER sip:192.168.43.135 SIP/2.0 Via: SIP/2.0/WSS df7jal23ls0d.invalid;branchz9hG4bKurFnCK9qJuXQlSrbszSL1S6wbCokKlLr;rport From: <…...
Java入门基础学习笔记22——程序流程控制
程序流程控制:控制程序的执行顺序。 程序有哪些执行顺序? 顺序、分支和循环。 分支结构: if、switch 循环: for、while、do-while 顺序结构是程序中最简单最基本的流程控制,没有特定的语法结构,按照代码…...
java医院信息系统HIS源码SaaS模式Java版云HIS系统 接口技术RESTful API + WebSocket + WebService
java医院信息系统HIS源码SaaS模式Java版云HIS系统 接口技术RESTful API WebSocket WebService 云HIS是基于云计算的医疗卫生信息系统(Cloud-Based Healthcare Information System),它运用云计算、大数据、物联网等新兴信息技术,…...
2024年成都高新区支持企业申报国家、省级、市级大数据产业发展、新一代信息技术与制造业融合发展、工业互联网推广应用等试点示范项目申报对象条件和奖补
一、申报对象 (一)本政策支持注册地址、税收关系在成都高新区,具有独立法人资格的企业。 (二)管理规范,无不良信用记录,自觉遵守安全生产、环境保护等方面的法律法规,近三年未发生…...
让《行列视》解放数据力量,提升业务洞察
在当今信息化浪潮下,数据已经成为企业发展的核心驱动力之一。如何更好地管理和利用数据,已成为企业发展过程中亟需解决的问题之一。而报表工具作为数据可视化和分析的利器,正逐渐受到企业的重视和青睐。 一、《行列视》作为报表工具的重要性…...
AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...
Unity3D中Gfx.WaitForPresent优化方案
前言 在Unity中,Gfx.WaitForPresent占用CPU过高通常表示主线程在等待GPU完成渲染(即CPU被阻塞),这表明存在GPU瓶颈或垂直同步/帧率设置问题。以下是系统的优化方案: 对惹,这里有一个游戏开发交流小组&…...
(二)TensorRT-LLM | 模型导出(v0.20.0rc3)
0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述,后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作,其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...
理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端
🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...
Leetcode 3577. Count the Number of Computer Unlocking Permutations
Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接:3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯,要想要能够将所有的电脑解锁&#x…...
Linux简单的操作
ls ls 查看当前目录 ll 查看详细内容 ls -a 查看所有的内容 ls --help 查看方法文档 pwd pwd 查看当前路径 cd cd 转路径 cd .. 转上一级路径 cd 名 转换路径 …...
在 Nginx Stream 层“改写”MQTT ngx_stream_mqtt_filter_module
1、为什么要修改 CONNECT 报文? 多租户隔离:自动为接入设备追加租户前缀,后端按 ClientID 拆分队列。零代码鉴权:将入站用户名替换为 OAuth Access-Token,后端 Broker 统一校验。灰度发布:根据 IP/地理位写…...
Keil 中设置 STM32 Flash 和 RAM 地址详解
文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...
Spring AI 入门:Java 开发者的生成式 AI 实践之路
一、Spring AI 简介 在人工智能技术快速迭代的今天,Spring AI 作为 Spring 生态系统的新生力量,正在成为 Java 开发者拥抱生成式 AI 的最佳选择。该框架通过模块化设计实现了与主流 AI 服务(如 OpenAI、Anthropic)的无缝对接&…...
微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...
