ampwait.c 17 KB

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