首页 > 系统相关 >linux总线设备驱动模型

linux总线设备驱动模型

时间:2024-08-28 14:23:53浏览次数:13  
标签:struct driver 总线 dev platform linux device 驱动 name

linux总线设备驱动模型

platform 平台驱动模型

linux自带I2C、 SPI、 USB 等总线。但是在 SOC 中有些外设是没有总线这个概念的,但是又要使用总线、驱动和设备模型该怎么办呢?为了解决此问题, Linux 提出了 platform 这个虚拟总线,相应
的就有 platform_driver 和 platform_device。
Linux系统内核使用 bus_type结构体表示总线

struct bus_type {
	const char *name; /* 总线名字 */
	const char *dev_name;
	struct device *dev_root;
	struct device_attribute *dev_attrs;
	const struct attribute_group **bus_groups; /* 总线属性 */
	const struct attribute_group **dev_groups; /* 设备属性 */
	const struct attribute_group **drv_groups; /* 驱动属性 */
	int (*match)(struct device *dev, struct device_driver *drv);
	int (*uevent)(struct device *dev, struct kobj_uevent_env *env);
	int (*probe)(struct device *dev);
	int (*remove)(struct device *dev);
	void (*shutdown)(struct device *dev);
	int (*online)(struct device *dev);
	int (*offline)(struct device *dev);
	int (*suspend)(struct device *dev, pm_message_t state);
	int (*resume)(struct device *dev);
	const struct dev_pm_ops *pm;
	const struct iommu_ops *iommu_ops;
	struct subsys_private *p;
	struct lock_class_key lock_key;
};

match 函数有两个参数: dev 和 drv,
这两个参数分别为 device 和 device_driver 类型,也就是设备和驱动
platform 总线是 bus_type 的一个具体实例

struct bus_type platform_bus_type = {
	.name = "platform",
	.dev_groups = platform_dev_groups,
	.match = platform_match,
	.uevent = platform_uevent,
	.pm = &platform_dev_pm_ops,
};

platform_bus_type 就是 platform 平台总线,其中 platform_match 就是匹配函数。我们来看
一下驱动和设备是如何匹配的

static int platform_match(struct device *dev,
struct device_driver *drv)
{
	struct platform_device *pdev = to_platform_device(dev);
	struct platform_driver *pdrv = to_platform_driver(drv);
	/*When driver_override is set,only bind to the matching driver*/
	//platform_device.driver_override 和 platform_driver.driver.name   第一比较
	if (pdev->driver_override)
		return !strcmp(pdev->driver_override, drv->name);
	/* Attempt an OF style match first */
	// OF 类型的匹配,也就是设备树采用的匹配方式,   设备树类型的比较          第二比较
	if (of_driver_match_device(dev, drv))
		return 1;
	/* Then try ACPI style match */
	if (acpi_driver_match_device(dev, drv))							//第三  不常用
		return 1;
	/* Then try to match against the id table */
	//第四种匹配方式, id_table 匹配,每个 platform_driver 结构体有一个 id_table成员变量,
	//顾名	思义,保存了很多 id 信息。这些 id 信息存放着这个 platformd 驱动所支持的驱动类型。
	if (pdrv->id_table)			
		return platform_match_id(pdrv->id_table, pdev) != NULL;
	/* fall-back to driver name match */
	//第五种匹配方式,如果第三种匹配方式的 id_table 不存在的话就直接比较驱动和设备的 name 字段,			    //看看是不是相等,如果相等的话就匹配成功。
	return (strcmp(pdev->name, drv->name) == 0);
}

platform_device

struct platform_device {
	const char *name;                  /* 设备名称 */
	int id;                           /* 设备唯一标识符 */
	bool id_auto;                     /* 标识是否自动生成 ID */
	struct device dev;                /* 通用设备结构体 */
	u32 num_resources;                /* 设备相关资源的数量 */
	struct resource *resource;        /* 指向资源数组的指针 */
	const struct platform_device_id *id_entry; /* 设备特定 ID 表的指针 */
	char *driver_override;            /* 强制匹配的驱动程序名称 */
	struct mfd_cell *mfd_cell;        /* 指向 MFD(多功能设备)单元的指针(如果适用) */
	struct pdev_archdata archdata;    /* 架构特定的数据和附加信息 */
};

platform_driver

struct platform_driver {
	int (*probe)(struct platform_device *); /* 设备探测函数 */
	int (*remove)(struct platform_device *); /* 设备移除函数 */
	void (*shutdown)(struct platform_device *); /* 设备关机函数 */
	int (*suspend)(struct platform_device *, pm_message_t state); /* 设备挂起函数 */
	int (*resume)(struct platform_device *); /* 设备恢复函数 */
	struct device_driver driver; /* 设备驱动程序结构体 */
	const struct platform_device_id *id_table; /* 平台设备 ID 表 */
	bool prevent_deferred_probe; /* 防止延迟探测的标志 */
};
struct platform_device_id {
	char name[PLATFORM_NAME_SIZE];
	kernel_ulong_t driver_data;
};
struct device_driver {
	const char *name;                  /* 驱动程序名称 */
	struct bus_type *bus;              /* 驱动程序所属的总线类型 */
	struct module *owner;              /* 模块所有者 */
	const char *mod_name;              /* 用于内建模块的名称 */
	bool suppress_bind_attrs;          /* 禁用通过 sysfs 绑定/解绑 */
	const struct of_device_id *of_match_table; /* 用于设备树匹配的设备 ID 表 */
	const struct acpi_device_id *acpi_match_table; /* 用于 ACPI 匹配的设备 ID 表 */
	int (*probe) (struct device *dev); /* 设备探测函数 */
	int (*remove) (struct device *dev); /* 设备移除函数 */
	void (*shutdown) (struct device *dev); /* 设备关机函数 */
	int (*suspend) (struct device *dev, pm_message_t state); /* 设备挂起函数 */
	int (*resume) (struct device *dev); /* 设备恢复函数 */
	const struct attribute_group **groups; /* 属性组数组 */
	const struct dev_pm_ops *pm;        /* 设备电源管理操作 */
	struct driver_private *p;          /* 驱动程序私有数据 */
};

struct of_device_id {
	char name[32];
	char type[32];
	char compatible[128];
	const void *data;
};

最先比较

platform_device.driver_override 和 platform_driver.driver.name
可以设置 platform_device 的 driver_override,强制选择某个 platform_driver。

设备树比较

device_driver 结构体(表示
设备驱动)中有个名为 of_match_table的成员变量,此成员变量保存着驱动的 compatible匹配表,
设备树中的每个设备节点的 compatible 属性会和 of_match_table 表中的所有成员比较,查看是
否有相同的条目,如果有的话就表示设备和此驱动匹配,设备和驱动匹配成功以后 probe 函数
就会执行。

idtable比较

platform_device. name 和 platform_driver.id_table[i].name,Platform_driver.id_table 是“platform_device_id”指针,表示该 drv 支持若干个 device,它里面列出了各个 device 的{.name, .driver_data},其中的“name”表示该drv 支持的设备的名字, driver_data 是些提供给该 device 的私有数据。

name比较name

platform_device.name 和 platform_driver.driver.name
platform_driver.id_table 可能为空,这时可以根据 platform_driver.driver.name 来寻找同名的 platform_device。

实测小细节

1、按顺序匹配,如果其中有一个匹配存在,并且没有匹配上,不会再去下面的选项再匹配
举例子,如果idtable设置了,但是没有匹配项,而你的name是匹配的,但是系统在idtable中匹配不到,他就不会再去匹配name。

标签:struct,driver,总线,dev,platform,linux,device,驱动,name
From: https://blog.csdn.net/qq_43587089/article/details/141469676

相关文章

  • linux编程——认识GCC编译器
    目录一、引言二、GCC编译器简介三、GCC的主要特点四、GCC的使用方法五、GCC的高级功能六、总结一、引言  在Linux编程的广阔世界中,GCC(GNUCompilerCollection)编译器无疑是一个至关重要的工具。对于广大的开发者来说,熟练掌握GCC编译器的使用,能够极大地......
  • 【Linux入门】shell基础篇——if、case、与for循环
    文章目录if的条件分支基本`if`语句包含`else`的`if`语句包含`elseif`(或`elif`)的`if`语句注意示例if的嵌套使用case`case`语句的基本语法:示例使用if语句结合casefor循环for循环的基本格式1.基于列表的`for`循环2.C语言风格的`for`循环注意其他循环基于文件的for循环......
  • 【Linux入门】shell基础篇——while循环与until循环
    文章目录while循环与until循环while循环while循环的基本格式示例:打印0到5的数字死循环的几种实现方式注意退出循环until循环、与while循环的区别until循环示例while循环方式注意有趣的实例批量建立用户批量删除用户猜价格游戏密码输入验证说明while循环与until......
  • 安卓驱动开发中的设备树
    安卓驱动开发中的设备树在安卓驱动开发领域,设备树(DeviceTree)是一个至关重要的概念和技术。设备树是一种描述硬件配置的数据结构,它为操作系统提供了一种标准化的方法来描述和管理硬件资源。本文将深入探讨设备树在安卓驱动开发中的重要性、工作原理以及实际应用。一、设备树......
  • linux中,解决使用sudo时,自定义path路径不生效的问题
    问题当我们在非root用户下,需要使用root权限的时候,通常在前面加sudo就可以了但是,今天遇到了一个小问题,就是我明明已经配置了go的bin目录在PATH变量中,还是配置在/etc/profile中,在root用户下,是可以执行go命令的按理说,用sudogo应该也是没问题的但是,我发现,当我......
  • 安装python教程详解-(Linux和Windows11安装python)
    一、Linux编译安装Python3.12.5python官网地址:WelcometoPython.org1.1安装python环境1.1.1安装开发工具包和依赖#yum-ygroupinstall"DevelopmentTools" #yum-yinstallgcczlibzlib-devellibffilibffi-develreadline-developenssl-developenssl11ope......
  • Linux上网配置-CentOS7
    上网配置思路1.确认虚拟机的网络适配器为桥接模式2.网卡配置文件是否存在并启用网卡接口3.网卡配置文件详解:静态IP地址、子网掩码、默认网关、物理地址以及DNS4.局域网的其他物理机能远程该虚拟机5.虚拟机ping通外网一、确认虚拟机的网络适配器为桥接模式问题1.为什么......
  • .net core下使用事件总线
    .netcore下使用事件总线        随着微服务的火热,DDD(领域驱动设计模式)思想风起云涌,冲击着整个软件生态系统。其中,事件总线那是必须知道的了,于是我便抱着一个学习DDD的心态搭建了一个博客网站,目前该网站正在建设阶段,后续会不断完善,这里我只是讲一下我里面所用到的......
  • 【计算机组成原理】五、中央处理器:2.数据通路、控制器(单总线结构、专用数据通路、硬布
    3.数据通路文章目录3.数据通路3.1单总线结构3.2专用数据通路4.控制器4.1硬布线控制器4.1.1基本结构4.1.2设计步骤==微操作总结==1)分析每个阶段的微操作序列2)安排微操作时序的原则3)电路设计4.1.3特点4.2微程序控制器==微指令包含关系==4.2.1基本结构4.2.2微指令格式4.......
  • Linux零基础到精通(二)-vmware虚拟机使用教程及Centos7操作系统安装
    目录前言Linux操作系统运用领域vmware虚拟机安装与使用电脑硬件环境要求vmware虚拟机软件安装创建一个虚拟机配置vmware的虚拟化网络通过vmware虚拟机安装操作系统下载Centos7系统镜像安装Centos7操作系统配置网络和主机名称信息配置系统分区软件包选择设置用户密码进......