Linux kernel 中的work queue原理

 

 

分成两大部分,第一部分是用来执行work queue中每个节点上挂载的函数的内核线程,第二部分是从驱动程序的角度看work queue的使用。

第一部分 worker_thread内核线程
Linux系统启动期间会创建一名为worker_thread线程,该线程创建之后就处于sleep状态。这里所谓的内核线程,从调度器的角度就是一可以调度的进程,从代码的表现形式看,就是一函数。系统创建的这个worker_thread线程基于一workqueue_struct结构变量上(该结构体变量的成员name为"events").

第二部分 work queue的使用
1.只考虑使用系统的keventd管理的工作队列
驱动程序调用schedule_work向工作队列递交新的工作节点,schedule_work内部会唤醒worker_thread内核线程(使之进程状态为可调度)。在下一次进程调度时刻,worker_thread被调度执行,其主要任务便是调用它所管理工作队列中每个工作节点上挂载的函数,调用完毕该工作节点会从任务队列中被删除。当所有节点上的函数调用完毕,worker_thread继续sleep,直到schedule_work再次被某个驱动程序调用。
与使用驱动程序自己创建的工作对列的区别是:schedule_work内部是调用queue_work(keventd_wq, work),而使用驱动程序自己创建的工作队列在调用queue_work时的第一个参数是驱动程序自己创建的工作队列。

2.驱动程序使用自己创建的工作队列
这种情况驱动程序调用create_workqueue。该函数的原理跟1中基本是一样的,只不过再会创建一个内核进程,该内核进程的内核函数名字依然为worker_thread,只不过这个worker_thread工作在新的属于驱动程序自己的工作队列。
使用方法是:
a. 调用create_workqueue生成属于驱动程序自己的work queue. struct workqueue_struct *my_workqueue = create_workqueue("my_workqueue");
b.调用queue_work象a中生成的my_workqueue中注册工作节点, queue_work(my_workqueue, work)

这两种情况下的内核线程其实都是利用了一个名为kthreadd的内核线程来创建工作队列的worker_thread,其本质是向全局列表kthread_create_list中加入相应的内核线程函数节点,由kthreadd来负责创建进程框架,然后运行kthread_create_list中加入的节点上的内核线程函数。
向kthread_create_list中每加入一个内核线程函数节点,都会唤醒kthreadd线程。

当等待队列是空时,worker_thread会睡眠在该等待队列中。当driver调用queue_work时,再向等待队列中加入工作节点的同时,会唤醒睡眠在该等待队列上的worker_thread线程。

这个问题可能是你想问的是 queue_delayed_work 函数,它是 Linux 内核定义的一个函数,用于向工作队列添加一个延迟工作(delayed work)。 queue_delayed_work 函数需要传入三个参数,分别是指向工作队列的指针、指向延迟工作的指针以及延迟时间。当指定的延迟时间到达后,内核会自动将指定的延迟工作添加到指定的工作队列,等待执行。 下面是一个简单的示例: ```c #include <linux/delay.h> #include <linux/init.h> #include <linux/kernel.h> #include <linux/module.h> #include <linux/workqueue.h> MODULE_LICENSE("GPL"); // 定义工作队列 static struct workqueue_struct *my_wq; // 定义延迟工作结构体 static struct delayed_work my_work; // 延迟工作函数 static void work_handler(struct work_struct *work) { printk(KERN_INFO "Delayed work executed!\n"); } // 模块加载函数 static int __init my_module_init(void) { printk(KERN_INFO "Module loaded!\n"); // 创建工作队列 my_wq = create_workqueue("my_work_queue"); // 初始化延迟工作结构体 INIT_DELAYED_WORK(&my_work, work_handler); // 延迟5秒执行工作 queue_delayed_work(my_wq, &my_work, msecs_to_jiffies(5000)); return 0; } // 模块卸载函数 static void __exit my_module_exit(void) { // 取消延迟工作 cancel_delayed_work(&my_work); // 销毁工作队列 destroy_workqueue(my_wq); printk(KERN_INFO "Module unloaded!\n"); } module_init(my_module_init); module_exit(my_module_exit); ``` 在这个示例,我们首先创建了一个名为my_wq的工作队列,然后定义了一个名为my_work的延迟工作结构体,并且定义了一个工作函数work_handler。在模块加载函数,我们使用queue_delayed_work函数将my_work延迟5秒后添加到my_wq工作队列。在模块卸载函数,我们使用cancel_delayed_work函数取消my_work的执行,并且销毁了my_wq工作队列。当模块加载时,内核会输出"Delayed work executed!",表示延迟工作已经被执行。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值