详解FreeRTOS中的信号量(semaphore)

news2025/1/17 21:34:29

信号,顾名思义最基础的作用是通知,量,表示数量,意思就是可以有多个信号。在不同的场景延伸下,还有同步和互斥访问资源的作用(这都是通知作用的延伸)。

当"量"没有限制时,它就是"计数型信号量"(Counting Semaphores)

当"量"只有0、1两个取值时,它就是"二进制信号量"(Binary Semaphores)

参考资料:

FreeRTOS全解析-8.信号量(semaphore)

目录

1.二进制信号量

1.1创建二进制信号量

1.2 give/take

1.3删除

1.4例子

2.计数信号量


1.二进制信号量

二进制信号量和我们裸机编程时经常设置的flag很像。比如某个数据好了,或者某个行为做了,我们就设置flag为1,类似的在FreeRTOS中就可以give给出一个信号量。flag触发了些操作后往往会被重置为0,类似的take信号量后信号量会减少为0。

take和give是信号量的术语。意思就是“获取信号量”和“给出信号量”。

为了彻底理解它,先来看一下他的同步作用的例子,上篇文章讲到推迟中断处理。

FreeRTOS全解析-7.中断安全API和推迟中断处理

可以将延迟处理任务的优先级设置最高,然后,在ISR中调用portYIELD_FROM_ISR(),ISR结束后会运行延迟处理的任务(因为它优先级最高)。这样可以确保整个事件处理及时连续地执行,就像所有事件处理都是在ISR本身中实现的一样。也可以使用二进制信号量实现这个效果。

Task2就是中断推迟处理任务。

t1时Task1运行,Task2阻塞等待信号量,t2时发生中断,ISR执行,give一个信号量解除Task2的阻塞态,t3时Task2运行,结束后继续等待下一个信号量。

在这种情况下,二进制信号量可以被认为是一个长度为1的队列。

队列在任何时候最多可以包含一个项,所以总是空的或满的(因此是二进制).

中断延迟处理任务通过调用xSemaphoreTake(),尝试从队列中读取数据,如果队列为空,则导致任务进入阻塞状态。当事件发生时,ISR使用xSemaphoreGiveFromISR()函数将一个信号量放入队列,使队列满。这将导致中断延迟处理任务退出Blocked状态并删除信号量,使队列再次为空。当任务完成处理后,它再次尝试从队列中读取,发现队列为空,重新进入Blocked状态以等待下一个事件。

1.1创建二进制信号量

在使用信号量之前,必须先创建信号量。

SemaphoreHandle_t xSemaphoreCreateBinary( void );

如果返回NULL,则不能创建信号量,因为没有足够的堆内存供FreeRTOS分配信号量数据结构。

返回非NULL值表示信号量已成功创建。返回值为创建的信号量的句柄。

也可以静态创建

SemaphoreHandle_t xSemaphoreCreateBinaryStatic( StaticSemaphore_t *pxSemaphoreBuffer );

1.2 give/take

二进制信号量、计数型信号量的give、take操作函数是一样的。这些函数也分为2个版本:给任务使用,给ISR使用。列表如下:

在任务中使用在ISR中使用
givexSemaphoreGivexSemaphoreGiveFromISR
takexSemaphoreTakexSemaphoreTakeFromISR
BaseType_t xSemaphoreGive( SemaphoreHandle_t xSemaphore );BaseType_t xSemaphoreGiveFromISR(                   SemaphoreHandle_t xSemaphore,                   BaseType_t *pxHigherPriorityTaskWoken);BaseType_t xSemaphoreTake(                   SemaphoreHandle_t xSemaphore,                   TickType_t xTicksToWait);BaseType_t xSemaphoreTakeFromISR(                   SemaphoreHandle_t xSemaphore,                   BaseType_t *pxHigherPriorityTaskWoken  );

xSemaphore就是要获取的信号量句柄。

xTicksToWait决定了,假如没有信号量时,任务进入阻塞态以等待有信号量的等待时间。如果xTicksToWait为零,那么如果信号量不可用,xSemaphoreTake()将立即返回。阻塞时间以滴答周期指定,宏pdMS_TO_TICKS()可用于将以毫秒为单位指定的时间转换为以tick为单位指定的时间。如果将xTicksToWait设置为portMAX_DELAY(前提是FreeRTOSConfig.h中INCLUDE_vTaskSuspend设置为1)将导致任务无限期地等待(没有超时)。

pxHigherPriorityTaskWoken在上一篇解释过了,用于判断是否有更高优先级的任务等待唤醒。

1.3删除

对于动态创建的信号量,不再需要它们时,可以删除它们以回收内存。

vSemaphoreDelete可以用来删除二进制信号量、计数型信号量,函数原型如下:

void vSemaphoreDelete( SemaphoreHandle_t xSemaphore );

1.4例子

如下代码所示:

vPeriodicTask周期性产生中断,产生中断前打印Periodic task - About to generate an interrupt,中断处理后打印Periodic task - Interrupt generated其ISR为ulExampleInterruptHandlergive信号量,该中断推迟处理任务为vHandlerTasktake信号量,并打印Handler task - Processing event​​​​​​​

#define mainINTERRUPT_NUMBER 3/*周期产生中断*/static void vPeriodicTask( void *pvParameters ){  const TickType_t xDelay500ms = pdMS_TO_TICKS( 500UL );  for( ;; ) {  /* 阻塞500ms. */  vTaskDelay( xDelay500ms );  vPrintString( "Periodic task - About to generate an interrupt.\r\n" );  /*该函数用于产生一个中断*/  vPortGenerateSimulatedInterrupt( mainINTERRUPT_NUMBER );  vPrintString( "Periodic task - Interrupt generated.\r\n\r\n\r\n" );  }} /*推迟中断处理任务*/static void vHandlerTask( void *pvParameters ){  for( ;; ) {    /*获取信号量*/    xSemaphoreTake( xBinarySemaphore, portMAX_DELAY );    /*用打印模拟处理*/    vPrintString( "Handler task - Processing event.\r\n" );  }}/*中断处理例程ISR*/static uint32_t ulExampleInterruptHandler( void ){  BaseType_t xHigherPriorityTaskWoken;  xHigherPriorityTaskWoken = pdFALSE;  /* 给出信号 */  xSemaphoreGiveFromISR( xBinarySemaphore, &xHigherPriorityTaskWoken );  portYIELD_FROM_ISR( xHigherPriorityTaskWoken );}int main( void ){  xBinarySemaphore = xSemaphoreCreateBinary();  if( xBinarySemaphore != NULL ) {    xTaskCreate( vHandlerTask, "Handler", 1000, NULL, 3, NULL );    xTaskCreate( vPeriodicTask, "Periodic", 1000, NULL, 1, NULL );     /*配置软中断*/    vPortSetInterruptHandler( mainINTERRUPT_NUMBER, ulExampleInterruptHandler );    vTaskStartScheduler();  }  for( ;; );}

时序图如下:

2.计数信号量

就像二进制信号量可以被认为是长度为1的队列一样,计数信号量可以被认为是长度大于1的队列。任务可以往这个队列放数据(give),也可以取走数据(take),但是只对数据的数目感兴趣,而无关是什么数据。

要使用计数信号量首先要将FreeRTOSConfig.h中的configUSE_COUNTING_SEMAPHORES必须设置为1。

计数信号量通常用于两种情况:

1.事件计数

在这种情况下,事件处理程序将在每次事件发生时“give”一个信号量——导致信号量的计数值在每次“give”时增加。任务每次处理一个事件时都会“take”一个信号量,导致信号量的计数值在每次“take”时递减。计数值是已发生的事件数与已处理的事件数之间的差值。用于对事件进行计数的计数信号量的初始计数值为0。

2. 资源管理。

在这种情况下,计数值表示可用资源的数量。要获得对资源的控制,任务必须首先“take”一个信号量——减去信号量的计数值。当计数值为0时,表示没有空闲资源。当一个任务完成对资源的处理后,它将信号量的计数值向后递增“give”信号量。

创建计数信号量:

/* 创建一个计数型信号量,返回它的句柄。 * 此函数内部会分配信号量结构体  * uxMaxCount: 最大计数值 * uxInitialCount: 初始计数值 * 返回值: 返回句柄,非NULL表示成功 */SemaphoreHandle_t xSemaphoreCreateCounting(UBaseType_t uxMaxCount, UBaseType_t uxInitialCount);
/* 创建一个计数型信号量,返回它的句柄。 * 此函数无需动态分配内存,所以需要先有一个StaticSemaphore_t结构体,并传入它的指针 * uxMaxCount: 最大计数值 * uxInitialCount: 初始计数值 * pxSemaphoreBuffer: StaticSemaphore_t结构体指针 * 返回值: 返回句柄,非NULL表示成功 */SemaphoreHandle_t xSemaphoreCreateCountingStatic( UBaseType_t uxMaxCount,                                                  UBaseType_t uxInitialCount,                                                  StaticSemaphore_t *pxSemaphoreBuffer );

其余函数都和二进制信号量一样。

更多的关于信号量的使用的例子将在今后的文章中提到。

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

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

相关文章

SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式,系统详解springcloud微服务技术栈(Eureka、Ribbon)

微服务技术栈导学 微服务技术是分布式架构(把服务做拆分)的一种 而springcloud仅仅是解决了拆分时的微服务治理的问题,其他更复杂的问题并没有给出解决方案 一个完整的微服务技术要包含的不仅仅是springcloud 微服务技术栈 包括什么 …

什么是远程办公,如何挑选远程办公软件

远程办公已经不再是新型工作模式,随着科技的发展和全球化的趋势,越来越多的企业和个人已经开始接受这种新型的工作模式。远程办公可以让员工在家里或者任何地方工作,不用去公司办公室,大大提高了员工的工作效率和生活质量。本文将…

hdfs命令行操作

文章目录1. 对文件夹进行操作1.1 ls:对路径进行访问1.2 mkdir:对路径进行创建1.3 rm:对路径进行删除2.对文件进行操作2.1在文件系统中创建空文件2.2上传本地文件到hdfs上2.3 从hdfs上下载文件到本地路径2.4 查看hdfs 上的文件内容2.5 对hdfs上的文件进行复制2.6 追加本地文件内…

微服务·入门·贰——注册中心nacos、eureka

文章目录1. 微服务问题解决服务治理问题1.1 问题抛出1.2 解决方法2 Eureka注册中心2.1 Eureka解决的问题2.2 Eureka的结构和作用2.2.1 Eureka的作用2.2.2 order-service如何得知user-service实例地址 ?2.2.3 order-service如何从多个user-service实例中选择具体实例…

家政服务小程序实战开发教程019-我的预约功能(已完结)

我们上一篇讲解了用户注册的功能,注册完毕后页面需要显示用户的头像和昵称,并显示我的预约的菜单,本篇我们介绍一下如何开发。 1 显示用户头像和昵称 在未注册时我们显示了一个默认的头像,已注册需要显示用户的头像。思路是将未…

【零基础入门 Nginx】——万字文章通俗易懂

一、Nginx 简介 1️⃣ Nginx 概述 Nginx(Engine X) 是一个高性能的HTTP和反向代理服务器,特点是占有内存少,并发能力强。同时也提供了IMAP/POP3/SMTP服务 nginx可以作为静态页面的web服务器,同时还支持CGI协议的动态…

WebRTC 系列(二、本地通话,H5、Android、iOS)

WebRTC 系列(一、简介)​​​​​​​ 一、整体流程 有了上一篇 WebRTC 简介的基础,我们知道了 WebRTC 的工作流程,接下来就是需要用代码去实现这个流程了。对于不同端,实现起来的难易程度可能略微不同(实…

RHCE第二次作业ssh远程连接和NTP时间服务器

1.配置ntp时间服务器,确保客户端主机能和服务主机同步时间 在服务器准备工作查看服务是否开启,查看是否运行 同步时间,编辑/etc/chrony.conf,层级优先级10,在允许客户机。 暂时关闭防火墙,关闭服务后,重启…

vue3-element-plus表单校验和多选表格table的基本使用

表单校验 <script setup> import { ref } from "vue"; // 登录的表单数据(绑定到最外层的from标签上) //里面的每个属性都与element-plus的表单标签进行双向绑定,具体可以看html代码 const loginForm ref({username: "",password: "",lo…

双向可控硅详细用法说明

可控硅作为功率开关器件&#xff0c;在各种需要控制功率的电子产品中经常用到&#xff0c;我所涉及的行业为家电产品研发&#xff0c;比如发热丝、发热管的控温&#xff0c;或者AC电机、水泵的控速等&#xff1b;由于双向可控硅是在单向可控硅的基础上发展而来且应用场景更广&a…

ucgui的触摸执行过程

在STM32上调试ucosucguI的触摸时&#xff0c;显示上下左右中5个button&#xff0c;但是按上button时触发的却是右button&#xff0c;调试发现显示区域大小正常&#xff0c;触摸区域大小正常。但就是触摸区域无法与实际的button相对应。 分析原因可能是xy轴不匹配&#xff0c;那…

手撕深度学习中的优化器

深度学习中的优化算法采用的原理是梯度下降法&#xff0c;选取适当的初值params&#xff0c;不断迭代&#xff0c;进行目标函数的极小化&#xff0c;直到收敛。由于负梯度方向时使函数值下降最快的方向&#xff0c;在迭代的每一步&#xff0c;以负梯度方向更新params的值&#…

web前端笔记

HTML(安装live server插件) 我们上网时所看到的所有内容其实就是body里面的内容&#xff01; &#xff01; tab 快速生成一个html模板&#xff1b; https://www.runoob.com/tags/html-elementsdoctypes.html html的菜鸟教程&#xff01; html中的元素分为两种&#xff0c;一种…

广州蓝景分享—13个Web开发人员都知道的基本JavaScript函数

各位编程爱好者&#xff0c;今天由小蓝与大家分享13个基本的JavaScript 函数&#xff0c;如果您是 Web前端开发人员&#xff0c;您应该熟悉这些函数。 您可以将本文所有 JavaScript 函数加入收藏至您的工具箱&#xff0c;以便在您的软件项目中尽可能使用这些片段。 1. 检索任…

恐怖的低代码平台,我 All in 了!

本文目录一、低代码平台是什么&#xff1f;二、目前低代码产品平台是如何分类的&#xff1f;三、低代码平台是怎么互相比较的&#xff1f;一个比喻就明白了&#xff01;四、iVX平台的恐怖优势&#xff01;我 All in 了&#xff01;五、iVX的学习成本&#xff1f;总结&#xff1…

百度CTO王海峰做客《中国经济大讲堂》:文心一言,读书破万亿

当下&#xff0c;大语言模型热度空前&#xff0c;诸如文心一言、ChatGPT 等已经能够与人对话互动、回答问题、协助创作&#xff0c;逐渐应用于人们的工作和生活&#xff0c;也引发了社会热议。近日&#xff0c;百度首席技术官、深度学习技术及应用国家工程研究中心主任王海峰再…

asp.net Core 6 从空建立一个MVC项目,Razor组件使用

Razor组件使用MVC项目创建创建空的Web项目添加MVC框架Razor组件使用准备封装Razor组件MVC项目创建 创建一个空的项目&#xff0c;然后添加MVC。 创建空的Web项目 添加MVC框架 1.添加文件夹 2.添加控制器 3.添加界面 4.修改program.cs文件内容 //原生的 //var builder …

python入门:cl.exe‘ failed with exit status 2错误通用解决方案

文章目录 错误一错误二pypi.org独立安装正确安装错误一 error: Microsoft Visual C++ 14.0 or greater is required. Get it with "Microsoft C++ Build Tools": https://visualstudio.microsoft.com/visual-cpp-build-tools/ 这个错误在windows系统上安装python工…

用64位的plsql developer 连接虚拟机中的64位oracle数据库

背景&#xff1a;为了学习oracle&#xff0c;我在虚拟机上安装了oracle。并在实体机上安装了oracle客户端及plsql developer。 开始之前&#xff0c;先回答两个问题 为什么不在本机安装oracle? 因为oracle比较消耗资源&#xff0c;而我不会一直用&#xff0c;所以放到虚拟机里…

使用VMware虚拟机创建Ubuntu的linux系统,用Xshell连接这个系统,VScode作为编辑器时遇到的问题

使用VMware虚拟机创建Ubuntu的linux系统&#xff0c;用Xshell连接这个系统&#xff0c;VScode作为编辑器时遇到的问题1.软件2.Xshell和Xftp软件的使用3.VScode中安装了Remote Development扩展之后&#xff0c;点击远程资源管理器&#xff0c;下拉框里没有SSH-Targets4.将VScode…