Linux 工作队列(Workqueue):概念与实现
目录
- 一、工作队列的概念
- 1.1 什么是工作队列
- 1.2 为什么使用工作队列
- 二、工作队列的实现
- 2.1 定义和初始化工作队列
- 2.2 工作队列API
- 三、工作队列的应用
- 3.1 延迟执行任务
- 3.2 处理复杂的中断任务
- 四、工作队列的类型
- 4.1 普通工作队列
- 4.2 高优先级工作队列
- 五、总结
在Linux内核中,工作队列(Workqueue)是一种用于延迟任务执行的机制。它允许内核代码在可中断上下文中执行任务,这对于需要在进程上下文中运行或者需要延迟执行的任务非常有用。在本文中,我们将详细介绍工作队列的概念、实现以及实际应用。
一、工作队列的概念
1.1 什么是工作队列
工作队列(Workqueue)是Linux内核提供的一种机制,用于将任务推迟到将来某个时间点在进程上下文中执行。与软中断和任务队列不同,工作队列允许任务在进程上下文中运行,这意味着它们可以睡眠和使用所有的内核API。
1.2 为什么使用工作队列
使用工作队列的主要原因是一些任务需要在中断上下文之外执行。中断处理程序通常需要尽可能快地完成,以避免延迟其他中断的处理。然而,有些任务需要较长的时间来完成,或者需要使用可能会睡眠的内核API。在这种情况下,可以使用工作队列将这些任务推迟到稍后在进程上下文中执行。
二、工作队列的实现
2.1 定义和初始化工作队列
在Linux内核中,工作队列由struct workqueue_struct
表示,工作项由struct work_struct
表示。你可以使用create_workqueue
函数创建一个新的工作队列,并使用INIT_WORK
宏初始化一个工作项。
struct workqueue_struct *my_wq;
struct work_struct my_work;void my_work_function(struct work_struct *work) {printk(KERN_INFO "Workqueue: Task executed\n");
}static int __init my_module_init(void) {my_wq = create_workqueue("my_workqueue");if (my_wq) {INIT_WORK(&my_work, my_work_function);queue_work(my_wq, &my_work);}return 0;
}static void __exit my_module_exit(void) {flush_workqueue(my_wq);destroy_workqueue(my_wq);
}module_init(my_module_init);
module_exit(my_module_exit);
2.2 工作队列API
以下是一些常用的工作队列API:
create_workqueue
:创建一个新的工作队列。destroy_workqueue
:销毁工作队列。INIT_WORK
:初始化一个工作项。queue_work
:将工作项添加到工作队列中。flush_workqueue
:等待工作队列中的所有工作项完成。cancel_work_sync
:取消一个工作项的执行。
三、工作队列的应用
3.1 延迟执行任务
工作队列非常适合需要延迟执行的任务。例如,在驱动程序中,可以使用工作队列来处理需要延迟执行的任务,如处理硬件中断或定期检查设备状态。
示例:延迟执行任务
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/workqueue.h>
#include <linux/delay.h>static struct workqueue_struct *my_wq;
static struct delayed_work my_delayed_work;void my_delayed_work_function(struct work_struct *work) {printk(KERN_INFO "Delayed Workqueue: Task executed after delay\n");
}static int __init my_module_init(void) {my_wq = create_workqueue("my_delayed_workqueue");if (my_wq) {INIT_DELAYED_WORK(&my_delayed_work, my_delayed_work_function);queue_delayed_work(my_wq, &my_delayed_work, msecs_to_jiffies(5000)); // 延迟5秒执行}return 0;
}static void __exit my_module_exit(void) {cancel_delayed_work_sync(&my_delayed_work);destroy_workqueue(my_wq);
}module_init(my_module_init);
module_exit(my_module_exit);MODULE_LICENSE("GPL");
MODULE_AUTHOR("OpenAI GPT-4");
MODULE_DESCRIPTION("A simple example of using workqueue with delayed execution in Linux kernel.");
3.2 处理复杂的中断任务
在中断处理程序中,我们通常尽量减少执行时间,以免阻塞其他中断。对于需要较长时间处理的任务,可以使用工作队列来延迟执行。
示例:在中断处理程序中使用工作队列
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/workqueue.h>static struct workqueue_struct *my_wq;
static struct work_struct my_work;irqreturn_t my_interrupt_handler(int irq, void *dev_id) {queue_work(my_wq, &my_work);return IRQ_HANDLED;
}void my_work_function(struct work_struct *work) {printk(KERN_INFO "Workqueue: Handling interrupt task\n");
}static int __init my_module_init(void) {int irq = 1; // 假设使用 IRQ 1my_wq = create_workqueue("my_interrupt_workqueue");if (my_wq) {INIT_WORK(&my_work, my_work_function);request_irq(irq, my_interrupt_handler, IRQF_SHARED, "my_interrupt_handler", NULL);}return 0;
}static void __exit my_module_exit(void) {int irq = 1; // 假设使用 IRQ 1free_irq(irq, NULL);flush_workqueue(my_wq);destroy_workqueue(my_wq);
}module_init(my_module_init);
module_exit(my_module_exit);MODULE_LICENSE("GPL");
MODULE_AUTHOR("OpenAI GPT-4");
MODULE_DESCRIPTION("A simple example of using workqueue in interrupt handler in Linux kernel.");
四、工作队列的类型
Linux内核提供了两种类型的工作队列:
4.1 普通工作队列
普通工作队列使用默认的系统工作队列来执行任务。示例中的create_workqueue
就是创建一个普通工作队列。
4.2 高优先级工作队列
在某些情况下,需要更高优先级的工作队列来执行关键任务。Linux内核提供了高优先级工作队列,可以通过alloc_workqueue
函数创建。
示例:创建高优先级工作队列
static struct workqueue_struct *my_highpri_wq;static int __init my_module_init(void) {my_highpri_wq = alloc_workqueue("my_highpri_workqueue", WQ_HIGHPRI, 0);if (my_highpri_wq) {INIT_WORK(&my_work, my_work_function);queue_work(my_highpri_wq, &my_work);}return 0;
}static void __exit my_module_exit(void) {flush_workqueue(my_highpri_wq);destroy_workqueue(my_highpri_wq);
}
五、总结
工作队列(Workqueue)是Linux内核中用于延迟执行任务的一种机制。它允许任务在进程上下文中运行,使得任务可以睡眠并使用所有的内核API。本文详细介绍了工作队列的概念、实现以及实际应用,并提供了多个示例代码。通过这些知识和示例,相信你能够在内核开发中更好地使用工作队列来管理延迟任务。
相关文章:
Linux 工作队列(Workqueue):概念与实现
目录 一、工作队列的概念1.1 什么是工作队列1.2 为什么使用工作队列 二、工作队列的实现2.1 定义和初始化工作队列2.2 工作队列API 三、工作队列的应用3.1 延迟执行任务3.2 处理复杂的中断任务 四、工作队列的类型4.1 普通工作队列4.2 高优先级工作队列 五、总结 在Linux内核中…...
前端页面是如何禁止被查看源码、被下载,被爬取,以及破解方法
文章目录 1.了解禁止查看,爬取原理1.1.JS代码,屏蔽屏蔽键盘和鼠标右键1.2.查看源码时,通过JS控制浏览器窗口变化2.百度文库是如何防止抓包2.1.HTPPS2.2. 动态加载为什么看不到?如何查看动态加载的内容?3.禁止复制,如果解决3.1.禁止复制原理3.2.如何破解1.了解禁止查看,爬…...

51单片机嵌入式开发:14、STC89C52RC 之HX1838红外解码NEC+数码管+串口打印+LED显示
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 STC89C52RC 之HX1838红外解码NEC数码管串口打印LED显示 STC89C52RC 之HX1838红外解码NEC数码管串口打印LED显示1 概述2 硬件电路2.1 遥控器2.2 红外接收器电路2.3 STC89C52单…...
在不同环境中,Java应用程序和MySQL等是如何与Docker进行交互和操作的?
1. 本地开发环境 在本地开发环境中,可以使用Docker Compose来管理和运行Java应用程序容器和MySQL容器。通常,会创建一个docker-compose.yml文件,定义需要的服务及其配置。 以下是一个示例docker-compose.yml文件: version: 3 services:app…...

《DRL》P10-P15-损失函数-优化(梯度下降和误差的反向传播)
文章目录 损失函数交叉熵损失多类别分类任务概述真实标签的独热编码交叉熵损失函数 L p 范式 \mathcal{L}_{p}\text{ 范式} Lp 范式均方误差平均绝对误差 优化梯度下降和误差的反向传播 简介 本文介绍了神经网络中的损失函数及其优化方法。损失函数用于衡量模型预测值与真实值…...

Spring Boot项目的404是如何发生的
问题 在日常开发中,假如我们访问一个Sping容器中并不存在的路径,通常会返回404的报错,具体原因是什么呢? 结论 错误的访问会调用两次DispatcherServlet:第一次调用无法找到对应路径时,会给Response设置一个…...

<数据集>手势识别数据集<目标检测>
数据集格式:VOCYOLO格式 图片数量:2400张 标注数量(xml文件个数):2400 标注数量(txt文件个数):2400 标注类别数:5 标注类别名称:[fist, no_gesture, like, ok, palm] 序号类别名称图片数框数1fist597…...
【Vue3】选项式 API
【Vue3】选项式 API 背景简介开发环境开发步骤及源码总结 背景 随着年龄的增长,很多曾经烂熟于心的技术原理已被岁月摩擦得愈发模糊起来,技术出身的人总是很难放下一些执念,遂将这些知识整理成文,以纪念曾经努力学习奋斗的日子。…...

2、如何发行自己的数字代币(truffle智能合约项目实战)
2、如何发行自己的数字代币(truffle智能合约项目实战) 1-Atom IDE插件安装2-truffle tutorialtoken3-tutorialtoken源码框架分析4-安装openzeppelin代币框架(代币发布成功) 1-Atom IDE插件安装 正式介绍基于web的智能合约开发 推…...

百日筑基第二十三天-23种设计模式-创建型总汇
百日筑基第二十三天-23种设计模式-创建型总汇 前言 设计模式可以说是对于七大设计原则的实现。 总体来说设计模式分为三大类: 创建型模式,共五种:单例模式、简单工厂模式、抽象工厂模式、建造者模式、原型模式。结构型模式,共…...

张量的基本使用
目录 1.张量的定义 2.张量的分类 3.张量的创建 3.1 根据已有数据创建张量 3.2 根据形状创建张量 3.3 创建指定类型的张量 1.张量的定义 张量(Tensor)是机器学习的基本构建模块,是以数字方式表示数据的形式。PyTorch就是将数据封装成张量…...
Oracle(14)什么是唯一键(Unique Key)?
唯一键(Unique Key)是数据库表中的一个或多个列,它们的值必须在整个表中唯一,但允许包含NULL值。唯一键的主要目的是确保表中每一行的数据在指定的列(或列组合)中是唯一的,以防止重复数据的出现…...

PostgreSQL的引号、数据类型转换和数据类型
一、单引号和双引号(重要): 1、在mysql没啥区别 2、在pgsql中,实际字符串用单引号,双引号相当于mysql的,用来包含关键字; -- 单引号,表示user_name的字符串实际值 insert into t_user(user_nam…...
Mad MAD Sum-Codeforces Round 960 (Div. 2)
题目在这里 大意: MAD函数返回出现次数 ≥ 2 \geq2 ≥2的最大整数 b i b_i bi M A D ( a [ 1 , 2 , . . . i ] ) MAD(a[1,2,...i]) MAD(a[1,2,...i]) 每次操作把 a i a_i ai进行上述操作,直到全变为0为止,对每次操作的数组进行求和,记…...
Flutter 插件之 package_info_plus
当使用Flutter开发应用时,通常需要获取应用程序的基本信息,例如包名、版本号和构建号。Flutter提供了一个名为 package_info_plus 的插件,它能方便地帮助我们获取这些信息。 1. 添加依赖 首先,需要在项目的 pubspec.yaml 文件中添加 package_info_plus 的依赖。打开 pubs…...

如何实现布隆过滤器?
1.布隆过滤器的场景 在Redis 缓存击穿(失效)、缓存穿透、缓存雪崩怎么解决?中我们说到可以使用布隆过滤器避免「缓存穿透」。 你会说我们只要记录了每个用户看过的历史记录,每次推荐的时候去查询数据库过滤存在的数据实现去重。 …...

运维团队如何高效监控容器化环境中的PID及其他关键指标
随着云计算和容器化技术的快速发展,越来越多的企业开始采用容器化技术来部署和管理应用程序。然而,容器化环境的复杂性和动态性给运维团队带来了前所未有的挑战。本文将从PID(进程标识符)监控入手,探讨运维团队如何高效…...

通过vue3 + TypeScript + uniapp + uni-ui 实现下拉刷新和加载更多的功能
效果图: 核心代码: <script lang="ts" setup>import { ref, reactive } from vue;import api from @/request/api.jsimport empty from @/component/empty.vueimport { onLoad,onShow, onPullDownRefresh, onReachBottom } from @dcloudio/uni-applet form …...

Pointnet++改进即插即用系列:全网首发WTConv2d大接受域的小波卷积|即插即用,提升特征提取模块性能
简介:1.该教程提供大量的首发改进的方式,降低上手难度,多种结构改进,助力寻找创新点!2.本篇文章对Pointnet++特征提取模块进行改进,加入WTConv2d,提升性能。3.专栏持续更新,紧随最新的研究内容。 目录 1.理论介绍 2.修改步骤 2.1 步骤一 2.2 步骤二 2.3 步骤三 1.理…...

4核16G服务器支持多少人?4C16G服务器性能测评
租赁4核16G服务器费用,目前4核16G服务器10M带宽配置70元1个月、210元3个月,那么能如何呢?配置为ECS经济型e实例4核16G、按固定带宽10Mbs、100GB ESSD Entry系统盘。 那么问题来了,4C16G10M带宽的云服务器可以支持多少人同时在线&…...
基于算法竞赛的c++编程(28)结构体的进阶应用
结构体的嵌套与复杂数据组织 在C中,结构体可以嵌套使用,形成更复杂的数据结构。例如,可以通过嵌套结构体描述多层级数据关系: struct Address {string city;string street;int zipCode; };struct Employee {string name;int id;…...
React Native 导航系统实战(React Navigation)
导航系统实战(React Navigation) React Navigation 是 React Native 应用中最常用的导航库之一,它提供了多种导航模式,如堆栈导航(Stack Navigator)、标签导航(Tab Navigator)和抽屉…...

MongoDB学习和应用(高效的非关系型数据库)
一丶 MongoDB简介 对于社交类软件的功能,我们需要对它的功能特点进行分析: 数据量会随着用户数增大而增大读多写少价值较低非好友看不到其动态信息地理位置的查询… 针对以上特点进行分析各大存储工具: mysql:关系型数据库&am…...

【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器
——全方位测试解决方案与代码实战 一、工具定位与核心能力 DevEco Testing是HarmonyOS官方推出的一体化测试平台,覆盖应用全生命周期测试需求,主要提供五大核心能力: 测试类型检测目标关键指标功能体验基…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)
概述 在 Swift 开发语言中,各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过,在涉及到多个子类派生于基类进行多态模拟的场景下,…...
服务器硬防的应用场景都有哪些?
服务器硬防是指一种通过硬件设备层面的安全措施来防御服务器系统受到网络攻击的方式,避免服务器受到各种恶意攻击和网络威胁,那么,服务器硬防通常都会应用在哪些场景当中呢? 硬防服务器中一般会配备入侵检测系统和预防系统&#x…...
VTK如何让部分单位不可见
最近遇到一个需求,需要让一个vtkDataSet中的部分单元不可见,查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行,是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示,主要是最后一个参数,透明度…...

【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...

Unsafe Fileupload篇补充-木马的详细教程与木马分享(中国蚁剑方式)
在之前的皮卡丘靶场第九期Unsafe Fileupload篇中我们学习了木马的原理并且学了一个简单的木马文件 本期内容是为了更好的为大家解释木马(服务器方面的)的原理,连接,以及各种木马及连接工具的分享 文件木马:https://w…...
Android第十三次面试总结(四大 组件基础)
Activity生命周期和四大启动模式详解 一、Activity 生命周期 Activity 的生命周期由一系列回调方法组成,用于管理其创建、可见性、焦点和销毁过程。以下是核心方法及其调用时机: onCreate() 调用时机:Activity 首次创建时调用。…...