int6kwait.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * plcwait.c - Qualcomm Atheros Powerline Device Procrastinator;
  11. *
  12. * wait for device events to start or finish before returning;
  13. *
  14. * Contributor(s):
  15. * Charles Maier <cmaier@qca.qualcomm.com>
  16. *
  17. *--------------------------------------------------------------------*/
  18. /*====================================================================*
  19. * system header files;
  20. *--------------------------------------------------------------------*/
  21. #include <unistd.h>
  22. #include <stdlib.h>
  23. #include <stdint.h>
  24. #include <limits.h>
  25. #include <sys/time.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/timer.h"
  38. #include "../tools/error.h"
  39. #include "../plc/plc.h"
  40. /*====================================================================*
  41. * custom source files;
  42. *--------------------------------------------------------------------*/
  43. #ifndef MAKEFILE
  44. #include "../tools/getoptv.c"
  45. #include "../tools/putoptv.c"
  46. #include "../tools/version.c"
  47. #include "../tools/uintspec.c"
  48. #include "../tools/hexdump.c"
  49. #include "../tools/hexencode.c"
  50. #include "../tools/hexdecode.c"
  51. #include "../tools/todigit.c"
  52. #include "../tools/checkfilename.c"
  53. #include "../tools/synonym.c"
  54. #include "../tools/error.c"
  55. #endif
  56. #ifndef MAKEFILE
  57. #include "../plc/Confirm.c"
  58. #include "../plc/Display.c"
  59. #include "../plc/Failure.c"
  60. #include "../plc/ReadMME.c"
  61. #include "../plc/Request.c"
  62. #include "../plc/SendMME.c"
  63. #include "../plc/Devices.c"
  64. #endif
  65. #ifndef MAKEFILE
  66. #include "../ether/openchannel.c"
  67. #include "../ether/closechannel.c"
  68. #include "../ether/readpacket.c"
  69. #include "../ether/sendpacket.c"
  70. #include "../ether/channel.c"
  71. #endif
  72. #ifndef MAKEFILE
  73. #include "../mme/MMECode.c"
  74. #include "../mme/EthernetHeader.c"
  75. #include "../mme/QualcommHeader.c"
  76. #include "../mme/QualcommHeader1.c"
  77. #include "../mme/UnwantedMessage.c"
  78. #endif
  79. #define RETRY 300
  80. #define POLL 5
  81. /*====================================================================*
  82. *
  83. * signed ResetAndWait (struct plc * plc);
  84. *
  85. * plc.h
  86. *
  87. * reset the device using a VS_RS_DEV Request message; continue to
  88. * request resets each second until the device accepts the request
  89. * or the wait period expires;
  90. *
  91. *--------------------------------------------------------------------*/
  92. signed ResetAndWait (struct plc * plc)
  93. {
  94. struct channel * channel = (struct channel *) (plc->channel);
  95. struct message * message = (struct message *) (plc->message);
  96. struct timeval ts;
  97. struct timeval tc;
  98. unsigned timer = 0;
  99. #ifndef __GNUC__
  100. #pragma pack (push,1)
  101. #endif
  102. struct __packed vs_rs_dev_request
  103. {
  104. struct ethernet_hdr ethernet;
  105. struct qualcomm_hdr qualcomm;
  106. }
  107. * request = (struct vs_rs_dev_request *) (message);
  108. struct __packed vs_rs_dev_confirm
  109. {
  110. struct ethernet_hdr ethernet;
  111. struct qualcomm_hdr qualcomm;
  112. uint8_t MSTATUS;
  113. }
  114. * confirm = (struct vs_rs_dev_confirm *) (message);
  115. #ifndef __GNUC__
  116. #pragma pack (pop)
  117. #endif
  118. Request (plc, "Reset when Ready");
  119. if (gettimeofday (& ts, NULL) == -1)
  120. {
  121. error (1, errno, CANT_START_TIMER);
  122. }
  123. for (timer = 0; timer < plc->timer; timer = SECONDS (ts, tc))
  124. {
  125. memset (message, 0, sizeof (* message));
  126. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  127. QualcommHeader (& request->qualcomm, 0, (VS_RS_DEV | MMTYPE_REQ));
  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_RS_DEV | MMTYPE_CNF)) < 0)
  135. {
  136. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  137. return (-1);
  138. }
  139. if (gettimeofday (& tc, NULL) == -1)
  140. {
  141. error (1, errno, CANT_RESET_TIMER);
  142. }
  143. if (plc->packetsize)
  144. {
  145. if (! confirm->MSTATUS)
  146. {
  147. Confirm (plc, "Resetting ...");
  148. return (0);
  149. }
  150. }
  151. }
  152. return (-1);
  153. }
  154. /*====================================================================*
  155. *
  156. * signed WaitForReset (struct plc * plc);
  157. *
  158. * plc.h
  159. *
  160. * send and receive VS_SW_VER messages until a confirmation is not
  161. * received within channel->timeout milliseconds to indicate that
  162. * the device is inactive; return 0 if the device resets within
  163. * plc->timer seconds; otherwise, return -1;
  164. *
  165. * this function cannot distinguish between a software reset and
  166. * hardware reset;
  167. *
  168. *--------------------------------------------------------------------*/
  169. signed WaitForReset (struct plc * plc)
  170. {
  171. struct channel * channel = (struct channel *) (plc->channel);
  172. struct message * message = (struct message *) (plc->message);
  173. struct timeval ts;
  174. struct timeval tc;
  175. unsigned timer = 0;
  176. #ifndef __GNUC__
  177. #pragma pack (push,1)
  178. #endif
  179. struct __packed vs_sw_ver_request
  180. {
  181. struct ethernet_hdr ethernet;
  182. struct qualcomm_hdr qualcomm;
  183. uint8_t MSTATUS;
  184. uint8_t MDEVICEID;
  185. uint8_t MVERLENGTH;
  186. char MVERSION [PLC_VERSION_STRING];
  187. }
  188. * request = (struct vs_sw_ver_request *) (message);
  189. #ifndef __GNUC__
  190. #pragma pack (pop)
  191. #endif
  192. Request (plc, "Allow %d seconds for Reset", plc->timer);
  193. if (gettimeofday (& ts, NULL) == -1)
  194. {
  195. error (1, errno, CANT_START_TIMER);
  196. }
  197. for (timer = 0; timer < plc->timer; timer = SECONDS (ts, tc))
  198. {
  199. memset (message, 0, sizeof (* message));
  200. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  201. QualcommHeader (& request->qualcomm, 0, (VS_SW_VER | MMTYPE_REQ));
  202. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  203. if (SendMME (plc) <= 0)
  204. {
  205. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  206. return (-1);
  207. }
  208. if (ReadMME (plc, 0, (VS_SW_VER | MMTYPE_CNF)) < 0)
  209. {
  210. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  211. return (-1);
  212. }
  213. if (gettimeofday (& tc, NULL) == -1)
  214. {
  215. error (1, errno, CANT_RESET_TIMER);
  216. }
  217. if (! plc->packetsize)
  218. {
  219. if (_allset (plc->flags, (PLC_WAITFORRESET | PLC_ANALYSE)))
  220. {
  221. Confirm (plc, "Waited %d seconds for Reset", timer);
  222. }
  223. return (0);
  224. }
  225. }
  226. if (_allset (plc->flags, (PLC_WAITFORRESET | PLC_ANALYSE)))
  227. {
  228. Confirm (plc, "Waited %d seconds for Reset", timer);
  229. }
  230. return (-1);
  231. }
  232. /*====================================================================*
  233. *
  234. * signed WaitForStart (struct plc * plc, char string [], size_t length);
  235. *
  236. * plc.h
  237. *
  238. * send VS_SW_VER messages until confirmation is received within
  239. * channel->timeout milliseconds to indicate that the device is
  240. * active; return 0 if the device responds within plc->timer
  241. * seconds; otherwise, return -1;
  242. *
  243. * poll the device using VS_SW_VER messages until it responds or
  244. * the allotted time expires; return 0 if the device responds within
  245. * the allotted time or -1 if it does not or if a transmission error
  246. * occurs;
  247. *
  248. *--------------------------------------------------------------------*/
  249. signed WaitForStart (struct plc * plc, char string [], size_t length)
  250. {
  251. struct channel * channel = (struct channel *) (plc->channel);
  252. struct message * message = (struct message *) (plc->message);
  253. struct timeval ts;
  254. struct timeval tc;
  255. unsigned timer = 0;
  256. #ifndef __GNUC__
  257. #pragma pack (push,1)
  258. #endif
  259. struct __packed vs_sw_ver_request
  260. {
  261. struct ethernet_hdr ethernet;
  262. struct qualcomm_hdr qualcomm;
  263. uint8_t MSTATUS;
  264. uint8_t MDEVICEID;
  265. uint8_t MVERLENGTH;
  266. char MVERSION [PLC_VERSION_STRING];
  267. }
  268. * request = (struct vs_sw_ver_request *) (message);
  269. struct __packed vs_sw_ver_confirm
  270. {
  271. struct ethernet_hdr ethernet;
  272. struct qualcomm_hdr qualcomm;
  273. uint8_t MSTATUS;
  274. uint8_t MDEVICEID;
  275. uint8_t MVERLENGTH;
  276. char MVERSION [PLC_VERSION_STRING];
  277. }
  278. * confirm = (struct vs_sw_ver_confirm *) (message);
  279. #ifndef __GNUC__
  280. #pragma pack (pop)
  281. #endif
  282. Request (plc, "Allow %d seconds for Start", plc->timer);
  283. if (gettimeofday (& ts, NULL) == -1)
  284. {
  285. error (1, errno, CANT_START_TIMER);
  286. }
  287. for (timer = 0; timer < plc->timer; timer = SECONDS (ts, tc))
  288. {
  289. memset (message, 0, sizeof (* message));
  290. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  291. QualcommHeader (& request->qualcomm, 0, (VS_SW_VER | MMTYPE_REQ));
  292. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  293. if (SendMME (plc) <= 0)
  294. {
  295. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  296. return (-1);
  297. }
  298. if (ReadMME (plc, 0, (VS_SW_VER | MMTYPE_CNF)) < 0)
  299. {
  300. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  301. return (-1);
  302. }
  303. if (gettimeofday (& tc, NULL) == -1)
  304. {
  305. error (1, errno, CANT_RESET_TIMER);
  306. }
  307. if (plc->packetsize)
  308. {
  309. if (confirm->MSTATUS)
  310. {
  311. Failure (plc, PLC_WONTDOIT);
  312. return (-1);
  313. }
  314. if (_allset (plc->flags, (PLC_WAITFORSTART | PLC_ANALYSE)))
  315. {
  316. Confirm (plc, "Waited %d seconds for Start", timer);
  317. }
  318. strncpy (string, confirm->MVERSION, length);
  319. return (0);
  320. }
  321. }
  322. if (_allset (plc->flags, (PLC_WAITFORSTART | PLC_ANALYSE)))
  323. {
  324. Confirm (plc, "Waited %d seconds for Start", timer);
  325. }
  326. return (-1);
  327. }
  328. /*====================================================================*
  329. *
  330. * signed WaitForAssoc (struct plc * plc);
  331. *
  332. * plc.h
  333. *
  334. * send and receive VS_NW_INFO messages until the device reports
  335. * that a network exists and has at least one station; return 0 if
  336. * a network forms within plc->timer seconds; otherwise, return
  337. * -1;
  338. *
  339. *--------------------------------------------------------------------*/
  340. signed WaitForAssoc (struct plc * plc)
  341. {
  342. extern const uint8_t broadcast [ETHER_ADDR_LEN];
  343. struct channel * channel = (struct channel *) (plc->channel);
  344. struct message * message = (struct message *) (plc->message);
  345. struct timeval ts;
  346. struct timeval tc;
  347. unsigned timer = 0;
  348. #ifndef __GNUC__
  349. #pragma pack (push,1)
  350. #endif
  351. struct __packed vs_nw_info_request
  352. {
  353. struct ethernet_hdr ethernet;
  354. struct qualcomm_hdr qualcomm;
  355. }
  356. * request = (struct vs_nw_info_request *) (message);
  357. struct __packed vs_nw_info_confirm
  358. {
  359. struct ethernet_hdr ethernet;
  360. struct qualcomm_hdr qualcomm;
  361. uint8_t data [1];
  362. }
  363. * confirm = (struct vs_nw_info_confirm *) (message);
  364. struct __packed station
  365. {
  366. uint8_t MAC [ETHER_ADDR_LEN];
  367. uint8_t TEI;
  368. uint8_t BDA [ETHER_ADDR_LEN];
  369. uint8_t AVGTX;
  370. uint8_t AVGRX;
  371. }
  372. * station;
  373. struct __packed network
  374. {
  375. uint8_t NID [7];
  376. uint8_t SNID;
  377. uint8_t TEI;
  378. uint8_t ROLE;
  379. uint8_t CCO_MAC [ETHER_ADDR_LEN];
  380. uint8_t CCO_TEI;
  381. uint8_t NUMSTAS;
  382. struct station stations [1];
  383. }
  384. * network;
  385. struct __packed networks
  386. {
  387. uint8_t NUMAVLNS;
  388. struct network networks [1];
  389. }
  390. * networks = (struct networks *) (confirm->data);
  391. #ifndef __GNUC__
  392. #pragma pack (pop)
  393. #endif
  394. Request (plc, "Allow %d seconds for Assoc", plc->timer);
  395. if (gettimeofday (& ts, NULL) == -1)
  396. {
  397. error (1, errno, CANT_START_TIMER);
  398. }
  399. for (timer = 0; timer < plc->timer; timer = SECONDS (ts, tc))
  400. {
  401. memset (message, 0, sizeof (* message));
  402. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  403. QualcommHeader (& request->qualcomm, 0, (VS_NW_INFO | MMTYPE_REQ));
  404. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  405. if (SendMME (plc) <= 0)
  406. {
  407. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  408. return (-1);
  409. }
  410. if (ReadMME (plc, 0, (VS_NW_INFO | MMTYPE_CNF)) < 0)
  411. {
  412. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  413. return (-1);
  414. }
  415. if (gettimeofday (& tc, NULL) == -1)
  416. {
  417. error (1, errno, CANT_RESET_TIMER);
  418. }
  419. if (plc->packetsize)
  420. {
  421. network = (struct network *) (& networks->networks);
  422. while (networks->NUMAVLNS--)
  423. {
  424. station = (struct station *) (& network->stations);
  425. while (network->NUMSTAS--)
  426. {
  427. if (memcmp (station->MAC, broadcast, sizeof (broadcast)))
  428. {
  429. if (_allset (plc->flags, (PLC_WAITFORASSOC | PLC_ANALYSE)))
  430. {
  431. Confirm (plc, "Waited %d seconds for Assoc", timer);
  432. }
  433. return (0);
  434. }
  435. station++;
  436. }
  437. network = (struct network *) (station);
  438. }
  439. }
  440. }
  441. if (_allset (plc->flags, (PLC_WAITFORASSOC | PLC_ANALYSE)))
  442. {
  443. Confirm (plc, "Waited %d seconds for Assoc", timer);
  444. }
  445. return (-1);
  446. }
  447. /*====================================================================*
  448. *
  449. * void function (struct plc * plc, char const * firmware);
  450. *
  451. * perform operations in a logical order;
  452. *
  453. *--------------------------------------------------------------------*/
  454. static void function (struct plc * plc, char const * firmware)
  455. {
  456. char string [PLC_VERSION_STRING];
  457. if (_anyset (plc->flags, PLC_RESET_DEVICE))
  458. {
  459. if (ResetAndWait (plc))
  460. {
  461. Failure (plc, "Device did not Reset.");
  462. }
  463. }
  464. if (_anyset (plc->flags, PLC_WAITFORRESET))
  465. {
  466. if (WaitForReset (plc))
  467. {
  468. Failure (plc, "Device did not Reset.");
  469. }
  470. }
  471. if (_anyset (plc->flags, PLC_WAITFORSTART))
  472. {
  473. if (WaitForStart (plc, string, sizeof (string)))
  474. {
  475. Failure (plc, "Device did not Start.");
  476. }
  477. if ((firmware) && (* firmware) && strcmp (firmware, string))
  478. {
  479. Failure (plc, "Started wrong firmware");
  480. }
  481. }
  482. if (_anyset (plc->flags, PLC_WAITFORASSOC))
  483. {
  484. if (WaitForAssoc (plc))
  485. {
  486. Failure (plc, "Device did not Assoc.");
  487. }
  488. }
  489. if (plc->sleep)
  490. {
  491. Request (plc, "Pause %d seconds", plc->sleep);
  492. sleep (plc->sleep);
  493. }
  494. return;
  495. }
  496. /*====================================================================*
  497. *
  498. * int main (int argc, char const * argv[]);
  499. *
  500. * parse command line, populate plc structure and perform selected
  501. * operations; show help summary if asked; see getoptv and putoptv
  502. * to understand command line parsing and help summary display; see
  503. * plc.h for the definition of struct plc;
  504. *
  505. * the command line accepts multiple MAC addresses and the program
  506. * performs the specified operations on each address, in turn; the
  507. * address order is significant but the option order is not; the
  508. * default address is a local broadcast that causes all devices on
  509. * the local H1 interface to respond but not those at the remote
  510. * end of the powerline;
  511. *
  512. * the default address is 00:B0:52:00:00:01; omitting the address
  513. * will automatically address the local device; some options will
  514. * cancel themselves if this makes no sense;
  515. *
  516. * the default interface is eth1 because most people use eth0 as
  517. * their principle network connection; you can specify another
  518. * interface with -i or define environment string PLC to make
  519. * that the default interface and save typing;
  520. *
  521. *--------------------------------------------------------------------*/
  522. int main (int argc, char const * argv [])
  523. {
  524. extern struct channel channel;
  525. static char const * optv [] =
  526. {
  527. "ac:ef:i:p:qrRstvw:x",
  528. "device [device] [...] [> stdout]",
  529. "Qualcomm Atheros Thunderbolt Powerline Device Procrastinator",
  530. "a\twait for device assoc",
  531. "c n\tpolling retry count [" LITERAL (RETRY) "]",
  532. "e\tredirect stderr to stdout",
  533. "f s\tconfirm firmware is revision s",
  534. #if defined (WINPCAP) || defined (LIBPCAP)
  535. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  536. #else
  537. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  538. #endif
  539. "p n\tpoll (n) times per second [" LITERAL (POLL) "]",
  540. "q\tquiet mode",
  541. "v\tverbose mode",
  542. "r\twait for device reset",
  543. "R\treset device and wait",
  544. "s\twait for device start",
  545. "t\tshow wait times",
  546. "w n\twait up to (n) seconds",
  547. "x\texit on error",
  548. (char const *) (0)
  549. };
  550. #include "../plc/plc.c"
  551. char const * firmware = "";
  552. unsigned retry = RETRY;
  553. unsigned poll = POLL;
  554. signed c;
  555. if (getenv (PLCDEVICE))
  556. {
  557. #if defined (WINPCAP) || defined (LIBPCAP)
  558. channel.ifindex = atoi (getenv (PLCDEVICE));
  559. #else
  560. channel.ifname = strdup (getenv (PLCDEVICE));
  561. #endif
  562. }
  563. optind = 1;
  564. while (~ (c = getoptv (argc, argv, optv)))
  565. {
  566. switch (c)
  567. {
  568. case 'a':
  569. _setbits (plc.flags, PLC_WAITFORASSOC);
  570. break;
  571. case 'c':
  572. retry = (unsigned) (uintspec (optarg, 1, UINT_MAX));
  573. break;
  574. case 'e':
  575. dup2 (STDOUT_FILENO, STDERR_FILENO);
  576. break;
  577. case 'f':
  578. firmware = optarg;
  579. break;
  580. case 'i':
  581. #if defined (WINPCAP) || defined (LIBPCAP)
  582. channel.ifindex = atoi (optarg);
  583. #else
  584. channel.ifname = optarg;
  585. #endif
  586. break;
  587. case 'p':
  588. poll = (unsigned) (uintspec (optarg, 1, 50));
  589. break;
  590. case 'q':
  591. _setbits (channel.flags, CHANNEL_SILENCE);
  592. _setbits (plc.flags, PLC_SILENCE);
  593. break;
  594. case 'r':
  595. _setbits (plc.flags, PLC_WAITFORRESET);
  596. break;
  597. case 'R':
  598. _setbits (plc.flags, PLC_RESET_DEVICE);
  599. break;
  600. case 's':
  601. _setbits (plc.flags, PLC_WAITFORSTART);
  602. break;
  603. case 't':
  604. _setbits (plc.flags, PLC_ANALYSE);
  605. break;
  606. case 'v':
  607. _setbits (channel.flags, CHANNEL_VERBOSE);
  608. _setbits (plc.flags, PLC_VERBOSE);
  609. break;
  610. case 'w':
  611. plc.sleep = (unsigned) (uintspec (optarg, 0, 3600));
  612. break;
  613. case 'x':
  614. _setbits (plc.flags, PLC_BAILOUT);
  615. break;
  616. default:
  617. break;
  618. }
  619. }
  620. argc -= optind;
  621. argv += optind;
  622. plc.timer = retry / poll;
  623. openchannel (& channel);
  624. if (! (plc.message = malloc (sizeof (* plc.message))))
  625. {
  626. error (1, errno, PLC_NOMEMORY);
  627. }
  628. if (! argc)
  629. {
  630. function (& plc, firmware);
  631. }
  632. while ((argc) && (* argv))
  633. {
  634. if (! hexencode (channel.peer, sizeof (channel.peer), synonym (* argv, devices, SIZEOF (devices))))
  635. {
  636. error (1, errno, PLC_BAD_MAC, * argv);
  637. }
  638. function (& plc, firmware);
  639. argv++;
  640. argc--;
  641. }
  642. free (plc.message);
  643. closechannel (& channel);
  644. exit (0);
  645. }