当前位置: 首页 > news >正文

Linux内核 -- INIT_WORK 使用与注意事项

Linux内核 – INIT_WORK 使用与注意事项

一、概述

在 Linux 内核中,workqueue(工作队列)机制用于将任务从中断上下文中异步转移到进程上下文中执行,降低中断处理负担。INIT_WORK 是工作队列机制的核心之一,用于初始化普通工作项(struct work_struct)。


二、INIT_WORK 简介

#define INIT_WORK(_work, _func) \__INIT_WORK((_work), (_func), 0)

INIT_WORK() 用于初始化一个工作项 struct work_struct 并指定回调函数 _func,此工作项随后可通过 schedule_work()queue_work() 提交到默认或指定工作队列中执行。


三、使用方法

3.1 示例代码

#include <linux/module.h>
#include <linux/init.h>
#include <linux/workqueue.h>static struct work_struct my_work;static void work_handler(struct work_struct *work)
{pr_info("Workqueue executed in process context\n");
}static int __init my_module_init(void)
{INIT_WORK(&my_work, work_handler);schedule_work(&my_work);pr_info("Work scheduled\n");return 0;
}static void __exit my_module_exit(void)
{cancel_work_sync(&my_work);pr_info("Module exited cleanly\n");
}module_init(my_module_init);
module_exit(my_module_exit);
MODULE_LICENSE("GPL");

3.2 关键 API 说明

接口说明
INIT_WORK()初始化普通工作项
schedule_work()提交工作项至默认队列 system_wq
queue_work(wq, w)提交到自定义队列
cancel_work_sync()等待工作项完成或取消
flush_work()等待工作项执行完毕(不适合模块卸载)
work_pending()检查工作项是否已调度

四、注意事项

4.1 同一 work_struct 不能重复调度

  • 若某个工作项已被调度但尚未执行完毕,再次调用 queue_work() 无效
  • work_struct 结构中有一个 PENDING 位标志,防止被重复添加至队列。
if (!work_pending(&my_work))queue_work(wq, &my_work);

4.2 回调函数中可重新调度自身

static void my_work_fn(struct work_struct *work)
{// do task ...queue_work(my_wq, work);  // 再次调度
}

适用于链式任务或轮询逻辑。但要加条件判断,避免死循环。

4.3 模块卸载前必须同步取消工作项

必须使用 cancel_work_sync() 来确保:

  • 若任务在队列中,则等待其完成或取消;
  • 避免模块卸载后,工作函数引用释放内存导致 UAF。

4.4 多个任务建议使用多个 work_struct 实例

#define MAX_WORKS 8
struct work_struct works[MAX_WORKS];for (int i = 0; i < MAX_WORKS; i++) {INIT_WORK(&works[i], my_work_fn);queue_work(my_wq, &works[i]);
}

避免同一任务被并发调度失败的问题。


五、进阶建议

5.1 使用 delayed_work 实现周期调度

struct delayed_work my_delayed_work;
INIT_DELAYED_WORK(&my_delayed_work, my_delayed_fn);
schedule_delayed_work(&my_delayed_work, msecs_to_jiffies(1000));

适合定时任务、轮询扫描、状态检查等场景。

5.2 创建自定义工作队列

struct workqueue_struct *my_wq;
my_wq = alloc_workqueue("my_queue", WQ_UNBOUND | WQ_HIGHPRI, 1);
queue_work(my_wq, &my_work);

卸载时使用 destroy_workqueue(my_wq);


六、调试建议

工具用途
ftrace跟踪工作函数调度和执行
cat /proc/workqueue查看工作队列状态
kmemleak检查 work_struct 生命周期泄漏

七、总结

问题是否允许
上一次工作执行完毕,再调度✅ 允许
上一次工作还在队列中❌ 不会调度
上一次工作正在执行❌ 不会调度
回调函数中再次调度自己✅ 可行
多个并发任务使用同一个 work_struct❌ 不推荐,需多个实例

八、参考建议

  • 避免 work_struct UAF,使用 cancel_work_sync 配合模块生命周期;
  • 使用 INIT_DELAYED_WORK 替代 INIT_WORK 实现定时、周期性处理;
  • 调试期间可加上 WARN_ON(work_pending(...)) 避免重复调度。

如需进一步深入,可以参考:

  • 《Linux Kernel Development》(Robert Love)
  • 《Linux Device Drivers, Third Edition》
  • Linux 源码目录:kernel/workqueue.c, include/linux/workqueue.h
http://www.xdnf.cn/news/966871.html

相关文章:

  • Windows 文件路径与文件名限制
  • 如何根据excel表生成sql的insert脚本
  • ABP vNext + Hive 集成:多租户大数据 SQL 查询与报表分析
  • 【iOS】cell的复用以及自定义cell
  • 使用NNI剪枝工具对VGG16网络进行剪枝,同时使用知识蒸馏对剪枝后结果进行优化。(以猫狗二分类为例)
  • 认证与授权的区别与联系
  • 看板任务描述不清如何解决
  • 数据库学习笔记(十五)--变量与定义条件与处理程序
  • 云蝠智能大模型语音智能体:构建心理咨询领域的智能助手
  • leetcode1034. 边界着色-medium
  • 使用mpu6500, PID,互补滤波实现一个简单的飞行自稳控制系统
  • 南昌市新建区委书记陈奕蒙会见深兰科技集团董事长陈海波一行
  • 如何使用 DeepSeek 帮助自己的工作
  • 机械制造系统中 PROFINET 与 PROFIBUS-DP 的融合应用及捷米科技解决方案
  • Matlab点云合并函数pcmerge全解析
  • 线程与协程
  • Prometheus + Grafana 监控 RabbitMQ 实践指南
  • Spring Boot 分层架构与数据流转详解
  • Word中如何对文献应用的格式数字连起来,如:【1-3】
  • 如何看容器的ip地址
  • 每日收获总结20250610
  • 循环结构使用
  • Java 通用实体验证框架:从业务需求到工程化实践【生产级 - 适用于订单合并前置校验】
  • B2B供应链交易平台多商户电商商城系统开发批发采购销售有哪些功能?发展现状如何?
  • 什么是库存周转?如何用进销存系统提高库存周转率?
  • 第五章 GPIO示例
  • PennyLane 是一个用于量子计算、量子机器学习和量子化学的跨平台 Python 库。由研究人员构建,用于研究
  • 向量数据库ChromaDB的使用
  • Vim 复制/剪切/粘贴命令完整学习笔记
  • java Condition类