首页 > 编程语言 >《解读 freeRTOS queue.c 源码:探寻任务间高效通信的核心机制》(建议收藏)

《解读 freeRTOS queue.c 源码:探寻任务间高效通信的核心机制》(建议收藏)

时间:2024-11-26 17:59:40浏览次数:10  
标签:信号量 函数 freeRTOS 队列 queue 互斥 任务 源码 数据

摘要:文章介绍了 FreeRTOS 中的队列机制,包括队列的基本概念、创建与初始化方式、数据操作(发送、接收、查看等)、状态查询与管理、在互斥量与信号量中的应用以及队列集的相关操作

建议:作者在这里建议,结合queue.c源码来看效果会更好一下

FreeRTOS 队列:任务间通信与同步的高效利器

在嵌入式系统开发中,任务间的通信与同步是至关重要的环节。FreeRTOS 作为一款广泛应用的实时操作系统,其队列机制为任务间的数据传递和协调提供了强大而灵活的解决方案。

一、队列的基本概念与结构

FreeRTOS 中的队列是一种用于任务间数据传递的数据结构。队列中的数据项通过复制的方式进行存储,而非引用。其核心数据结构Queue_t(在旧版本中为xQUEUE)包含了多个重要成员:

  • pcHead:指向队列存储区域的起始位置,当队列用于表示互斥量时,有特殊用途。
  • pcTail:指向队列存储区域的末尾字节,用于标记队列的边界。
  • pcWriteTo:指示下一个可写入数据的位置。
  • u.pcReadFrom(或u.uxRecursiveCallCount,取决于队列用途):在队列用于存储数据时,指向最后一个读取数据的位置;当作为互斥量时,用于记录递归调用次数。
  • xTasksWaitingToSendxTasksWaitingToReceive:分别是等待发送数据到队列等待从队列接收数据的任务列表,这些任务按照优先级顺序存储
  • uxMessagesWaiting:记录当前队列中存储的数据项数量。
  • uxLength:队列的长度,即能够容纳的数据项数量。
  • uxItemSize:每个数据项的大小。
  • cRxLockcTxLock:用于记录队列在锁定状态下接收和发送数据项的数量,初始值为queueUNLOCKED(int8_t) -1)。

此外,还有一些与队列相关的宏定义,如queueUNLOCKED表示队列未锁定,queueLOCKED_UNMODIFIED表示队列锁定但未修改等,这些宏定义的使用是用来更清晰地表达队列的状态和操作逻辑。

二、队列的创建与初始化

FreeRTOS 提供了多种方式创建队列,以满足不同的内存分配需求。

  • configSUPPORT_STATIC_ALLOCATIONconfigSUPPORT_DYNAMIC_ALLOCATION都为1时,可以使用xQueueGenericCreateStatic函数进行静态创建队列。该函数要求传入队列长度uxQueueLength、数据项大小uxItemSize、队列存储区域指针pucQueueStorage以及队列类型ucQueueType等参数。在创建过程中,会对传入的参数进行严格的断言检查,确保参数的有效性。例如,队列长度必须大于0,并且根据数据项大小是否为0来确定是否提供队列存储区域。创建成功后,会调用prvInitialiseNewQueue函数对队列进行初始化,设置队列的各项成员变量,包括pcHeaduxLengthuxItemSize等,并根据configUSE_TRACE_FACILITY的设置来配置队列的跟踪信息。
  • configSUPPORT_DYNAMIC_ALLOCATION1时,可以使用xQueueGenericCreate函数进行动态创建队列。该函数同样需要传入队列长度和数据项大小等参数。根据数据项大小计算所需的队列存储区域大小,如果数据项大小为0,则不分配存储区域。然后使用pvPortMalloc函数动态分配队列结构和存储区域的内存空间。成功分配后,也会调用prvInitialiseNewQueue函数进行初始化,并设置ucStaticallyAllocated成员变量为pdFALSE,表示队列是动态分配的。

对于互斥量这种特殊类型的队列,当configUSE_MUTEXES1时,创建互斥量队列的过程稍有不同。首先会创建一个普通队列,然后调用prvInitialiseMutex函数对其进行配置。在prvInitialiseMutex函数中,会将pxMutexHolder设置为NULLuxQueueType设置为queueQUEUE_IS_MUTEX,并将u.uxRecursiveCallCount初始化为0,同时调用traceCREATE_MUTEX函数进行跟踪记录,并通过xQueueGenericSend函数将互斥量初始化为可用状态。

三、队列的数据操作

(一)数据发送

  • xQueueGenericSend函数用于向队列发送数据。在发送数据之前,会先检查队列是否有空间。如果队列未满或者发送模式为覆盖(queueOVERWRITE且队列长度为1),则可以直接将数据复制到队列中。数据复制通过prvCopyDataToQueue函数完成,该函数会根据队列的类型和发送位置进行相应的处理。如果是普通队列且发送到队列末尾(queueSEND_TO_BACK),则将数据复制到pcWriteTo指向的位置,并更新pcWriteTo指针;如果是发送到队列头部,则将数据复制到u.pcReadFrom指向的位置,并更新u.pcReadFrom指针。在复制数据完成后,会检查队列是否属于某个队列集(configUSE_QUEUE_SETS1pxQueue->pxQueueSetContainer不为NULL),如果是,则调用prvNotifyQueueSetContainer函数通知队列集。如果队列中有任务正在等待接收数据,且该任务的优先级高于当前任务,则会进行任务切换(通过queueYIELD_IF_USING_PREEMPTION宏实现)。如果队列已满且设置了阻塞时间,则当前任务会进入阻塞状态,等待队列有空间或者阻塞时间超时。在阻塞期间,任务会被挂起,并加入到xTasksWaitingToSend列表中。
  • xQueueGenericSendFromISR函数用于在中断服务程序中向队列发送数据。与xQueueGenericSend函数类似,但不能阻塞。如果队列有空间,则直接复制数据到队列,并根据队列是否属于队列集以及是否有任务等待接收数据来决定是否进行任务切换(通过设置pxHigherPriorityTaskWoken标志)。如果队列已满,则直接返回errQUEUE_FULL

(二)数据接收

  • xQueueReceive函数用于从队列接收数据。首先检查队列中是否有数据,如果有数据,则调用prvCopyDataFromQueue函数将数据从队列复制到接收缓冲区,并更新队列的相关状态,如uxMessagesWaiting1。如果队列中有任务正在等待发送数据到队列,且该任务的优先级高于当前任务,则会进行任务切换。如果队列为空且设置了阻塞时间,则当前任务会进入阻塞状态,等待队列中有数据或者阻塞时间超时。在阻塞期间,任务会被挂起,并加入到xTasksWaitingToReceive列表中。
  • xQueueReceiveFromISR函数用于在中断服务程序中从队列接收数据。该函数不能阻塞,直接检查队列中是否有数据,如果有数据,则复制数据到接收缓冲区,并根据队列是否锁定以及是否有任务等待发送数据来决定是否进行任务切换(通过设置pxHigherPriorityTaskWoken标志)。如果队列为空,则返回pdFAIL

(三)其他操作

  • xQueuePeek函数用于查看队列头部的数据,但不将其从队列中移除。其操作过程与
  • xQueueReceive函数类似,但在复制数据后会将读取指针恢复到原来的位置,以保证数据仍然留在队列中。
  • xQueuePeekFromISR函数用于在中断服务程序中查看队列头部的数据,同样不将数据移除。如果队列中有数据,则直接复制数据到接收缓冲区并返回pdPASS,否则返回pdFAIL

四、队列的状态查询与管理

  • uxQueueMessagesWaiting函数用于获取队列中当前等待处理的数据项数量。通过进入临界区,读取uxMessagesWaiting成员变量的值并返回。
  • taskENTER_CRITICALtaskEXIT_CRITICAL的基本原理(临界区)
    • 在许多实时操作系统(如 FreeRTOS)中,taskENTER_CRITICALtaskEXIT_CRITICAL是用于保护临界区的宏或函数。它们的实现方式通常是通过关闭中断来实现临界区的保护。
    • taskENTER_CRITICAL被调用时,系统会暂时禁止中断(具体的实现可能因操作系统和硬件平台而异)。这就防止了其他中断服务程序或任务切换等操作打断当前的关键代码序列。在这个例子中,从taskENTER_CRITICALtaskEXIT_CRITICAL之间的代码就是临界区。
    • 在临界区内,代码可以安全地访问和修改共享资源(如队列相关的数据结构),而不用担心被其他任务或中断干扰导致数据不一致。
  • uxQueueSpacesAvailable函数用于获取队列中剩余的可用空间数量。通过计算队列长度与uxMessagesWaiting的差值得到。
  • xQueueIsQueueEmptyFromISRxQueueIsQueueFullFromISR函数分别用于在中断服务程序中检查队列是否为空和是否已满,直接判断uxMessagesWaiting的值与0或队列长度的关系并返回相应结果。
  • vQueueDelete函数用于删除队列。在删除队列之前,会先进行一些断言检查,确保队列指针有效。如果configQUEUE_REGISTRY_SIZE大于0,则会调用vQueueUnregisterQueue函数将队列从队列注册表中移除。然后根据队列的内存分配方式,如果是动态分配且configSUPPORT_DYNAMIC_ALLOCATION1,则使用vPortFree函数释放队列占用的内存空间;如果是静态分配且configSUPPORT_STATIC_ALLOCATION1,则根据ucStaticallyAllocated成员变量的值来决定是否释放内存(如果为pdFALSE则释放)。

五、队列在互斥量与信号量中的应用

(一)互斥量

互斥量在 FreeRTOS 中是一种特殊的队列,用于保护共享资源。当一个任务获取互斥量时,其他任务将无法获取该互斥量,从而保证了共享资源的互斥访问。在创建互斥量时,通过prvInitialiseMutex函数对队列进行特殊配置,将pxMutexHolder设置为NULL表示互斥量未被持有,uxQueueType设置为queueQUEUE_IS_MUTEX,并初始化u.uxRecursiveCallCount0。在任务获取互斥量时,会检查pxMutexHolder是否为当前任务,如果是,则增加u.uxRecursiveCallCount,表示递归获取互斥量;如果不是,则通过xQueueSemaphoreTake函数尝试获取互斥量,并根据获取结果更新u.uxRecursiveCallCount。当任务释放互斥量时,会检查u.uxRecursiveCallCount的值,如果为0,则表示互斥量可以被其他任务获取,通过xQueueGenericSend函数将互斥量释放,并自动唤醒等待该互斥量的任务。

(二)信号量

信号量也是基于队列实现的一种同步机制。当configUSE_COUNTING_SEMAPHORES1时,可以创建计数信号量。计数信号量的队列长度表示信号量的最大计数值,uxMessagesWaiting成员变量表示当前信号量的计数值。通过xQueueGiveFromISRxQueueSemaphoreTake等函数对信号量进行操作,实现任务间的同步和资源的计数管理。

六、队列集的操作

configUSE_QUEUE_SETS1时,FreeRTOS 支持队列集的操作。队列集可以将多个队列或信号量组合在一起,方便任务对多个事件源进行监听和处理。

  • xQueueCreateSet函数用于创建队列集,其内部调用xQueueGenericCreate函数创建一个队列,该队列的每个数据项为Queue_t *类型,用于存储队列集成员的指针。
  • xQueueAddToSet函数用于将队列或信号量添加到队列集中。在添加之前,会检查队列或信号量是否已经属于其他队列集,以及其是否为空(uxMessagesWaiting0)。如果满足条件,则将其pxQueueSetContainer成员变量设置为队列集指针,表示该队列或信号量已加入队列集。
  • xQueueRemoveFromSet函数用于从队列集中移除队列或信号量。在移除之前,会检查队列或信号量是否属于指定的队列集,以及其是否为空。如果满足条件,则将其pxQueueSetContainer成员变量设置为NULL,表示该队列或信号量已从队列集中移除。
  • xQueueSelectFromSetxQueueSelectFromSetFromISR函数分别用于在任务和中断服务程序中从队列集中选择一个有事件发生的队列或信号量。这两个函数内部调用xQueueReceivexQueueReceiveFromISR函数,从队列集中接收数据,并返回对应的队列或信号量句柄。

FreeRTOS 的队列机制提供了丰富而强大的功能,涵盖了任务间数据传递、同步以及资源保护等多个方面。无论是普通的数据队列,还是特殊的互斥量、信号量和队列集,都为嵌入式系统开发者提供了灵活高效的工具,有助于构建稳定、可靠且高效的嵌入式应用程序。通过深入理解和熟练运用 FreeRTOS 队列机制,可以更好地发挥 FreeRTOS 在嵌入式系统中的优势,满足各种复杂的任务间通信和同步需求。

标签:信号量,函数,freeRTOS,队列,queue,互斥,任务,源码,数据
From: https://blog.csdn.net/2401_83606346/article/details/144042898

相关文章

  • php毕业设计购物商城在线购物系统日用品购物商城手工艺系统日用品系统手工艺网站php+m
    一,功能介绍        前台主要包括网站首页、商品推荐、最新商品、新闻咨询、商品分类、商品资讯、评论、登录、注册、加入购物车、结算、个人中心等功能模块商品推荐、最新商品在商品推荐、最新商品模块,用户可以查看全部商品信息,选择商品进行添加购物车等操作,购物......
  • springboot毕设基于Android的考勤签到系统源码+论文+部署
    本系统(程序+源码)带文档lw万字以上 文末可获取一份本项目的java源码和数据库参考。系统程序文件列表开题报告内容一、研究背景随着信息技术的飞速发展,移动设备在企业管理中的应用日益广泛。Android系统作为移动设备的主流操作系统,具有开放性、易用性和广泛的用户基础等特......
  • springboot毕设基于B_S架构的作业自动评阅系统源码+论文+部署
    本系统(程序+源码)带文档lw万字以上 文末可获取一份本项目的java源码和数据库参考。系统程序文件列表开题报告内容一、研究背景随着信息技术的飞速发展,教育领域也在不断寻求数字化转型。在传统的作业评阅模式下,教师需要花费大量的时间和精力来批改学生作业,效率较低且容易出......
  • servlet职称评审系统-计算机设计毕业源码00122
    目录摘要1绪论1.1选题背景与意义1.2国内外研究现状1.3论文结构与章节安排2系统分析2.1可行性分析2.2系统流程分析2.2.1系统开发流程2.2.2用户登录流程2.2.3系统操作流程2.2.4添加信息流程2.2.5修改信息流程2.2.6删除信息流程2.3 系统......
  • springboot某企业自主招聘系统-计算机设计毕业源码90393
    目录摘要1绪论1.1选题背景与意义1.2国内外研究现状2系统分析2.1.1技术可行性分析2.1.2 经济可行性分析2.1.3法律可行性分析2.2系统流程分析2.2.1添加信息流程2.2.2修改信息流程2.2.3删除信息流程2.3 系统功能分析2.3.1功能性分析2.3......
  • 帝国CMS7.5之帝国十合一源码图文安装说明!
    帝国CMS十合一源码/字典/成语/古诗词/二十四节气/英语单词/百家姓/范文文库/词语等首先测试环境是宝塔,nginx环境,php5.6+mysql5.7!不能超出这个环境否则安装了会出现各种问题。第一步:上传到根目录解压,然后导入数据库!(导入完有3G,如果导入不了就在本地搞个数据库导入,然后把数据库文件......
  • 数字人直播带货前景及入局方式曝光!手把手教你获取高质量源码!
    随着数字人直播带货模式的出现频率不断升高,让越来越多的人产生入局想法的同时,也让与之相关的话题,如数字人直播带货前景怎么样和如何顺利完成入局等的讨论热度与日俱增。本期,我们就以上述的两大话题为切入点,为大家带来数字人直播带货项目的详细介绍。一、数字人直播带货前景怎......
  • springboot高校志愿服务信息管理平台-计算机毕业设计源码12823
    摘 要本课题的研究对象是高校志愿服务信息管理平台,该系统实现了系统用户、活动类型管理、志愿活动管理、活动报名管理、活动参与管理、积分礼品管理、兑换记录管理、组织申请管理、留言管理等功能。本系统在设计上,考虑到系统内容以及系统的受众群体,在系统的美工方面采用了比......
  • SSM酒店后台管理系统-计算机毕业设计源码13123
    目 录摘要Abstract1绪论1.1课题目的及意义1.2研究背景1.3研究方法1.4论文结构与章节安排2 酒店后台管理系统系统分析2.1可行性分析2.2系统功能分析2.2.1功能性分析2.3.2非功能性分析2.4 系统用例分析2.5本章小结3酒店后台管理系统总......
  • 树的双亲表示法与孩子表示法(C++源码)
    目录一、双亲表示法:二、孩子表示法:一、双亲表示法:#include<iostream>usingnamespacestd;#defineMAX_TREE_SIZE100template<typenameDataType>structPNode{ DataTypedata; intparent;};//树的存储结构:双亲表示法template<typenameDataType......