Linux驱动三:platform总线之match( ) 函数

Linux 内核为 platform 总线定义了一个 bus_type 的实例 platform_bus_type ,其定义位于 drivers/base/platform.c 下。

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

这里重点关注 match 成员函数,正是此成员函数确定了 platform_device 和 platform_driver 之间是如何进行匹配的。

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 */
	if (pdev->driver_override)
		return !strcmp(pdev->driver_override, drv->name);

	/* Attempt an OF style match first */
	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 */
	if (pdrv->id_table)
		return platform_match_id(pdrv->id_table, pdev) != NULL;

	/* fall-back to driver name match */
	return (strcmp(pdev->name, drv->name) == 0);
}

首先看platform_match函数的前两行,找到 to_platform_device 定义的位置,发现这是一个宏:

#define to_platform_device(x) container_of((x), struct platform_device, dev)

进一步查找 container_of ,发现 container_of 也是一个宏,且在内核中有很多定义。
在这里插入图片描述
container_of 定义的比较多,具体使用哪个定义也不太好找,所以直接通过预编译查看宏展开后的结果。如下所示:

宏展开前:
struct platform_device *pdev = to_platform_device(dev);
宏展开后:
struct platform_device *pdev = (
{
 const typeof( ((struct platform_device *)0)->dev ) *__mptr = ((dev)); 
 (struct platform_device *)( (char *)__mptr - __builtin_offsetof(struct platform_device, dev) );
});
---------------------------------------------------------
其中 “typeof( ((struct platform_device *)0)->dev )” 可以等价于 “struct device” 
因此宏展开的代码块中第一句就相当于:
const struct device *__mptr = ((dev));
第二句相当于:
(struct platform_device *)( (char *)__mptr - ((char *)pdev - (char *)dev) )

总之可以看到函数 platform_match 的前两行代码是通过结构中成员的地址找到了结构的初始地址。

接着往下看,就可以看到 platform_match 中定义的 platform_device 和 platform_driver 之间的5种匹配方式,这里只关注最后一种“ strcmp(pdev->name, drv->name) ”,即结构 platform_device 的元素 name 与结构 platform_driver 成员 driver 的元素name的匹配。
当两个name都相同时, platform总线上的驱动与设备匹配成功。

struct platform_device {
	const char	*name;
	......
};
struct platform_driver {
	......
	struct device_driver driver;
	......
};
struct device_driver {
	const char		*name;
	......
};
  • 6
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
回答: 在Linux驱动模型中,设备和驱动之间的匹配是通过match函数来实现的。match函数platformmatch函数,它用于完成设备和驱动之间的匹配。总线使用match函数来根据注册的设备来查找对应的驱动,或者根据注册的驱动来查找相应的设备。每一条总线都必须实现此函数match函数有两个参数:dev和drv,分别为device和device_driver类型,即设备和驱动。在match函数中,首先尝试OF风格的匹配,然后尝试ACPI风格的匹配,接着尝试根据id table进行匹配,最后回退到驱动名称的匹配。\[1\] 在Linux驱动模型中,还有一个重要的概念是设备树(device tree),它是配置板级信息的地方。Linux驱动模型分为设备、驱动总线三个部分,设备树是管理这些部分的基础。设备树是一棵树状结构,用于描述硬件设备的信息。了解和理解设备树是正确开发硬件驱动的前提。\[2\] 在编写驱动时,我们会使用一些注册函数,比如platform_driver_register、spi_register_driver、i2c_add_driver等。这些函数用于将驱动和设备进行匹配,并调用probe函数。以platform_driver为例,设备驱动的匹配是通过设备和驱动match函数来实现的。\[3\] 综上所述,Linux驱动的匹配规则是通过match函数来实现的,该函数根据设备和驱动的信息进行匹配,以确定设备和驱动之间的对应关系。设备树是配置板级信息的地方,它是管理设备、驱动总线的基础。在编写驱动时,我们使用注册函数驱动和设备进行匹配,并调用相应的函数进行处理。 #### 引用[.reference_title] - *1* [platform设备驱动](https://blog.csdn.net/weixin_52849254/article/details/127811193)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [Linux设备驱动和设备匹配过程](https://blog.csdn.net/tiantianhaoxinqing__/article/details/124843344)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值