int6ktone.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * plctone.c - Qualcomm Atheros INT6x00 Tone Map Dump Utility
  11. *
  12. * Contributor(s):
  13. * Charles Maier <cmaier@qca.qualcomm.com>
  14. *
  15. *--------------------------------------------------------------------*/
  16. /*====================================================================*
  17. * system header files;
  18. *--------------------------------------------------------------------*/
  19. #include <limits.h>
  20. #include <unistd.h>
  21. #include <stdlib.h>
  22. #include <stdint.h>
  23. /*====================================================================*
  24. * custom header files;
  25. *--------------------------------------------------------------------*/
  26. #include "../tools/getoptv.h"
  27. #include "../tools/putoptv.h"
  28. #include "../tools/memory.h"
  29. #include "../tools/symbol.h"
  30. #include "../tools/number.h"
  31. #include "../tools/types.h"
  32. #include "../tools/flags.h"
  33. #include "../tools/files.h"
  34. #include "../tools/error.h"
  35. #include "../plc/plc.h"
  36. /*====================================================================*
  37. * custom source files;
  38. *--------------------------------------------------------------------*/
  39. #ifndef MAKEFILE
  40. #include "../plc/Devices.c"
  41. #include "../plc/Failure.c"
  42. #include "../plc/ReadMME.c"
  43. #include "../plc/SendMME.c"
  44. #include "../plc/mod2bits.c"
  45. #include "../plc/mod2db.c"
  46. #endif
  47. #ifndef MAKEFILE
  48. #include "../tools/error.c"
  49. #include "../tools/getoptv.c"
  50. #include "../tools/putoptv.c"
  51. #include "../tools/version.c"
  52. #include "../tools/uintspec.c"
  53. #include "../tools/hexdump.c"
  54. #include "../tools/hexencode.c"
  55. #include "../tools/hexdecode.c"
  56. #include "../tools/hexstring.c"
  57. #include "../tools/todigit.c"
  58. #include "../tools/synonym.c"
  59. #endif
  60. #ifndef MAKEFILE
  61. #include "../ether/openchannel.c"
  62. #include "../ether/closechannel.c"
  63. #include "../ether/readpacket.c"
  64. #include "../ether/sendpacket.c"
  65. #include "../ether/channel.c"
  66. #endif
  67. #ifndef MAKEFILE
  68. #include "../mme/MMECode.c"
  69. #include "../mme/EthernetHeader.c"
  70. #include "../mme/QualcommHeader.c"
  71. #include "../mme/UnwantedMessage.c"
  72. #endif
  73. /*====================================================================*
  74. *
  75. * signed thunderbolt_tonemaps (struct plc * plc);
  76. *
  77. * plc.h
  78. *
  79. * See the Atheros HomePlug AV Firmware Technical Reference Manual
  80. * for more information;
  81. *
  82. * Contributor(s):
  83. * Charles Maier <cmaier@qca.qualcomm.com>
  84. *
  85. *--------------------------------------------------------------------*/
  86. signed thunderbolt_tonemaps (struct plc * plc)
  87. {
  88. extern uint8_t const mod2bits [PLC_BITS_PER_TONE];
  89. struct channel * channel = (struct channel *) (plc->channel);
  90. struct message * message = (struct message *) (plc->message);
  91. byte tonemap [PLC_TIME_SLOTS] [INT_CARRIERS >> 1];
  92. uint16_t carriers = 0;
  93. uint16_t carrier = 0;
  94. uint8_t slots = 0;
  95. uint8_t slot = 0;
  96. #ifndef __GNUC__
  97. #pragma pack (push,1)
  98. #endif
  99. struct __packed vs_tone_slot_char_request
  100. {
  101. struct ethernet_hdr ethernet;
  102. struct qualcomm_hdr qualcomm;
  103. uint8_t MACADDRESS [ETHER_ADDR_LEN];
  104. uint8_t TMSLOT;
  105. }
  106. * request = (struct vs_tone_slot_char_request *) (message);
  107. struct __packed vs_tone_slot_char_confirm
  108. {
  109. struct ethernet_hdr ethernet;
  110. struct qualcomm_hdr qualcomm;
  111. uint8_t MSTATUS;
  112. uint8_t TMSLOT;
  113. uint8_t NUMTMS;
  114. uint16_t TMNUMACTCARRIERS;
  115. uint8_t MOD_CARRIER [INT_CARRIERS / 2];
  116. uint8_t GIL;
  117. uint8_t AGC;
  118. }
  119. * confirm = (struct vs_tone_slot_char_confirm *) (message);
  120. #ifndef __GNUC__
  121. #pragma pack (pop)
  122. #endif
  123. memset (message, 0, sizeof (* message));
  124. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  125. QualcommHeader (& request->qualcomm, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_REQ));
  126. memcpy (request->MACADDRESS, plc->RDA, sizeof (request->MACADDRESS));
  127. request->TMSLOT = 0;
  128. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  129. if (SendMME (plc) <= 0)
  130. {
  131. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  132. return (-1);
  133. }
  134. if (ReadMME (plc, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_CNF)) <= 0)
  135. {
  136. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  137. return (-1);
  138. }
  139. carriers = LE16TOH (confirm->TMNUMACTCARRIERS);
  140. slots = confirm->NUMTMS;
  141. if (! slots)
  142. {
  143. error (PLC_EXIT (plc), ECANCELED, "No Tone Maps Available");
  144. return (-1);
  145. }
  146. memset (tonemap, 0, sizeof (tonemap));
  147. for (slot = 0; slot < slots; slot++)
  148. {
  149. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  150. QualcommHeader (& request->qualcomm, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_REQ));
  151. memcpy (request->MACADDRESS, plc->RDA, sizeof (request->MACADDRESS));
  152. request->TMSLOT = slot;
  153. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  154. if (SendMME (plc) <= 0)
  155. {
  156. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  157. return (-1);
  158. }
  159. if (ReadMME (plc, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_CNF)) <= 0)
  160. {
  161. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  162. return (-1);
  163. }
  164. if (confirm->MSTATUS)
  165. {
  166. Failure (plc, PLC_WONTDOIT);
  167. return (-1);
  168. }
  169. memcpy (& tonemap [slot] [0], confirm->MOD_CARRIER, sizeof (confirm->MOD_CARRIER));
  170. }
  171. while (carrier < carriers)
  172. {
  173. unsigned value = 0;
  174. unsigned scale = 0;
  175. unsigned index = carrier >> 1;
  176. printf ("% 04d", carrier);
  177. for (slot = 0; slot < PLC_TIME_SLOTS; slot++)
  178. {
  179. value = tonemap [slot] [index];
  180. if ((carrier & 1))
  181. {
  182. value >>= 4;
  183. }
  184. value &= 0x0F;
  185. printf (" %02d", mod2bits [value]);
  186. value *= value;
  187. scale += value;
  188. }
  189. if (slots)
  190. {
  191. scale /= slots;
  192. }
  193. if (_anyset (plc->flags, PLC_GRAPH))
  194. {
  195. printf (" %03d ", scale);
  196. while (scale--)
  197. {
  198. printf ("#");
  199. }
  200. }
  201. printf ("\n");
  202. carrier++;
  203. }
  204. return (0);
  205. }
  206. /*====================================================================*
  207. *
  208. * signed thunderbolt_snrdata (struct plc * plc);
  209. *
  210. * plc.h
  211. *
  212. * Contributor(s):
  213. * Charles Maier <cmaier@qca.qualcomm.com>
  214. *
  215. *--------------------------------------------------------------------*/
  216. signed thunderbolt_snrdata (struct plc * plc)
  217. {
  218. extern uint8_t const mod2db [PLC_BITS_PER_TONE];
  219. extern uint8_t const mod2bits [PLC_BITS_PER_TONE];
  220. struct channel * channel = (struct channel *) (plc->channel);
  221. struct message * message = (struct message *) (plc->message);
  222. struct tonemap
  223. {
  224. byte GIL;
  225. byte AGC;
  226. double SNR;
  227. double BPC;
  228. byte DATA [AMP_CARRIERS >> 1];
  229. }
  230. tonemap [PLC_TIME_SLOTS];
  231. double AvgSNR = 0;
  232. double AvgBPC = 0;
  233. unsigned carriers = 0;
  234. unsigned carrier = 0;
  235. unsigned slots = 0;
  236. unsigned slot = 0;
  237. #ifndef __GNUC__
  238. #pragma pack (push,1)
  239. #endif
  240. struct __packed vs_tone_slot_char_request
  241. {
  242. struct ethernet_hdr ethernet;
  243. struct qualcomm_hdr qualcomm;
  244. uint8_t MACADDRESS [ETHER_ADDR_LEN];
  245. uint8_t TMSLOT;
  246. }
  247. * request = (struct vs_tone_slot_char_request *) (message);
  248. struct __packed vs_tone_slot_char_confirm
  249. {
  250. struct ethernet_hdr ethernet;
  251. struct qualcomm_hdr qualcomm;
  252. uint8_t MSTATUS;
  253. uint8_t TMSLOT;
  254. uint8_t NUMTMS;
  255. uint16_t TMNUMACTCARRIERS;
  256. uint8_t MOD_CARRIER [(INT_CARRIERS + 1) >> 1];
  257. uint8_t GIL;
  258. uint8_t AGC;
  259. }
  260. * confirm = (struct vs_tone_slot_char_confirm *) (message);
  261. #ifndef __GNUC__
  262. #pragma pack (pop)
  263. #endif
  264. memset (message, 0, sizeof (* message));
  265. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  266. QualcommHeader (& request->qualcomm, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_REQ));
  267. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  268. memcpy (request->MACADDRESS, plc->RDA, sizeof (request->MACADDRESS));
  269. request->TMSLOT = 0;
  270. if (SendMME (plc) <= 0)
  271. {
  272. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  273. return (-1);
  274. }
  275. if (ReadMME (plc, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_CNF)) <= 0)
  276. {
  277. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  278. return (-1);
  279. }
  280. carriers = LE16TOH (confirm->TMNUMACTCARRIERS);
  281. slots = confirm->NUMTMS;
  282. if (! slots)
  283. {
  284. error (PLC_EXIT (plc), ECANCELED, "No Tone Maps Available");
  285. return (-1);
  286. }
  287. memset (tonemap, 0, sizeof (tonemap));
  288. for (slot = 0; slot < slots; slot++)
  289. {
  290. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  291. QualcommHeader (& request->qualcomm, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_REQ));
  292. memcpy (request->MACADDRESS, plc->RDA, sizeof (request->MACADDRESS));
  293. request->TMSLOT = slot;
  294. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  295. if (SendMME (plc) <= 0)
  296. {
  297. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  298. return (-1);
  299. }
  300. if (ReadMME (plc, 0, (VS_RX_TONE_MAP_CHAR | MMTYPE_CNF)) <= 0)
  301. {
  302. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  303. return (-1);
  304. }
  305. if (confirm->MSTATUS)
  306. {
  307. Failure (plc, PLC_WONTDOIT);
  308. return (-1);
  309. }
  310. memcpy (& tonemap [slot].DATA [0], confirm->MOD_CARRIER, sizeof (confirm->MOD_CARRIER));
  311. tonemap [slot].GIL = confirm->GIL;
  312. tonemap [slot].AGC = confirm->AGC;
  313. }
  314. while (carrier < carriers)
  315. {
  316. unsigned value = 0;
  317. unsigned index = carrier >> 1;
  318. for (slot = 0; slot < PLC_TIME_SLOTS; slot++)
  319. {
  320. value = tonemap [slot].DATA [index];
  321. if ((carrier & 1))
  322. {
  323. value >>= 4;
  324. }
  325. value &= 0x0F;
  326. tonemap [slot].BPC += mod2bits [value];
  327. tonemap [slot].SNR += mod2db [value];
  328. AvgBPC += mod2bits [value];
  329. AvgSNR += mod2db [value];
  330. }
  331. carrier++;
  332. }
  333. AvgBPC /= carriers;
  334. AvgBPC /= slots;
  335. AvgSNR /= carriers;
  336. AvgSNR /= slots;
  337. printf (" SNR");
  338. for (slot = 0; slot < slots; slot++)
  339. {
  340. printf (" %8.3f", tonemap [slot].SNR / carriers);
  341. }
  342. printf (" %8.3f", AvgSNR);
  343. printf (" \n");
  344. printf (" ATN");
  345. for (slot = 0; slot < slots; slot++)
  346. {
  347. printf (" %8.3f", tonemap [slot].SNR / carriers - 60);
  348. }
  349. printf (" %8.3f", AvgSNR - 60);
  350. printf (" \n");
  351. printf (" BPC");
  352. for (slot = 0; slot < slots; slot++)
  353. {
  354. printf (" %8.3f", tonemap [slot].BPC / carriers);
  355. }
  356. printf (" %8.3f", AvgBPC);
  357. printf (" \n");
  358. printf (" AGC");
  359. for (slot = 0; slot < slots; slot++)
  360. {
  361. printf (" %02d", tonemap [slot].AGC);
  362. }
  363. printf (" \n");
  364. printf (" GIL");
  365. for (slot = 0; slot < slots; slot++)
  366. {
  367. printf (" %02d", tonemap [slot].GIL);
  368. }
  369. printf (" \n");
  370. return (0);
  371. }
  372. /*====================================================================*
  373. *
  374. * int main (int argc, char const * argv[]);
  375. *
  376. *--------------------------------------------------------------------*/
  377. int main (int argc, char const * argv [])
  378. {
  379. extern struct channel channel;
  380. static char const * optv [] =
  381. {
  382. "ehi:qst:vx",
  383. "node peer [> stdout]",
  384. "Qualcomm Atheros INT6x00 Tone Map Dump Utility",
  385. "e\tredirect stderr to stdout",
  386. "h\tprint mean-square histogram",
  387. #if defined (WINPCAP) || defined (LIBPCAP)
  388. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  389. #else
  390. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  391. #endif
  392. "q\tquiet mode",
  393. "s\tcompute signal-to-noise and bits-per-carrier ratios",
  394. "t n\tread timeout is (n) milliseconds [" LITERAL (CHANNEL_TIMEOUT) "]",
  395. "v\tverbose mode",
  396. "x\texit on error",
  397. (char const *) (0)
  398. };
  399. #include "../plc/plc.c"
  400. signed (* function) (struct plc *) = thunderbolt_tonemaps;
  401. signed c;
  402. if (getenv (PLCDEVICE))
  403. {
  404. #if defined (WINPCAP) || defined (LIBPCAP)
  405. channel.ifindex = atoi (getenv (PLCDEVICE));
  406. #else
  407. channel.ifname = strdup (getenv (PLCDEVICE));
  408. #endif
  409. }
  410. optind = 1;
  411. while (~ (c = getoptv (argc, argv, optv)))
  412. {
  413. switch (c)
  414. {
  415. case 'e':
  416. dup2 (STDOUT_FILENO, STDERR_FILENO);
  417. break;
  418. case 'h':
  419. _setbits (plc.flags, PLC_GRAPH);
  420. break;
  421. case 'i':
  422. #if defined (WINPCAP) || defined (LIBPCAP)
  423. channel.ifindex = atoi (optarg);
  424. #else
  425. channel.ifname = optarg;
  426. #endif
  427. break;
  428. case 'q':
  429. _setbits (channel.flags, CHANNEL_SILENCE);
  430. _setbits (plc.flags, PLC_SILENCE);
  431. break;
  432. case 's':
  433. _setbits (plc.flags, PLC_ANALYSE);
  434. function = thunderbolt_snrdata;
  435. break;
  436. case 't':
  437. channel.timeout = (signed) (uintspec (optarg, 0, UINT_MAX));
  438. break;
  439. case 'v':
  440. _setbits (channel.flags, CHANNEL_VERBOSE);
  441. _setbits (plc.flags, PLC_VERBOSE);
  442. break;
  443. case 'x':
  444. _setbits (plc.flags, PLC_BAILOUT);
  445. break;
  446. default:
  447. break;
  448. }
  449. }
  450. argc -= optind;
  451. argv += optind;
  452. if (! argc || ! argv)
  453. {
  454. error (1, ECANCELED, "No node address given");
  455. }
  456. if (! hexencode (channel.peer, sizeof (channel.peer), synonym (* argv, devices, SIZEOF (devices))))
  457. {
  458. error (1, errno, PLC_BAD_MAC, * argv);
  459. }
  460. argc--;
  461. argv++;
  462. if (! argc || ! argv)
  463. {
  464. error (1, ECANCELED, "No peer address given");
  465. }
  466. if (! hexencode (plc.RDA, sizeof (plc.RDA), synonym (* argv, devices, SIZEOF (devices))))
  467. {
  468. error (1, errno, PLC_BAD_MAC, * argv);
  469. }
  470. argc--;
  471. argv++;
  472. openchannel (& channel);
  473. if (! (plc.message = malloc (sizeof (* plc.message))))
  474. {
  475. error (1, errno, PLC_NOMEMORY);
  476. }
  477. function (& plc);
  478. free (plc.message);
  479. closechannel (& channel);
  480. return (0);
  481. }