一. 简介
PHY 子系统就是用于 PHY 设备相关内容的,分为 PHY 设备和 PHY 驱动,和 platform 总线一样,PHY 子系统也是一个设备、总线和驱动模型。
前面一篇文章学习了 PHY子系统中的 PHY设备。文章如下:
Linux 内核中PHY子系统(网络):PHY设备-CSDN博客
本文继续学习 PHY子系统中内容,具体学习 Linux内核中PHY子系统中的PHY驱动。
二. Linux 内核中PHY子系统(网络):PHY驱动
1. phy_driver结构体
PHY
驱动使用结构体
phy_driver
表示,结构体也定义在
include/linux/phy.h
文件中,结构体
内容如下
(
为了缩小篇幅,省略了注释部分
)
:
struct phy_driver {
u32 phy_id; /* PHY ID */
char *name;
unsigned int phy_id_mask; /* PHY ID 掩码 */
u32 features;
u32 flags;
const void *driver_data;
int (*soft_reset)(struct phy_device *phydev);
int (*config_init)(struct phy_device *phydev);
int (*probe)(struct phy_device *phydev);
int (*suspend)(struct phy_device *phydev);
int (*resume)(struct phy_device *phydev);
int (*config_aneg)(struct phy_device *phydev);
int (*aneg_done)(struct phy_device *phydev);
int (*read_status)(struct phy_device *phydev);
int (*ack_interrupt)(struct phy_device *phydev);
int (*config_intr)(struct phy_device *phydev);
int (*did_interrupt)(struct phy_device *phydev);
void (*remove)(struct phy_device *phydev);
int (*match_phy_device)(struct phy_device *phydev);
int (*ts_info)(struct phy_device *phydev, struct ethtool_ts_info *ti);
int (*hwtstamp)(struct phy_device *phydev, struct ifreq *ifr);
bool (*rxtstamp)(struct phy_device *dev, struct sk_buff *skb, int type);
void (*txtstamp)(struct phy_device *dev, struct sk_buff *skb,int type);
int (*set_wol)(struct phy_device *dev,struct ethtool_wolinfo *wol);
void (*get_wol)(struct phy_device *dev,struct ethtool_wolinfo *wol);
void (*link_change_notify)(struct phy_device *dev);
int (*read_mmd_indirect)(struct phy_device *dev, int ptrad, int devnum, int regnum);
void (*write_mmd_indirect)(struct phy_device *dev, int ptrad,int devnum, int regnum, u32 val);
int (*module_info)(struct phy_device *dev,struct ethtool_modinfo *modinfo);
int (*module_eeprom)(struct phy_device *dev,struct ethtool_eeprom *ee, u8 *data);
struct device_driver driver;
};
可以看出,phy_driver 重点是大量的函数,编写 PHY 驱动的主要工作就是实现这些函数, 但是不一定全部实现,稍后我们会简单分析一下 Linux 内核通用 PHY 驱动。
2. 注册 PHY 驱动
(1) 注册一个PHY驱动
phy_driver
结构体初始化完成以后,就需要向
Linux
内核注册,
PHY
驱动的注册使用
phy_driver_register
函数,注册
phy
驱动时候会设置驱动的总线为
mdio_bus_type
,也就是
MDIO
总线,关于
MDIO
总线稍后会讲解,函数原型如下:
int phy_driver_register(struct phy_driver *new_driver)
函数参数和返回值含义如下:
new_driver
:
需要注册的
PHY
驱动。
返回值:
0
成功,负值 失败。
(2) 连续注册多个 PHY 驱动
一个厂家会生产多种
PHY
芯片,这些
PHY
芯片内部差别一般不大,如果一个个的去注册驱动将会导致一堆重复的驱动文件,因此,
Linux
内核提供了一个连续注册多个
PHY
驱动的函数:
phy_drivers_register
函数。
首先准备一个
phy_driver
数组,一个数组元素就表示一个
PHY
芯片的驱
动,然后调用
phy_drivers_register
一次性注册整个数组中的所有驱动,函数原型如下:
int phy_drivers_register(struct phy_driver *new_driver, int n)
函数参数和返回值含义如下:
new_driver
:
需要注册的多个
PHY
驱动数组。
n
:
要注册的驱动数量。
返回值:
0
成功,负值 失败。
卸载
PHY
驱动的话使用
phy_driver_unregister
函数,函数原型如下:
void phy_driver_unregister(struct phy_driver *drv)
函数参数和返回值含义如下:
new_driver
:
需要卸载的
PHY
驱动。
返回值:
无。
主控需要配置或读取 PHY 芯片,也就是读写 PHY 的内部寄存器,所以还需要一个控制接口,叫做 MIDO。
MDIO 总线最主要的工作就是匹配 PHY 设备和 PHY 驱动。
接下来学习 MIDO接口总线。