mdio.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * (C) Copyright 2011 Freescale Semiconductor, Inc
  4. * Andy Fleming
  5. */
  6. /*
  7. * MDIO Commands
  8. */
  9. #include <common.h>
  10. #include <command.h>
  11. #include <miiphy.h>
  12. #include <phy.h>
  13. static char last_op[2];
  14. static uint last_data;
  15. static uint last_addr_lo;
  16. static uint last_addr_hi;
  17. static uint last_devad_lo;
  18. static uint last_devad_hi;
  19. static uint last_reg_lo;
  20. static uint last_reg_hi;
  21. static int extract_range(char *input, int *plo, int *phi)
  22. {
  23. char *end;
  24. *plo = simple_strtol(input, &end, 16);
  25. if (end == input)
  26. return -1;
  27. if ((*end == '-') && *(++end))
  28. *phi = simple_strtol(end, NULL, 16);
  29. else if (*end == '\0')
  30. *phi = *plo;
  31. else
  32. return -1;
  33. return 0;
  34. }
  35. static int mdio_write_ranges(struct phy_device *phydev, struct mii_dev *bus,
  36. int addrlo,
  37. int addrhi, int devadlo, int devadhi,
  38. int reglo, int reghi, unsigned short data,
  39. int extended)
  40. {
  41. int addr, devad, reg;
  42. int err = 0;
  43. for (addr = addrlo; addr <= addrhi; addr++) {
  44. for (devad = devadlo; devad <= devadhi; devad++) {
  45. for (reg = reglo; reg <= reghi; reg++) {
  46. if (!extended)
  47. err = bus->write(bus, addr, devad,
  48. reg, data);
  49. else
  50. err = phydev->drv->writeext(phydev,
  51. addr, devad, reg, data);
  52. if (err)
  53. goto err_out;
  54. }
  55. }
  56. }
  57. err_out:
  58. return err;
  59. }
  60. static int mdio_read_ranges(struct phy_device *phydev, struct mii_dev *bus,
  61. int addrlo,
  62. int addrhi, int devadlo, int devadhi,
  63. int reglo, int reghi, int extended)
  64. {
  65. int addr, devad, reg;
  66. printf("Reading from bus %s\n", bus->name);
  67. for (addr = addrlo; addr <= addrhi; addr++) {
  68. printf("PHY at address %x:\n", addr);
  69. for (devad = devadlo; devad <= devadhi; devad++) {
  70. for (reg = reglo; reg <= reghi; reg++) {
  71. int val;
  72. if (!extended)
  73. val = bus->read(bus, addr, devad, reg);
  74. else
  75. val = phydev->drv->readext(phydev, addr,
  76. devad, reg);
  77. if (val < 0) {
  78. printf("Error\n");
  79. return val;
  80. }
  81. if (devad >= 0)
  82. printf("%d.", devad);
  83. printf("%d - 0x%x\n", reg, val & 0xffff);
  84. }
  85. }
  86. }
  87. return 0;
  88. }
  89. /* The register will be in the form [a[-b].]x[-y] */
  90. static int extract_reg_range(char *input, int *devadlo, int *devadhi,
  91. int *reglo, int *reghi)
  92. {
  93. char *regstr;
  94. /* use strrchr to find the last string after a '.' */
  95. regstr = strrchr(input, '.');
  96. /* If it exists, extract the devad(s) */
  97. if (regstr) {
  98. char devadstr[32];
  99. strncpy(devadstr, input, regstr - input);
  100. devadstr[regstr - input] = '\0';
  101. if (extract_range(devadstr, devadlo, devadhi))
  102. return -1;
  103. regstr++;
  104. } else {
  105. /* Otherwise, we have no devad, and we just got regs */
  106. *devadlo = *devadhi = MDIO_DEVAD_NONE;
  107. regstr = input;
  108. }
  109. return extract_range(regstr, reglo, reghi);
  110. }
  111. static int extract_phy_range(char *const argv[], int argc, struct mii_dev **bus,
  112. struct phy_device **phydev,
  113. int *addrlo, int *addrhi)
  114. {
  115. struct phy_device *dev = *phydev;
  116. if ((argc < 1) || (argc > 2))
  117. return -1;
  118. /* If there are two arguments, it's busname addr */
  119. if (argc == 2) {
  120. *bus = miiphy_get_dev_by_name(argv[0]);
  121. if (!*bus)
  122. return -1;
  123. return extract_range(argv[1], addrlo, addrhi);
  124. }
  125. /* It must be one argument, here */
  126. /*
  127. * This argument can be one of two things:
  128. * 1) Ethernet device name
  129. * 2) Just an address (use the previously-used bus)
  130. *
  131. * We check all buses for a PHY which is connected to an ethernet
  132. * device by the given name. If none are found, we call
  133. * extract_range() on the string, and see if it's an address range.
  134. */
  135. dev = mdio_phydev_for_ethname(argv[0]);
  136. if (dev) {
  137. *addrlo = *addrhi = dev->addr;
  138. *bus = dev->bus;
  139. return 0;
  140. }
  141. /* It's an address or nothing useful */
  142. return extract_range(argv[0], addrlo, addrhi);
  143. }
  144. /* ---------------------------------------------------------------- */
  145. static int do_mdio(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  146. {
  147. char op[2];
  148. int addrlo, addrhi, reglo, reghi, devadlo, devadhi;
  149. unsigned short data;
  150. int pos = argc - 1;
  151. struct mii_dev *bus;
  152. struct phy_device *phydev = NULL;
  153. int extended = 0;
  154. if (argc < 2)
  155. return CMD_RET_USAGE;
  156. /*
  157. * We use the last specified parameters, unless new ones are
  158. * entered.
  159. */
  160. op[0] = argv[1][0];
  161. addrlo = last_addr_lo;
  162. addrhi = last_addr_hi;
  163. devadlo = last_devad_lo;
  164. devadhi = last_devad_hi;
  165. reglo = last_reg_lo;
  166. reghi = last_reg_hi;
  167. data = last_data;
  168. bus = mdio_get_current_dev();
  169. if (flag & CMD_FLAG_REPEAT)
  170. op[0] = last_op[0];
  171. if (strlen(argv[1]) > 1) {
  172. op[1] = argv[1][1];
  173. if (op[1] == 'x') {
  174. phydev = mdio_phydev_for_ethname(argv[2]);
  175. if (phydev) {
  176. addrlo = phydev->addr;
  177. addrhi = addrlo;
  178. bus = phydev->bus;
  179. extended = 1;
  180. } else {
  181. return -1;
  182. }
  183. if (!phydev->drv ||
  184. (!phydev->drv->writeext && (op[0] == 'w')) ||
  185. (!phydev->drv->readext && (op[0] == 'r'))) {
  186. puts("PHY does not have extended functions\n");
  187. return -1;
  188. }
  189. }
  190. }
  191. switch (op[0]) {
  192. case 'w':
  193. if (pos > 1)
  194. data = simple_strtoul(argv[pos--], NULL, 16);
  195. case 'r':
  196. if (pos > 1)
  197. if (extract_reg_range(argv[pos--], &devadlo, &devadhi,
  198. &reglo, &reghi))
  199. return -1;
  200. default:
  201. if (pos > 1)
  202. if (extract_phy_range(&argv[2], pos - 1, &bus,
  203. &phydev, &addrlo, &addrhi))
  204. return -1;
  205. break;
  206. }
  207. if (op[0] == 'l') {
  208. mdio_list_devices();
  209. return 0;
  210. }
  211. /* Save the chosen bus */
  212. miiphy_set_current_dev(bus->name);
  213. switch (op[0]) {
  214. case 'w':
  215. mdio_write_ranges(phydev, bus, addrlo, addrhi, devadlo, devadhi,
  216. reglo, reghi, data, extended);
  217. break;
  218. case 'r':
  219. mdio_read_ranges(phydev, bus, addrlo, addrhi, devadlo, devadhi,
  220. reglo, reghi, extended);
  221. break;
  222. }
  223. /*
  224. * Save the parameters for repeats.
  225. */
  226. last_op[0] = op[0];
  227. last_addr_lo = addrlo;
  228. last_addr_hi = addrhi;
  229. last_devad_lo = devadlo;
  230. last_devad_hi = devadhi;
  231. last_reg_lo = reglo;
  232. last_reg_hi = reghi;
  233. last_data = data;
  234. return 0;
  235. }
  236. /***************************************************/
  237. U_BOOT_CMD(
  238. mdio, 6, 1, do_mdio,
  239. "MDIO utility commands",
  240. "list - List MDIO buses\n"
  241. "mdio read <phydev> [<devad>.]<reg> - "
  242. "read PHY's register at <devad>.<reg>\n"
  243. "mdio write <phydev> [<devad>.]<reg> <data> - "
  244. "write PHY's register at <devad>.<reg>\n"
  245. "mdio rx <phydev> [<devad>.]<reg> - "
  246. "read PHY's extended register at <devad>.<reg>\n"
  247. "mdio wx <phydev> [<devad>.]<reg> <data> - "
  248. "write PHY's extended register at <devad>.<reg>\n"
  249. "<phydev> may be:\n"
  250. " <busname> <addr>\n"
  251. " <addr>\n"
  252. " <eth name>\n"
  253. "<addr> <devad>, and <reg> may be ranges, e.g. 1-5.4-0x1f.\n"
  254. );