函数 INIT_WORK() / schedule_work()
来源:互联网 发布:mac系统测试软件 编辑:程序博客网 时间:2024/06/05 04:04
1. 初始化工作队列INIT_WORK();
2. 调度工作队列 schedule_work();
转自http://blog.csdn.net/xiaopohaibebo/article/details/45648089
工作队列一般用来做滞后的工作,比如在中断里面要做很多事,但是比较耗时,这时就可以把耗时的工作放到工作队列。说白了就是系统延时调度的一个自定义函数。
1、定义struct work_struct irq_queue;
2、初始化INIT_WORK(&irq_queue,do_irq_queuework);
3、调用方法:schedule_work(&rq_queue);
注,调用完毕后系统会释放此函数,所以如果想再次执行的话,就再次调用schedule_work()即可。
另外,内核必须挂载文件系统才可以使用工作队列。我的理解是:工作队列也属于调度,如果内核挂了,他就不调度了,当然就不能用工作队列了。
Linux2.6内核使用了不少工作队列来处理任务,他在使用上和 tasklet最大的不同是工作队列的函数可以使用休眠,而tasklet的函数是不允许使用休眠的。
工作队列的使用又分两种情况,一种是利用系统共享的工作队列来添加自己的工作,这种情况处理函数不能消耗太多时间,这样会影响共享队列中其他任务的处理;另外一种是创建自己的工作队列并添加工作。
(一)利用系统共享的工作队列添加工作:
第一步:声明或编写一个工作处理函数
void my_func();
第二步:创建一个工作结构体变量,并将处理函数和参数的入口地址赋给这个工作结构体变量
DECLARE_WORK(my_work,my_func,&data); //编译时创建名为my_work的结构体变量并把函数入口地址和参数地址赋给它;
如果不想要在编译时就用DECLARE_WORK()创建并初始化工作结构体变量,也可以在程序运行时再用INIT_WORK()创建
struct work_struct my_work; //创建一个名为my_work的结构体变量,创建后才能使用INIT_WORK()
INIT_WORK(&my_work,my_func,&data); //初始化已经创建的my_work,其实就是往这个结构体变量中添加处理函数的入口地址和data的地址,通常在驱动的open函数中完成
第三步:将工作结构体变量添加入系统的共享工作队列
schedule_work(&my_work); //添加入队列的工作完成后会自动从队列中删除
或schedule_delayed_work(&my_work,tick); //延时tick个滴答后再提交工作
(二)创建自己的工作队列来添加工作
第一步:声明工作处理函数和一个指向工作队列的指针
void my_func();
struct workqueue_struct *p_queue;
第二步:创建自己的工作队列和工作结构体变量(通常在open函数中完成)
p_queue=create_workqueue("my_queue"); //创建一个名为my_queue的工作队列并把工作队列的入口地址赋给声明的指针
struct work_struct my_work;
INIT_WORK(&my_work, my_func, &data); //创建一个工作结构体变量并初始化,和第一种情况的方法一样
第三步:将工作添加入自己创建的工作队列等待执行
queue_work(p_queue, &my_work);
//作用与schedule_work()类似,不同的是将工作添加入p_queue指针指向的工作队列而不是系统共享的工作队列
第四步:删除自己的工作队列
destroy_workqueue(p_queue); //一般是在close函数中删除
内核中,有关函数的定义说明如下,
1. INIT_WORK()
#define INIT_WORK(_work, _func)\do {\__INIT_WORK((_work), (_func), 0);\} while (0)
/* * initialize all of a work item in one go * * NOTE! No point in using "atomic_long_set()": using a direct * assignment of the work data initializer allows the compiler * to generate better code. */#ifdef CONFIG_LOCKDEP#define __INIT_WORK(_work, _func, _onstack)\do {\static struct lock_class_key __key;\\__init_work((_work), _onstack);\(_work)->data = (atomic_long_t) WORK_DATA_INIT();\lockdep_init_map(&(_work)->lockdep_map, #_work, &__key, 0); \INIT_LIST_HEAD(&(_work)->entry);\PREPARE_WORK((_work), (_func));\} while (0)#else#define __INIT_WORK(_work, _func, _onstack)\do {\__init_work((_work), _onstack);\(_work)->data = (atomic_long_t) WORK_DATA_INIT();\INIT_LIST_HEAD(&(_work)->entry);\PREPARE_WORK((_work), (_func));\} while (0)#endif
void __init_work(struct work_struct *work, int onstack){if (onstack)debug_object_init_on_stack(work, &work_debug_descr);elsedebug_object_init(work, &work_debug_descr);}
2. schedule_work()
/** * schedule_work - put work task in global workqueue * @work: job to be done * * Returns %false if @work was already on the kernel-global workqueue and * %true otherwise. * * This puts a job in the kernel-global workqueue if it was not already * queued and leaves it in the same position on the kernel-global * workqueue otherwise. */static inline bool schedule_work(struct work_struct *work){return queue_work(system_wq, work);}
- 函数 INIT_WORK() / schedule_work()
- INIT_WORK
- INIT_DELAYED_WORK和INIT_WORK定时器中断函数
- 工作队列 schedule_delayed_work/schedule_work_on/schedule_work函数内核实现
- schedule_work 工作队列
- schedule_work 工作队列
- schedule_work 工作队列
- linux 工作队列之schedule_work
- INIT_WORK的暂时理解
- INIT_WORK宏定义解释
- 对INIT_WORK的理解
- linux INIT_WORK 工作队列
- INIT_WORK和INIT_DELAYED_WORK详解
- 对INIT_WORK的理解
- 对INIT_WORK的理解
- INIT_WORK和INIT_DELAYED_WORK详解
- 对INIT_WORK的理解
- INIT_WORK和INIT_DELAYED_WORK详解
- 写作
- C# Array
- 认识html标签(第三部分)
- CentOS 安装svn服务And eclipse安装svn
- 2016苹果开发者账号注册申请流程链接
- 函数 INIT_WORK() / schedule_work()
- 机房收费系统——报表的使用
- 重启/关闭Windows
- 上传图片压缩工具类
- Redis搭建(二)
- java.lang.LinkageError: loader constraint violation
- 活动Activity的栈的存储方式
- GCov交叉编译覆盖率测试要点
- SSL,SSH,OpenSSL,OpenSSH,HTTPS的区别