当前位置: 首页 > news >正文

Linux驱动-中断-工作队列传参

Linux驱动-中断-工作队列传参 知识点

文章目录

  • 前言
  • 一、参考资料
  • 二、实现方案-container_of和自定义结构体
    • 自定义结构体-work_data
  • 三、驱动代码程序
    • 思考
    • 程序分析
    • 知识点-container_of
  • 总结


前言

  • 假使已经对中断基本内容熟悉,中断共享工作队列、中断自定义工作队列
  • 按照以往经验,传参其实就是定义一个结构体,将必要数据、必要结构体封装一层,都是通过结构体来实现的。
  • 通过结构体,这个结构体要么是系统自定义、要么是系统提供的一个结构体数据结构。参数传递和结构体里面的结构体调用、赋值、获取 都是通过这个封装的结构体调用来实现。
  • container_of,来实现结构体成员指针来获取包含该成员的结构体的指针,进而输出传递过来的参数值

比如如下结构体:

struct work_data {
struct work_struct test_work;
int a;
int b;
};
struct work_data test_workqueue_work;

一、参考资料

Linux驱动-中断-自定义工作队列

Linux内核INIT_WORK如何传参数

内核工作队列

linux中断下文工作队列之工作队列传参

RK3568驱动指南|第五期-中断-第47章 工作队列传参实验

二、实现方案-container_of和自定义结构体

自定义结构体-work_data

在 Linux 内核的工作队列中, 可以通过使用工作项的方式向工作队列传递参数。 工作项是一个抽象的结构, 可以用于封装需要执行的工作及其相关的参数。

首先我们定义工作项结构, 如下所示, 在结构体 struct work_data 中定义了需要传递给工作
项处理函数的参数 a 和 b, 然后定义一个类型为 struct work_data 的变量 test_workqueue_work

struct work_data {
struct work_struct test_work;
int a;
int b;
};
struct work_data test_workqueue_work;

接下来在模块初始化函数 interrupt_irq_init 中创建了一个工作队列 test_workqueue 和一个
工作项 test_workqueue_work

test_workqueue = create_workqueue("test_workqueue"); // 创建工作队列
INIT_WORK(&test_workqueue_work.test_work, test_work); // 初始化工作项

然后在模块初始化函数中, 为工作项的参数 a 和 b 赋值。

test_workqueue_work.a = 1;
test_workqueue_work.b = 2;

三、驱动代码程序

#include <linux/module.h>
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/gpio.h>
#include <linux/delay.h>
#include <linux/workqueue.h>int irq;struct work_data
{struct work_struct test_work;int a;int b;
};struct work_data test_workqueue_work;struct workqueue_struct *test_workqueue;// 工作项处理函数
void test_work(struct work_struct *work)
{struct work_data *pdata;pdata = container_of(work, struct work_data, test_work);printk("a is %d", pdata->a);printk("b is %d", pdata->b);
}// 中断处理函数
irqreturn_t test_interrupt(int irq, void *args)
{printk("This is test_interrupt\n");// 提交工作项到工作队列queue_work(test_workqueue, &test_workqueue_work.test_work);return IRQ_RETVAL(IRQ_HANDLED);
}static int interrupt_irq_init(void)
{int ret;irq = gpio_to_irq(101); // 将GPIO映射为中断号printk("irq is %d\n", irq);// 请求中断ret = request_irq(irq, test_interrupt, IRQF_TRIGGER_RISING, "test", NULL);if (ret < 0){printk("request_irq is error\n");return -1;}// 创建工作队列test_workqueue = create_workqueue("test_workqueue");// 初始化工作项INIT_WORK(&test_workqueue_work.test_work, test_work);test_workqueue_work.a = 1;test_workqueue_work.b = 2;return 0;
}static void interrupt_irq_exit(void)
{free_irq(irq, NULL);                              // 释放中断cancel_work_sync(&test_workqueue_work.test_work); // 取消工作项flush_workqueue(test_workqueue);                  // 刷新工作队列destroy_workqueue(test_workqueue);                // 销毁工作队列printk("bye bye\n");
}module_init(interrupt_irq_init);
module_exit(interrupt_irq_exit);MODULE_LICENSE("GPL");
MODULE_AUTHOR("fangchen");

实验结果如下

在这里插入图片描述

思考

你看日志打印,为什么 This is test_interrupt 一直打印,This is test_work 只是打印了两次。 这里要搞清楚 msleep(1000); 用法。 如:msleep 阻塞导致调度延迟如果频繁调用 schedule_work(),而工作项中的 msleep(1000) 尚未完成,后续调度可能被合并或丢弃。

程序分析

  • 如前言分析,这个程序代码就是自定义队列的程序,只是定义了一个work_data 结构体,把 工作项work_struct 放在了这个结构体里面而已。
  • 对于工作项的调用,都是通过结构体指针指向工作项方式实现
  • 当我们在工作项处理函数里面获取传递参数时候,就是通过工作项+container_of 来获取封装工作项的结构体,然后通过结构体指针获取传递的参数

知识点-container_of

container_of 的作用是通过结构体成员的指针来获取包含该成员的结构体的指针。
参数解释

  • ptr:结构体成员的指针
  • type:包含该成员的结构体类型
  • member:该成员在结构体中的名称

总结

  • 如果对中断工作队列了解的话,这里参数传递很容易理解了。封装一次结构体,携带传递参数而已
http://www.dtcms.com/a/293706.html

相关文章:

  • mysql中varchar可以放多少个中文字符?
  • shiro的SecurityUtils.getSubject() 使用说明
  • 【按下电源键后,电脑里发生了什么?——BIOS:启动世界的“第一把钥匙”】
  • AI产品经理面试宝典第49天:智能客服、教育产品与医疗应用设计题相关解析
  • 深入解析Sqoop数据导入的并行切分机制与主键分区算法
  • AM1.5G AAA稳态太阳光模拟器特点
  • 西安电子科技大学金融学431考研经历分享
  • VSCode 开发 STM32 - clangd 带来的极致补全体验
  • FastAdmin 中生成插件
  • Python笔记之跨文件实例化、跨文件调用、导入库
  • 算法竞赛备赛——【图论】拓扑排序
  • PAT 甲级题目讲解:1002《A+B for Polynomials》
  • 二分查找----2.搜索二维矩阵
  • (13)机器学习小白入门YOLOv:YOLOv8-cls中用TensorBoard实时监控指标
  • 深入浅出理解 Reactor:响应式编程的利器​
  • Github上传文件流程图
  • Docker 应用数据备份、迁移方案
  • Redis原理之分布式锁
  • 武德物业和浑元科技-《软件方法》第2章业务建模之愿景03
  • Oracle物化视图详解
  • 计算机网络第四章(4)——网络层《无分类编址CIDR、路由聚合》
  • ESP32-CAM实战:DIY基于OpenAI的AI视觉识别相机
  • 校园后勤服务平台小程序的设计与实现
  • Android埋点实现方案深度分析
  • 新手向:Idea的使用技巧
  • Android 架构演进:从 MVC 到 MVVM 的设计之道
  • SpringCloud seata全局事务
  • python在windows电脑找回WiFi密码
  • LinkedList的模拟实现+LinkedList和ArrayList的区别
  • 使用 Maven 的 `maven-assembly-plugin` 插件打包zip