中断下半部之 work queue

Posted Li-Yongjun

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了中断下半部之 work queue相关的知识,希望对你有一定的参考价值。

作用

work queue 和 softirq 和 tasklet 一样,是常用的下半部机制之一。不过,和 softirq 以及 tasklet 不同的是,work queue 的本质是把 work 交给一个内核线程,在进程上下文调度的时候执行。因为这个特点,work queue 允许重新调度和睡眠。这种异步执行的进程上下文,能解决因为 softirq 和 tasklet 执行时间长而导致的系统实时性下降等问题。
当驱动程序在进程上下文中有异步执行的工作任务时,可以用 work 来描述工作任务。把 work 添加到一个链表 worklist 中,然后由一个内核线程 worker 遍历链表,串行地执行挂入worklist 中的所有 work,如果 worklist 中没有 work,那么内核线程 worker 就会变成 IDLE 状态,如果有 work,则执行 work 中的回调函数。

数据结构

work_struct :
描述一个 work

struct work_struct 
	atomic_long_t data;
	struct list_head entry;
	work_func_t func;
#ifdef CONFIG_LOCKDEP
	struct lockdep_map lockdep_map;
#endif
;
  • data:低 bit 存放状态位,高 bit 存放 worker_pool 的 ID 或者 pool_workqueue 的指针
  • entry:用于添加到其它队列上
  • func:work 的处理函数,在内核线程 work 中被调用执行

使用

一、初始化
内核推荐驱动开发者使用默认的 work queue,而不是新建 work queue。要使用系统默认的 work queue,首先需要初始化 work,内核提供了相应的宏 INIT_WORK()。
二、调度
初始化 work 后,就可以调用 schedule_work() 函数把 work 挂入系统默认的 work queue 中。

示例

irq_workqueue.c

#include <linux/gpio.h>
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/module.h>

#define BUTTON_PIN 12 /* GPIO 12 */

int flag = 0;

static struct work_struct work;

void workqueue_fn(struct work_struct *work)	 // 下半部

	printk("%s(), %s\\n", __func__, current->comm);


static irqreturn_t irq_handler(int irq, void *dev)	// 上半部

	printk("%s(): enter\\n", __FUNCTION__);

	schedule_work(&work);

	printk("%s(): exit\\n", __FUNCTION__);

	return IRQ_HANDLED;


static int led_init(void)

	int err;
	int irq;

	printk("%s()\\n", __FUNCTION__);

	INIT_WORK(&work, workqueue_fn);

	err = gpio_request_one(BUTTON_PIN, GPIOF_IN, "Button");
	if (err)
		return err;

	irq = gpio_to_irq(BUTTON_PIN);
	// enable_irq(irq); // why crash ?
	err = request_irq(irq, irq_handler, IRQ_TYPE_EDGE_BOTH, "LED Test", NULL);
	if (err < 0) 
		printk("request irq (%d) failed!\\n", irq);
		return err;
	

	printk("request irq (%d) success!\\n", irq);

	flag = 1;

	return 0;


static void led_exit(void)

	printk("%s()\\n", __FUNCTION__);

	if (flag)
		free_irq(gpio_to_irq(BUTTON_PIN), NULL);

	gpio_free(BUTTON_PIN);

	return;


module_init(led_init);
module_exit(led_exit);

MODULE_LICENSE("Dual BSD/GPL");

Makefile

obj-m = irq_workqueue.o

KDIR=/home/liyongjun/project/board/buildroot/RPi3/build/linux-custom
CROSS_COMPILE=/home/liyongjun/project/board/buildroot/RPi3/host/bin/arm-buildroot-linux-gnueabihf-

all:
	make -C $(KDIR) M=$(PWD) ARCH=arm CROSS_COMPILE=$(CROSS_COMPILE) modules

clean:
	make -C $(KDIR) M=$(PWD) ARCH=arm CROSS_COMPILE=$(CROSS_COMPILE) clean

gpioout.c

#include "bcm2835.h"
#include <unistd.h>

int main(int argc, char *argv[])

	int n = atoi(argv[1]);
	bcm2835_init();
	bcm2835_gpio_fsel(21, BCM2835_GPIO_FSEL_OUTP);
	while (n--) 
		bcm2835_gpio_set(21);
		sleep(1);
		bcm2835_gpio_clr(21);
		sleep(1);
	

	return 0;


/home/liyongjun/project/board/buildroot/RPi3/host/bin/arm-buildroot-linux-gnueabihf-gcc gpioout.c -o gpioout.out -L ../bcm2835-1.71/src/ -I ../bcm2835-1.71/src/ -lbcm2835

测试

在树莓派上执行,树莓派 GPIO12 引脚和 GPIO21 使用杜邦线连接。gpioout.out 会操作 21 引脚,使其输出一次高电平和一次低电平,通过杜邦线连接到 12 引脚,进而触发 12 引脚的边沿中断。
产生中断后,会调用中断处理函数 irq_handler(),在中断处理函数中调用 schedule_work(&work) 将 work 挂入系统默认的 worklist,最终由 worker 内核线程执行。

# 
# insmod irq_workqueue.ko 
[ 3907.271811] led_init()
[ 3907.275666] request irq (200) success!
# 
# ./gpioout.out 1
[ 3911.812154] irq_handler(): enter
[ 3911.816768] irq_handler(): exit
[ 3911.821380] workqueue_fn(), kworker/0:1
[ 3912.812235] irq_handler(): enter
[ 3912.816763] irq_handler(): exit
[ 3912.821208] workqueue_fn(), kworker/0:1
# 
# rmmod irq_workqueue.ko 
[ 3916.560012] led_exit()
# 

以上是关于中断下半部之 work queue的主要内容,如果未能解决你的问题,请参考以下文章

linux驱动: 中断下半部之tasklet&workqueue

Linux(内核剖析):23---中断下半部之(下半部总体概述)

中断下半部之 tasklet

中断下半部之 tasklet

把握linux内核设计思想:下半部机制之软中断

Linux中断底半部机制