• 嵌入式linux开发,Linux下访问PHY芯片寄存器,获取phyID号,获取phy的link状态


    嵌入式linux开发,Linux下访问PHY芯片寄存器,获取phyID号,获取phy的link状态

    一、说明

    1、在嵌入式linux开发中,可以通过smi/mdio总线通信访问PHY芯片寄存器,从而获取PHY芯片的状态、ID号等相关信息。

    2、部分寄存器定义,如lan8720i

    SMI Register Map
    REGISTER INDEX
    (DECIMAL) 
    REGISTER NAMEGROUP
    0Basic Control RegisterBasic
    1Basic Status RegisterBasic
    2PHY Identifier 1Extended
    3PHY Identifier 2Extended
    4Auto-Negotiation Advertisement RegisterExtended
    5Auto-Negotiation Link Partner Ability RegisterExtended
    6Auto-Negotiation Expansion RegisterExtended
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17Mode Control/Status RegisterVendor-specific
    18Special ModesVendor-specific
    19
    20
    21
    22
    23
    24
    25
    26Symbol Error Counter RegisterVendor-specific
    27Control / Status Indication RegisterVendor-specific
    28
    29Interrupt Source RegisterVendor-specific
    30Interrupt Mask RegisterVendor-specific
    31PHY Special Control/Status RegisterVendor-specific

    3、phy的link状态

    通过状态寄存器Basic Status Register获取,取值mii->val_out& 0x0004。

    1. Link Status
    2. 0 = link is down
    3. 1 = link is up

    大多数phy芯片的寄存器0为控制寄存器, 寄存器1 为状态寄存器。寄存器3和4为 Identifiier Register,为phy芯片产商的识别码。

    寄存器1都是 Status register,有16个bit, 第2个bit为link 状态, 第5个bit为自动协商, 一般这个状态寄存器的数值为: 0x796d。部分bit位说明如下

    1. 14bit : 有能力实现全双工100BASE-X工作模式
    2. 13bit :  有能力实现半双工 100BASE-X工作模式
    3. 12bit : 有能力实现全双工 10BASE-T工作模式
    4. 11bit : 有能力实现半双工 10BASE-T工作模式
    5. 8bit : 扩展信息描述在寄存器15.
    6. 6bit :  MF报头抑制
    7. 5bit : 自动协商完成
    8. 3bit : 有能力自动协商
    9. 2bit : link 状态: up
    10. 0bit : 有扩展寄存器

    二、使用

    1、查看帮助

    ./a.out -h

    2、读取

    ./a.out eth0 1          	读取phy寄存器1的数值
    sudo ./a.out ens33 0

    3、写入

    ./a.out eth0 0 0x1120      将0x1120写入 phy寄存器1

    三、源码

    1. /*************************************************************************
    2. > File Name:
    3. > Author:
    4. > Mail:
    5. > Created Time:
    6. ************************************************************************/
    7. /*
    8. Linux 下smi/mdio总线通信
    9. 下面代码描述了在用户层访问smi/mdio总线, 读写phy芯片寄存器的通用代码。Linux内核2.6以上通用。
    10. 将下面代码编译后,将可执行文件a.out 重命名为mdio
    11. mdio eth0 1 读取phy寄存器1的数值
    12. mdio eth0 0 0x1120 将0x1120写入 phy寄存器1
    13. eth0 为mac层控制器的名称, 一般为eth0或mgmt0或ens33。
    14. */
    15. #include <stdio.h>
    16. #include <stdlib.h>
    17. #include <string.h>
    18. #include <linux/mii.h>
    19. #include <sys/types.h>
    20. #include <sys/socket.h>
    21. #include <sys/ioctl.h>
    22. #include <net/if.h>
    23. #include <linux/sockios.h>
    24. #include <linux/types.h>
    25. #include <netinet/in.h>
    26. // /* This structure is used in all SIOCxMIIxxx ioctl calls */
    27. // struct mii_ioctl_data {
    28. // __u16 phy_id;
    29. // __u16 reg_num;
    30. // __u16 val_in;
    31. // __u16 val_out;
    32. // };
    33. #define reteck(ret) \
    34. if(ret < 0){ \
    35. printf("%m! \"%s\" : line: %d\n", __func__, __LINE__); \
    36. goto lab; \
    37. }
    38. #define help() \
    39. printf("mdio:\n"); \
    40. printf("read operation: mdio reg_addr\n"); \
    41. printf("write operation: mdio reg_addr value\n"); \
    42. printf("For example:\n"); \
    43. printf("mdio eth0 1\n"); \
    44. printf("mdio eth0 0 0x12\n\n"); \
    45. exit(0);
    46. int sockfd;
    47. int main(int argc, char *argv[]){
    48. if(argc == 1 || !strcmp(argv[1], "-h")){
    49. help();
    50. }
    51. struct mii_ioctl_data *mii = NULL;
    52. struct ifreq ifr;
    53. int ret;
    54. memset(&ifr, 0, sizeof(ifr));
    55. strncpy(ifr.ifr_name, argv[1], IFNAMSIZ - 1);
    56. sockfd = socket(PF_LOCAL, SOCK_DGRAM, 0);
    57. reteck(sockfd);
    58. //get phy address in smi bus
    59. ret = ioctl(sockfd, SIOCGMIIPHY, &ifr);
    60. reteck(ret);
    61. mii = (struct mii_ioctl_data*)&ifr.ifr_data;
    62. if(argc == 3){
    63. mii->reg_num = (uint16_t)strtoul(argv[2], NULL, 0);
    64. ret = ioctl(sockfd, SIOCGMIIREG, &ifr);
    65. reteck(ret);
    66. printf("read phy addr: 0x%x reg: 0x%x value : 0x%x\n", mii->phy_id, mii->reg_num, mii->val_out);
    67. if (mii->reg_num == 0x1)
    68. {
    69. printf("Link Status\n");
    70. if(mii->val_out& 0x0004){
    71. printf("link is up\n");
    72. }else{
    73. printf("link is down\n");
    74. }
    75. }
    76. }else if(argc == 4){
    77. mii->reg_num = (uint16_t)strtoul(argv[2], NULL, 0);
    78. mii->val_in = (uint16_t)strtoul(argv[3], NULL, 0);
    79. ret = ioctl(sockfd, SIOCSMIIREG, &ifr);
    80. reteck(ret);
    81. printf("write phy addr: 0x%x reg: 0x%x value : 0x%x\n", mii->phy_id, mii->reg_num, mii->val_in);
    82. }
    83. lab:
    84. close(sockfd);
    85. return 0;
    86. }

  • 相关阅读:
    Docker目录映射
    Python数据库编程全指南SQLite和MySQL实践
    MongoDB升级步骤
    如何与Excel聊天?
    Nginx详细学习记录
    Linux系统安全与应用【一】
    如何像用自来水一样使用数据库|腾讯云数据库TDSQL-C
    思维模型 劳力辨证
    Qt 调SDK图像处理显示
    【ubuntu】本地访问ubuntu服务器的jupyter文件,.ipynb误删除找回、恢复
  • 原文地址:https://blog.csdn.net/weixin_43782998/article/details/125452500