Linux下访问PHY芯片寄存器

在嵌入式开发中,可以通过smi/mdio总线通信访问PHY芯片寄存器,获取PHY芯片的状态。

总体思路:

可通过ioctl函数向内核发出控制命令读取phy芯片的寄存器。首先打开一个socket,然后通过系统管理员输入的参数初始化一个数据结构,并通过ioctl调用将数据传送到内核。SIOCSIFMTU是命令标识符。

struct ifreq data;
fd = socket(PF_INET, SOCK_DGRAM, 0);
< ... initialize "data" ...>
err = ioctl(fd, SIOCSIFMTU, &data);

ifreq结构定义在/usr/include/net/if.h,用来配置ip地址,激活接口,配置MTU等接口信息的。其中包含了一个接口的名字和具体内容——(是个共用体,有可能是IP地址,广播地址,子网掩码,MAC号,MTU或其他内容)。ifreq包含在ifconf结构中。而ifconf结构通常是用来保存所有接口的信息的。

使用ioctl命令时,可以看到上述两个命令的存在,

在头文件include/linux/sockios.h,定义如下:

#define **SIOCETHTOOL**    0x8946        /* Ethtool interface    */ethtool 接口
#define **SIOCGMIIPHY**    0x8947        /* Get address of MII PHY in use. */获取MII phy的地址
#define **SIOCGMIIREG**    0x8948        /* Read MII PHY register.    */读取 MII phy 寄存器
#define **SIOCSMIIREG**    0x8949        /* Write MII PHY register.    */写MII phy 寄存器

内核中对于上述两个命令的处理定义在文件 /drivers/net/phy/phy.c 的函数 phy_mii_ioctl中

关于MII补充:

MII即“媒体独立接口”,也叫“独立于介质的接口”。它是IEEE-802.3定义的以太网行业标准。它包括一个数据接口,以及一个MAC和PHY之间的管理接口。

RMII全称为“简化的媒体独立接口”,是IEEE-802.3u标准中除MII接口之外的另一种实现。

独立于介质的接口(MII) 用于MAC与外接的PHY互联,支持10Mbit/s和100Mbit/s数据传输模式。

C语言实现:

/*
Linux 下smi/mdio总线通信
下面代码描述了在用户层访问smi/mdio总线, 读写phy芯片寄存器的通用代码。Linux内核2.6以上通用。
将下面代码编译后,将可执行文件a.out 重命名为mdio
mdio eth0 1             读取phy寄存器1的数值
mdio eth0 0 0x1120      将0x1120写入 phy寄存器1
eth0 为mac层控制器的名称, 一般为eth0或mgmt0或ens33。
*/
 
#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>
 
 
#define reteck(ret)     \
        if(ret < 0){    \
            printf("%m! \"%s\" : line: %d\n", __func__, __LINE__);   \
            goto lab;   \
        }
 
#define help() \
    printf("mdio:\n");                  \
    printf("read operation: mdio reg_addr\n");          \
    printf("write operation: mdio reg_addr value\n");    \
    printf("For example:\n");            \
    printf("mdio eth0 1\n");             \
    printf("mdio eth0 0 0x12\n\n");      \
    exit(0);
 
int sockfd;
 
int main(int argc, char *argv[]){
        
    if(argc == 1 || !strcmp(argv[1], "-h")){
        help();
    }
    
    struct mii_ioctl_data *mii = NULL;
    struct ifreq ifr;
    int ret;
 
    memset(&ifr, 0, sizeof(ifr));
    strncpy(ifr.ifr_name, argv[1], IFNAMSIZ - 1);
 
    sockfd = socket(PF_LOCAL, SOCK_DGRAM, 0);
    reteck(sockfd);
 
    //get phy address in smi bus
    ret = ioctl(sockfd, SIOCGMIIPHY, &ifr);
    reteck(ret);
 
    mii = (struct mii_ioctl_data*)&ifr.ifr_data;
 
    if(argc == 3){
 
        mii->reg_num    = (uint16_t)strtoul(argv[2], NULL, 0);
        
        ret = ioctl(sockfd, SIOCGMIIREG, &ifr);
        reteck(ret);
    
        printf("read phy addr: 0x%x  reg: 0x%x   value : 0x%x\n\n", mii->phy_id, mii->reg_num, mii->val_out);
    }else if(argc == 4){
 
        mii->reg_num    = (uint16_t)strtoul(argv[2], NULL, 0);
        mii->val_in     = (uint16_t)strtoul(argv[3], NULL, 0);
 
        ret = ioctl(sockfd, SIOCSMIIREG, &ifr);
        reteck(ret);
 
        printf("write phy addr: 0x%x  reg: 0x%x  value : 0x%x\n\n", mii->phy_id, mii->reg_num, mii->val_in);
    }
 
lab:
    close(sockfd);
    return 0;
}

在read操作里面判断phy的link状态

if(mii->val_out& 0x0004){
    printf("linkup\n");
    }else{
    printf("linkdown\n");
}

补充知识

大多数phy芯片的寄存器0为控制寄存器, 寄存器1 为状态寄存器,寄存器3和4为 Identifiier Register,这里的内容为phy芯片产商的识别码。举个实例,marvell 88E1116,无论是光口模式还是电口模式,寄存器1都是 Status register。一般寄存器有16个bit, 第2个bit为link 状态, 第5个bit为自动协商, 一般这个状态寄存器的数值为: 0x796d。部分bit位说明如下

14bit : 有能力实现全双工100BASE-X工作模式

13bit : 有能力实现半双工 100BASE-X工作模式

12bit : 有能力实现全双工 10BASE-T工作模式

11 bit : 有能力实现半双工 10BASE-T工作模式

8bit : 扩展信息描述在寄存器

15.6bit : MF报头抑制

5bit : 自动协商完成

3bit : 有能力自动协商

2bit : link 状态: up

0bit : 有扩展寄存器

  • 1
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值