RT-Thread workqueue 详解

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了RT-Thread workqueue 详解相关的知识,希望对你有一定的参考价值。

参考技术A

在学习之前可以先去了解一下工作队列的使用场景: 工作队列 ( workqueue ) 。

简而言之,工作队列就是将一些工作任务的执行延迟,交由内核线程异步执行。

最简单的使用方式就是开启 RT-Thread 的系统工作线程(System workqueue),而我们往系统工作线程里提交工作项(work item)即可。

RT-Thread 其实给我们提供了一个系统工作线程了,但很少有人知道。配置选项路径如下图所示:

依次选中上述这些选项,就能够开启系统工作队列了。而且还可以看到工作队列线程的栈大小默认为 2048,优先级为 23 。

这样系统在初始化的时候就创建了系统工作队列了,名字叫作 sys_work ,在终端输入 ps 能够看到该线程。

如何向系统工作线程里添加工作项呢?

rt_work_submit() 用于向系统工作队列添加工作项, rt_work_cancel() 用于从系统工作队列中取消某一个工作项。

当然,在提交工作项时,需要初始化该工作项,绑定相应的回调函数和用户指针,接口如下:

这样,我们就可以随时随地地提交工作任务执行了,极大地方便了程序的组织。

用一个小例程测试一下:

在 qemu 项目里的 main.c 里输入:

然后执行就能看到下述效果,与工作项绑定的任务被异步执行了,而且工作项 1 延迟了 2 个 tick 才执行。

rt_workqueue 的接口有很多,我们只需要关注常用的即可。

首先使用 rt_workqueue_create() 创建一个工作队列,然后使用 rt_workqueue_submit_work() 提交工作项,使用 rt_workqueue_cancel_work() 取消工作项,当然还可以使用 rt_workqueue_destroy() 销毁一个工作队列。其他的接口有兴趣的可以了解,但常用的就是上面这四种。

这里提交任务与上述使用系统工作队列的唯一不同之处就是我们需要手动指定工作队列,其他的都是一模一样的。

用一个小例程测试一下:

在 qemu 项目里的 main.c 里输入:

然后执行就能看到下述效果,与工作项绑定的任务被异步执行了,而且工作项 1 延迟了 2 个 tick 才执行。

关于实现部分我这里不介绍具体细节,做了一些动画给大家展示一下内部过程

工作队列里面有一个线程(workthread),这个线程的任务就是不断地从挂载链表(worklist)里提取工作项执行,若没有则休眠。

然后提交工作项时,若延迟时间 time 大于 0,则启动该工作项的定时器,定时结束后再加入挂载链表(worklist)。

若提交工作项时延迟实际等于 0,则直接将该工作项挂加入到挂载链表(worklist)。

当然,工作项的定时器超时后,会自动将该工作项加入到挂载链表(worklist)。

以上是关于RT-Thread workqueue 详解的主要内容,如果未能解决你的问题,请参考以下文章

__attribute__之section详解 ------ 把函数指定到具体某个section 之 RT-thread 实例详解

RT-Thread快速入门-初探RT-Thread

RT-Thread

RT-Thread RTOS的RT-Thread / uCOS / FreeRTOS 简单比较

RT-Thread 内核学习笔记 - 设备模型rt_device的理解

RT-Thread 应用笔记 - RTC Alarm 组件的使用