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;
	......
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值