Chinaunix首页 | 论坛 | 博客
  • 博客访问: 134405
  • 博文数量: 20
  • 博客积分: 266
  • 博客等级: 二等列兵
  • 技术积分: 317
  • 用 户 组: 普通用户
  • 注册时间: 2012-11-28 14:15
文章存档

2013年(4)

2012年(16)

我的朋友

分类: LINUX

2012-11-28 17:10:18

在处理内核相关工作中, 我们经常看到工作队列(workqueue)的身影. 本文描述何为 linux workqueue.
本文基于 2.6.32 的内核, 此时的工作队列还不是 cmwq.

为什么使用 workqueue

在内核代码中, 经常希望延缓部分工作到将来某个时间执行, 这样做的原因很多, 比如

  • 在持有锁时做大量(或者说费时的)工作不合适.
  • 希望将工作聚集以获取批处理的性能.
  • 调用了一个可能导致睡眠的函数使得在此时执行新调度非常不合适.
  • ...

内核中提供了许多机制来提供延迟执行, 使用最多则是 workqueue.

  • 如中断的下半部处理可延迟中断上下文中的部分工作;
  • 定时器可指定延迟一定时间后执行某工作;
  • workqueue 则允许在进程上下文环境下延迟执行;
  • 内核中曾短暂出现过的慢工作机制 (slow work mechanism);
  • 异步函数调用(asynchronous function calls);
  • 各种私有实现的线程池;
  • ...
术语
  • workqueue: 所有工作项(需要被执行的工作)被排列于该队列.
  • worker thread: 是一个用于执行 workqueue 中各个工作项的内核线程, 当 workqueue 中没有工作项时, 该线程将变为 idle 状态.
  • single threaded(ST): worker thread 的表现形式之一, 在系统范围内, 只有一个 worker thread 为 workqueue 服务.
  • multi threaded(MT): worker thread 的表现形式之一, 在多 CPU 系统上每个 CPU 上都有一个 worker thread 为 workqueue 服务.
使用步骤
  1. 创建 workqueue(如果使用内核默认的 workqueue, 此步骤略过).
  2. 创建工作项 work_struct.
  3. 向 workqueue 提交工作项.
工作项

数据结构(略有调整):

struct work_struct {
atomic_long_t data;
struct list_head entry;
work_func_t func;
struct lockdep_map lockdep_map;
};
struct delayed_work {
struct work_struct work;
struct timer_list timer;
};

静态地创建工作项:

DECLARE_WORK(n, f)
DECLARE_DELAYED_WORK(n, f)

动态地创建工作项:

INIT_WORK(struct work_struct work, work_func_t func);
PREPARE_WORK(struct work_struct work, work_func_t func);
INIT_DELAYED_WORK(struct delayed_work work, work_func_t func);
PREPARE_DELAYED_WORK(struct delayed_work work, work_func_t func);
内核默认的 workqueue

查阅源码可知, 内核默认的全局 workqueue 应为:

// 定义
static struct workqueue_struct *keventd_wq __read_mostly;

// 初始化
...
keventd_wq = create_workqueue("events"); // MT worker thread 模式.
...

// 确认对应的内核线程数目, 应等于 CPU 核数
$ lscpu
Architecture: x86_64
CPU(s): 3
Thread(s) per core: 1
$ ps aux | grep "events"
root 9 0.0 0.0 0 0 ? S Feb09 1:15 [events/0]
root 10 0.0 0.0 0 0 ? S Feb09 0:59 [events/1]
root 11 0.0 0.0 0 0 ? S Feb09 0:59 [events/2]

工作项加入 keventd_wq 由下面两个函数之一完成:

int schedule_work(struct work_struct *work)
{
return queue_work(keventd_wq, work);
}
int schedule_delayed_work(struct delayed_work *dwork, unsigned long delay)
{
return queue_delayed_work(keventd_wq, dwork, delay);
}
用户自定义的 workqueue

创建 workqueue:

create_singlethread_workqueue(name) // 仅对应一个内核线程
create_workqueue(name) // 对应多个内核线程, 同上文.

向 workqueue 中提交工作项:

int queue_work(workqueue_t *queue, work_t *work);
int queue_delayed_work(workqueue_t *queue, work_t *work, unsigned long delay);

取消 workqueue 中挂起的工作项以及释放 workqueue 此处略过.

工作队列的优点
  • 使用简单.
  • 执行在进程上下文, 从而可以睡眠, 被调度和抢占.
  • 在多核环境下使用也非常友好.
example
#include
#include
#include

static struct workqueue_struct *queue = NULL;
static struct work_struct work;

static void work_handler(struct work_struct *data)
{
    printk(KERN_ALERT "work handler for work_item in queue helloworkqueue\n");
    // workqueue 中的每个工作完成之后就被移除 workqueue.
    // 下面的语句会造成"死机", 原因可能是该 workqueue 占据了所有的 CPU 时间.
    // queue_work(queue, &work);
}

static int __init test_init(void)
{
    queue = create_singlethread_workqueue("helloworkqueue");
    if (!queue)
    {
        goto err;
    }
    INIT_WORK(&work, work_handler);
    queue_work(queue, &work);

    return 0;
err:
    return -1;
}
static void __exit test_exit(void)
{
    destroy_workqueue(queue);
}

MODULE_LICENSE("GPL");
module_init(test_init);
module_exit(test_exit);
brought to you by .
阅读(2136) | 评论(0) | 转发(1) |
给主人留下些什么吧!~~