phy.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * (C) Copyright 2014
  4. * Dirk Eibach, Guntermann & Drunck GmbH, dirk.eibach@gdsys.cc
  5. */
  6. #include <common.h>
  7. #include <miiphy.h>
  8. enum {
  9. MIICMD_SET,
  10. MIICMD_MODIFY,
  11. MIICMD_VERIFY_VALUE,
  12. MIICMD_WAIT_FOR_VALUE,
  13. };
  14. struct mii_setupcmd {
  15. u8 token;
  16. u8 reg;
  17. u16 data;
  18. u16 mask;
  19. u32 timeout;
  20. };
  21. /*
  22. * verify we are talking to a 88e1518
  23. */
  24. struct mii_setupcmd verify_88e1518[] = {
  25. { MIICMD_SET, 22, 0x0000 },
  26. { MIICMD_VERIFY_VALUE, 2, 0x0141, 0xffff },
  27. { MIICMD_VERIFY_VALUE, 3, 0x0dd0, 0xfff0 },
  28. };
  29. /*
  30. * workaround for erratum mentioned in 88E1518 release notes
  31. */
  32. struct mii_setupcmd fixup_88e1518[] = {
  33. { MIICMD_SET, 22, 0x00ff },
  34. { MIICMD_SET, 17, 0x214b },
  35. { MIICMD_SET, 16, 0x2144 },
  36. { MIICMD_SET, 17, 0x0c28 },
  37. { MIICMD_SET, 16, 0x2146 },
  38. { MIICMD_SET, 17, 0xb233 },
  39. { MIICMD_SET, 16, 0x214d },
  40. { MIICMD_SET, 17, 0xcc0c },
  41. { MIICMD_SET, 16, 0x2159 },
  42. { MIICMD_SET, 22, 0x00fb },
  43. { MIICMD_SET, 7, 0xc00d },
  44. { MIICMD_SET, 22, 0x0000 },
  45. };
  46. /*
  47. * default initialization:
  48. * - set RGMII receive timing to "receive clock transition when data stable"
  49. * - set RGMII transmit timing to "transmit clock internally delayed"
  50. * - set RGMII output impedance target to 78,8 Ohm
  51. * - run output impedance calibration
  52. * - set autonegotiation advertise to 1000FD only
  53. */
  54. struct mii_setupcmd default_88e1518[] = {
  55. { MIICMD_SET, 22, 0x0002 },
  56. { MIICMD_MODIFY, 21, 0x0030, 0x0030 },
  57. { MIICMD_MODIFY, 25, 0x0000, 0x0003 },
  58. { MIICMD_MODIFY, 24, 0x8000, 0x8000 },
  59. { MIICMD_WAIT_FOR_VALUE, 24, 0x4000, 0x4000, 2000 },
  60. { MIICMD_SET, 22, 0x0000 },
  61. { MIICMD_MODIFY, 4, 0x0000, 0x01e0 },
  62. { MIICMD_MODIFY, 9, 0x0200, 0x0300 },
  63. };
  64. /*
  65. * turn off CLK125 for PHY daughterboard
  66. */
  67. struct mii_setupcmd ch1fix_88e1518[] = {
  68. { MIICMD_SET, 22, 0x0002 },
  69. { MIICMD_MODIFY, 16, 0x0006, 0x0006 },
  70. { MIICMD_SET, 22, 0x0000 },
  71. };
  72. /*
  73. * perform copper software reset
  74. */
  75. struct mii_setupcmd swreset_88e1518[] = {
  76. { MIICMD_SET, 22, 0x0000 },
  77. { MIICMD_MODIFY, 0, 0x8000, 0x8000 },
  78. { MIICMD_WAIT_FOR_VALUE, 0, 0x0000, 0x8000, 2000 },
  79. };
  80. /*
  81. * special one for 88E1514:
  82. * Force SGMII to Copper mode
  83. */
  84. struct mii_setupcmd mii_to_copper_88e1514[] = {
  85. { MIICMD_SET, 22, 0x0012 },
  86. { MIICMD_MODIFY, 20, 0x0001, 0x0007 },
  87. { MIICMD_MODIFY, 20, 0x8000, 0x8000 },
  88. { MIICMD_SET, 22, 0x0000 },
  89. };
  90. /*
  91. * turn off SGMII auto-negotiation
  92. */
  93. struct mii_setupcmd sgmii_autoneg_off_88e1518[] = {
  94. { MIICMD_SET, 22, 0x0001 },
  95. { MIICMD_MODIFY, 0, 0x0000, 0x1000 },
  96. { MIICMD_MODIFY, 0, 0x8000, 0x8000 },
  97. { MIICMD_SET, 22, 0x0000 },
  98. };
  99. /*
  100. * invert LED2 polarity
  101. */
  102. struct mii_setupcmd invert_led2_88e1514[] = {
  103. { MIICMD_SET, 22, 0x0003 },
  104. { MIICMD_MODIFY, 17, 0x0030, 0x0010 },
  105. { MIICMD_SET, 22, 0x0000 },
  106. };
  107. static int process_setupcmd(const char *bus, unsigned char addr,
  108. struct mii_setupcmd *setupcmd)
  109. {
  110. int res;
  111. u8 reg = setupcmd->reg;
  112. u16 data = setupcmd->data;
  113. u16 mask = setupcmd->mask;
  114. u32 timeout = setupcmd->timeout;
  115. u16 orig_data;
  116. unsigned long start;
  117. debug("mii %s:%u reg %2u ", bus, addr, reg);
  118. switch (setupcmd->token) {
  119. case MIICMD_MODIFY:
  120. res = miiphy_read(bus, addr, reg, &orig_data);
  121. if (res)
  122. break;
  123. debug("is %04x. (value %04x mask %04x) ", orig_data, data,
  124. mask);
  125. data = (orig_data & ~mask) | (data & mask);
  126. /* fallthrough */
  127. case MIICMD_SET:
  128. debug("=> %04x\n", data);
  129. res = miiphy_write(bus, addr, reg, data);
  130. break;
  131. case MIICMD_VERIFY_VALUE:
  132. res = miiphy_read(bus, addr, reg, &orig_data);
  133. if (res)
  134. break;
  135. if ((orig_data & mask) != (data & mask))
  136. res = -1;
  137. debug("(value %04x mask %04x) == %04x? %s\n", data, mask,
  138. orig_data, res ? "FAIL" : "PASS");
  139. break;
  140. case MIICMD_WAIT_FOR_VALUE:
  141. res = -1;
  142. start = get_timer(0);
  143. while ((res != 0) && (get_timer(start) < timeout)) {
  144. res = miiphy_read(bus, addr, reg, &orig_data);
  145. if (res)
  146. continue;
  147. if ((orig_data & mask) != (data & mask))
  148. res = -1;
  149. }
  150. debug("(value %04x mask %04x) == %04x? %s after %lu ms\n", data,
  151. mask, orig_data, res ? "FAIL" : "PASS",
  152. get_timer(start));
  153. break;
  154. default:
  155. res = -1;
  156. break;
  157. }
  158. return res;
  159. }
  160. static int process_setup(const char *bus, unsigned char addr,
  161. struct mii_setupcmd *setupcmd, unsigned int count)
  162. {
  163. int res = 0;
  164. unsigned int k;
  165. for (k = 0; k < count; ++k) {
  166. res = process_setupcmd(bus, addr, &setupcmd[k]);
  167. if (res) {
  168. printf("mii cmd %u on bus %s addr %u failed, aborting setup\n",
  169. setupcmd[k].token, bus, addr);
  170. break;
  171. }
  172. }
  173. return res;
  174. }
  175. int setup_88e1518(const char *bus, unsigned char addr)
  176. {
  177. int res;
  178. res = process_setup(bus, addr,
  179. verify_88e1518, ARRAY_SIZE(verify_88e1518));
  180. if (res)
  181. return res;
  182. res = process_setup(bus, addr,
  183. fixup_88e1518, ARRAY_SIZE(fixup_88e1518));
  184. if (res)
  185. return res;
  186. res = process_setup(bus, addr,
  187. default_88e1518, ARRAY_SIZE(default_88e1518));
  188. if (res)
  189. return res;
  190. if (addr) {
  191. res = process_setup(bus, addr,
  192. ch1fix_88e1518, ARRAY_SIZE(ch1fix_88e1518));
  193. if (res)
  194. return res;
  195. }
  196. res = process_setup(bus, addr,
  197. swreset_88e1518, ARRAY_SIZE(swreset_88e1518));
  198. if (res)
  199. return res;
  200. return 0;
  201. }
  202. int setup_88e1514(const char *bus, unsigned char addr)
  203. {
  204. int res;
  205. res = process_setup(bus, addr,
  206. verify_88e1518, ARRAY_SIZE(verify_88e1518));
  207. if (res)
  208. return res;
  209. res = process_setup(bus, addr,
  210. fixup_88e1518, ARRAY_SIZE(fixup_88e1518));
  211. if (res)
  212. return res;
  213. res = process_setup(bus, addr,
  214. mii_to_copper_88e1514,
  215. ARRAY_SIZE(mii_to_copper_88e1514));
  216. if (res)
  217. return res;
  218. res = process_setup(bus, addr,
  219. sgmii_autoneg_off_88e1518,
  220. ARRAY_SIZE(sgmii_autoneg_off_88e1518));
  221. if (res)
  222. return res;
  223. res = process_setup(bus, addr,
  224. invert_led2_88e1514,
  225. ARRAY_SIZE(invert_led2_88e1514));
  226. if (res)
  227. return res;
  228. res = process_setup(bus, addr,
  229. default_88e1518, ARRAY_SIZE(default_88e1518));
  230. if (res)
  231. return res;
  232. if (addr) {
  233. res = process_setup(bus, addr,
  234. ch1fix_88e1518, ARRAY_SIZE(ch1fix_88e1518));
  235. if (res)
  236. return res;
  237. }
  238. res = process_setup(bus, addr,
  239. swreset_88e1518, ARRAY_SIZE(swreset_88e1518));
  240. if (res)
  241. return res;
  242. return 0;
  243. }