mii.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * (C) Copyright 2001
  4. * Gerald Van Baren, Custom IDEAS, vanbaren@cideas.com
  5. */
  6. /*
  7. * MII Utilities
  8. */
  9. #include <common.h>
  10. #include <command.h>
  11. #include <miiphy.h>
  12. typedef struct _MII_reg_desc_t {
  13. ushort regno;
  14. char * name;
  15. } MII_reg_desc_t;
  16. static const MII_reg_desc_t reg_0_5_desc_tbl[] = {
  17. { MII_BMCR, "PHY control register" },
  18. { MII_BMSR, "PHY status register" },
  19. { MII_PHYSID1, "PHY ID 1 register" },
  20. { MII_PHYSID2, "PHY ID 2 register" },
  21. { MII_ADVERTISE, "Autonegotiation advertisement register" },
  22. { MII_LPA, "Autonegotiation partner abilities register" },
  23. };
  24. typedef struct _MII_field_desc_t {
  25. ushort hi;
  26. ushort lo;
  27. ushort mask;
  28. char * name;
  29. } MII_field_desc_t;
  30. static const MII_field_desc_t reg_0_desc_tbl[] = {
  31. { 15, 15, 0x01, "reset" },
  32. { 14, 14, 0x01, "loopback" },
  33. { 13, 6, 0x81, "speed selection" }, /* special */
  34. { 12, 12, 0x01, "A/N enable" },
  35. { 11, 11, 0x01, "power-down" },
  36. { 10, 10, 0x01, "isolate" },
  37. { 9, 9, 0x01, "restart A/N" },
  38. { 8, 8, 0x01, "duplex" }, /* special */
  39. { 7, 7, 0x01, "collision test enable" },
  40. { 5, 0, 0x3f, "(reserved)" }
  41. };
  42. static const MII_field_desc_t reg_1_desc_tbl[] = {
  43. { 15, 15, 0x01, "100BASE-T4 able" },
  44. { 14, 14, 0x01, "100BASE-X full duplex able" },
  45. { 13, 13, 0x01, "100BASE-X half duplex able" },
  46. { 12, 12, 0x01, "10 Mbps full duplex able" },
  47. { 11, 11, 0x01, "10 Mbps half duplex able" },
  48. { 10, 10, 0x01, "100BASE-T2 full duplex able" },
  49. { 9, 9, 0x01, "100BASE-T2 half duplex able" },
  50. { 8, 8, 0x01, "extended status" },
  51. { 7, 7, 0x01, "(reserved)" },
  52. { 6, 6, 0x01, "MF preamble suppression" },
  53. { 5, 5, 0x01, "A/N complete" },
  54. { 4, 4, 0x01, "remote fault" },
  55. { 3, 3, 0x01, "A/N able" },
  56. { 2, 2, 0x01, "link status" },
  57. { 1, 1, 0x01, "jabber detect" },
  58. { 0, 0, 0x01, "extended capabilities" },
  59. };
  60. static const MII_field_desc_t reg_2_desc_tbl[] = {
  61. { 15, 0, 0xffff, "OUI portion" },
  62. };
  63. static const MII_field_desc_t reg_3_desc_tbl[] = {
  64. { 15, 10, 0x3f, "OUI portion" },
  65. { 9, 4, 0x3f, "manufacturer part number" },
  66. { 3, 0, 0x0f, "manufacturer rev. number" },
  67. };
  68. static const MII_field_desc_t reg_4_desc_tbl[] = {
  69. { 15, 15, 0x01, "next page able" },
  70. { 14, 14, 0x01, "(reserved)" },
  71. { 13, 13, 0x01, "remote fault" },
  72. { 12, 12, 0x01, "(reserved)" },
  73. { 11, 11, 0x01, "asymmetric pause" },
  74. { 10, 10, 0x01, "pause enable" },
  75. { 9, 9, 0x01, "100BASE-T4 able" },
  76. { 8, 8, 0x01, "100BASE-TX full duplex able" },
  77. { 7, 7, 0x01, "100BASE-TX able" },
  78. { 6, 6, 0x01, "10BASE-T full duplex able" },
  79. { 5, 5, 0x01, "10BASE-T able" },
  80. { 4, 0, 0x1f, "xxx to do" },
  81. };
  82. static const MII_field_desc_t reg_5_desc_tbl[] = {
  83. { 15, 15, 0x01, "next page able" },
  84. { 14, 14, 0x01, "acknowledge" },
  85. { 13, 13, 0x01, "remote fault" },
  86. { 12, 12, 0x01, "(reserved)" },
  87. { 11, 11, 0x01, "asymmetric pause able" },
  88. { 10, 10, 0x01, "pause able" },
  89. { 9, 9, 0x01, "100BASE-T4 able" },
  90. { 8, 8, 0x01, "100BASE-X full duplex able" },
  91. { 7, 7, 0x01, "100BASE-TX able" },
  92. { 6, 6, 0x01, "10BASE-T full duplex able" },
  93. { 5, 5, 0x01, "10BASE-T able" },
  94. { 4, 0, 0x1f, "xxx to do" },
  95. };
  96. typedef struct _MII_field_desc_and_len_t {
  97. const MII_field_desc_t *pdesc;
  98. ushort len;
  99. } MII_field_desc_and_len_t;
  100. static const MII_field_desc_and_len_t desc_and_len_tbl[] = {
  101. { reg_0_desc_tbl, ARRAY_SIZE(reg_0_desc_tbl) },
  102. { reg_1_desc_tbl, ARRAY_SIZE(reg_1_desc_tbl) },
  103. { reg_2_desc_tbl, ARRAY_SIZE(reg_2_desc_tbl) },
  104. { reg_3_desc_tbl, ARRAY_SIZE(reg_3_desc_tbl) },
  105. { reg_4_desc_tbl, ARRAY_SIZE(reg_4_desc_tbl) },
  106. { reg_5_desc_tbl, ARRAY_SIZE(reg_5_desc_tbl) },
  107. };
  108. static void dump_reg(
  109. ushort regval,
  110. const MII_reg_desc_t *prd,
  111. const MII_field_desc_and_len_t *pdl);
  112. static int special_field(
  113. ushort regno,
  114. const MII_field_desc_t *pdesc,
  115. ushort regval);
  116. static void MII_dump_0_to_5(
  117. ushort regvals[6],
  118. uchar reglo,
  119. uchar reghi)
  120. {
  121. ulong i;
  122. for (i = 0; i < 6; i++) {
  123. if ((reglo <= i) && (i <= reghi))
  124. dump_reg(regvals[i], &reg_0_5_desc_tbl[i],
  125. &desc_and_len_tbl[i]);
  126. }
  127. }
  128. static void dump_reg(
  129. ushort regval,
  130. const MII_reg_desc_t *prd,
  131. const MII_field_desc_and_len_t *pdl)
  132. {
  133. ulong i;
  134. ushort mask_in_place;
  135. const MII_field_desc_t *pdesc;
  136. printf("%u. (%04hx) -- %s --\n",
  137. prd->regno, regval, prd->name);
  138. for (i = 0; i < pdl->len; i++) {
  139. pdesc = &pdl->pdesc[i];
  140. mask_in_place = pdesc->mask << pdesc->lo;
  141. printf(" (%04hx:%04x) %u.",
  142. mask_in_place,
  143. regval & mask_in_place,
  144. prd->regno);
  145. if (special_field(prd->regno, pdesc, regval)) {
  146. }
  147. else {
  148. if (pdesc->hi == pdesc->lo)
  149. printf("%2u ", pdesc->lo);
  150. else
  151. printf("%2u-%2u", pdesc->hi, pdesc->lo);
  152. printf(" = %5u %s",
  153. (regval & mask_in_place) >> pdesc->lo,
  154. pdesc->name);
  155. }
  156. printf("\n");
  157. }
  158. printf("\n");
  159. }
  160. /* Special fields:
  161. ** 0.6,13
  162. ** 0.8
  163. ** 2.15-0
  164. ** 3.15-0
  165. ** 4.4-0
  166. ** 5.4-0
  167. */
  168. static int special_field(
  169. ushort regno,
  170. const MII_field_desc_t *pdesc,
  171. ushort regval)
  172. {
  173. if ((regno == MII_BMCR) && (pdesc->lo == 6)) {
  174. ushort speed_bits = regval & (BMCR_SPEED1000 | BMCR_SPEED100);
  175. printf("%2u,%2u = b%u%u speed selection = %s Mbps",
  176. 6, 13,
  177. (regval >> 6) & 1,
  178. (regval >> 13) & 1,
  179. speed_bits == BMCR_SPEED1000 ? "1000" :
  180. speed_bits == BMCR_SPEED100 ? "100" :
  181. "10");
  182. return 1;
  183. }
  184. else if ((regno == MII_BMCR) && (pdesc->lo == 8)) {
  185. printf("%2u = %5u duplex = %s",
  186. pdesc->lo,
  187. (regval >> pdesc->lo) & 1,
  188. ((regval >> pdesc->lo) & 1) ? "full" : "half");
  189. return 1;
  190. }
  191. else if ((regno == MII_ADVERTISE) && (pdesc->lo == 0)) {
  192. ushort sel_bits = (regval >> pdesc->lo) & pdesc->mask;
  193. printf("%2u-%2u = %5u selector = %s",
  194. pdesc->hi, pdesc->lo, sel_bits,
  195. sel_bits == PHY_ANLPAR_PSB_802_3 ?
  196. "IEEE 802.3" :
  197. sel_bits == PHY_ANLPAR_PSB_802_9 ?
  198. "IEEE 802.9 ISLAN-16T" :
  199. "???");
  200. return 1;
  201. }
  202. else if ((regno == MII_LPA) && (pdesc->lo == 0)) {
  203. ushort sel_bits = (regval >> pdesc->lo) & pdesc->mask;
  204. printf("%2u-%2u = %u selector = %s",
  205. pdesc->hi, pdesc->lo, sel_bits,
  206. sel_bits == PHY_ANLPAR_PSB_802_3 ?
  207. "IEEE 802.3" :
  208. sel_bits == PHY_ANLPAR_PSB_802_9 ?
  209. "IEEE 802.9 ISLAN-16T" :
  210. "???");
  211. return 1;
  212. }
  213. return 0;
  214. }
  215. static char last_op[2];
  216. static uint last_data;
  217. static uint last_addr_lo;
  218. static uint last_addr_hi;
  219. static uint last_reg_lo;
  220. static uint last_reg_hi;
  221. static uint last_mask;
  222. static void extract_range(
  223. char * input,
  224. unsigned char * plo,
  225. unsigned char * phi)
  226. {
  227. char * end;
  228. *plo = simple_strtoul(input, &end, 16);
  229. if (*end == '-') {
  230. end++;
  231. *phi = simple_strtoul(end, NULL, 16);
  232. }
  233. else {
  234. *phi = *plo;
  235. }
  236. }
  237. /* ---------------------------------------------------------------- */
  238. static int do_mii(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  239. {
  240. char op[2];
  241. unsigned char addrlo, addrhi, reglo, reghi;
  242. unsigned char addr, reg;
  243. unsigned short data, mask;
  244. int rcode = 0;
  245. const char *devname;
  246. if (argc < 2)
  247. return CMD_RET_USAGE;
  248. #if defined(CONFIG_MII_INIT)
  249. mii_init ();
  250. #endif
  251. /*
  252. * We use the last specified parameters, unless new ones are
  253. * entered.
  254. */
  255. op[0] = last_op[0];
  256. op[1] = last_op[1];
  257. addrlo = last_addr_lo;
  258. addrhi = last_addr_hi;
  259. reglo = last_reg_lo;
  260. reghi = last_reg_hi;
  261. data = last_data;
  262. mask = last_mask;
  263. if ((flag & CMD_FLAG_REPEAT) == 0) {
  264. op[0] = argv[1][0];
  265. if (strlen(argv[1]) > 1)
  266. op[1] = argv[1][1];
  267. else
  268. op[1] = '\0';
  269. if (argc >= 3)
  270. extract_range(argv[2], &addrlo, &addrhi);
  271. if (argc >= 4)
  272. extract_range(argv[3], &reglo, &reghi);
  273. if (argc >= 5)
  274. data = simple_strtoul(argv[4], NULL, 16);
  275. if (argc >= 6)
  276. mask = simple_strtoul(argv[5], NULL, 16);
  277. }
  278. if (addrhi > 31 && strncmp(op, "de", 2)) {
  279. printf("Incorrect PHY address. Range should be 0-31\n");
  280. return CMD_RET_USAGE;
  281. }
  282. /* use current device */
  283. devname = miiphy_get_current_dev();
  284. /*
  285. * check info/read/write.
  286. */
  287. if (op[0] == 'i') {
  288. unsigned char j, start, end;
  289. unsigned int oui;
  290. unsigned char model;
  291. unsigned char rev;
  292. /*
  293. * Look for any and all PHYs. Valid addresses are 0..31.
  294. */
  295. if (argc >= 3) {
  296. start = addrlo; end = addrhi;
  297. } else {
  298. start = 0; end = 31;
  299. }
  300. for (j = start; j <= end; j++) {
  301. if (miiphy_info (devname, j, &oui, &model, &rev) == 0) {
  302. printf("PHY 0x%02X: "
  303. "OUI = 0x%04X, "
  304. "Model = 0x%02X, "
  305. "Rev = 0x%02X, "
  306. "%3dbase%s, %s\n",
  307. j, oui, model, rev,
  308. miiphy_speed (devname, j),
  309. miiphy_is_1000base_x (devname, j)
  310. ? "X" : "T",
  311. (miiphy_duplex (devname, j) == FULL)
  312. ? "FDX" : "HDX");
  313. }
  314. }
  315. } else if (op[0] == 'r') {
  316. for (addr = addrlo; addr <= addrhi; addr++) {
  317. for (reg = reglo; reg <= reghi; reg++) {
  318. data = 0xffff;
  319. if (miiphy_read (devname, addr, reg, &data) != 0) {
  320. printf(
  321. "Error reading from the PHY addr=%02x reg=%02x\n",
  322. addr, reg);
  323. rcode = 1;
  324. } else {
  325. if ((addrlo != addrhi) || (reglo != reghi))
  326. printf("addr=%02x reg=%02x data=",
  327. (uint)addr, (uint)reg);
  328. printf("%04X\n", data & 0x0000FFFF);
  329. }
  330. }
  331. if ((addrlo != addrhi) && (reglo != reghi))
  332. printf("\n");
  333. }
  334. } else if (op[0] == 'w') {
  335. for (addr = addrlo; addr <= addrhi; addr++) {
  336. for (reg = reglo; reg <= reghi; reg++) {
  337. if (miiphy_write (devname, addr, reg, data) != 0) {
  338. printf("Error writing to the PHY addr=%02x reg=%02x\n",
  339. addr, reg);
  340. rcode = 1;
  341. }
  342. }
  343. }
  344. } else if (op[0] == 'm') {
  345. for (addr = addrlo; addr <= addrhi; addr++) {
  346. for (reg = reglo; reg <= reghi; reg++) {
  347. unsigned short val = 0;
  348. if (miiphy_read(devname, addr,
  349. reg, &val)) {
  350. printf("Error reading from the PHY");
  351. printf(" addr=%02x", addr);
  352. printf(" reg=%02x\n", reg);
  353. rcode = 1;
  354. } else {
  355. val = (val & ~mask) | (data & mask);
  356. if (miiphy_write(devname, addr,
  357. reg, val)) {
  358. printf("Error writing to the PHY");
  359. printf(" addr=%02x", addr);
  360. printf(" reg=%02x\n", reg);
  361. rcode = 1;
  362. }
  363. }
  364. }
  365. }
  366. } else if (strncmp(op, "du", 2) == 0) {
  367. ushort regs[6];
  368. int ok = 1;
  369. if ((reglo > 5) || (reghi > 5)) {
  370. printf(
  371. "The MII dump command only formats the "
  372. "standard MII registers, 0-5.\n");
  373. return 1;
  374. }
  375. for (addr = addrlo; addr <= addrhi; addr++) {
  376. for (reg = reglo; reg < reghi + 1; reg++) {
  377. if (miiphy_read(devname, addr, reg, &regs[reg]) != 0) {
  378. ok = 0;
  379. printf(
  380. "Error reading from the PHY addr=%02x reg=%02x\n",
  381. addr, reg);
  382. rcode = 1;
  383. }
  384. }
  385. if (ok)
  386. MII_dump_0_to_5(regs, reglo, reghi);
  387. printf("\n");
  388. }
  389. } else if (strncmp(op, "de", 2) == 0) {
  390. if (argc == 2)
  391. miiphy_listdev ();
  392. else
  393. miiphy_set_current_dev (argv[2]);
  394. } else {
  395. return CMD_RET_USAGE;
  396. }
  397. /*
  398. * Save the parameters for repeats.
  399. */
  400. last_op[0] = op[0];
  401. last_op[1] = op[1];
  402. last_addr_lo = addrlo;
  403. last_addr_hi = addrhi;
  404. last_reg_lo = reglo;
  405. last_reg_hi = reghi;
  406. last_data = data;
  407. last_mask = mask;
  408. return rcode;
  409. }
  410. /***************************************************/
  411. U_BOOT_CMD(
  412. mii, 6, 1, do_mii,
  413. "MII utility commands",
  414. "device - list available devices\n"
  415. "mii device <devname> - set current device\n"
  416. "mii info <addr> - display MII PHY info\n"
  417. "mii read <addr> <reg> - read MII PHY <addr> register <reg>\n"
  418. "mii write <addr> <reg> <data> - write MII PHY <addr> register <reg>\n"
  419. "mii modify <addr> <reg> <data> <mask> - modify MII PHY <addr> register <reg>\n"
  420. " updating bits identified in <mask>\n"
  421. "mii dump <addr> <reg> - pretty-print <addr> <reg> (0-5 only)\n"
  422. "Addr and/or reg may be ranges, e.g. 2-7."
  423. );