mii_phy.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. #include <common.h>
  2. #include <mii_phy.h>
  3. #include "rpxsuper.h"
  4. #define MII_MDIO 0x01
  5. #define MII_MDCK 0x02
  6. #define MII_MDIR 0x04
  7. void
  8. mii_discover_phy(void)
  9. {
  10. int known;
  11. unsigned short phy_reg;
  12. unsigned long phy_id;
  13. known = 0;
  14. printf("Discovering phy @ 0: ");
  15. phy_id = mii_phy_read(2) << 16;
  16. phy_id |= mii_phy_read(3);
  17. if ((phy_id & 0xFFFFFC00) == 0x00137800) {
  18. printf("Level One ");
  19. if ((phy_id & 0x000003F0) == 0xE0) {
  20. printf("LXT971A Revision %d\n", (int)(phy_id & 0xF));
  21. known = 1;
  22. }
  23. else printf("unknown type\n");
  24. }
  25. else printf("unknown OUI = 0x%08lX\n", phy_id);
  26. phy_reg = mii_phy_read(1);
  27. if (!(phy_reg & 0x0004)) printf("Link is down\n");
  28. if (!(phy_reg & 0x0020)) printf("Auto-negotiation not complete\n");
  29. if (phy_reg & 0x0002) printf("Jabber condition detected\n");
  30. if (phy_reg & 0x0010) printf("Remote fault condition detected \n");
  31. if (known) {
  32. phy_reg = mii_phy_read(17);
  33. if (phy_reg & 0x0400)
  34. printf("Phy operating at %d MBit/s in %s-duplex mode\n",
  35. phy_reg & 0x4000 ? 100 : 10,
  36. phy_reg & 0x0200 ? "full" : "half");
  37. else
  38. printf("bad link!!\n");
  39. /*
  40. left off: no link, green 100MBit, yellow 10MBit
  41. right off: no activity, green full-duplex, yellow half-duplex
  42. */
  43. mii_phy_write(20, 0x0452);
  44. }
  45. }
  46. unsigned short
  47. mii_phy_read(unsigned short reg)
  48. {
  49. int i;
  50. unsigned short tmp, val = 0, adr = 0;
  51. t_rpx_regs *regs = (t_rpx_regs*)CONFIG_SYS_REGS_BASE;
  52. tmp = 0x6002 | (adr << 7) | (reg << 2);
  53. regs->bcsr4 = 0xC3;
  54. for (i = 0; i < 64; i++) {
  55. regs->bcsr4 ^= MII_MDCK;
  56. }
  57. for (i = 0; i < 16; i++) {
  58. regs->bcsr4 &= ~MII_MDCK;
  59. if (tmp & 0x8000) regs->bcsr4 |= MII_MDIO;
  60. else regs->bcsr4 &= ~MII_MDIO;
  61. regs->bcsr4 |= MII_MDCK;
  62. tmp <<= 1;
  63. }
  64. regs->bcsr4 |= MII_MDIR;
  65. for (i = 0; i < 16; i++) {
  66. val <<= 1;
  67. regs->bcsr4 = MII_MDIO | (regs->bcsr4 | MII_MDCK);
  68. if (regs->bcsr4 & MII_MDIO) val |= 1;
  69. regs->bcsr4 = MII_MDIO | (regs->bcsr4 &= ~MII_MDCK);
  70. }
  71. return val;
  72. }
  73. void
  74. mii_phy_write(unsigned short reg, unsigned short val)
  75. {
  76. int i;
  77. unsigned short tmp, adr = 0;
  78. t_rpx_regs *regs = (t_rpx_regs*)CONFIG_SYS_REGS_BASE;
  79. tmp = 0x5002 | (adr << 7) | (reg << 2);
  80. regs->bcsr4 = 0xC3;
  81. for (i = 0; i < 64; i++) {
  82. regs->bcsr4 ^= MII_MDCK;
  83. }
  84. for (i = 0; i < 16; i++) {
  85. regs->bcsr4 &= ~MII_MDCK;
  86. if (tmp & 0x8000) regs->bcsr4 |= MII_MDIO;
  87. else regs->bcsr4 &= ~MII_MDIO;
  88. regs->bcsr4 |= MII_MDCK;
  89. tmp <<= 1;
  90. }
  91. for (i = 0; i < 16; i++) {
  92. regs->bcsr4 &= ~MII_MDCK;
  93. if (val & 0x8000) regs->bcsr4 |= MII_MDIO;
  94. else regs->bcsr4 &= ~MII_MDIO;
  95. regs->bcsr4 |= MII_MDCK;
  96. val <<= 1;
  97. }
  98. }