FreeRTOS之链表源码分析

文章目录

  • 前言
  • 一、结构体
    • 1、链表List_t
    • 2、链表项xLIST_ITEM
    • 3、头节点xMINI_LIST_ITEM
    • 4、链表示意图
  • 二、函数分析
    • 1、初始化函数vListInitialise
    • 2、初始化链表项vListInitialiseItem
    • 3、链表尾部添加节点vListInsertEnd
    • 4、按序插入节点vListInsert
    • 5、删除节点uxListRemove
  • 总结


前言

① 本文章学习完韦东山老师的 《freeRTOS系列教程:FreeRTOS的内部机制》之后做的总结。大家可前往看视频。
② 代码是云途芯片的SDK,关于FreeRTOS的源码都大同小异,无需纠结。

韦东山freeRTOS系列教程:FreeRTOS的内部机制

一、结构体

1、链表List_t

在这里插入图片描述

  • uxNumberOfItems:表示该链表下有多少个链表节点(例如:优先级为1的任务都存在链表pxReadyTaskLists[1]中,uxNumberOfItems 等于 2,表示有两个优先级为1的任务在Ready链表中等待)
  • pxIndex:指向当前正在使用的链表项item,这个pxIndex被用来遍历链表。
  • xListEnd:是一个MiniListItem_t节点,是链表的最后一个节点。因为FreeRTOS定义的链表是循环双向链表,因此xListEnd也是链表的第一节点。

2、链表项xLIST_ITEM

在这里插入图片描述

  • xItemValue:每个链表项(节点)的一个计数值,一般用来做排序。(比如把链表中的节点都按序排列,就离不开它)
  • pxNext:指向后继节点。
  • pxPrevious:指向前一个节点。
  • pvOwner:指向拥有该节点的内核对象,一般是指TCB结构体。
  • pxContainer:指向节点所在的链表xLIST。

3、头节点xMINI_LIST_ITEM

在这里插入图片描述

  • xItemValue:每个链表项(节点)的一个计数值,一般用来做排序。(比如把链表中的节点都按序排列,就离不开它)
  • pxNext:指向后继节点。
  • pxPrevious:指向前一个节点。
    (这个链表项比上面的链表项少了pvOwner和pxContainer,可以把它当成链表中的头节点/尾节点xListEnd)

4、链表示意图

在这里插入图片描述
在上图中,链表中元素是顺序是:item1、item2、item3、xListEnd。
list中有一个pxIndex,指向当前真在使用的item。链表的遍历过程如下:

  • pxIndex初始时指向链表list中的xListEnd(数据类型MiniListItem_t)
  • 要取出第一个元素时,pxIndex就会指向item1
  • 再取出下一个元素时,pxIndex就会指向item2
  • 再取出下一个元素时,pxIndex就会指向item3
  • 再取出下一个元素时,pxIndex就会指向xListEnd
  • 发现它是xListEnd时,继续去下一个元素,pxIndex就会指向item1

~

二、函数分析

1、初始化函数vListInitialise

在这里插入图片描述
List_t中有一个Item: xListEnd,初始化链表后,结果如下:

  • uxNumberOfItems等于0,表示链表尾空,链表项的数量为0。
  • pxIndex指向xListEnd。
  • xListEnd中的pxNext、pxPrevious指向xListEnd(表示链表是空的)。
  • xListEnd中的xItemValue等于portMAX_DELAY。
    可以用下图来简述:
    在这里插入图片描述

2、初始化链表项vListInitialiseItem

在这里插入图片描述
pxContainer指向节点所在的链表xLIST为NULL,表示不存在于任何链表。

3、链表尾部添加节点vListInsertEnd

在这里插入图片描述
① pxIndex = pxList->pxIndex; 指向当前正在使用的节点itme2(若链表是空的,那么就是指向xListEnd头节点)
② pxNewListItem->pxNext = pxIndex; 新增节点item4的下一个节点是当前正在使用的节点itme2(若链表是空的,那么就是指向xListEnd头节点)
③ pxNewListItem->pxPrevious = pxIndex->pxPrevious;新增节点item4的上一个节点是当前正在使用的节点itme2的前节点item1(若链表是空的,那么就是指向xListEnd头节点)
④ pxIndex->pxPrevious->pxNext = pxNewListItem;当前正在使用的节点itme2的前节点item1的后继节点是新增节点item4(若链表是空的,那么就是xListEnd头节点指向后继节点是新增节点item4)
⑤ pxIndex->pxPrevious = pxNewListItem; 当前正在使用的节点itme2的前一节点是item4(若链表是空的,那么就是xListEnd头节点指向前一个结点是新增节点item4)
⑥ pxNewListItem->pxContainer = pxList; 新插入节点item4指向节点所在的链表pxList.

看下图更好理解:
在这里插入图片描述
所谓的尾部并不是指链表中的尾部节点,而是指插入到pxIndex指向的item2节点之前。
(Tips:因为pxIndex指向的节点表示正在使用的节点,item2使用完之后,pxIndex指向item3,接着再pxIndex指向item1。这样插入结点之后,item4就是最后一个被使用的节点。)

4、按序插入节点vListInsert

新建一个PwmFtm_0,
for循环中,当pxIterator指向item2的时候,退出循环。
pxNewListItem->pxNext = pxIterator->pxNext; 新节点item4的后继节点是item2的后继节点item3
pxNewListItem->pxNext->pxPrevious = pxNewListItem; 新节点item4的后继节点item3的前一节点指向新节点item4
pxNewListItem->pxPrevious = pxIterator; 新节点item4的前一节点是item2(pxIterator指向的节点,来源于for循环的遍历结果)
pxIterator->pxNext = pxNewListItem; item2的后继节点是新节点item4
如果是初始值portMAX_DELAY,那就加入到最后一个位置,即xListEnd的前一项。(下面代码段)

if( xValueOfInsertion == portMAX_DELAY )
    {
        pxIterator = pxList->xListEnd.pxPrevious;
    }

看下图更好理解:
在这里插入图片描述
怎么插入呢?
a)先找到位置item
item(item2)->xItemValue < item_new(item4)->xItemValue
item(item3)->next(item3)->xItemValue < item_new(item4)->xItemValue
b)在item2之后插入

~

5、删除节点uxListRemove

在这里插入图片描述
① pxItemToRemove->pxNext->pxPrevious = pxItemToRemove->pxPrevious; 删除项item2后继节点(item3)的前一结点是删除项item2的前一节点(item1)。
② pxItemToRemove->pxPrevious->pxNext = pxItemToRemove->pxNext; 删除项item2前一节点(item1)的后继节点是item2的后继节点(item3)
如果删除项item2正处于被使用状态时候,就将索引指针pxList->pxIndex指向item2的前一节点(item1)
④ pxItemToRemove->pxContainer = NULL; item2都被删除啦,所以它的链表信息应该也被删除。

if( pxList->pxIndex == pxItemToRemove )
    {
        pxList->pxIndex = pxItemToRemove->pxPrevious;
    }

⑤ ( pxList->uxNumberOfItems ) = ( UBaseType_t ) ( pxList->uxNumberOfItems - 1U ); 链表中的节点个数减1

看下图更好理解:
在这里插入图片描述

~

总结

博主是小白,刚开始接触FreeRTOS,如果哪里表达的有问题,还请大佬们指点指点哈。接下来,让我们在后面的博文再相会哈~

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

预测未来 | MATLAB实现Transformer时间序列预测未来

预测未来 | MATLAB实现Transformer时间序列预测未来 预测效果 基本介绍 1.Matlab实现Transformer时间序列预测未来&#xff1b; 2.运行环境Matlab2023b及以上&#xff0c;data为数据集&#xff0c;单变量时间序列预测&#xff1b; 3.递归预测未来数据&#xff0c;可以控制预…

怎么样才算得上熟悉高并发编程?

提到并发编程很多人就会头疼了&#xff1b;首先就是一些基础概念&#xff1a;并发&#xff0c;并行&#xff0c;同步&#xff0c;异步&#xff0c;临界区&#xff0c;阻塞&#xff0c;非阻塞还有各种锁全都砸你脸上&#xff0c;随之而来的就是要保证程序运行时关键数据在多线程…

最新 Blender 4.2 保姆级安装教程(附安装包)

目录 Blender介绍&#xff1a; Blender下载&#xff1a; Blender改进功能&#xff1a; Blender介绍&#xff1a; Blender是一款开源的跨平台全能三维动画制作软件&#xff0c;提供从建模、渲染、动画、特效、合成到音频处理、视频剪辑等一系列动画短片制作解决方案。它支持…

web安全之信息收集

在信息收集中,最主要是就是收集服务器的配置信息和网站的敏感信息,其中包括域名及子域名信息,目标网站系统,CMS指纹,目标网站真实IP,开放端口等。换句话说,只要是与目标网站相关的信息,我们都应该去尽量搜集。 1.1收集域名信息 知道目标的域名之后,获取域名的注册信…

网络原理(一)—— http

什么是 http http 是一个应用层协议&#xff0c;全称为“超文本传输协议”。 http 自 1991 年诞生&#xff0c;目前已经发展为最主流使用的一种应用层协议。 HTTP 往往基于传输层的 TCP 协议实现的&#xff0c;例如 http1.0&#xff0c;http1.0&#xff0c;http2.0 http3 是…

第四十二篇 EfficientNet:重新思考卷积神经网络的模型缩放

文章目录 摘要1、简介2、相关工作3、复合模型缩放3.1、 问题公式化3.2、扩展维度3.3、复合比例 4、EfficientNet架构5、实验5.1、扩展MobileNets和ResNets5.2、EfficientNet的ImageNet结果5.3、EfficientNet的迁移学习结果 6、讨论7、结论 摘要 卷积神经网络(ConvNets)通常在固…

典型组合逻辑电路设计

目录 行为级描述方式基本运算电路 一、半加器&#xff08;Half Adder&#xff09; 二、全加器&#xff08;Full Adder&#xff09; 1、逻辑门构成加法器 2、集成全加器 3、串行加法器 4、超前进位加法器 三、全减器(Full Deductor) 数值比较电路 一、一位比较器 二、…

【论文阅读】三平面相关与变体

文章目录 1. 【CVPR2023】Tri-Perspective View for Vision-Based 3D Semantic Occupancy Prediction动机可视化方法Pipeline 2. 【2023/08/31】PointOcc: Cylindrical Tri-Perspective View for Point-based 3D Semantic Occupancy Prediction动机&#xff08;针对雷达点云、与…

修改bag的frame_id的工具srv_tools

在使用数据集导航或者建图时&#xff0c;bag中的点云或者其他话题的frame_id没有和需要的对应 1.创建工作空间 2.cd xxxx/src 3.git clone https://github.com/srv/srv_tools.git cd .. catkin_make source ./devel/setup.bash rosrun bag_tools change_frame_id.py -t /要改…

hue 4.11容器化部署,已结合Hive与Hadoop

配合《Hue 部署过程中的报错处理》食用更佳 官方配置说明页面&#xff1a; https://docs.gethue.com/administrator/configuration/connectors/ 官方配置hue.ini页面 https://github.com/cloudera/hue/blob/master/desktop/conf.dist/hue.ini docker部署 注意&#xff1a; …

如何用Excel做数据可视化自动化报表?

作为一个经常需要做数据报表的人&#xff0c;我最常用的工具是Excel&#xff0c;对于我来说用Excel处理繁琐冗杂的数据并不难&#xff0c;但是我发现身边很多人用Excel做的数据报表非常的耗时&#xff0c;而且最后的成品也是难以直视&#xff0c;逻辑和配色等都非常的“灾难”。…

layui table 纵向滚动条导致单元格表头表体错位问题

我用的时layui2.6.8版本 历史项目维护&#xff0c;bug给我让我做了&#xff0c;本来利用前端手段强解决&#xff0c;后来发现很多table 找了解决办法 打开layui-v2.6.8/lay/modules/table.js 如果打开后时压缩的代码 直接搜索 e.find(".layui-table-patch") …

C语言学习笔记:流程控制和数据输入输出

流程控制和数据的输入输出 算法 著名计算机科学家沃思提出了一个公式&#xff1a; 数据结构 算法 程序 数据结构&#xff1a;对数据的描述 算法&#xff1a;对操作步骤的描述 算法定义 广义的说&#xff0c;为解决一个问题而采取的方法和有限的步骤&#xff0c;就称为“…

旋转图像(java)

题目描述&#xff1a; 给定一个 n n 的二维矩阵 matrix 表示一个图像。请你将图像顺时针旋转 90 度。 你必须在 原地 旋转图像&#xff0c;这意味着你需要直接修改输入的二维矩阵。请不要 使用另一个矩阵来旋转图像。 代码思路&#xff1a; class Solution {public void ro…

windows 应用 UI 自动化实战

UI 自动化技术架构选型 UI 自动化是软件测试过程中的重要一环&#xff0c;网络上也有很多 UI 自动化相关的知识或资料&#xff0c;具体到 windows 端的 UI 自动化&#xff0c;我们需要从以下几个方面考虑&#xff1a; 开发语言 毋庸置疑&#xff0c;在 UI 自动化测试领域&am…

【R语言管理】Pycharm配置R语言及使用Anaconda管理R语言虚拟环境

目录 使用Anaconda创建R语言虚拟环境1. 安装Anaconda2. 创建R语言虚拟环境 Pycharm配置R语言1. 安装Pycharm2. R Language for IntelliJ插件 参考 使用Anaconda创建R语言虚拟环境 1. 安装Anaconda Anaconda的安装可参见另一博客-【Python环境管理工具】Anaconda安装及使用教程…

C语言进程编程

getpid函数&#xff1a; 原型&#xff1a;pid_t getpid(void) 特性&#xff1a;返回值是PID值 用途&#xff1a;获取当前进程PID 用法例 #include<stdio.h> #include <sys/types.h> #include<unistd.h> int main() {pid_t pid;pid getpid();printf(&qu…

SpringMVC |(一)SpringMVC概述

文章目录 &#x1f4da;SpringMVC概述&#x1f407;三层架构&#x1f407;异步调用 &#x1f4da;SpringMVC入门案例&#x1f407;入门案例&#x1f407;注意事项 &#x1f4da;小结 学习来源&#xff1a;黑马程序员SSM框架教程_SpringSpringMVCMaven高级SpringBootMyBatisPlus…

Android 桌面窗口新功能推进,聊一聊 Android 桌面化的未来

Android 桌面化支持可以说是 Android 15 里被多次提及的 new features&#xff0c;例如在 Android 15 QPR1 Beta 2 里就提到为 Pixel 平板引入了桌面窗口支持&#xff0c;桌面窗口允许用户在自由窗口同时运行多个应用&#xff0c;同时可以像在传统 PC 平台上一样调整这些窗口的…

Vue+Vite 组件开发的环境准备(零基础搭建)

一、什么是Vite Vue3作为一款现代化的JavaScript框架&#xff0c;配合Vite这样的构建工具&#xff0c;极大地简化了流程&#xff0c;提升了效率。Vite 是一个基于现代浏览器原生的 ES 模块系统&#xff0c;能够以原生模块导入的方式运行源代码的开发服务器。它被设计用来替代传…