mac和phy的结构
phy芯片通过mii接口和集成到cpu的mac相连;
phy有大量的模拟器件,所以外置;
mii总线
mii总线
7+7+2+2 18线
tx_clk
tx_err
tx_en
tx_dat[3:0]
rx_clk
rx_err
rx_dv
rx_dat[3:0]
crs 监测phy
col 监测phy
mdc
mkid
缺点:线太多
100M时钟25M
10M时钟2.5M
rmii
mii的精简版;
ref_clk 时钟,由mac提供或由外部晶振提供(50M)
tx_en
tx_dat[1:0]
rx_err
rx_dat[1:0]
crs_dv 载波和接收数据有效(监测)
mdc
mdio
100M下每个时钟传输一次数据;
10M下每10个时钟传一次数据;
smii
串行收发???
ref_clk
tx
rx
sync
mdc
mdio
GMII
千兆mii接口;
比mii多了一倍的数据线和一个千兆时钟线;
gtx_clk
tx_clk
tx_en
tx_err
tx_dat[7:0]
rx_clk
rx_dv
rx_err
rx_dat[7:0]
crs
col
mdc
mdio
RGMII
简化的gmii
tx_clk
tx_ctrl
tx_dat[3:0]
rx_clk
rx_ctrl
rx_dat[3:0]
mdc
mdio
10M时钟2.5M
100M时钟25M
1000兆时钟125M
1000M时的时钟和gmii一样;但是rgmii数据线减少了一半,控制线也是由两根合并成了一根;
解决方法是在上升沿和下降沿都有数据收发;
调试
注意点:
1、引脚配置,确认接口;
2、芯片id(0,1,2,3,4…)
3、时钟(25M,50M,125M)
mii-tool命令
参考博客:
Linux以太网卡架构解析-MAC层和PHY层
以太网详解(一)-MAC/PHY/MII/RMII/GMII/RGMII基本介绍
千兆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>
#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;
}