linux 中断之工作队列workqueue (自己创建的工作队列,减小默认线程负担)

本文主要是介绍linux 中断之工作队列workqueue (自己创建的工作队列,减小默认线程负担),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

默认工作队列和自己创建工作队列对比介绍

1、上一节我们用的是schedule_work(&btn_work);  和 schedule_delayed_work(&btn_dwork, 5*HZ);调度任务

这两个调度函数都是把任务交个内核默认线程(event)管理执行,一旦向内核默认线程提交的任务太多,将导致内核默认线程负担太重,影响实时性

2、这一节我们使用的是queue_work(btn_wq, &btn_work); queue_delayed_work(btn_wq, &btn_dwork, 3*HZ);调度任务

这两个函数将会将我们的工作任务添加到自己的新创建的线程中,减小默认线程的负担

1、创建自己的工作队列

//分配工作队列的指针
static struct workqueue_struct *btn_wq;//创建自己的工作队列和自己的内核线程btn_wq = create_workqueue("mybuttons");

2、登记调度

    //将自己的工作和自己的工作队列进行管理,然后再登记queue_work(btn_wq, &btn_work);//将自己延时的工作和自己的工作队列进行关联,然后再登记queue_delayed_work(btn_wq, &btn_dwork, 3*HZ);

参考代码

#include <linux/init.h>
#include <linux/module.h>
#include <linux/irq.h>
#include <linux/interrupt.h>//定义按键硬件相关的数据结构
struct button_resource {int irq;    //中断号char *name; //中断名称
};//初始化按键信息
static struct button_resource btn_info[] = {[0] = {.irq = IRQ_EINT(0),.name = "KEY_UP"},[1] = {.irq = IRQ_EINT(1),.name = "KEY_DOWN"}
};//分配工作和延时工作
static struct work_struct btn_work;
static struct delayed_work btn_dwork;//分配工作队列的指针
static struct workqueue_struct *btn_wq;//工作处理函数
static void btn_work_func(struct work_struct *work)
{printk("%s: %#x\n", __func__, work);
}//延时工作处理函数
static void btn_dwork_func(struct work_struct *work)
{printk("%s: %#x\n", __func__, work);
}//中断处理函数就是顶半部
static irqreturn_t button_isr(int irq, void *dev_id)
{//登记工作,内核会在适当的时候执行工作对应的处理函数//schedule_work(&btn_work);//登记延时工作,并且指定延时的时间间隔为3S//schedule_delayed_work(&btn_dwork, 3*HZ);//将自己的工作和自己的工作队列进行管理,然后再登记queue_work(btn_wq, &btn_work);//将自己延时的工作和自己的工作队列进行关联,然后再登记queue_delayed_work(btn_wq, &btn_dwork, 3*HZ);printk("%s\n", __func__);return IRQ_HANDLED; //处理完毕
}static int btn_init(void)
{int i;printk("register irq!\n");for (i = 0; i < ARRAY_SIZE(btn_info); i++)request_irq(btn_info[i].irq, button_isr, IRQF_TRIGGER_RISING|IRQF_TRIGGER_FALLING,btn_info[i].name, &btn_info[i]);//初始化工作和延时的工作INIT_WORK(&btn_work, btn_work_func); //指定工作处理函数INIT_DELAYED_WORK(&btn_dwork, btn_dwork_func); //指定延时工作处理函数printk("%s: %#x, %#x\n", __func__, &btn_work, &btn_dwork);//创建自己的工作队列和自己的内核线程btn_wq = create_workqueue("mybuttons");return 0;
}static void btn_exit(void)
{int i;printk("unregister irq!\n");//注意注册中断传递的参数和释放中断传递的参数一定要一致!for(i = 0; i < ARRAY_SIZE(btn_info); i++)free_irq(btn_info[i].irq, &btn_info[i]);//销毁自己的工作队列和内核线程destroy_workqueue(btn_wq);
}
module_init(btn_init);
module_exit(btn_exit);
MODULE_LICENSE("GPL");

 

这篇关于linux 中断之工作队列workqueue (自己创建的工作队列,减小默认线程负担)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/439051

相关文章

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

Java中如何正确的停掉线程

《Java中如何正确的停掉线程》Java通过interrupt()通知线程停止而非强制,确保线程自主处理中断,避免数据损坏,线程池的shutdown()等待任务完成,shutdownNow()强制中断... 目录为什么不强制停止为什么 Java 不提供强制停止线程的能力呢?如何用interrupt停止线程s

RabbitMQ 延时队列插件安装与使用示例详解(基于 Delayed Message Plugin)

《RabbitMQ延时队列插件安装与使用示例详解(基于DelayedMessagePlugin)》本文详解RabbitMQ通过安装rabbitmq_delayed_message_exchan... 目录 一、什么是 RabbitMQ 延时队列? 二、安装前准备✅ RabbitMQ 环境要求 三、安装延时队

linux系统上安装JDK8全过程

《linux系统上安装JDK8全过程》文章介绍安装JDK的必要性及Linux下JDK8的安装步骤,包括卸载旧版本、下载解压、配置环境变量等,强调开发需JDK,运行可选JRE,现JDK已集成JRE... 目录为什么要安装jdk?1.查看linux系统是否有自带的jdk:2.下载jdk压缩包2.解压3.配置环境

Linux搭建ftp服务器的步骤

《Linux搭建ftp服务器的步骤》本文给大家分享Linux搭建ftp服务器的步骤,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录ftp搭建1:下载vsftpd工具2:下载客户端工具3:进入配置文件目录vsftpd.conf配置文件4:

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

Spring创建Bean的八种主要方式详解

《Spring创建Bean的八种主要方式详解》Spring(尤其是SpringBoot)提供了多种方式来让容器创建和管理Bean,@Component、@Configuration+@Bean、@En... 目录引言一、Spring 创建 Bean 的 8 种主要方式1. @Component 及其衍生注解

Nginx中配置使用非默认80端口进行服务的完整指南

《Nginx中配置使用非默认80端口进行服务的完整指南》在实际生产环境中,我们经常需要将Nginx配置在其他端口上运行,本文将详细介绍如何在Nginx中配置使用非默认端口进行服务,希望对大家有所帮助... 目录一、为什么需要使用非默认端口二、配置Nginx使用非默认端口的基本方法2.1 修改listen指令

Linux实现查看某一端口是否开放

《Linux实现查看某一端口是否开放》文章介绍了三种检查端口6379是否开放的方法:通过lsof查看进程占用,用netstat区分TCP/UDP监听状态,以及用telnet测试远程连接可达性... 目录1、使用lsof 命令来查看端口是否开放2、使用netstat 命令来查看端口是否开放3、使用telnet

Linux系统管理与进程任务管理方式

《Linux系统管理与进程任务管理方式》本文系统讲解Linux管理核心技能,涵盖引导流程、服务控制(Systemd与GRUB2)、进程管理(前台/后台运行、工具使用)、计划任务(at/cron)及常用... 目录引言一、linux系统引导过程与服务控制1.1 系统引导的五个关键阶段1.2 GRUB2的进化优