news 2026/4/17 1:18:18

Linux MDIO子系统深度剖析:从原理到实践(5)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Linux MDIO子系统深度剖析:从原理到实践(5)

接前一篇文章:Linux MDIO子系统深度剖析:从原理到实践(4)

五、用户空间访问实例

1. 使用ioctl访问PHY寄存器

虽然大多数情况下PHY的管理由内核驱动自动处理,但在调试或特殊应用场景中,用户空间程序可能需要直接访问PHY寄存器。Linux提供了通过socket ioctl接口访问PHY寄存器的能力,这为用户空间工具提供了底层的PHY管理功能。

主要的ioctl命令包括:

  • SIOCGMIIPHY:获取MDIO总线上的当前PHY地址。
  • SIOCGMIIREG:读取指定PHY寄存器的值。
  • SIOCSMIIREG:设置指定PHY寄存器的值。

这些ioctl命令使用struct mii_ioctl_data作为数据传输结构,其定义如下:

struct mii_ioctl_data { __u16 phy_id; // PHY 设备地址 __u16 reg_num; // 寄存器地址 __u16 val_in; // 写入的值(用于写操作) __u16 val_out; // 读取的值(用于读操作) };

这种用户空间的访问机制类似于提供了一个硬件调试终端,让开发者和系统管理员能够直接与PHY芯片"对话",检查状态、修改配置或进行故障诊断。

2. 完整代码示例

以下是一个完整的用户空间MDIO工具实现,支持读取和写入PHY寄存器。这个工具可以编译为命令行程序,用于调试和配置PHY设备。

#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: 一个用户空间 MDIO 调试工具\n"); \ printf("读取操作: mdio 接口名 寄存器地址\n"); \ printf("写入操作: mdio 接口名 寄存器地址 数值\n"); \ printf("示例:\n"); \ printf("mdio eth0 1\t# 读取 eth0 的 PHY 寄存器 1\n"); \ printf("mdio eth0 0 0x1120\t# 向 eth0 的 PHY 寄存器 0 写入 0x1120\n\n"); \ int sockfd; int main(int argc, char *argv[]){ if(argc == 1 || !strcmp(argv[1], "-h")){ help(); return 0; } 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); // 获取 PHY 地址 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("读取 PHY 地址: 0x%x 寄存器: 0x%x 值: 0x%x\n", mii->phy_id, mii->reg_num, mii->val_out); // 检查链路状态(寄存器 1 的第 2 位) if (mii->reg_num == 1) { if (mii->val_out & 0x0004) { printf("链路状态: UP\n"); } else { printf("链路状态: DOWN\n"); } } } 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(sockwd, SIOCSMIIREG, &ifr); reteck(ret); printf("写入 PHY 地址: 0x%x 寄存器: 0x%x 值: 0x%x\n", mii->phy_id, mii->reg_num, mii->val_in); } else { help(); } lab: if (sockfd >= 0) close(sockfd); return 0; }

3. 编译与运行

这个MDIO工具可以很容易地编译和使用:

(1)编译

gcc -o mdio mdio_tool.c

(2)安装(可选)

sudo cp mdio /usr/local/bin/

(3)使用示例

  • 读取PHY状态寄存器(寄存器 1)
./mdio eth0 1

输出结果:

读取 PHY 地址: 0x1 寄存器: 0x1 值: 0x796d 链路状态: UP
  • 写入PHY控制寄存器(寄存器0)
./mdio eth0 0 0x1140

输出结果:

写入 PHY 地址: 0x1 寄存器: 0x0 值: 0x1140
  • 检查链路状态
./mdio eth0 1

然后检查输出中的值,寄存器1的第2位(从0开始计数)表示链路状态:1表示链路UP,0表示链路DOWN。

这个工具提供了对PHY寄存器的底层访问能力,对于调试网络连接问题、验证PHY配置或学习 MDIO总线工作原理都非常有用。但需要注意的是,不当的寄存器修改可能会导致网络连接中断,因此在生产环境中使用需要格外小心。

更多内容请看下回。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 1:03:42

如何甄别一家靠谱的背调公司?

如何选择一家可靠的背调服务商&#xff0c;是企业招聘中关键却常被忽视的一环。在信息繁杂的市场中&#xff0c;甄别需要回归核心&#xff0c;关注以下几个务实维度。首先&#xff0c;核查其数据源的合法性与广度。专业的背调公司应具备多元、合规的信息获取渠道&#xff0c;而…

作者头像 李华
网站建设 2026/4/16 13:02:54

山海鲸占用空间越来越大?软件瘦身攻略来了

一、概述 山海鲸可视化在使用过程中&#xff0c;我们可能会下载大量数据、组件和模型&#xff0c;最终导致山海鲸可视化占用存储空间越来越大。 下面就为大家详细介绍一下&#xff0c;应该如何减少山海鲸可视化对存储空间的占用。 二、项目清理 大量过往项目没有及时得到清理…

作者头像 李华
网站建设 2026/4/15 16:33:32

2026年,还能继续入网络安全行业吗?那些行业不敢说的秘密

收藏&#xff01;2026年网安行业生存指南&#xff1a;不推荐入行&#xff1f;先了解真相再决定 文章揭示2026年网络安全行业面临就业难、收入不稳定、企业重视度低等挑战。企业需要能发现漏洞、写工具、理解攻防逻辑的人才&#xff0c;而非只会使用工具的"工具人"。…

作者头像 李华
网站建设 2026/4/16 11:10:35

交流调压那些事儿:单相、三相及带中性线三相交流调压

单相交流调压&#xff0c;三相交流调压&#xff0c;三相交流调压带中性线。 &#xff08;改变触发角和负载三相桥式半控整流电路仿真&#xff0c;改变触发角或负载得到不同的波形&#xff09; 在电力电子领域&#xff0c;交流调压是一项极为重要的技术&#xff0c;它广泛应用于…

作者头像 李华
网站建设 2026/4/16 13:00:03

探索 S7 - 200 组态王构建全自动洗衣机控制系统

S7-200组态王基于PLC的全自动洗衣机工业家用变频洗衣机控制系统 带解释的梯形图程序&#xff0c;接线图原理图图纸&#xff0c;io分配&#xff0c;组态画面在工业与家用领域&#xff0c;基于 PLC 的全自动洗衣机控制系统越来越受到关注。今天咱们就来聊聊利用 S7 - 200 与组态王…

作者头像 李华