我是靠谱客的博主 哭泣秋天,最近开发中收集的这篇文章主要介绍Linux 下访问PHY芯片寄存器,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

Linux smi/mdio总线通信
韩大卫@吉林师范大学
下面代码描述了在用户层访问smimdio总线, 读写phy芯片寄存器的通用代码。Linux内核2.6以上通用。
将下面代码编译后,将可执行文件a.out 重命名为mdio
mdio eth0 1
读取phy寄存器1的数值
mdio eth0 0 0x1120
0x1120写入 phy寄存器1
eth0 mac层控制器的名称, 一般为eth0 mgmt0
#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: %dn", __func__, __LINE__);

goto lab;

}
#define help() 
printf("mdio:n");

printf("read operation: mdio reg_addrn");

printf("write operation: mdio reg_addr valuen");

printf("For example:n");

printf("mdio eth0 1n");

printf("mdio eth0 0 0x12nn");

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%xnn", 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%xnn", mii->phy_id, mii->reg_num, mii->val_in);
}
lab:
close(sockfd);
return 0;
}

很多人在read操作里面判断phylink状态,

if(mii->val_out& 0x0004){

printf("linkupn");

}else{

printf("linkdownn");

}

其实这个做法是比较通用可行的。


解释一下,关于mii->val_out& 0x0004

大多数phy芯片的寄存器0为控制寄存器, 寄存器1 为状态寄存器, 
寄存器34Identifiier Register
, 这里的内容为phy芯片产商的识别码。
举个实例,marvell 88E1116, 无论是光口模式还是电口模式, 寄存器1都是 Status register
一般寄存器有16bit, 第2bitlink 状态, 第5bit为自动协商, 
一般这个状态寄存器的数值为: 0x796d
意思是:
14bit : 有能力实现全双工100BASE-X工作模式
13bit :
有能力实现半双工 100BASE-X工作模式
12bit :
有能力实现全双工 10BASE-T工作模式
11 bit :
有能力实现半双工 10BASE-T工作模式
8bit : 扩展信息描述在寄存器15.
6bit :
MF报头抑制
5bit : 自动协商完成
3bit : 有能力自动协商
2bit link 状态: up
0bit : 有扩展寄存器

最后

以上就是哭泣秋天为你收集整理的Linux 下访问PHY芯片寄存器的全部内容,希望文章能够帮你解决Linux 下访问PHY芯片寄存器所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(39)

评论列表共有 0 条评论

立即
投稿
返回
顶部