背景
在DM8168的EVM板中,两个EMAC模块分别连接到两个PHY上,用同一个MDIO模块管理,两个PHY配置为不同地址。内核启动后可以注册为两个eth设备。
然而在新研FXX板中,DM8168改变了网口连接方式,其EMAC0未引连接PHY,而是直接用GMII接口连接到对端某芯片上。因此需要修改内核驱动代码来完成适配。
分析
分析内核与网口有关的驱动可知,EMAC和MDIO都作为内核platform设备由platform虚拟总线来管理。因此对驱动的分析分为两部分,一是一是device部分,一是device_driver部分。在内核启动过程中,两者先后注册到总线上,然后适配在一起共同工作。
网口设备
两个网口和一个MDIO都是platform_device的设备。设备注册和初始化的调用关系为:
omap2_init_devices()
–>ti81xx_ethernet_init()
–>ti816x_ethernet_init()
。
先看omap2_init_devices()
,它被声明为:
======== arch/arm/mach-omap2/devices.c 2208 2208 ========
arch_initcall(omap2_init_devices);
因此是在系统初始化的某一个阶段调用的。
再看ti816x_ethernet_init()
函数:
- 把MAC地址等存入ti816x_emac1_pdata私有数据变量中
- 把ti816x_emac1_device设备的私有信息指针指向ti816x_emac1_pdata
- 调用
platform_device_register()
注册ti816x_emac1_device设备 - 调用
platform_device_register()
注册ti816x_mdio_device设备 - 与1~3步同样的方式注册ti816x_emac2_device设备
- 调用
ti816x_emac_mux()
把EMAC1的引脚配置为EMAC功能。
我们来看一看静态的网口私有变量结构体:
======== arch/arm/mach-omap2/devices.c 1125 1133 ========
static struct emac_platform_data ti816x_emac1_pdata = {
.rmii_en = 0,
.phy_id = "0:01",
};
static struct emac_platform_data ti816x_emac2_pdata = {
.rmii_en = 0