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;
......
};