Linux应用层通过mdio访问phy的3种方式

文章介绍了在Linux环境中访问PHY寄存器的三种方法:通过/dev/mdio节点、映射/dev/mem以及使用ioctl命令。每种方法都有其适用场景,其中ioctl是通用方法,通常在内核网络驱动中提供支持。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

        在某些网络使用环境中,由于驱动的异常或特殊需求需要访问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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Absorbed_w

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值