mdiodump.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * mdiodump.c - Qualcomm Atheros MDIO Custom Module Analyser
  11. *
  12. * Contributor(s):
  13. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  14. * Charles Maier <charles.maier@qca.qualcomm.com>
  15. * Marc Bertola <marc.bertola@qca.qualcomm.com>
  16. *
  17. *--------------------------------------------------------------------*/
  18. #define _GETOPT_H
  19. /*====================================================================*
  20. * system header files;
  21. *--------------------------------------------------------------------*/
  22. #include <errno.h>
  23. #include <fcntl.h>
  24. #include <stdio.h>
  25. #include <stdlib.h>
  26. #include <unistd.h>
  27. /*====================================================================*
  28. * custom header files;
  29. *--------------------------------------------------------------------*/
  30. #include "../tools/getoptv.h"
  31. #include "../tools/flags.h"
  32. #include "../tools/error.h"
  33. #include "../tools/files.h"
  34. #include "../tools/endian.h"
  35. #include "../tools/symbol.h"
  36. /*====================================================================*
  37. * custom source files;
  38. *--------------------------------------------------------------------*/
  39. #ifndef MAKEFILE
  40. #include "../tools/getoptv.c"
  41. #include "../tools/putoptv.c"
  42. #include "../tools/version.c"
  43. #include "../tools/hexstring.c"
  44. #include "../tools/hexdecode.c"
  45. #include "../tools/codelist.c"
  46. #include "../tools/error.c"
  47. #include "../tools/lookup.c"
  48. #include "../tools/assist.c"
  49. #endif
  50. /*====================================================================*
  51. * program constants;
  52. *--------------------------------------------------------------------*/
  53. #define MDIODUMP_SUMMARY (1 << 0)
  54. #define MDIODUMP_VERBOSE (1 << 1)
  55. #define MDIO32_NORMAL 0x00
  56. #define MDIO32_ACCESS_USING_HIGH 0x02
  57. #define MDIO32_SET_HIGH 0x03
  58. /*====================================================================*
  59. * supported PHY types;
  60. *--------------------------------------------------------------------*/
  61. #define PHY_GENERIC 0
  62. #define PHY_AR8236 1
  63. struct _code_ switches [] =
  64. {
  65. {
  66. PHY_GENERIC,
  67. "generic"
  68. },
  69. {
  70. PHY_AR8236,
  71. "ar8236"
  72. }
  73. };
  74. /*====================================================================*
  75. * command structure;
  76. *--------------------------------------------------------------------*/
  77. #ifndef __GNUC__
  78. #pragma pack (push,1)
  79. #endif
  80. struct __packed command
  81. {
  82. uint16_t ctrl;
  83. uint16_t data;
  84. uint16_t mask;
  85. };
  86. #ifndef __GNUC__
  87. #pragma pack (pop)
  88. #endif
  89. /*====================================================================*
  90. * register & memmap struction
  91. *--------------------------------------------------------------------*/
  92. #define PHY_REGISTER 0
  93. #define GBL_REGISTER 1
  94. struct reg
  95. {
  96. uint32_t address;
  97. uint8_t phy;
  98. uint8_t reg;
  99. uint32_t content;
  100. uint8_t type;
  101. };
  102. struct memmap
  103. {
  104. unsigned size;
  105. unsigned used;
  106. struct reg * reg;
  107. };
  108. /*====================================================================*
  109. *
  110. * signed write_phy_reg (struct memmap * memmap, uint8_t phy, uint8_t reg, uint16_t data, uint16_t mask);
  111. *
  112. *
  113. *--------------------------------------------------------------------*/
  114. static signed write_phy_reg (struct memmap * memmap, uint8_t phy, uint8_t reg, uint16_t data, uint16_t mask)
  115. {
  116. unsigned i;
  117. for (i = 0; i < memmap->used; ++ i)
  118. {
  119. if (memmap->reg [i].type != PHY_REGISTER)
  120. {
  121. continue;
  122. }
  123. if (memmap->reg [i].phy != phy)
  124. {
  125. continue;
  126. }
  127. if (memmap->reg [i].reg == reg)
  128. {
  129. continue;
  130. }
  131. memmap->reg [i].content &= mask;
  132. memmap->reg [i].content |= mask & data;
  133. return (0);
  134. }
  135. if (memmap->used < memmap->size)
  136. {
  137. memmap->reg [i].phy = phy;
  138. memmap->reg [i].reg = reg;
  139. memmap->reg [i].content = mask & data;
  140. memmap->reg [i].type = PHY_REGISTER;
  141. memmap->used++;
  142. return (0);
  143. }
  144. error (1, 0, "not enough registers to run simulation");
  145. return (- 1);
  146. }
  147. /*====================================================================*
  148. *
  149. * signed write_gbl_reg (struct memmap *memmap, uint32_t address, uint8_t upper, uint16_t data, uint16_t mask);
  150. *
  151. *
  152. *--------------------------------------------------------------------*/
  153. static signed write_gbl_reg (struct memmap * memmap, uint32_t address, uint8_t upper, uint16_t data, uint16_t mask)
  154. {
  155. unsigned i;
  156. for (i = 0; i < memmap->used; ++ i)
  157. {
  158. if (memmap->reg [i].type != GBL_REGISTER)
  159. {
  160. continue;
  161. }
  162. if (memmap->reg [i].address != address)
  163. {
  164. continue;
  165. }
  166. if (upper)
  167. {
  168. memmap->reg [i].content &= (mask << 16) | 0x0000FFFF;
  169. memmap->reg [i].content |= (mask & data) << 16;
  170. }
  171. else
  172. {
  173. memmap->reg [i].content &= mask | 0xFFFF0000;
  174. memmap->reg [i].content |= mask & data;
  175. }
  176. return (0);
  177. }
  178. if (memmap->used < memmap->size)
  179. {
  180. memmap->reg [i].address = address;
  181. memmap->reg [i].content = mask & data;
  182. if (upper)
  183. {
  184. memmap->reg [i].content <<= 16;
  185. }
  186. memmap->reg [i].type = GBL_REGISTER;
  187. memmap->used++;
  188. return (0);
  189. }
  190. error (1, 0, "not enough registers to run simulation");
  191. return (- 1);
  192. }
  193. #if 0
  194. /*====================================================================*
  195. *
  196. * signed read_phy_reg (struct memmap * memmap, uint8_t phy, uint8_t reg, uint32_t * data);
  197. *
  198. *
  199. *--------------------------------------------------------------------*/
  200. static signed read_phy_reg (struct memmap * memmap, uint8_t phy, uint8_t reg, uint32_t * data)
  201. {
  202. unsigned i;
  203. for (i = 0; i < memmap->used; ++ i)
  204. {
  205. if (memmap->reg [i].type != PHY_REGISTER)
  206. {
  207. continue;
  208. }
  209. if (memmap->reg [i].phy != phy)
  210. {
  211. continue;
  212. }
  213. if (memmap->reg [i].reg != reg)
  214. {
  215. continue;
  216. }
  217. * data = memmap->reg [i].content;
  218. return (0);
  219. }
  220. return (- 1);
  221. }
  222. /*====================================================================*
  223. *
  224. * signed read_gbl_reg (struct memmap * memmap, uint32_t address, uint32_t * content);
  225. *
  226. *
  227. *--------------------------------------------------------------------*/
  228. static signed read_gbl_reg (struct memmap * memmap, uint32_t address, uint32_t * content)
  229. {
  230. unsigned i;
  231. for (i = 0; i < memmap->used; ++ i)
  232. {
  233. if (memmap->reg [i].type != GBL_REGISTER)
  234. {
  235. continue;
  236. }
  237. if (memmap->reg [i].address != address)
  238. {
  239. continue;
  240. }
  241. * content = memmap->reg [i].content;
  242. return (0);
  243. }
  244. return (- 1);
  245. }
  246. #endif
  247. /*====================================================================*
  248. *
  249. * void print_memmap (struct memmap * memmap);
  250. *
  251. *
  252. *
  253. *--------------------------------------------------------------------*/
  254. static void print_memmap (struct memmap * memmap)
  255. {
  256. unsigned i;
  257. for (i = 0; i < memmap->used; ++ i)
  258. {
  259. if (memmap->reg [i].type == PHY_REGISTER)
  260. {
  261. printf ("phy 0x%02x, reg 0x%02x: 0x%04x\n", memmap->reg [i].phy, memmap->reg [i].reg, memmap->reg [i].content);
  262. }
  263. if (memmap->reg [i].type == GBL_REGISTER)
  264. {
  265. printf ("0x%08x: 0x%08x\n", memmap->reg [i].address, memmap->reg [i].content);
  266. }
  267. }
  268. return;
  269. }
  270. /*====================================================================*
  271. *
  272. * void print_command (struct command * command);
  273. *
  274. *
  275. *
  276. *--------------------------------------------------------------------*/
  277. static void print_command (struct command * command)
  278. {
  279. union __packed
  280. {
  281. uint16_t data;
  282. struct __packed
  283. {
  284. uint16_t start: 2;
  285. uint16_t operation: 2;
  286. uint16_t phy_address: 5;
  287. uint16_t reg_address: 5;
  288. uint16_t turnaround: 2;
  289. }
  290. bits;
  291. }
  292. ctrl;
  293. ctrl.data = command->ctrl;
  294. printf ("%02x %02x %04x %04x;\n", ctrl.bits.phy_address, ctrl.bits.reg_address, command->data, command->mask);
  295. return;
  296. }
  297. /*====================================================================*
  298. *
  299. * signed init_memmap (unsigned count, struct memmap * memmap);
  300. *
  301. *
  302. *
  303. *--------------------------------------------------------------------*/
  304. static signed init_memmap (unsigned count, struct memmap * memmap)
  305. {
  306. memmap->reg = calloc (count, sizeof (struct reg));
  307. if (memmap->reg == NULL)
  308. {
  309. error (1, errno, "could not allocate reg memory");
  310. }
  311. memmap->size = count;
  312. memmap->used = 0;
  313. return (0);
  314. }
  315. /*====================================================================*
  316. *
  317. * void free_memmap (struct memmap * memmap);
  318. *
  319. *
  320. *
  321. *--------------------------------------------------------------------*/
  322. static void free_memmap (struct memmap * memmap)
  323. {
  324. free (memmap->reg);
  325. return;
  326. }
  327. /*====================================================================*
  328. *
  329. * signed phy_ar8236 (char const * filename, unsigned commands, flag_t flags);
  330. *
  331. *--------------------------------------------------------------------*/
  332. static signed phy_ar8236 (char const * filename, unsigned commands, flag_t flags)
  333. {
  334. struct command command;
  335. struct memmap memmap;
  336. signed ar8236_code;
  337. signed set_high_addr = 0;
  338. uint16_t high_addr = 0;
  339. uint32_t address;
  340. uint16_t low_address;
  341. if (init_memmap (commands, & memmap))
  342. {
  343. error (1, 0, "could not allocate memory for simulation");
  344. }
  345. while (commands--)
  346. {
  347. if (read (STDIN_FILENO, & command, sizeof (struct command)) != sizeof (struct command))
  348. {
  349. error (0, errno, FILE_CANTREAD, filename);
  350. return (- 1);
  351. }
  352. command.ctrl = LE16TOH (command.ctrl);
  353. command.data = LE16TOH (command.data);
  354. command.mask = LE16TOH (command.mask);
  355. ar8236_code = (command.ctrl & 0x180) >> 7;
  356. switch (ar8236_code)
  357. {
  358. case MDIO32_NORMAL:
  359. if (_anyset (flags, MDIODUMP_VERBOSE))
  360. {
  361. printf ("Normal MDIO Operation:\n");
  362. printf ("\tPhy Address: 0x%02x\n", (command.ctrl & 0x1F0) >> 4);
  363. printf ("\tRegister Address: 0x%02x\n", (command.ctrl & 0x3E00) >> 9);
  364. }
  365. if ((command.ctrl & 0x0C) >> 2 == 0x01)
  366. {
  367. write_phy_reg (& memmap, (command.ctrl & 0x1F0) >> 4, (command.ctrl & 0x3E00) >> 9, command.data, command.mask);
  368. }
  369. break;
  370. case MDIO32_SET_HIGH:
  371. set_high_addr = 1;
  372. high_addr = command.data & 0x3FF & command.mask;
  373. if ((command.ctrl & 0x0C) >> 2 == 0x01)
  374. {
  375. if (_anyset (flags, MDIODUMP_VERBOSE))
  376. {
  377. printf ("Set High Address to 0x%03x:\n", high_addr);
  378. }
  379. }
  380. else
  381. {
  382. if (_anyset (flags, MDIODUMP_VERBOSE))
  383. {
  384. printf ("Read High Address:\n");
  385. }
  386. }
  387. break;
  388. case MDIO32_ACCESS_USING_HIGH:
  389. if (! set_high_addr)
  390. {
  391. error (0, 0, "warning: high address bits not set when attempting to do a 32 bit read, assuming high address bits are 0");
  392. high_addr = 0;
  393. }
  394. low_address = (command.ctrl & 0x3E00) >> 9;
  395. low_address |= (command.ctrl & 0x070) << 1;
  396. address = high_addr << 9;
  397. address |= (low_address << 1) & 0xFFFFFFFC;
  398. if (low_address & 0x01)
  399. {
  400. if (_anyset (flags, MDIODUMP_VERBOSE))
  401. {
  402. printf ("Access bits 31:16 using address 0x%08x:\n", address);
  403. }
  404. write_gbl_reg (& memmap, address, 1, command.data, command.mask);
  405. }
  406. else
  407. {
  408. if (_anyset (flags, MDIODUMP_VERBOSE))
  409. {
  410. printf ("Access bits 15:0 using address 0x%08x:\n", address);
  411. }
  412. write_gbl_reg (& memmap, address, 0, command.data, command.mask);
  413. }
  414. break;
  415. }
  416. if ((command.ctrl & 0x03) != 0x01)
  417. {
  418. error (1, ECANCELED, "start command must be 0x01");
  419. }
  420. if (_anyset (flags, MDIODUMP_VERBOSE))
  421. {
  422. printf ("\tStart: 0x%02x\n", command.ctrl & 0x03);
  423. printf ("\tOperation: 0x%02x (%s)\n", (command.ctrl & 0x0C) >> 2, ((command.ctrl & 0x0C) >> 2 == 0x01)? "write": "read");
  424. printf ("\tTurnaround: 0x%02x\n", (command.ctrl & 0xC000) >> 14);
  425. printf ("\tData: 0x%04x\n", command.data);
  426. printf ("\tMask: 0x%04x\n", command.mask);
  427. printf ("\n");
  428. }
  429. }
  430. if (_anyset (flags, MDIODUMP_SUMMARY))
  431. {
  432. printf ("Memory after execution:\n");
  433. print_memmap (& memmap);
  434. }
  435. free_memmap (& memmap);
  436. return (0);
  437. }
  438. /*====================================================================*
  439. *
  440. * signed phy_generic (char const * filename, unsigned commands, flag_t flags);
  441. *
  442. * assume instructions are 16-bit and display them in human readable
  443. * format on stdout;
  444. *
  445. *--------------------------------------------------------------------*/
  446. static signed phy_generic (char const * filename, unsigned commands, flag_t flags)
  447. {
  448. struct command command;
  449. struct memmap memmap;
  450. unsigned count = 0;
  451. unsigned group = 3;
  452. if (init_memmap (commands, & memmap))
  453. {
  454. error (1, 0, "could not allocate memory for simulation");
  455. }
  456. for (count = 1; commands--; count++)
  457. {
  458. if (read (STDIN_FILENO, & command, sizeof (command)) != sizeof (command))
  459. {
  460. error (0, errno, FILE_CANTREAD, filename);
  461. return (- 1);
  462. }
  463. command.ctrl = LE16TOH (command.ctrl);
  464. command.data = LE16TOH (command.data);
  465. command.mask = LE16TOH (command.mask);
  466. if ((command.ctrl & 0x03) != 0x01)
  467. {
  468. error (1, ECANCELED, "start command must be 0x01");
  469. }
  470. if (_anyset (flags, MDIODUMP_VERBOSE))
  471. {
  472. printf ("Start: 0x%02x\n", command.ctrl & 0x03);
  473. printf ("Operation: 0x%02x (%s)\n", (command.ctrl & 0x0C) >> 2, ((command.ctrl & 0x0C) >> 2 == 0x01)? "write": "read");
  474. printf ("Phy Address: 0x%02x\n", (command.ctrl & 0x1F0) >> 4);
  475. printf ("Register Address: 0x%02x\n", (command.ctrl & 0x3E00) >> 9);
  476. printf ("Turnaround: 0x%02x\n", (command.ctrl & 0xC000) >> 14);
  477. printf ("Data: 0x%04x\n", command.data);
  478. printf ("Mask: 0x%04x\n", command.mask);
  479. printf ("\n");
  480. continue;
  481. }
  482. if ((command.ctrl & 0x0C) >> 2 == 0x01)
  483. {
  484. if (_anyset (flags, MDIODUMP_SUMMARY))
  485. {
  486. write_phy_reg (& memmap, (command.ctrl & 0x1F0) >> 4, (command.ctrl & 0x3E00) >> 9, command.data, command.mask);
  487. continue;
  488. }
  489. print_command (& command);
  490. if ((count % group) == 0)
  491. {
  492. printf ("\n");
  493. }
  494. continue;
  495. }
  496. }
  497. if (_anyset (flags, MDIODUMP_SUMMARY))
  498. {
  499. printf ("Memory after execution:\n");
  500. print_memmap (& memmap);
  501. }
  502. free_memmap (& memmap);
  503. return (0);
  504. }
  505. /*====================================================================*
  506. *
  507. * signed function (char const * filename, unsigned phy_code, flag_t flags);
  508. *
  509. * read the MDIO block header to determine the number of MDIO
  510. * instructions in the program block; call appropriate function
  511. * to interpret instructions and display them in human readable
  512. * format;
  513. *
  514. *--------------------------------------------------------------------*/
  515. static signed function (char const * filename, unsigned phy_code, flag_t flags)
  516. {
  517. uint16_t mdio_header;
  518. unsigned commands;
  519. if (read (STDIN_FILENO, & mdio_header, sizeof (mdio_header)) != sizeof (mdio_header))
  520. {
  521. error (0, errno, FILE_CANTREAD, filename);
  522. return (- 1);
  523. }
  524. mdio_header = LE16TOH (mdio_header);
  525. printf ("# ------- %s -------\n\n", filename);
  526. commands = (mdio_header & 0xFFC0) >> 6;
  527. if (_anyset (flags, MDIODUMP_SUMMARY))
  528. {
  529. printf ("Enabled: %s\n", (mdio_header & 0x0001)? "yes": "no");
  530. printf ("Number of Commands: %d\n", commands);
  531. }
  532. if (phy_code == PHY_GENERIC)
  533. {
  534. return (phy_generic (filename, commands, flags));
  535. }
  536. if (phy_code == PHY_AR8236)
  537. {
  538. return (phy_ar8236 (filename, commands, flags));
  539. }
  540. return (0);
  541. }
  542. /*====================================================================*
  543. *
  544. * int main (int argc, const char * argv []);
  545. *
  546. *--------------------------------------------------------------------*/
  547. int main (int argc, const char * argv [])
  548. {
  549. static const char * optv [] =
  550. {
  551. "st:v",
  552. "file [file] [...]",
  553. "Qualcomm Atheros MDIO Custom Module Analyser",
  554. "s\tprint summary information",
  555. "t s\tinterpret MDIO commands for phy type (s) [generic]",
  556. "v\tprint complete module dump, not just the summary",
  557. (const char *) (0)
  558. };
  559. unsigned phy_code = PHY_GENERIC;
  560. flag_t flags = (flag_t) (0);
  561. signed state = 0;
  562. signed c;
  563. optind = 1;
  564. while (~ (c = getoptv (argc, argv, optv)))
  565. {
  566. switch ((char) (c))
  567. {
  568. case 's':
  569. _setbits (flags, MDIODUMP_SUMMARY);
  570. break;
  571. case 't':
  572. if ((c = lookup (optarg, switches, SIZEOF (switches))) == - 1)
  573. {
  574. assist (optarg, "type", switches, SIZEOF (switches));
  575. }
  576. _setbits (flags, MDIODUMP_SUMMARY);
  577. phy_code = (unsigned) (c);
  578. break;
  579. case 'b':
  580. _clrbits (flags, MDIODUMP_SUMMARY);
  581. break;
  582. case 'v':
  583. _setbits (flags, MDIODUMP_VERBOSE);
  584. break;
  585. default:
  586. break;
  587. }
  588. }
  589. argc -= optind;
  590. argv += optind;
  591. if (! argc)
  592. {
  593. function ("stdin", phy_code, flags);
  594. }
  595. while ((argc) && (* argv))
  596. {
  597. if (! freopen (* argv, "rb", stdin))
  598. {
  599. error (0, errno, "%s", * argv);
  600. state = 1;
  601. errno = 0;
  602. }
  603. else if (function (* argv, phy_code, flags))
  604. {
  605. state = 1;
  606. }
  607. argc--;
  608. argv++;
  609. }
  610. return (state);
  611. }