可以在电脑做公务员题的网站微信小程序排名关键词优化
一. 简介
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接口总线。