pts.c 32 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2016 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * pts.c -
  11. *
  12. *
  13. *
  14. *--------------------------------------------------------------------*/
  15. /*====================================================================*
  16. * system header files;
  17. *--------------------------------------------------------------------*/
  18. #include <unistd.h>
  19. #include <stdlib.h>
  20. #include <stdint.h>
  21. #include <limits.h>
  22. #include <sys/time.h>
  23. #include <memory.h>
  24. #include <errno.h>
  25. #include "../tools/timer.h"
  26. /*====================================================================*
  27. * custom header files;
  28. *--------------------------------------------------------------------*/
  29. #include "../tools/getoptv.h"
  30. #include "../tools/putoptv.h"
  31. #include "../tools/memory.h"
  32. #include "../tools/number.h"
  33. #include "../tools/symbol.h"
  34. #include "../tools/types.h"
  35. #include "../tools/flags.h"
  36. #include "../tools/files.h"
  37. #include "../tools/error.h"
  38. #include "../ether/channel.h"
  39. #include "../key/HPAVKey.h"
  40. #include "../key/keys.h"
  41. #include "../ram/sdram.h"
  42. #include "../pib/pib.h"
  43. #include "../nvm/nvm.h"
  44. #include "../plc/plc.h"
  45. #include "../plc/pts.h"
  46. /*====================================================================*
  47. * custom source files;
  48. *--------------------------------------------------------------------*/
  49. #ifndef MAKEFILE
  50. #include "../plc/Attributes2.c"
  51. #include "../plc/Confirm.c"
  52. #include "../plc/Devices.c"
  53. #include "../plc/Display.c"
  54. #include "../plc/FactoryDefaults.c"
  55. #include "../plc/Failure.c"
  56. #include "../plc/FlashDevice2.c"
  57. #include "../plc/FlashFirmware.c"
  58. #include "../plc/FlashParameters.c"
  59. #include "../plc/FlashSoftloader.c"
  60. #include "../plc/HostActionResponse.c"
  61. #include "../plc/Identity1.c"
  62. #include "../plc/Identity2.c"
  63. #include "../plc/LinkStatus.c"
  64. #include "../plc/ModuleCommit.c"
  65. #include "../plc/ModuleRead.c"
  66. #include "../plc/ModuleSession.c"
  67. #include "../plc/ModuleSpec.c"
  68. #include "../plc/ModuleWrite.c"
  69. #include "../plc/NVRAMInfo.c"
  70. #include "../plc/NetInfo2.c"
  71. #include "../plc/PLCSelect.c"
  72. #include "../plc/PushButton.c"
  73. #include "../plc/ReadMFG.c"
  74. #include "../plc/ReadMME.c"
  75. #include "../plc/ReadFirmware2.c"
  76. #include "../plc/ReadParameters2.c"
  77. #include "../plc/Request.c"
  78. #include "../plc/ResetDevice.c"
  79. #include "../plc/RemoteHosts.c"
  80. #include "../plc/SDRAMInfo.c"
  81. #include "../plc/SendMME.c"
  82. #include "../plc/SetNMK.c"
  83. #include "../plc/StartFirmware1.c"
  84. #include "../plc/StationRole.c"
  85. #include "../plc/VersionInfo2.c"
  86. #include "../plc/WaitForReset.c"
  87. #include "../plc/WaitForStart.c"
  88. #include "../plc/WaitForRestart.c"
  89. #include "../plc/WatchdogReport.c"
  90. #include "../plc/chipset.c"
  91. #include "../plc/pts.c"
  92. #endif
  93. #ifndef MAKEFILE
  94. #include "../tools/error.c"
  95. #include "../tools/getoptv.c"
  96. #include "../tools/putoptv.c"
  97. #include "../tools/synonym.c"
  98. #include "../tools/uintspec.c"
  99. #include "../tools/basespec.c"
  100. #include "../tools/version.c"
  101. #include "../tools/hexdump.c"
  102. #include "../tools/hexencode.c"
  103. #include "../tools/hexdecode.c"
  104. #include "../tools/hexstring.c"
  105. #include "../tools/decstring.c"
  106. #include "../tools/decdecode.c"
  107. #include "../tools/hexout.c"
  108. #include "../tools/todigit.c"
  109. #include "../tools/checkfilename.c"
  110. #include "../tools/checksum32.c"
  111. #include "../tools/fdchecksum32.c"
  112. #include "../tools/strfbits.c"
  113. #include "../tools/typename.c"
  114. #endif
  115. #ifndef MAKEFILE
  116. #include "../ether/channel.c"
  117. #include "../ether/openchannel.c"
  118. #include "../ether/closechannel.c"
  119. #include "../ether/readpacket.c"
  120. #include "../ether/sendpacket.c"
  121. #endif
  122. #ifndef MAKEFILE
  123. #include "../ram/nvram.c"
  124. #include "../ram/sdramfile.c"
  125. #include "../ram/sdrampeek.c"
  126. #endif
  127. #ifndef MAKEFILE
  128. #include "../nvm/panther_nvm_file.c"
  129. #endif
  130. #ifndef MAKEFILE
  131. #include "../pib/lightning_pib_peek.c"
  132. #include "../pib/panther_pib_peek.c"
  133. #include "../pib/panther_pib_file.c"
  134. #endif
  135. #ifndef MAKEFILE
  136. #include "../mme/EthernetHeader.c"
  137. #include "../mme/QualcommHeader.c"
  138. #include "../mme/QualcommHeader1.c"
  139. #include "../mme/UnwantedMessage.c"
  140. #include "../mme/MMECode.c"
  141. #endif
  142. #ifndef MAKEFILE
  143. #include "../key/keys.c"
  144. #endif
  145. #define PTS_PROGRAM
  146. int isEqual( uint8_t *MAC1 , uint8_t *MAC2 );
  147. extern int relay (unsigned atn_grd, unsigned atn_line, char const * name );
  148. const char *pts_messages[] = {"\nSetting Attenuator LG=%ddB LN=%ddB",\
  149. "\ntwo","\nthree"};
  150. /*====================================================================*
  151. * program constants;
  152. *--------------------------------------------------------------------*/
  153. #define PLCTOOL_WAIT 0
  154. #define PLCTOOL_LOOP 1
  155. #define BUTTONS (sizeof (buttons) / sizeof (struct _term_))
  156. /*====================================================================*
  157. * program variables;
  158. *--------------------------------------------------------------------*/
  159. #if 0
  160. static const struct _term_ buttons [] =
  161. {
  162. {
  163. "none",
  164. "0"
  165. },
  166. {
  167. "GN->DUT",
  168. "1"
  169. },
  170. {
  171. "leave",
  172. "2"
  173. },
  174. {
  175. "GN<->DUT",
  176. "3"
  177. }
  178. };
  179. #endif
  180. // PTS parameters
  181. struct pts pts = { {
  182. 0x00,
  183. 0xB0,
  184. 0x52,
  185. 0x00,
  186. 0x00,
  187. 0x01
  188. },{
  189. 0x00,
  190. 0xB0,
  191. 0x52,
  192. 0x00,
  193. 0x00,
  194. 0x01
  195. }, 1, PTSCTL_PORT, PTSCTL_LINE_ATTN, PTSCTL_LINE_ATTN, 50,0,0,0,0,0, 120, 0, 0 };
  196. /*====================================================================*
  197. *
  198. * signed Get_GN_MAC (struct plc * plc);
  199. *
  200. * Issue GET_VERSION MME to local device to determine the MAC address of the PLC device
  201. * The local device is considered to be the Golden Node (GN).
  202. *
  203. *--------------------------------------------------------------------*/
  204. signed Get_GN_MAC (struct plc * plc)
  205. {
  206. int i;
  207. struct channel * channel = (struct channel *) (plc->channel);
  208. struct message * message = (struct message *) (plc->message);
  209. #ifndef __GNUC__
  210. #pragma pack (push,1)
  211. #endif
  212. struct __packed vs_sw_ver_request
  213. {
  214. struct ethernet_hdr ethernet;
  215. struct qualcomm_hdr qualcomm;
  216. uint32_t COOKIE;
  217. }
  218. * request = (struct vs_sw_ver_request *) (message);
  219. struct __packed vs_sw_ver_confirm
  220. {
  221. struct ethernet_hdr ethernet;
  222. struct qualcomm_hdr qualcomm;
  223. uint8_t MSTATUS;
  224. uint8_t MDEVICE_CLASS;
  225. uint8_t MVERLENGTH;
  226. char MVERSION [254];
  227. uint32_t IDENT;
  228. uint32_t STEPPING_NUM;
  229. uint32_t COOKIE;
  230. uint32_t RSVD [6];
  231. }
  232. * confirm = (struct vs_sw_ver_confirm *) (message);
  233. #ifndef __GNUC__
  234. #pragma pack (pop)
  235. #endif
  236. memset (message, 0, sizeof (* message));
  237. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  238. QualcommHeader (& request->qualcomm, 0, (VS_SW_VER | MMTYPE_REQ));
  239. request->COOKIE = HTOLE32 (plc->cookie);
  240. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  241. if (SendMME (plc) <= 0)
  242. {
  243. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  244. return (-1);
  245. }
  246. while (ReadMME (plc, 0, (VS_SW_VER | MMTYPE_CNF)) > 0)
  247. {
  248. if (confirm->MSTATUS)
  249. {
  250. Failure (plc, PLC_WONTDOIT);
  251. continue;
  252. }
  253. // save GN MAC address
  254. for (i=0;i<ETHER_ADDR_LEN;i++)
  255. {
  256. pts.gn[i] = plc->message->ethernet.OSA[i];
  257. }
  258. }
  259. return (0);
  260. }
  261. /*====================================================================*
  262. *
  263. * signed Get_DUT_MAC (struct plc * plc);
  264. *
  265. * Issue GET_VERSION MME to local device to determine the MAC address of the PLC device
  266. * The local device is considered to be the Golden Node (GN).
  267. *
  268. *--------------------------------------------------------------------*/
  269. signed Get_DUT_MAC (struct plc * plc)
  270. {
  271. int i;
  272. struct channel * channel1 = (struct channel *) (plc->channel1);
  273. struct message * message = (struct message *) (plc->message);
  274. #ifndef __GNUC__
  275. #pragma pack (push,1)
  276. #endif
  277. struct __packed vs_sw_ver_request
  278. {
  279. struct ethernet_hdr ethernet;
  280. struct qualcomm_hdr qualcomm;
  281. uint32_t COOKIE;
  282. }
  283. * request = (struct vs_sw_ver_request *) (message);
  284. struct __packed vs_sw_ver_confirm
  285. {
  286. struct ethernet_hdr ethernet;
  287. struct qualcomm_hdr qualcomm;
  288. uint8_t MSTATUS;
  289. uint8_t MDEVICE_CLASS;
  290. uint8_t MVERLENGTH;
  291. char MVERSION [254];
  292. uint32_t IDENT;
  293. uint32_t STEPPING_NUM;
  294. uint32_t COOKIE;
  295. uint32_t RSVD [6];
  296. }
  297. * confirm = (struct vs_sw_ver_confirm *) (message);
  298. #ifndef __GNUC__
  299. #pragma pack (pop)
  300. #endif
  301. memset (message, 0, sizeof (* message));
  302. EthernetHeader (& request->ethernet, channel1->peer, channel1->host, channel1->type);
  303. QualcommHeader (& request->qualcomm, 0, (VS_SW_VER | MMTYPE_REQ));
  304. request->COOKIE = HTOLE32 (plc->cookie);
  305. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  306. if (SendMME1 (plc) <= 0)
  307. {
  308. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  309. return (-1);
  310. }
  311. while (ReadMME1 (plc, 0, (VS_SW_VER | MMTYPE_CNF)) > 0)
  312. {
  313. if (confirm->MSTATUS)
  314. {
  315. Failure (plc, PLC_WONTDOIT);
  316. continue;
  317. }
  318. // save DUT MAC address
  319. for (i=0;i<ETHER_ADDR_LEN;i++)
  320. {
  321. pts.dut[i] = plc->message->ethernet.OSA[i];
  322. }
  323. }
  324. return (0);
  325. }
  326. unsigned TestGNToDUT(struct plc * plc )
  327. {
  328. struct timeval ts;
  329. struct timeval tc;
  330. unsigned timer = 0;
  331. struct channel * channel = (struct channel *) (plc->channel);
  332. struct message * message = (struct message *) (plc->message);
  333. if (gettimeofday (& ts, NULL) == -1)
  334. {
  335. error (1, errno, CANT_START_TIMER);
  336. }
  337. // transmit for time duration as specified (-dxx)
  338. for (timer = 0; timer < pts.timeout; timer = SECONDS (ts, tc))
  339. {
  340. // transmit from GN to DUT using host interface (-i)
  341. memset (message, 0xAA, sizeof (* message));
  342. EthernetHeader (message, pts.dut, pts.gn, ETHERTYPE_IP);
  343. // this is a work around for a buffer overrun condition that may happen nder OSX
  344. if (sendpacket (channel, message, sizeof (* message)) == -1)
  345. {
  346. // we may be overrunning the buffer so need to wait a bit
  347. sleep( 1 );
  348. }
  349. // Send netinfo MME to DUT, look for GN MAC address and read RX PHY Rate, save in pts->AVGRX_DUT
  350. GetDUTPhyRates (plc, &pts );
  351. if( pts.AVGRX_DUT >= pts.phy_rate_limit )
  352. {
  353. // test passed
  354. return( 0 );
  355. }
  356. if (gettimeofday (& tc, NULL) == -1)
  357. {
  358. error (1, errno, CANT_RESET_TIMER);
  359. }
  360. }
  361. // test failed, did not get min. PHY rate within test time specified
  362. return( 1 );
  363. }
  364. unsigned TestDUTToGN( struct plc * plc)
  365. {
  366. struct timeval ts;
  367. struct timeval tc;
  368. unsigned timer = 0;
  369. struct channel * channel1 = (struct channel *) (plc->channel1);
  370. struct message * message = (struct message *) (plc->message);
  371. if (gettimeofday (& ts, NULL) == -1)
  372. {
  373. error (1, errno, CANT_START_TIMER);
  374. }
  375. for (timer = 0; timer < pts.timeout; timer = SECONDS (ts, tc))
  376. {
  377. // transmit from GN to DUT using host interface (-i)
  378. memset (message, 0xAA, sizeof (* message));
  379. EthernetHeader (message, pts.gn, pts.dut, ETHERTYPE_IP);
  380. // this is a work around for a buffer overrun condition that may happen nder OSX
  381. if (sendpacket (channel1, message, sizeof (* message)) == -1)
  382. {
  383. // we may be overrunning the buffer so need to wait a bit
  384. sleep( 1 );
  385. }
  386. // send netifo MME to GN, look for DUT MAC address and read RX PHY Rate, save in pts->AVGRX_GN
  387. GetGNPhyRates (plc, &pts );
  388. if( pts.AVGRX_GN >= pts.phy_rate_limit )
  389. {
  390. //test passed
  391. return( 0 );
  392. }
  393. if (gettimeofday (& tc, NULL) == -1)
  394. {
  395. error (1, errno, CANT_RESET_TIMER);
  396. }
  397. }
  398. // test failed, did not get min. PHY rate within test time specified
  399. return( 1 );
  400. }
  401. unsigned TestBiDi(struct plc * plc)
  402. {
  403. struct timeval ts;
  404. struct timeval tc;
  405. unsigned timer = 0;
  406. struct channel * channel = (struct channel *) (plc->channel);
  407. struct channel * channel1 = (struct channel *) (plc->channel1);
  408. struct message * message = (struct message *) (plc->message);
  409. if (gettimeofday (& ts, NULL) == -1)
  410. {
  411. error (1, errno, CANT_START_TIMER);
  412. }
  413. for (timer = 0; timer < pts.timeout; timer = SECONDS (ts, tc))
  414. {
  415. // transmit from GN to DUT using host interface (-i)
  416. memset (message, 0xAA, sizeof (* message));
  417. EthernetHeader (message, pts.dut, pts.gn, ETHERTYPE_IP);
  418. // this is a work around for a buffer overrun condition that may happen nder OSX
  419. if (sendpacket (channel, message, sizeof (* message)) == -1)
  420. {
  421. // we may be overrunning the buffer so need to wait a bit
  422. sleep( 1 );
  423. }
  424. // transmit from DUT to DN using host interface (-j)
  425. EthernetHeader (message, pts.gn, pts.dut, ETHERTYPE_IP);
  426. // this is a work around for a buffer overrun condition that may happen nder OSX
  427. if (sendpacket (channel1, message, sizeof (* message)) == -1)
  428. {
  429. // we may be overrunning the buffer so need to wait a bit
  430. sleep( 1 );
  431. }
  432. // Send netinfo MME to DUT, look for GN MAC address and read RX PHY Rate, save in pts->AVGRX_DUT
  433. GetDUTPhyRates (plc, &pts );
  434. GetGNPhyRates (plc, &pts );
  435. // must get min PHY rate as specified from both the DUT and GN
  436. if( (pts.AVGRX_DUT >= pts.phy_rate_limit) && (pts.AVGRX_GN >= pts.phy_rate_limit) )
  437. {
  438. //test passed
  439. return( 0 );
  440. }
  441. if (gettimeofday (& tc, NULL) == -1)
  442. {
  443. error (1, errno, CANT_RESET_TIMER);
  444. }
  445. }
  446. // test failed, did not get min. PHY rate within test time specified
  447. return( 1 );
  448. }
  449. signed ClearToneMaps (struct plc * plc, struct pts * pts)
  450. {
  451. struct channel * channel = (struct channel *) (plc->channel);
  452. struct channel * channel1 = (struct channel *) (plc->channel1);
  453. struct message * message = (struct message *) (plc->message);
  454. #ifndef __GNUC__
  455. #pragma pack (push,1)
  456. #endif
  457. struct __packed vs_tonemap_op_request
  458. {
  459. struct ethernet_hdr ethernet;
  460. struct qualcomm_hdr qualcomm;
  461. uint8_t Mode;
  462. uint8_t Reserved1;
  463. uint8_t MACADDRESS [ETHER_ADDR_LEN];
  464. uint8_t DIRECTION;
  465. }
  466. * request = (struct vs_tonemap_op_request *) (message);
  467. #ifndef __GNUC__
  468. #pragma pack (pop)
  469. #endif
  470. // channel = GN->DUT
  471. memset (message, 0, sizeof (* message));
  472. EthernetHeader (& request->ethernet, pts->gn, channel->host, channel->type);
  473. QualcommHeader (& request->qualcomm, 0, (VS_TONE_MAP_OPER | MMTYPE_REQ));
  474. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  475. request->Mode = 0; //clean-up tonemap operation
  476. request->DIRECTION = 0; // clean both TX and RX tonemaps
  477. memcpy (request->MACADDRESS, pts->dut, sizeof (request->MACADDRESS));
  478. SendMME( plc );
  479. ReadMME (plc, 0, (VS_TONE_MAP_OPER | MMTYPE_CNF));
  480. // channel = DUT->GN
  481. memset (message, 0, sizeof (* message));
  482. EthernetHeader (& request->ethernet, pts->dut, channel1->host, channel1->type);
  483. QualcommHeader (& request->qualcomm, 0, (VS_TONE_MAP_OPER | MMTYPE_REQ));
  484. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  485. request->Mode = 0; //clean-up tonemap operation
  486. request->DIRECTION = 0; // clean both TX and RX tonemaps
  487. memcpy (request->MACADDRESS, pts->gn, sizeof (request->MACADDRESS));
  488. SendMME1( plc );
  489. ReadMME1 (plc, 0, (VS_TONE_MAP_OPER | MMTYPE_CNF));
  490. return (0);
  491. }
  492. /*====================================================================*
  493. *
  494. * void manager (struct plc * plc, signed count, signed pause);
  495. *
  496. * perform operations in logical order despite any order specfied
  497. * on the command line; for example read PIB before writing PIB;
  498. *
  499. * operation order is controlled by the order of "if" statements
  500. * shown here; the entire operation sequence can be repeated with
  501. * an optional pause between each iteration;
  502. *
  503. *--------------------------------------------------------------------*/
  504. static void manager (struct plc * plc, signed count, signed pause)
  505. {
  506. unsigned status = 0;
  507. char sourcename [ETHER_ADDR_LEN * 3];
  508. char targetname [ETHER_ADDR_LEN * 3];
  509. // Discover the GN MAC address
  510. Get_GN_MAC (plc);
  511. // Discover the DUT MAC address
  512. Get_DUT_MAC (plc);
  513. if (_anyset (plc->flags, PLC_VERBOSE))
  514. {
  515. hexdecode (pts.gn, ETHER_ADDR_LEN, sourcename, sizeof (sourcename));
  516. hexdecode (pts.dut, ETHER_ADDR_LEN, targetname, sizeof (targetname));
  517. printf ("GN = %s DUT = %s\n", sourcename, targetname);
  518. }
  519. // see if we need to program the attenuator
  520. if (_anyset (plc->flags, PLC_PUSH_BUTTON))
  521. {
  522. if (_anyset (plc->flags, PLC_VERBOSE))
  523. {
  524. printf(pts_messages[0], pts.atn_grnd, pts.atn_line );
  525. }
  526. relay( pts.atn_grnd, pts.atn_line, pts.name );
  527. }
  528. // clear the tonemaps, unless "-c1" option set
  529. if( pts.tonemaps == 0 )
  530. {
  531. if (_anyset (plc->flags, PLC_VERBOSE))
  532. {
  533. printf ("\nClearing tonemaps between GN and DUT\n");
  534. }
  535. ClearToneMaps( plc, &pts );
  536. }
  537. switch ( pts.traffic )
  538. {
  539. case 0:
  540. if (_anyset (plc->flags, PLC_VERBOSE))
  541. {
  542. printf("Sending traffic from GN to DUT...\n");
  543. }
  544. status = TestGNToDUT( plc );
  545. if (_anyset (plc->flags, PLC_VERBOSE))
  546. {
  547. if( status )
  548. {
  549. printf("Test FAILED\nMinimum PHY Rate Expected = %dMbps Actual DUT PHY Rate = %dMbps\n", pts.phy_rate_limit, pts.AVGRX_DUT);
  550. }
  551. else
  552. {
  553. printf("Test PASSED\nMinimum PHY Rate Expected = %dMbps Actual DUT PHY Rate = %dMbps\n", pts.phy_rate_limit, pts.AVGRX_DUT);
  554. }
  555. }
  556. break;
  557. case 1:
  558. if (_anyset (plc->flags, PLC_VERBOSE))
  559. {
  560. printf("Sending traffic from DUT to GN...\n");
  561. }
  562. status = TestDUTToGN( plc );
  563. if (_anyset (plc->flags, PLC_VERBOSE))
  564. {
  565. if( status )
  566. {
  567. printf("Test FAILED\nMinimum Expected PHY Rate = %dMbps Actual GN PHY Rate = %dMbps\n", pts.phy_rate_limit, pts.AVGRX_GN);
  568. }
  569. else
  570. {
  571. printf("Test PASSED\nMinimum Expected PHY Rate = %dMbps Actual GN PHY Rate = %dMbps\n", pts.phy_rate_limit, pts.AVGRX_GN);
  572. }
  573. }
  574. break;
  575. case 2:
  576. if (_anyset (plc->flags, PLC_VERBOSE))
  577. {
  578. printf ("Sending traffic between GN and DUT\n");
  579. }
  580. status = TestBiDi( plc );
  581. if (_anyset (plc->flags, PLC_VERBOSE))
  582. {
  583. if( status )
  584. {
  585. printf("Test FAILED...Minimum PHY Rate Expected = %dMbps Actual GN PHY Rate = %dMbps Actual DUT PHY Rate = %dMbps\n", pts.phy_rate_limit, pts.AVGRX_GN, pts.AVGRX_DUT);
  586. }
  587. else
  588. {
  589. printf("Test PASSED...Minimum PHY Rate Expected = %dMbps Actual GN PHY Rate = %dMbps Actual DUT PHY Rate = %dMbps\n", pts.phy_rate_limit, pts.AVGRX_GN, pts.AVGRX_DUT);
  590. }
  591. }
  592. break;
  593. }
  594. if (!status)
  595. {
  596. if (_anyset (plc->flags, PLC_VERBOSE))
  597. {
  598. printf ("Test PASSED returning 0\n" );
  599. }
  600. exit( 0 );
  601. }
  602. if( pts.exit_option == 1 )
  603. {
  604. if (_anyset (plc->flags, PLC_VERBOSE))
  605. {
  606. printf ("Test FAILED bypass error returning 0\n");
  607. }
  608. // Bypass failure and return success
  609. exit( 0 );
  610. }
  611. if (_anyset (plc->flags, PLC_VERBOSE))
  612. {
  613. printf ("Test FAILED returning 1\n" );
  614. }
  615. // Return Failed Status
  616. exit( 1 );
  617. }
  618. signed GetDUTPhyRates (struct plc * plc, struct pts * pts )
  619. {
  620. struct channel * channel1 = (struct channel *) (plc->channel1);
  621. struct message * message = (struct message *) (plc->message);
  622. #ifndef __GNUC__
  623. #pragma pack (push,1)
  624. #endif
  625. struct __packed vs_nw_info_request
  626. {
  627. struct ethernet_hdr ethernet;
  628. struct qualcomm_fmi qualcomm;
  629. }
  630. * request = (struct vs_nw_info_request *) (message);
  631. struct __packed vs_nw_info_confirm
  632. {
  633. struct ethernet_hdr ethernet;
  634. struct qualcomm_fmi qualcomm;
  635. uint8_t SUB_VERSION;
  636. uint8_t Reserved;
  637. uint16_t DATA_LEN;
  638. uint8_t DATA [1];
  639. }
  640. * confirm = (struct vs_nw_info_confirm *) (message);
  641. struct __packed station
  642. {
  643. uint8_t MAC [ETHER_ADDR_LEN];
  644. uint8_t TEI;
  645. uint8_t Reserved [3];
  646. uint8_t BDA [ETHER_ADDR_LEN];
  647. uint16_t AVGTX;
  648. uint8_t COUPLING;
  649. uint8_t Reserved3;
  650. uint16_t AVGRX;
  651. uint16_t Reserved4;
  652. }
  653. * station;
  654. struct __packed network
  655. {
  656. uint8_t NID [7];
  657. uint8_t Reserved1 [2];
  658. uint8_t SNID;
  659. uint8_t TEI;
  660. uint8_t Reserved2 [4];
  661. uint8_t ROLE;
  662. uint8_t CCO_MAC [ETHER_ADDR_LEN];
  663. uint8_t CCO_TEI;
  664. uint8_t Reserved3 [3];
  665. uint8_t NUMSTAS;
  666. uint8_t Reserved4 [5];
  667. struct station stations [1];
  668. }
  669. * network;
  670. struct __packed networks
  671. {
  672. uint8_t Reserved;
  673. uint8_t NUMAVLNS;
  674. struct network networks [1];
  675. }
  676. * networks = (struct networks *) (confirm->DATA);
  677. #ifndef __GNUC__
  678. #pragma pack (pop)
  679. #endif
  680. memset (message, 0, sizeof (* message));
  681. EthernetHeader (& request->ethernet, localcast, channel1->host, channel1->type);
  682. QualcommHeader1 (& request->qualcomm, 1, (VS_NW_INFO | MMTYPE_REQ));
  683. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  684. if (SendMME1 (plc) <= 0)
  685. {
  686. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  687. return (-1);
  688. }
  689. while (ReadMME1 (plc, 1, (VS_NW_INFO | MMTYPE_CNF)) > 0)
  690. {
  691. network = (struct network *) (& networks->networks);
  692. while (networks->NUMAVLNS--)
  693. {
  694. station = (struct station *) (& network->stations);
  695. while (network->NUMSTAS--)
  696. {
  697. //look for the MAC address of the GN
  698. if (memcmp (station->MAC, pts->gn, ETHER_ADDR_LEN) == 0)
  699. {
  700. // Save the RX PHY rate (coded)
  701. station->AVGTX = LE16TOH (station->AVGTX);
  702. station->AVGRX = LE16TOH (station->AVGRX);
  703. // uncoded PHY rates
  704. //station->AVGTX = ((station->AVGTX * 21) >> 4);
  705. //station->AVGRX = ((station->AVGRX * 21) >> 4);
  706. pts->AVGRX_DUT = station->AVGRX;
  707. return( 0 );
  708. }
  709. station++;
  710. }
  711. network = (struct network *) (station);
  712. }
  713. }
  714. return (0);
  715. }
  716. signed GetGNPhyRates (struct plc * plc, struct pts * pts )
  717. {
  718. struct channel * channel = (struct channel *) (plc->channel);
  719. struct message * message = (struct message *) (plc->message);
  720. #ifndef __GNUC__
  721. #pragma pack (push,1)
  722. #endif
  723. struct __packed vs_nw_info_request
  724. {
  725. struct ethernet_hdr ethernet;
  726. struct qualcomm_fmi qualcomm;
  727. }
  728. * request = (struct vs_nw_info_request *) (message);
  729. struct __packed vs_nw_info_confirm
  730. {
  731. struct ethernet_hdr ethernet;
  732. struct qualcomm_fmi qualcomm;
  733. uint8_t SUB_VERSION;
  734. uint8_t Reserved;
  735. uint16_t DATA_LEN;
  736. uint8_t DATA [1];
  737. }
  738. * confirm = (struct vs_nw_info_confirm *) (message);
  739. struct __packed station
  740. {
  741. uint8_t MAC [ETHER_ADDR_LEN];
  742. uint8_t TEI;
  743. uint8_t Reserved [3];
  744. uint8_t BDA [ETHER_ADDR_LEN];
  745. uint16_t AVGTX;
  746. uint8_t COUPLING;
  747. uint8_t Reserved3;
  748. uint16_t AVGRX;
  749. uint16_t Reserved4;
  750. }
  751. * station;
  752. struct __packed network
  753. {
  754. uint8_t NID [7];
  755. uint8_t Reserved1 [2];
  756. uint8_t SNID;
  757. uint8_t TEI;
  758. uint8_t Reserved2 [4];
  759. uint8_t ROLE;
  760. uint8_t CCO_MAC [ETHER_ADDR_LEN];
  761. uint8_t CCO_TEI;
  762. uint8_t Reserved3 [3];
  763. uint8_t NUMSTAS;
  764. uint8_t Reserved4 [5];
  765. struct station stations [1];
  766. }
  767. * network;
  768. struct __packed networks
  769. {
  770. uint8_t Reserved;
  771. uint8_t NUMAVLNS;
  772. struct network networks [1];
  773. }
  774. * networks = (struct networks *) (confirm->DATA);
  775. #ifndef __GNUC__
  776. #pragma pack (pop)
  777. #endif
  778. memset (message, 0, sizeof (* message));
  779. EthernetHeader (& request->ethernet, localcast, channel->host, channel->type);
  780. QualcommHeader1 (& request->qualcomm, 1, (VS_NW_INFO | MMTYPE_REQ));
  781. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  782. if (SendMME (plc) <= 0)
  783. {
  784. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  785. return (-1);
  786. }
  787. while (ReadMME (plc, 1, (VS_NW_INFO | MMTYPE_CNF)) > 0)
  788. {
  789. network = (struct network *) (& networks->networks);
  790. while (networks->NUMAVLNS--)
  791. {
  792. station = (struct station *) (& network->stations);
  793. while (network->NUMSTAS--)
  794. {
  795. //look for the MAC address of the DUT
  796. if (memcmp (station->MAC, pts->dut, ETHER_ADDR_LEN) == 0)
  797. {
  798. // Save the RX PHY rate (coded)
  799. station->AVGTX = LE16TOH (station->AVGTX);
  800. station->AVGRX = LE16TOH (station->AVGRX);
  801. // uncoded PHY rates
  802. //station->AVGTX = ((station->AVGTX * 21) >> 4);
  803. //station->AVGRX = ((station->AVGRX * 21) >> 4);
  804. pts->AVGRX_GN = station->AVGRX;
  805. return( 0 );
  806. }
  807. station++;
  808. }
  809. network = (struct network *) (station);
  810. }
  811. }
  812. return (0);
  813. }
  814. // transmit from GN to DUT
  815. signed Transmit4 (struct plc * plc, byte source [], byte target [])
  816. {
  817. struct channel * channel = (struct channel *) (plc->channel);
  818. struct message * message = (struct message *) (plc->message);
  819. struct timeval ts;
  820. struct timeval tc;
  821. unsigned timer = 0;
  822. memset (message, 0xAA, sizeof (* message));
  823. EthernetHeader (message, target, source, ETHERTYPE_IP);
  824. if (gettimeofday (& ts, NULL) == -1)
  825. {
  826. error (1, errno, CANT_START_TIMER);
  827. }
  828. for (timer = 0; timer < 5; timer = SECONDS (ts, tc))
  829. {
  830. // this is a work around for a buffer overrun condition that may happen nder OSX
  831. if (sendpacket (channel, message, sizeof (* message)) == -1)
  832. {
  833. // we may be overrunning the buffer so need to wait a bit
  834. sleep( 1 );
  835. }
  836. if (gettimeofday (& tc, NULL) == -1)
  837. {
  838. error (1, errno, CANT_RESET_TIMER);
  839. }
  840. }
  841. return (0);
  842. }
  843. // transmit from DUT to GN
  844. signed Transmit5 (struct plc * plc, byte source [], byte target [])
  845. {
  846. struct channel * channel1 = (struct channel *) (plc->channel1);
  847. struct message * message = (struct message *) (plc->message);
  848. struct timeval ts;
  849. struct timeval tc;
  850. unsigned timer = 0;
  851. memset (message, 0x55, sizeof (* message));
  852. EthernetHeader (message, target, source, ETHERTYPE_IP);
  853. if (gettimeofday (& ts, NULL) == -1)
  854. {
  855. error (1, errno, CANT_START_TIMER);
  856. }
  857. // Transmit for at least 5 seconds before returning and asking for the PHY rates
  858. for (timer = 0; timer < 5; timer = SECONDS (ts, tc))
  859. {
  860. // this is a work around for a buffer overrun condition that may happen under OSX
  861. if (sendpacket (channel1, message, sizeof (* message)) == -1)
  862. {
  863. // we may be overrunning the buffer so need to wait a bit
  864. sleep( 1 );
  865. }
  866. if (gettimeofday (& tc, NULL) == -1)
  867. {
  868. error (1, errno, CANT_RESET_TIMER);
  869. }
  870. }
  871. return (0);
  872. }
  873. // transmit bi-di
  874. signed Transmit6 (struct plc * plc, byte source [], byte target [])
  875. {
  876. struct channel * channel = (struct channel *) (plc->channel);
  877. struct channel * channel1 = (struct channel *) (plc->channel1);
  878. struct message * message = (struct message *) (plc->message);
  879. struct timeval ts;
  880. struct timeval tc;
  881. unsigned timer = 0;
  882. if (_allclr (plc->flags, PLC_SILENCE))
  883. {
  884. char sourcename [ETHER_ADDR_LEN * 3];
  885. char targetname [ETHER_ADDR_LEN * 3];
  886. hexdecode (source, ETHER_ADDR_LEN, sourcename, sizeof (sourcename));
  887. hexdecode (target, ETHER_ADDR_LEN, targetname, sizeof (targetname));
  888. }
  889. memset (message, 0xA5, sizeof (* message));
  890. if (gettimeofday (& ts, NULL) == -1)
  891. {
  892. error (1, errno, CANT_START_TIMER);
  893. }
  894. for (timer = 0; timer < 5; timer = SECONDS (ts, tc))
  895. {
  896. EthernetHeader (message, target, source, ETHERTYPE_IP);
  897. // this is a work around for a buffer overrun condition observed under OSX
  898. if (sendpacket (channel, message, sizeof (* message)) == -1)
  899. {
  900. // we may be overrunning the buffer so need to wait a bit
  901. sleep( 1 );
  902. }
  903. EthernetHeader (message, source, target, ETHERTYPE_IP);
  904. if (sendpacket (channel1, message, sizeof (* message)) == -1)
  905. {
  906. // we may be overrunning the buffer so need to wait a bit
  907. sleep( 1 );
  908. }
  909. if (gettimeofday (& tc, NULL) == -1)
  910. {
  911. error (1, errno, CANT_RESET_TIMER);
  912. }
  913. }
  914. return (0);
  915. }
  916. /*====================================================================*
  917. *
  918. * int main (int argc, char const * argv[]);
  919. *
  920. * parse command line, populate plc structure and perform selected
  921. * operations; show help summary if asked; see getoptv and putoptv
  922. * to understand command line parsing and help summary display; see
  923. * plc.h for the definition of struct plc;
  924. *
  925. * the command line accepts multiple MAC addresses and the program
  926. * performs the specified operations on each address, in turn; the
  927. * address order is significant but the option order is not; the
  928. * default address is a local broadcast that causes all devices on
  929. * the local H1 interface to respond but not those at the remote
  930. * end of the powerline;
  931. *
  932. * the default address is 00:B0:52:00:00:01; omitting the address
  933. * will automatically address the local device; some options will
  934. * cancel themselves if this makes no sense;
  935. *
  936. * the default interface is eth1 because most people use eth0 as
  937. * their principle network connection; you can specify another
  938. * interface with -i or define environment string PLC to make
  939. * that the default interface and save typing;
  940. *
  941. *--------------------------------------------------------------------*/
  942. int main (int argc, char const * argv [])
  943. {
  944. extern struct channel channel;
  945. extern struct channel channel1;
  946. static char const * optv [] =
  947. {
  948. "f:g:n:c:e:i:j:p:d:t:v",
  949. "device [device] [...]",
  950. "Qualcomm Atheros Production Test Utility",
  951. "f f\tport is (f) [" PTSCTL_PORT "]",
  952. "g n\tline ground attenuation is (n) [" LITERAL (PTSCTL_GRND_ATTN) "]",
  953. "n n\tline neutral attenuation is (n) [" LITERAL (PTSCTL_LINE_ATTN) "]",
  954. "c n\tclear tonemap option (pts speed-up) 0=clear tonemaps, 1=tonemaps not cleared [0] ",
  955. "e n\texit option 0=return 0 for test pass,1 for test failure, 1=always return 0 (test pass) [0] ",
  956. #if defined (WINPCAP) || defined (LIBPCAP)
  957. "i n\thost interface (GN) is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  958. "j n\thost interface (DUT) is (n) [" LITERAL (3) "]",
  959. #else
  960. "i s\thost interface (GN) is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  961. "j s\thost interface (DUT) is (s) [" LITERAL (3) "]",
  962. #endif
  963. "p n\tPHY rate test limit (n) Mbps",
  964. "t n\tMaximum time to run test (n) seconds",
  965. "d n\tTraffic direction [0|1|2|'GN->DUT'|'DUT->GN'|'GN<->DUT']",
  966. "v\tOutput Progress Messages",
  967. (char const *) (0)
  968. };
  969. #include "../plc/plc.c"
  970. signed loop = PLCTOOL_LOOP;
  971. signed wait = PLCTOOL_WAIT;
  972. signed c;
  973. optind = 1;
  974. if (getenv (PLCDEVICE))
  975. {
  976. #if defined (WINPCAP) || defined (LIBPCAP)
  977. channel.ifindex = atoi (getenv (PLCDEVICE));
  978. #else
  979. channel.ifname = strdup (getenv (PLCDEVICE));
  980. #endif
  981. }
  982. if (getenv ("PLC1"))
  983. {
  984. #if defined (WINPCAP) || defined (LIBPCAP)
  985. channel1.ifindex = atoi (getenv ("PLC1"));
  986. #else
  987. channel1.ifname = strdup (getenv ("PLC1"));
  988. #endif
  989. }
  990. plc.timer = PLC_ECHOTIME;
  991. //optind = 1;
  992. while (~ (c = getoptv (argc, argv, optv)))
  993. {
  994. switch (c)
  995. {
  996. case 'd':
  997. pts.traffic = atoi (optarg);
  998. //pts.traffic = (unsigned) (uintspec (synonym (optarg, buttons, BUTTONS), 0, UCHAR_MAX));
  999. // check range
  1000. if ( (pts.traffic < 0) || (pts.traffic >2) )
  1001. {
  1002. error (1, errno, "Invalid Traffic Direction Type %d", pts.traffic );
  1003. }
  1004. break;
  1005. case 'f':
  1006. pts.name = optarg;
  1007. break;
  1008. case 'g':
  1009. pts.atn_grnd = (unsigned) (uintspec (optarg, 0, 0x7F));
  1010. _setbits (plc.flags, PLC_PUSH_BUTTON);
  1011. break;
  1012. case 'n':
  1013. pts.atn_line = (unsigned) (uintspec (optarg, 0, 0x7F));
  1014. _setbits (plc.flags, PLC_PUSH_BUTTON);
  1015. break;
  1016. case 'c':
  1017. pts.tonemaps = atoi (optarg);
  1018. break;
  1019. case 'e':
  1020. pts.exit_option = atoi (optarg);
  1021. break;
  1022. case 'p':
  1023. pts.phy_rate_limit = atoi (optarg);
  1024. break;
  1025. case 't':
  1026. pts.timeout = atoi (optarg);
  1027. break;
  1028. case 'i':
  1029. // GN interface
  1030. #if defined (WINPCAP) || defined (LIBPCAP)
  1031. channel.ifindex = atoi (optarg);
  1032. #else
  1033. channel.ifname = optarg;
  1034. #endif
  1035. break;
  1036. case 'j':
  1037. // DUT interface
  1038. #if defined (WINPCAP) || defined (LIBPCAP)
  1039. channel1.ifindex = atoi (optarg);
  1040. #else
  1041. channel1.ifname = optarg;
  1042. #endif
  1043. break;
  1044. case 'v':
  1045. //_setbits (channel.flags, CHANNEL_VERBOSE);
  1046. _setbits (plc.flags, PLC_VERBOSE);
  1047. break;
  1048. default:
  1049. break;
  1050. }
  1051. }
  1052. argc -= optind;
  1053. argv += optind;
  1054. openchannel (& channel);
  1055. openchannel (& channel1);
  1056. if (! (plc.message = malloc (sizeof (* plc.message))))
  1057. {
  1058. error (1, errno, PLC_NOMEMORY);
  1059. }
  1060. manager (& plc, loop, wait);
  1061. argc--;
  1062. argv++;
  1063. free (plc.message);
  1064. closechannel (& channel);
  1065. closechannel (& channel1);
  1066. exit (0);
  1067. }