dts解析过程(ZZ)

1. dtb文件解析,生成资源单项列表
-------------------------------------------------------------------
start_kernel() --> setup_arch() --> unflatten_device_tree()
该函数可以解析dtb文件,构建一个由device_node结构连接而成的单项链表。如下在此函数执行过后,在内存中会存在一个如下的链表

 

 

 

后面所有的函数,如果需要从of tree结构上读取设备资料的,都将从这个链表中遍历并读取。



2. 将allnodes显示在/proc/device-tree目录下
-------------------------------------------------------------------
linux-2.6.26
start_kernel() -->
#ifdef CONFIG_PROC_FS
proc_root_init(); -->
#endif
#ifdef CONFIG_PROC_DEVICETREE
proc_device_tree_init();-->
#endif
proc_device_tree_add_node(root, proc_device_tree);

此时将在/proc/device-tree目录下生成设备节点
# ls -al /proc/device-tree

3. of_platform_bus_type的注册/初始化
-------------------------------------------------------------------
linux2.6.28/arch/powerpc/kernel/of_platform.c
struct bus_type of_platform_bus_type = {
       .uevent   = of_device_uevent,
};
EXPORT_SYMBOL(of_platform_bus_type);

static int __init of_bus_driver_init(void)
{
   return of_bus_type_init(&of_platform_bus_type, "of_platform");
}
postcore_initcall(of_bus_driver_init);

of_platform_bus_type总线注册完毕。
此时/sys/bus/目录下将会有of_platform


4. 将allnodes设备节点添加到总线of_platform_bus_type
-------------------------------------------------------------------
arch/powerpc/platform/83xx/mpc831x_rdb.c
machine_device_initcall(mpc831x_rdb, declare_of_platform_devices);

declare_of_platform_devices() -->
of_platform_bus_probe(NULL, of_bus_ids, NULL)

arch/powerpc/kernel/of_platform.c
遍历第一步中在内存中生成链表的所有soc的子节点,将所有的soc子节点设备添加到of_platform总线。

-->
-->
of_device_register() -->
device_add()

of_platform总线上的所有设备添加完毕,e0024000.ethernet,e0024520.mdio等设备现在都在总线上。至此设备节点将出现在/sys/devices/platform/

 

3 mdio总线的注册
--------------------------------------------
/driver/net/phy_device.c

subsys_initcall(phy_init)
phy_init --> mdio_bus_init --> bus_register(&mdio_bus_type)

总线注册后,在总线上注册了一个默认的phy的驱动 genphy_driver:

.phy_id      = 0xffffffff,
.phy_id_mask = 0xffffffff,
.name        = "Generic phy",

mdio总线注册完毕。
/sys/bus/mdio


5 mdio总线上驱动的添加
--------------------------------------------

/driver/net/phy/marvell.c
module_init(marvell_init)
marvell_init() -->
phy_driver_register(&marvell_drivers[i]) -->
driver_register()

前面第三步,注册mdio总线后,已经添加了一个默认的phy的驱动,现在要将所有的phy驱动添加到总线上,这里将所有的marvell的phy都添加。

这步过后,内核的/sys/bus/mdio/driver里面就有了各种phy的驱动,但这时还没有和具体的设备绑定。


6 of_platform总线上mdio设备驱动该驱动的目的是在mdio总线上添加phy设备)的添加,并绑定设备:e0024520.mdio和e0025520.mdio

/driver/net/fsl_pq_mdio.c
module_init(fsl_pq_mdio_init)
fsl_pq_mdio_init --> of_register_platform_driver (&fsl_pq_mdio_driver) --> of_register_driver --> driver_register --> bus_add_driver --> driver_attach

遍历整个of_platform总线,寻找与之相匹配的设备,找到 e0024520.mdio
driver_attach --> __driver_attach --> driver_match_device
将driver的match_table里的信息和dev_nod中的做比较,若符合就进入driver的probe,也就是fsl_pq_mdio_probe。

现在of_platform总线上的设备e0024520.mdio和e0025520.mdio已经绑定了驱动。



7 mdio总线上的 设备的添加,寻找并绑定相应的驱动。
/driver/net/fsl_pq_mdio.c
fsl_pq_mdio_probe --> of_mdiobus_register --> phy_device_register --> device_register(&phydev->dev) --> device_add --> bus_probe_device --> device_attach -->bus_for_each_drv
扫描mdio总线上的所有的驱动,若找到匹配的,就绑定,并probe。
__device_attach --> driver_probe_device --> really_probe --> phy_probe

将所有的phy和tbi-phy的设备都添加到mdio总线上,并且两个phy设备和两个tbi-phy设备都会根据其自己的phyID找到各自的驱动

你可能感兴趣的:(dts解析过程(ZZ))