linux driver probe deferral 机制

1. 背景介绍

在偶然的一次实验中(具体是pinctrl实验),我发现有些平台的pincontroller驱动起得很晚,而pinctrl client驱动却起得很早,在设备驱动模型中probe之前又会进行管脚复用的相关设置,按照常理来讲,这就产生了某种依赖性: pincontroller必须尽早启动,否则pinctrl client无法使用管脚复用功能,但实际上的效果并非如此,尽管pincontroller驱动起得很晚,但是client仍然能够正常使用pinctrl子系统提供的复用功能,这就是延迟probe机制。

  

2. 提交说明

我在github上找到了probe延迟机制的提交,最原始的提交如下链接,后续有人陆续在上面修改BUG:

drivercore: Add driver probe deferral mechanism · torvalds/linux@d1c3414 · GitHub

我们可以看看他的提交描述,引入这一机制解决什么样的问题: 

 为了解决驱动间的顺序依赖,引入了该机制后驱动间的顺序依赖就解耦了。

3. probe延迟机制的具体说明

该机制的主要引入是引入在drivers/base/dd.c中的,并且为struct device结构引入了一个链表节点来挂载被延迟probe的设备 :

简述一下该机制:

1).  在dev与drv匹配成功后的really_probe()中,如果驱动与设备因为某种原因无法probe成功,那么probe返回-EPROBE_DEFER表示自己需要延迟probe。

2). 这个时候调用driver_deferred_probe_add(dev)将设备加入延迟probe的链表中。

3). 处理deferred_probe_pending_list链表有两个时间点

第一个时间点是某些dev和drv probe成功后的driver_bound()中:

但是这个时机一般driver_deferred_probe_trigger()是无效的:

 

第二个时间点是late_initcall(deferred_probe_initcall):

 

在这个时机不仅真正创建了执行延迟probe的工作队列deferwq,还真正处理了deferred_probe_pending_list中挂载的节点进行延迟probe。

可以看到这个时机真的很晚了(late_initcall), 正如描述中所说: "this initcall makes sure that deferred probing is delayed until late_initcall time"。

deferred_probe_initcall()

    ->driver_deferred_probe_trigger()

        ->deferred_probe_work_func()

            ->bus_probe_device()

4). late_initcall之后

类似于ko这种场景,deferred probe就靠driver_bound()去触发,因为此时driver_deferred_probe_enable为true,且工人队列已经建立,driver_deferred_probe_trigger()就生效了。

4. 实验

为了模拟上述deferred probe机制,我构造了两个驱动: driver1.c与driver2.c,并且使他们产生依赖: driver2.c依赖driver1.c,否则probe不成功。

代码如下:driver1.c

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_gpio.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#include <linux/platform_device.h>extern int g_val;static int driver1_dummy_probe(struct platform_device *dev)
{printk("[LJW]driver1_dummy_probe=====>\n");/* driver1 modified the g_val */g_val = 1;printk("[LJW]driver1_dummy_probe<=====\n");return 0;    
}static int driver1_dummy_remove(struct platform_device *dev)
{printk("[LJW]driver1_dummy_remove=====>\n");printk("[LJW]driver1_dummy_remove<=====\n");return 0;  
}static struct platform_device driver1_dummy_device = {.name = "driver1_compatible",
};static struct platform_driver driver1_dummy_driver = {.probe = driver1_dummy_probe,.remove = driver1_dummy_remove,.driver = {.name = "driver1_compatible",},
};static int __init driver1_init(void)
{int ret;printk("[LJW]driver1_init=====>\n");ret = platform_device_register(&driver1_dummy_device);if (ret < 0) {printk("[FAILED]platform_device_register failed for driver1_dummy_device!\n");return -1;}printk("[SUCCESS]platform_device_register for driver1_dummy_device\n");ret = platform_driver_register(&driver1_dummy_driver);if (ret < 0) {printk("[FAILED]platform_driver_register failed for driver1_dummy_driver!\n");platform_device_unregister(&driver1_dummy_device);return -1;}printk("[SUCCESS]platform_driver_register for driver1_dummy_driver\n");return 0;
}static void __exit driver1_exit(void)
{platform_driver_unregister(&driver1_dummy_driver);platform_device_unregister(&driver1_dummy_device);return;
}module_init(driver1_init);
module_exit(driver1_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("liaojunwu");

代码如下: driver2.c

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_gpio.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#include <linux/platform_device.h>int g_val = 0;
EXPORT_SYMBOL(g_val);static int driver2_dummy_probe(struct platform_device *dev)
{printk("[LJW]driver2_dummy_probe=====>\n");/* driver1 modified the g_val */if (g_val == 0) {printk("[LJW]driver2 probe failed, return -EPROBE_DEFER!\n");return -EPROBE_DEFER;}printk("[LJW]driver2_dummy_probe<=====\n");return 0;    
}static int driver2_dummy_remove(struct platform_device *dev)
{printk("[LJW]driver2_dummy_remove=====>\n");printk("[LJW]driver2_dummy_remove<=====\n");return 0;  
}static struct platform_device driver2_dummy_device = {.name = "driver2_compatible",
};static struct platform_driver driver2_dummy_driver = {.probe = driver2_dummy_probe,.remove = driver2_dummy_remove,.driver = {.name = "driver2_compatible",},
};static int __init driver2_init(void)
{int ret;printk("[LJW]driver2_init=====>\n");ret = platform_device_register(&driver2_dummy_device);if (ret < 0) {printk("[FAILED]platform_device_register failed for driver2_dummy_device!\n");return -1;}printk("[SUCCESS]platform_device_register for driver2_dummy_device\n");ret = platform_driver_register(&driver2_dummy_driver);if (ret < 0) {printk("[FAILED]platform_driver_register failed for driver2_dummy_driver!\n");platform_device_unregister(&driver2_dummy_device);return -1;}printk("[SUCCESS]platform_driver_register for driver2_dummy_driver\n");return 0;
}static void __exit driver2_exit(void)
{platform_driver_unregister(&driver2_dummy_driver);platform_device_unregister(&driver2_dummy_device);return;
}module_init(driver2_init);
module_exit(driver2_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("liaojunwu");

makefile:

.PHONY: build clean#Why change the SHELL, because default /bin/sh not support source command
SHELL := /bin/bash
KERNELDIR := /home/liaojunwu/linux/code/sdk_ori_code/sdk_oriCURRENT_PATH := $(shell pwd)obj-m := driver1.o
obj-m += driver2.obuild: pre_build kernel_modulespre_build:source /opt/fsl-imx-x11/4.1.15-2.1.0/environment-setup-cortexa7hf-neon-poky-linux-gnueabikernel_modules:$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) modulesclean:$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) cleanrm -rf *.mod.c *.o *.ko *.order *.symvers

实验结果:

通过实验结果可以很清楚地看到deferred probe的整个过程,可以看到虽然先加载driver2.ko但是driver2的probe是等到driver1的probe执行完毕才执行的,正是因为deferred probe机制为其保证了正确的执行顺序,同时也可以看到driver2的probe实际上是在工作队列中执行的,这时候与driver1的某些代码是并发关系(原始的提交在处理这种并发关系上有一些BUG,后人陆续有修改,具体可以去追溯github),最后在driver2的probe成功后在driver_bound()里面又触发了一次probe的defer。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/28256.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Sa-Token + SpringBoot 实现登录鉴权

1. 技术选型 今天最近在做登录、授权的功能,一开始考虑到的是spring boot + spring security,但spring security太重,而我们是轻量级的项目,所以,spring security不适合我们。 而后考虑spring boot + shiro,但shiro自带的aop会影响spring boot的aop,所以,shiro也不适…

【代码随想录 | Leetcode | 第五天】链表 | 移除链表元素 | 设计链表

前言 欢迎来到小K的Leetcode|代码随想录|专题化专栏&#xff0c;今天将为大家带来移除链表元素和设计链表的分享✨ 目录 前言203. 移除链表元素707. 设计链表总结 203. 移除链表元素 ✨题目链接点这里 给你一个链表的头节点 head 和一个整数 val &#xff0c;请你删除链表中所…

【kubernetes系列】Kubernetes之配置dashboard安装使用

概述 Dashboard 是基于网页的 Kubernetes 用户界面。 你可以使用 Dashboard 将容器应用部署到 Kubernetes 集群中&#xff0c;也可以对容器应用排错&#xff0c;还能管理集群资源。 你可以使用 Dashboard 获取运行在集群中的应用的概览信息&#xff0c;也可以创建或者修改 Kub…

10.6.5 【Linux】分区命令: split

如果你有文件太大&#xff0c;导致一些携带式设备无法复制的问题&#xff0c;split可以将一个大文件&#xff0c;依据文件大小或行数来分区&#xff0c;可以将大文件分区成为小文件&#xff0c;快速有效。 将文件分区的话&#xff0c;使用-b size来将一个分区的文件限制其大小&…

java模拟MQTT客户端发送消息及EMQX配置

EMQX配置 登录地址 首先打开EMQX的管理界面&#xff0c;界面的地址如下&#xff0c; http://192.168.1.110:18083/ 规则是IP就是MQTT的IP&#xff0c;端口是固定的18083&#xff0c;输入该地址后&#xff0c;展示界面如下&#xff1a; 然后输入用户名和密码&#xff0c;用户…

python_day7_画图

json数据与python字典的相互转换 import json列表&#xff0c;其中每个元素均为一个字典 data [{"name": "张三", "age": 10},{"name": "李四", "age": 13},{"name": "jay", "age&qu…

科技云报道:数字化转型完成后,制造业如何走向“数智”时代?

科技云报道原创。 随着我国数字化转型行动的深入推进和智能制造工程的大力实施&#xff0c;制造业正朝着“数智”时代迈进&#xff0c;生成式AI被视为推动制造业智能化发展的关键驱动力。 据预测&#xff0c;到2027年&#xff0c;将有30%的制造业采用生成式AI来提升产品研发效…

汇编实现1-100累加(ARMv7)

汇编实现1-100累加 代码.text .globl _start _start:mov r0,#0 summationmov r1,#0 autoIncrementbl funadd funadd:cmp r1,#100addccs r0,r1,r0addccs r1,r1,#1mov pc,lr stop:b stop .end运行效果

【LangChain】数据连接(Data connection)

概要 许多LLM申请需要特定于用户的数据&#xff0c;这些数据不属于模型训练集的一部分。 LangChain 为您提供了通过以下方式加载、转换、存储和查询数据的构建块&#xff1a; Document loaders &#xff1a; 从许多不同来源加载文档Document transformers&#xff1a;拆分文档…

React native 已有项目升级兼容web

基础 概念 | webpack 中文文档 | webpack 中文文档 | webpack 中文网 深入理解Webpack及Babel的使用 - 掘金 Introduction to React Native for Web // React Native for Web Webpack 是一个现代的 JavaScript 应用程序的静态模块打包工具&#xff0c;它将应用程序所依赖的各…

Docker本地镜像发布到阿里云

我们构建了自己的镜像后&#xff0c;可以发布到远程镜像提供给其他人使用&#xff0c;比如发布到阿里云 使用build/commit生成新的镜像&#xff0c;并生成自己镜像的版本标签tag&#xff0c;此新的镜像在自己的本地库中&#xff0c;使用push可以将镜像提交到阿里云公有库/私有库…

确认应答机制与超时重发机制【TCP原理(笔记一)】

文章目录 通过序列号与确认应答提高可靠性正常的数据传输数据包丢失的情况确认应答丢失的情况发送的数据 重发超时如何确定 通过序列号与确认应答提高可靠性 在TCP中&#xff0c;当发送端的数据到达接收主机时&#xff0c;接收端主机会返回一个已收到消息的通知。这个消息叫做…