- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
如何处理进入阻塞状态的任务?
为了让 RTOS 支持多优先级,我们创建了多个就绪链表(数组形式),用每一个就绪链表表示一个优先级,对于阻塞状态的任务显然要从就绪链表中移除,但是阻塞状态的任务并不是永久阻塞了,等待一段时间后应该从阻塞状态恢复,所以我们需要创建一个阻塞链表用来存放进入阻塞状态的任务 。
还有一个问题,xTicksToDelay 是一个 32 位的变量,如何处理其潜在的溢出问题?
假设使用一个 32 位的 xNextTaskUnblockTime 变量表示任务下次解除阻塞的时间,其一般应该由如下所示的程序代码计算 。
// 任务下次解除阻塞的时间 = 当前滴答定时器计数值 + 要延时的滴答次数
xNextTaskUnblockTime = xConstTickCount + xTicksToWait;
可以看出 xNextTaskUnblockTime 变量随着运行时间流逝存在溢出风险,因此我们需要再定义一个溢出阻塞链表用来存放所有下次解除阻塞的时间溢出的任务,这样我们就拥有两个阻塞链表,在滴答定时器中断服务函数中如果一旦发现滴答定时器计数值全局变量溢出,就通过链表指针将这两个链表交换,保证永远处理的是正确的阻塞链表 。
/* task.c */
// 阻塞链表和其指针
static List_t xDelayed_Task_List1;
static List_t volatile *pxDelayed_Task_List;
// 溢出阻塞链表和其指针
static List_t xDelayed_Task_List2;
static List_t volatile *pxOverflow_Delayed_Task_List;
由于新增加了阻塞链表和溢出阻塞链表,因此在链表初始化函数中除了需要初始化就绪链表数组外,还需要增加对阻塞链表和溢出阻塞链表的初始化操作,如下所示 。
/* task.c */
// 就绪列表初始化函数
void prvInitialiseTaskLists(void)
{
// 省略未修改部分
......
// 初始化延时阻塞链表
vListInitialise(&xDelayed_Task_List1);
vListInitialise(&xDelayed_Task_List2);
// 初始化指向延时阻塞链表的指针
pxDelayed_Task_List = &xDelayed_Task_List1;
pxOverflow_Delayed_Task_List = &xDelayed_Task_List2;
}
为什么需要阻塞链表和溢出阻塞链表需要交换?
阅读 ” 0.2、思考二“ 小节内容 。
阻塞链表和溢出阻塞链表是如何实现交换的?
利用两个指针进行交换 。
/* task.c */
// 记录溢出次数
static volatile BaseType_t xNumOfOverflows = (BaseType_t)0;
// 延时阻塞链表和溢出延时阻塞链表交换
#define taskSWITCH_DELAYED_LISTS()\
{\
List_t volatile *pxTemp;\
pxTemp = pxDelayed_Task_List;\
pxDelayed_Task_List = pxOverflow_Delayed_Task_List;\
pxOverflow_Delayed_Task_List = pxTemp;\
xNumOfOverflows++;\
prvResetNextTaskUnblockTime();\
}
由于将任务插入溢出阻塞链表时不会更新 xNextTaskUnblockTime 变量,只有在将任务插入阻塞链表中时才会更新xNextTaskUnblockTime 变量,所以对于溢出阻塞链表中存在的任务没有对应的唤醒时间,因此当心跳溢出切换阻塞链表时候,需要重设 xNextTaskUnblockTime 变量的值 。
/* task.c */
// 记录下个任务解除阻塞时间
static volatile TickType_t xNextTaskUnblockTime = (TickType_t)0U;
// 函数声明
static void prvResetNextTaskUnblockTime(void);
// 重设 xNextTaskUnblockTime 变量值
static void prvResetNextTaskUnblockTime(void)
{
TCB_t *pxTCB;
// 切换阻塞链表后,阻塞链表为空
if(listLIST_IS_EMPTY(pxDelayed_Task_List) != pdFALSE)
{
// 下次解除延时的时间为可能的最大值
xNextTaskUnblockTime = portMAX_DELAY;
}
else
{
// 如果阻塞链表不为空,下次解除延时的时间为链表头任务的阻塞时间
(pxTCB) = (TCB_t *)listGET_OWNER_OF_HEAD_ENTRY(pxDelayed_Task_List);
xNextTaskUnblockTime=listGET_LIST_ITEM_VALUE(&((pxTCB)->xStateListItem));
}
}
将当前任务加入到阻塞链表中,具体流程可以参看程序注释,对于延时到期时间未溢出的任务会插入到阻塞链表中,而对于延时到期时间溢出的任务会插入溢出阻塞链表中 。
/* task.c */
// 将当前任务添加到阻塞链表中
static void prvAddCurrentTaskToDelayedList(TickType_t xTicksToWait)
{
TickType_t xTimeToWake;
// 当前滴答定时器中断次数
const TickType_t xConstTickCount = xTickCount;
// 成功从就绪链表中移除该阻塞任务
if(uxListRemove((ListItem_t *)&(pxCurrentTCB->xStateListItem)) == 0)
{
// 将当前任务的优先级从优先级位图中删除
portRESET_READY_PRIORITY(pxCurrentTCB->uxPriority, uxTopReadyPriority);
}
// 计算延时到期时间
xTimeToWake = xConstTickCount + xTicksToWait;
// 将延时到期值设置为阻塞链表中节点的排序值
listSET_LIST_ITEM_VALUE(&(pxCurrentTCB->xStateListItem), xTimeToWake);
// 如果延时到期时间会溢出
if(xTimeToWake < xConstTickCount)
{
// 将其插入溢出阻塞链表中
vListInsert((List_t *)pxOverflow_Delayed_Task_List,
(ListItem_t *)&(pxCurrentTCB->xStateListItem));
}
// 没有溢出
else
{
// 插入到阻塞链表中
vListInsert((List_t *)pxDelayed_Task_List,
(ListItem_t *) &( pxCurrentTCB->xStateListItem));
// 更新下一个任务解锁时刻变量 xNextTaskUnblockTime 的值
if(xTimeToWake < xNextTaskUnblockTime)
{
xNextTaskUnblockTime = xTimeToWake;
}
}
}
/* task.c */
void vTaskStartScheduler(void)
{
// 省略创建空闲任务函数
......
// 初始化滴答定时器计数值,感觉有点儿多余?全局变量定义时候已被初始化为 0
xTickCount = (TickType_t)0U;
if(xPortStartScheduler() != pdFALSE){}
}
阻塞延时函数,当任务调用阻塞延时函数时会将任务从就绪链表中删除,然后加入到阻塞链表中 。
/* task.c */
// 阻塞延时函数
void vTaskDelay(const TickType_t xTicksToDelay)
{
// 将当前任务加入到阻塞链表
prvAddCurrentTaskToDelayedList(xTicksToDelay);
// 任务切换
taskYIELD();
}
利用 RTOS 的心跳(滴答定时器中断服务函数)对阻塞任务进行处理,具体流程如下所示 。
/* task.c */
// 任务阻塞延时处理
BaseType_t xTaskIncrementTick(void)
{
TCB_t *pxTCB = NULL;
TickType_t xItemValue;
BaseType_t xSwitchRequired = pdFALSE;
// 更新系统时基计数器 xTickCount
const TickType_t xConstTickCount = xTickCount + 1;
xTickCount = xConstTickCount;
// 如果 xConstTickCount 溢出,则切换延时列表
if(xConstTickCount == (TickType_t)0U)
{
taskSWITCH_DELAYED_LISTS();
}
// 最近的延时任务延时到期
if(xConstTickCount >= xNextTaskUnblockTime)
{
for(;;)
{
// 延时阻塞链表为空则跳出 for 循环
if(listLIST_IS_EMPTY(pxDelayed_Task_List) != pdFALSE)
{
// 设置下个任务解除阻塞时间为最大值,也即永不解除阻塞
xNextTaskUnblockTime = portMAX_DELAY;
break;
}
else
{
// 依次获取延时阻塞链表头节点
pxTCB=(TCB_t *)listGET_OWNER_OF_HEAD_ENTRY(pxDelayed_Task_List);
// 依次获取延时阻塞链表中所有节点解除阻塞的时间
xItemValue = listGET_LIST_ITEM_VALUE(&(pxTCB->xStateListItem));
// 当阻塞链表中所有延时到期的任务都被移除则跳出 for 循环
if(xConstTickCount < xItemValue)
{
xNextTaskUnblockTime = xItemValue;
break;
}
// 将任务从延时列表移除,消除等待状态
(void)uxListRemove(&(pxTCB->xStateListItem));
// 将解除等待的任务添加到就绪列表
prvAddTaskToReadyList(pxTCB);
#if(configUSE_PREEMPTION == 1)
// 如果解除阻塞状态的任务优先级比当前任务优先级高,则需要进行任务调度
if(pxTCB->uxPriority >= pxCurrentTCB->uxPriority)
{
xSwitchRequired = pdTRUE;
}
#endif
}
}
}
return xSwitchRequired;
}
/* task.h */
// 修改函数声明
BaseType_t xTaskIncrementTick(void);
/* FreeRTOSConfig.h */
// 支持抢占优先级
#define configUSE_PREEMPTION 1
无其他变化,只是将任务切换从函数体内修改到函数体外 。
/* port.c */
// SysTick 中断
void xPortSysTickHandler(void)
{
// 关中断
vPortRaiseBASEPRI();
// 更新系统时基
if(xTaskIncrementTick() != pdFALSE)
{
taskYIELD();
}
// 开中断
vPortSetBASEPRI(0);
}
与 FreeRTOS简单内核实现6 优先级 文章中 "3.1、测试" 小节内容一致 。
如果使用的开发环境为 Keil 且程序工作不正常,可以勾选 Use MicroLIB 试试,如下图所示 。
当前 RTOS 简单内核已实现的功能有 。
当前 RTOS 简单内核存在的缺点有 。
最后此篇关于FreeRTOS简单内核实现7阻塞链表的文章就讲到这里了,如果你想了解更多关于FreeRTOS简单内核实现7阻塞链表的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
总的来说,我对 Linux 内核和操作系统非常感兴趣。我想知道的是,内核的文件类型或扩展名是什么?它显然没有 .exe 或 .out 扩展名,因为它们用于安装在操作系统上的应用程序。 内核只是一个二进
我需要为 Raspbian Linux 内核添加一个自己的系统调用。现在我在搜索了大约 2 天以找到解决方案后陷入困境。 要加一个系统调用,我基本上是按照大纲来的( http://elinux.org
对于一个学术项目,我希望将源文件 (myfile.c) 添加到 kernel/目录,与exit.c相同的目录和 fork.c .构建系统似乎不会自动获取新文件,因为我在 myfile.c 中定义的函数
浏览器排行榜 浏览器市占率排行榜全球榜 。 浏览器市占率排行榜中国榜 -快科技 。 如果按照浏览器内核来看, Chromium 内核的市场占有率无疑是最大的,一家独大
给定一个进程或线程的任务结构,迭代属于同一进程的所有其他线程的习惯用法是什么? 最佳答案 Linux 不区分进程(任务)和线程。库调用 fork() 和 pthread_create() 使用相同的系
我正在用c(不是linux。完全从头开始)从头开始制作一个内核,但我遇到了一些问题。我有这个代码: #include "timer.h" int ms = 0; void timer_handler(
我正在从头开始制作一个 C 内核,我实际上只是从网站上复制了这段代码,因为我的代码无法工作,所以我很困惑。 void kmain(void) { const char *str = "my f
我不确定,如果我完全理解上述差异,所以我想自己解释一下,你可以打断我,只要我有错:“内核是创建内核线程的初始代码段。内核线程是由内核管理的进程。用户线程是进程的一部分。如果你有一个单线程进程,那么整个
看一下struct file 定义from this code Linux 内核版本 2.6.18。 我正在尝试比较代码中的两个 struct file 变量,并确定它们是否指的是同一个文件。该结构中
我试图在 Linux 启动时使嵌入式设备中的 LED 闪烁。基本上,LED 闪烁表明 Linux 正在启动。为了使 LED 闪烁,我正在做以下事情 在 init/main.c 中创建了一个全局定时器(
我有一些在 FreeBSD 和 Linux 上运行的特定硬件。 我必须做一个用户空间应用程序,它将使用内核/用户空间应用程序之间的共享内存与驱动程序一起工作。我的应用程序对来自用户空间的共享内存进行忙
我在哪里可以找到 linux 内核中相应函数的解释,特别是对于 ICMPv4? 例如:icmp_reply、icmp_send等 感谢您的帮助。 最好的,阿里木 最佳答案 探索 Linux 内核中的
我在 Linux Kernel 3.4 上工作,我有以下代码: /* Proximity sensor calibration values */ unsigned int als_kadc;
我正在阅读“罗伯特·洛夫 (Robert Love) 撰写的 Linux 内核开发第 3 版”,以大致了解 Linux 内核的工作原理..(2.6.2.3) 我对等待队列的工作方式感到困惑,例如这段代
我之前也问过同样的问题,但是我的帖子不知为何被删除了。 无论如何,我正在尝试使用 C++ 并编写一个允许我直接访问内存并向其中写入内容的程序。我听说我需要对内核做一些事情,因为它是连接操作系统和应用程
在尝试了解 Ruby 执行方法时,我找到了这篇关于在 Ruby 中运行命令的五种方法的博文 http://mentalized.net/journal/2010/03/08/5_ways_to_run
是否有 Linux 发行版(Minix 除外)包含良好的源代码文档?或者,是否有一些好的文档来描述一般的 Linux 源代码? 我已经下载了内核源代码,但是(不出所料)我有点不知所措,我想知道是否有一
有谁知道 linux 中的哪个函数或文件包含查找用于 bind() 系统调用的随机端口的算法?我到处寻找,在 Linux 源代码中找不到包含此算法的方法。 谢谢! 最佳答案 这是一段又长又复杂的代码,
前言 首先,对于有科班背景的读者,可以跳过本系列文章。这些文章的主要目的是通过简单易懂的汇总,帮助非科班出身的读者理解底层知识,进一步了解为什么在面试中会涉及这些底层问题。否则,某些概念将始终
CentOS7.2与CentOS6区别及特点 Linux 操作系统的启动首先从 BIOS 开始,接下来进入 boot loader,由 bootloader 载入内核,进行内核初始化。内核初始化的
我是一名优秀的程序员,十分优秀!