在某些网络使用环境中,由于驱动的异常或特殊需求需要访问phy寄存器,去进行配置、状态获取或问题定位。本篇总结了3中访问方式。
方式一
/dev/mdio :有些原厂会提供mdio访问节点,应用层下使用open,write,read即可访问mdio总线,进行读写控制,该方法要求原厂驱动提供,非通用方式。如果你的系统存在该节点,那么恭喜你,可以直接访问!(具体访问方式可能因cpu厂家而异)
方式二
/dev/mem :mem节点可以用来映射物理寄存器,当你的系统中存在/dev/mem节点,而此时你又知道cpu的mdio访问寄存器地址和定义,那么可以使用mmap函数映射该地址,然后直接通过/dev/mem节点访问映射后的地址(mdio访问方式也会因cpu厂家而异,使用此种方式需要知道mdio访问流程,相当于将驱动层的访问方式移到应用层来做)
如果你的系统不存在/dev/mem节点,不要紧,可以通过内核配置解决该问题:
总结:该方式相当于将驱动层挪到应用层来实现,算是万金油的方式,不过需要知道的信息要全,要调试,并不是一种通用且省力的方式,属于万不得已才用的。
方式三
ioctrl :这是一种通用方法,一般在内核中对cpu mac(网络驱动)初始化的文件中,会提供一组ops函数用于应用层访问:
看一下内核ioctrl函数的内容,接收到SIOCGMIIPHY(获取phy地址),SIOCGMIIREG(读phy寄存器),SIOCSMIIREG(写phy)寄存器后最终调用了phy_mii_ioctrl函数,这是内核phy初始化中的标准函数:
phy_mii_ioctrl函数:可以看到在该函数中会根据cmd执行不同的操作,这里的mdiobus_read 函数最终还是要调用网络驱动初始化中定义好的访问接口,也就是方式二要做的访问接口函数,即最终操作寄存器的接口。(所以只要内核存在mdio的驱动接口都可以使用本方法)
要使用方式三,要保证网卡和phy是绑定的,否则不能正确访问,具体的做法则是在设备树中包含phy_handle属性:
如果在当前MAC下通过某种方式扩展了多个PHY,不要紧,仅定义其中一个phy_handle即可,这样就保证了mdio总线是和mac绑定在了一起
应用例程:
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <linux/mii.h> #include <sys/types.h> #include <sys/socket.h> #include <sys/ioctl.h> #include <net/if.h> #include <linux/sockios.h> #include <linux/types.h> #include <netinet/in.h> #include <unistd.h> //#include <QDebug> int main(int argc, char *argv[]) { int sockfd; struct mii_ioctl_data *mii = NULL; struct ifreq ifr; memset(&ifr, 0, sizeof(ifr)); strncpy(ifr.ifr_name, "eth0", IFNAMSIZ - 1); sockfd = socket(PF_LOCAL, SOCK_DGRAM, 0); ioctl(sockfd, SIOCGMIIPHY, &ifr); mii = (struct mii_ioctl_data*)&ifr.ifr_data; if(argc == 4) { strncpy(ifr.ifr_name, argv[1], IFNAMSIZ - 1); sockfd = socket(PF_LOCAL, SOCK_DGRAM, 0); ioctl(sockfd, SIOCGMIIPHY, &ifr); mii = (struct mii_ioctl_data*)&ifr.ifr_data; mii->phy_id = (uint16_t)strtoul(argv[2], NULL, 0); mii->reg_num = (uint16_t)strtoul(argv[3], NULL, 0); ioctl(sockfd, SIOCGMIIREG, &ifr); printf("read --- value : 0x%x", mii->val_out); } else if(argc == 5) { strncpy(ifr.ifr_name, argv[1], IFNAMSIZ - 1); sockfd = socket(PF_LOCAL, SOCK_DGRAM, 0); ioctl(sockfd, SIOCGMIIPHY, &ifr); mii = (struct mii_ioctl_data*)&ifr.ifr_data; mii->phy_id = (uint16_t)strtoul(argv[2], NULL, 0); mii->reg_num = (uint16_t)strtoul(argv[3], NULL, 0); mii->val_in = (uint16_t)strtoul(argv[4], NULL, 0); ioctl(sockfd, SIOCSMIIREG, &ifr); }else{ printf("mdio ethX phyId addr value\n"); } close(sockfd); return 0; }
使用方式:
写:
./mdio ethx phyaddr phyreg value
读:
./mdio ethx phyaddr phyreg