FreeRTOS Semaphore信号量-笔记

FreeRTOS Semaphore信号量-笔记

      • **一、信号量与互斥量的核心区别**
      • **二、二值信号量(Binary Semaphore)**
        • **1. 功能与使用场景**
        • **2. 示例:ADC中断与任务同步**
      • **三、计数信号量(Counting Semaphore)**
        • **1. 功能与使用场景**
        • **2. 示例:ADC双缓冲区管理**
      • **四、互斥量(Mutex)**
        • **1. 功能与使用场景**
        • **2. 示例:保护共享资源**
      • **五、关键注意事项**
      • **六、总结**

在这里插入图片描述


队列的功能是将进程间需要传递的数据存在其中。所以在有的RTOS系统里,队列也被称为邮箱。

一、信号量与互斥量的核心区别

特性信号量(Semaphore)互斥量(Mutex)
用途进程间同步(如事件通知)或资源计数(如ADC双缓冲区)互斥访问共享资源(如保护全局变量或外设)
所有权无所有权(任何任务或ISR均可释放)有所有权(仅持有者可释放)
优先级继承机制(可能导致优先级翻转)(缓解优先级翻转)
是否可中断使用(通过 xSemaphoreGiveFromISR不可(ISR中无法使用)
初始值二值信号量(0或1)或计数信号量(任意值)固定为1(表示资源可用)

信号量和互斥量。信号量和互斥量的实现都是基于队列的,信号量更适用于进程间同步,而互斥量更适用于共享资源的互斥性访问。
在这里插入图片描述


二、二值信号量(Binary Semaphore)

在这里插入图片描述
如果不使用二值信号量,而是使用一个自定义标志变量来实现以上的同步过程,则任务需要不断的查询标志变量的值,而不是像使用二值信号那样可以使任务进入阻塞动态状态。所以使用二值信号量进行进程间同步的效率更高。

1. 功能与使用场景
  • 功能:作为“标志”实现任务同步,例如:
    • ADC中断通知任务:当ADC中断写入缓冲区后,释放二值信号量通知任务处理数据。
    • 任务等待事件:任务阻塞等待信号量,避免忙等待。
  • 创建函数
  // 动态分配xSemaphoreCreateBinary() xQueueGenericCreate( ( UBaseType_t ) 1, semSEMAPHORE_QUEUE_ITEM_LENGTH, queueQUEUE_TYPE_BINARY_SEMAPHORE );//这是一个宏函数。调用的函数类似于队列(xQueueGenericCreate)创建队列时,调用的也是这个函数。

参数解析

  • ( UBaseType_t ) 1

    • 队列长度:表示队列最多能容纳的“消息”数量。
    • 二进制信号量特性:二进制信号量只能处于“空”(0)或“满”(1)状态,因此队列长度设为 1。
  • semSEMAPHORE_QUEUE_ITEM_LENGTH

    • 队列项大小:定义队列中每个消息的字节长度。
    • 二进制信号量特殊性:该宏被定义为 0(#define semSEMAPHORE_QUEUE_ITEM_LENGTH ( ( uint8_t ) 0U )),表示信号量不存储实际数据,仅用队列的空/满状态表示信号量状态。
  • queueQUEUE_TYPE_BINARY_SEMAPHORE

    • 队列类型:指定此队列用于二进制信号量,而非普通队列或其他类型(如互斥锁或计数信号量)。

  // 静态分配SemaphoreHandle_t xSemaphoreCreateBinaryStatic(StaticSemaphore_t *pxSemaphoreBuffer);
  • 操作函数
    • 释放信号量(任务):
    xSemaphoreGive( xSemaphore )    xQueueGenericSend( ( QueueHandle_t ) ( xSemaphore ), NULL, semGIVE_BLOCK_TIME, queueSEND_TO_BACK )

参数解析

  • (QueueHandle_t)(xSemaphore)

    • 信号量即队列:FreeRTOS 的信号量句柄 xSemaphore 实际上是队列句柄(QueueHandle_t)的别名。信号量通过队列实现,其行为由队列的配置参数控制(如长度、项大小等)。
  • NULL

    • 无实际数据:信号量本身不传递数据,仅用队列的空/满状态表示信号量状态。因此,发送的“消息”无需有效数据,使用 NULL 即可。
  • semGIVE_BLOCK_TIME

    • 阻塞时间:定义为 0(#define semGIVE_BLOCK_TIME (0)),表示调用 xSemaphoreGive 时不等待。如果队列已满(信号量已处于“可用”状态),则直接返回错误 errQUEUE_FULL。
  • queueSEND_TO_BACK

    • 入队位置:将“消息”添加到队列的尾部。信号量的顺序无关紧要,因此使用尾部入队是合理的。
  • 返回类型为BaseType_t,pdFALSE或者pdTRUE。


  • 释放信号量(ISR):
xSemaphoreGiveFromISR( xSemaphore, pxHigherPriorityTaskWoken )    xQueueGiveFromISR( ( QueueHandle_t ) ( xSemaphore ), ( pxHigherPriorityTaskWoken ) )

参数解析

  • xSemaphore
    • 信号量句柄:实际是一个队列句柄(QueueHandle_t)的别名。信号量通过队列实现,其行为由队列的配置参数控制(如长度、项大小等)。
  • pxHigherPriorityTaskWoken
    • 优先级唤醒标志:
    • 类型:BaseType_t *(指向布尔值的指针)。
    • 功能:如果释放信号量导致更高优先级的任务被唤醒,该指针会被设置为 pdTRUE,表示需要在退出中断前请求上下文切换。
    • 可选参数:从 FreeRTOS V7.3.0 开始,此参数可以设为 NULL。
    • 注意:如果释放信号量导致了一个任务解锁,而解锁的任务比当前任务的优先级高,这里就会返回pdTRUE。这就需要在退出ISR之前申请任务调度,以便及时的解锁高优先级的任务。

在这里插入图片描述


  • 获取信号量(任务):
xSemaphoreTake( xSemaphore, xBlockTime )    xQueueSemaphoreTake( ( xSemaphore ), ( xBlockTime ) )

参数解析

  • xSemaphore
  • 信号量句柄:实际是一个队列句柄(QueueHandle_t)的别名。信号量通过队列实现,其行为由队列的配置参数控制(如长度、项大小等)。
  • 类型:SemaphoreHandle_t(底层为 QueueHandle_t)。
    ● xBlockTime
  • 阻塞时间:以 Tick 为单位指定任务等待信号量的最长时间。
  • portMAX_DELAY:无限期等待,直到信号量可用。
  • 0:非阻塞模式,立即返回。
  • 转换:使用 pdMS_TO_TICKS(ms) 将毫秒转换为 Tick(例如 pdMS_TO_TICKS(100))。

2. 示例:ADC中断与任务同步
// 创建二值信号量
SemaphoreHandle_t xADCSemaphore = xSemaphoreCreateBinary();// ADC中断服务程序
void ADC_IRQHandler(void) {// 写入数据到缓冲区WriteDataToBuffer();// 释放信号量通知任务xSemaphoreGiveFromISR(xADCSemaphore, NULL);
}// 数据处理任务
void vDataProcessingTask(void *pvParameters) {while (1) {// 等待信号量xSemaphoreTake(xADCSemaphore, portMAX_DELAY);// 处理缓冲区数据ProcessData();}
}

三、计数信号量(Counting Semaphore)

在这里插入图片描述
资源类比成一个餐馆中的四个餐桌。
管理多个共享资源。例如ADC连续数据采集时,一般使用双缓冲区,就可以使用计数信号量来进行管理。

1. 功能与使用场景
  • 功能:管理多个同类型资源(如ADC双缓冲区),允许同时访问多个资源。
  • 创建函数
    // 动态分配
    SemaphoreHandle_t xSemaphoreCreateCounting(UBaseType_t uxMaxCount, UBaseType_t uxInitialCount);
    // 静态分配
    SemaphoreHandle_t xSemaphoreCreateCountingStatic(UBaseType_t uxMaxCount, UBaseType_t uxInitialCount, StaticSemaphore_t *pxSemaphoreBuffer);
    
  • 操作函数
    • 释放资源
      xSemaphoreGive(xSemaphore);  // 资源计数+1
      
    • 获取资源
      xSemaphoreTake(xSemaphore, xTicksToWait);  // 资源计数-1
      
2. 示例:ADC双缓冲区管理
// 创建计数信号量(2个缓冲区)
SemaphoreHandle_t xADCBufferSemaphore = xSemaphoreCreateCounting(2, 2);// ADC中断服务程序
void ADC_IRQHandler(void) {// 获取一个缓冲区if (xSemaphoreTakeFromISR(xADCBufferSemaphore, NULL) == pdTRUE) {// 写入数据到缓冲区WriteDataToBuffer();// 释放信号量(资源可用)xSemaphoreGiveFromISR(xADCBufferSemaphore, NULL);}
}// 数据处理任务
void vDataProcessingTask(void *pvParameters) {while (1) {// 获取缓冲区资源xSemaphoreTake(xADCBufferSemaphore, portMAX_DELAY);// 处理缓冲区数据ProcessData();// 释放缓冲区资源xSemaphoreGive(xADCBufferSemaphore);}
}

四、互斥量(Mutex)

在这里插入图片描述
在这里插入图片描述

二值信号量更适用于进程间同步,而互斥量更适用于控制对互斥型资源的访问。二值信号量没有优先级继承机制,将二值信号量用于互斥型资源访问时,容易出现优先级翻转问题。而互斥量有优先级继承机制,可以减缓优先级翻转问题。

1. 功能与使用场景
  • 功能:保护共享资源的独占访问(如串口),避免数据竞争。
  • 创建函数
    // 动态分配
    SemaphoreHandle_t xSemaphoreCreateMutex(void);
    // 静态分配
    SemaphoreHandle_t xSemaphoreCreateMutexStatic(StaticSemaphore_t *pxSemaphoreBuffer);
    
  • 操作函数
    • 释放互斥量(可以释放二值信号量、计数信号量或者是互斥量。)
      xSemaphoreGive(xSemaphore);
      
    • 获取互斥量
      xSemaphoreTake(xSemaphore, xTicksToWait);
      
    • 递归互斥量(允许任务多次获取同一互斥量):
      // 创建
      SemaphoreHandle_t xSemaphoreCreateRecursiveMutex(void);
      // 释放
      xSemaphoreGiveRecursive(xSemaphore);
      // 获取
      xSemaphoreTakeRecursive(xSemaphore, xTicksToWait);
      
2. 示例:保护共享资源
// 创建互斥量
SemaphoreHandle_t xSharedResourceMutex = xSemaphoreCreateMutex();// 任务A
void vTaskA(void *pvParameters) {while (1) {xSemaphoreTake(xSharedResourceMutex, portMAX_DELAY);// 访问共享资源SharedResource++;xSemaphoreGive(xSharedResourceMutex);}
}// 任务B
void vTaskB(void *pvParameters) {while (1) {xSemaphoreTake(xSharedResourceMutex, portMAX_DELAY);// 访问共享资源SharedResource--;xSemaphoreGive(xSharedResourceMutex);}
}

五、关键注意事项

  1. 优先级翻转问题

    • 二值信号量:无优先级继承机制,可能导致低优先级任务阻塞高优先级任务。
    • 互斥量:通过优先级继承机制缓解问题,但仍需谨慎设计任务优先级。
  2. ISR中的使用限制

    • 信号量:可在ISR中使用(xSemaphoreGiveFromISR)。
    • 互斥量禁止在ISR中使用(因其依赖优先级继承)。
  3. 内存分配选择

    • 动态分配:适合资源充足的系统,但可能产生内存碎片。
    • 静态分配:适合资源受限的嵌入式系统,需手动管理内存。
  4. 调试工具

    • uxSemaphoreGetCount():获取信号量当前值(适用于计数信号量)。
    • xSemaphoreGetMutexHolder():检查互斥量当前持有者。

六、总结

其他相关函数
● xSemaphoreGiveRecursive(用于递归互斥锁的释放)。
●xSemaphoreTakeRecursive()获取二值信号量、计数信号量或者是互斥量。释放递归互斥量依然有一个专用的函数。
●uxSemaphoreGetCount()返回传进去的这个信号量的当前值。
●xSemaphoreGetMutexHolder()作用:返回当前持有指定互斥锁(mutex)的任务的句柄(TaskHandle)。如果互斥锁未被任何任务持有,则返回 NULL。
xSemaphoreTakeFromISR()

  • 选择信号量还是互斥量
    • 同步需求(如事件通知)→ 信号量
    • 资源互斥访问(如保护共享变量)→ 互斥量
  • 避免常见错误
    • 不要在ISR中释放互斥量。
    • 确保每次获取互斥量后最终释放。
    • 使用递归互斥量时,获取与释放需严格配对。

通过合理使用信号量和互斥量,可以高效实现FreeRTOS中的任务同步与资源共享,提升系统的实时性和稳定性。

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

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

相关文章

音频类网站或者资讯总结

我爱音频网: 我爱音频网 - 我们只谈音频,丰富的TWS真无线蓝牙耳机拆解报告 (52audio.com) 其他更多资讯 音频行业全品类深度剖析,2024市场趋势解读汇总-EDN 电子技术设计 (ednchina.com)

16.Excel:数据收集

一 使用在线协作工具 简道云。 excel的在线表格协作在国内无法使用,而数据采集最需要在线协作。 二 使用 excel 1.制作表格 在使用excel进行数据采集的时候,会制作表头给填写人,最好还制作一个示例。 1.输入提示 当点击某个单元格的时候&am…

JAVA虚拟机(JVM)总结,很清晰,很好理解!!

目录 java编译相关知识 Java文件编译过程 java的可跨平台性 JVM内存结构 运行期数据区域(JDK8之后) 本地方法栈 虚拟方法栈 程序计数器 堆 本地内存 栈帧里面的局部变量表和方法区(元空间的区别) 类加载器 启动类加载…

前端项目中单元测试与集成测试的管理实践

前端项目中单元测试与集成测试的管理实践 在现代前端工程化中,单元测试(Unit Test)和集成测试(Integration Test)已成为保障项目质量的重要手段。合理地组织和管理测试代码,不仅有助于持续集成&#xff0c…

【Redis】缓存和分布式锁

🔥个人主页: 中草药 🔥专栏:【中间件】企业级中间件剖析 一、缓存(Cache) 概述 Redis最主要的应用场景便是作为缓存。缓存(Cache)是一种用于存储数据副本的技术或组件,…

深入解析路由策略:从流量控制到策略实施

一、网络流量双平面解析 在路由策略的设计中,必须明确区分两个关键平面: 1. 控制层面(Control Plane) ​​定义​​:路由协议传递路由信息形成的逻辑平面(如OSPF的LSA、RIP的Response报文)​…

从杰夫・托尔纳看 BPLG 公司的技术创新与发展

在科技与商业紧密交织的时代,企业的技术领导者在推动组织前行、应对复杂多变的市场环境中扮演着极为关键的角色。《对话 CTO,驾驭高科技浪潮》的第 6 章聚焦于杰夫・托尔纳及其所在的 BPLG 公司,为我们展现了一幅技术驱动企业发展的生动图景&…

UniRepLknet助力YOLOv8:高效特征提取与目标检测性能优化

文章目录 一、引言二、UniRepLknet 的框架原理(一)架构概述(二)架构优势 三、UniRepLknet 在 YOLOv8 中的集成(一)集成方法(二)代码实例 四、实验与对比(一)对…

比较Facebook与其他社交平台的隐私保护策略

在这个数字化的时代,隐私保护已成为用户和社交平台共同关注的核心议题。Facebook,作为全球最大的社交网络平台之一,其隐私保护策略一直受到广泛的关注和讨论。本文将对Facebook的隐私保护策略与其他社交平台进行比较,以帮助用户更…

数据结构--树

一、树的概念 树是由n(n≥0)个节点组成的有限集合,它满足以下条件: 1. 当n0时,称为空树 2. 当n>0时,有且仅有一个特定的节点称为根节点(root) 3. 其余节点可分为m(m≥0)个互不相交的有限集合,每个集合本身又是一…

Linux `ifconfig` 指令深度解析与替代方案指南

Linux `ifconfig` 指令深度解析与替代方案指南 一、核心功能与现状1. 基础作用2. 版本适配二、基础语法与常用操作1. 标准语法2. 常用操作速查显示所有接口信息启用/禁用接口配置IPv4地址修改MAC地址(临时)三、高级配置技巧1. 虚拟接口创建2. MTU调整3. 多播配置4. ARP控制四…

什么是分布式光伏系统?屋顶分布式光伏如何并网?

政策窗口倒计时!分布式光伏如何破局而立? 2025年,中国分布式光伏行业迎来关键转折: ▸ "430"落幕——抢装潮收官,但考验才刚开始; ▸ "531"生死线——新增项目全面市场化交易启动&…

Cluster Interconnect in Oracle RAC

Cluster Interconnect in Oracle RAC (文档 ID 787420.1)​编辑转到底部 In this Document Purpose Scope Details Physical Layout of the Private Interconnect Why Do We Need a Private Interconnect ? Interconnect Failure Interconnect High Availability Private Inte…

.Net HttpClient 使用准则

HttpClient 使用准则 System.Net.Http.HttpClient 类用于发送 HTTP 请求以及从 URI 所标识的资源接收 HTTP 响应。 HttpClient 实例是应用于该实例执行的所有请求的设置集合,每个实例使用自身的连接池,该池将其请求与其他请求隔离开来。 从 .NET Core …

【PostgreSQL】数据库主从库备份与高可用部署

文章目录 一、架构设计原理二、部署清单示例2.1 StatefulSet配置片段2.2 Service配置三、配置详解3.1 主节点postgresql.conf3.2 从节点配置四、初始化流程4.1 创建复制用户4.2 配置pg_hba.conf五、故障转移示例5.1 自动切换脚本5.2 手动提升从节点六、监控与维护6.1 关键监控指…

JavaScript 数组去重:11 种方法对比与实战指南

文章目录 前言一、使用 Set 数据结构二、使用 filter indexOf三、使用 reduce 累加器四、双重 for 循环五、利用对象属性唯一性六、先排序后去重七、使用 Map 数据结构八、使用 includes 方法九、优化处理 NaN 的 filter 方法十、利用 findIndex十一.利用Set和展开运算符处理多…

ai agent(智能体)开发 python3基础14:在python 中 总能看到方法里面套方法,那什么时候用这种方式合适呢?

让人头疼的方法嵌套还是要去了解的 在 Python 中,方法内部嵌套方法(即在类的方法中定义另一个函数)是一种常见的代码组织技巧,它可以在特定场景下带来以下好处: 1. 代码复用与逻辑封装 如果某个方法内部有重复的逻辑…

Yocto项目实战经验总结:从入门到高级的全面概览

本文面向开发者和实际项目经验者,分享经过大量实战积累的 Yocto 项目工程经验和基础技巧。本文简明但精彩,应用和观察相结合,充分适合做为全面进阶 Yocto 项目开发的实用指南。 一、入门理解:Yocto 是什么?规划如何开始…

添加物体.

在cesium中我们可以添加物体进入地图.我们以广州塔为例 //生成广州塔的位置var position2 Cesium.Cartesian3.fromDegrees(113.3191,23.109,100)viewer.camera.setView({//指定相机位置destination: position2, 运行后如图 我们使用cesium官网提供的代码为广州塔在地图上标点…

正则表达式非捕获分组?:

一个使用 Java 正则表达式的具体例子,展示了 (ab) 和 (?:ab) 的不同: 示例 1:使用 (ab)(捕获分组) import java.util.regex.*; public class RegexExample { public static void main(String[] args) { …