int6klog.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * plclog.c - Qualcomm Atheros INT6x00 Log Retrieval Utility;
  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 <stdint.h>
  22. #include <stdlib.h>
  23. #include <ctype.h>
  24. /*====================================================================*
  25. * custom header files;
  26. *--------------------------------------------------------------------*/
  27. #include "../tools/getoptv.h"
  28. #include "../tools/putoptv.h"
  29. #include "../tools/memory.h"
  30. #include "../tools/number.h"
  31. #include "../tools/symbol.h"
  32. #include "../tools/format.h"
  33. #include "../tools/base64.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 "../plc/plc.h"
  40. /*====================================================================*
  41. * custom source files;
  42. *--------------------------------------------------------------------*/
  43. #ifndef MAKEFILE
  44. #include "../plc/Devices.c"
  45. #include "../plc/Failure.c"
  46. #include "../plc/SendMME.c"
  47. #include "../plc/ReadMME.c"
  48. #endif
  49. #ifndef MAKEFILE
  50. #include "../tools/error.c"
  51. #include "../tools/getoptv.c"
  52. #include "../tools/putoptv.c"
  53. #include "../tools/version.c"
  54. #include "../tools/uintspec.c"
  55. #include "../tools/hexdump.c"
  56. #include "../tools/hexencode.c"
  57. #include "../tools/hexdecode.c"
  58. #include "../tools/hexstring.c"
  59. #include "../tools/todigit.c"
  60. #include "../tools/strfbits.c"
  61. #include "../tools/synonym.c"
  62. #include "../tools/lookup.c"
  63. #include "../tools/assist.c"
  64. #include "../tools/codelist.c"
  65. #include "../tools/b64dump.c"
  66. #endif
  67. #ifndef MAKEFILE
  68. #include "../ether/openchannel.c"
  69. #include "../ether/closechannel.c"
  70. #include "../ether/readpacket.c"
  71. #include "../ether/sendpacket.c"
  72. #include "../ether/channel.c"
  73. #endif
  74. #ifndef MAKEFILE
  75. #include "../mme/MMECode.c"
  76. #include "../mme/EthernetHeader.c"
  77. #include "../mme/QualcommHeader.c"
  78. #include "../mme/UnwantedMessage.c"
  79. #endif
  80. /*====================================================================*
  81. * constants
  82. *--------------------------------------------------------------------*/
  83. #define MSTATUS_STATUS (7 << 0)
  84. #define MSTATUS_MAJORVERSION (1 << 3)
  85. #define MSTATUS_BUFFERISLOCKED (1 << 4)
  86. #define MSTATUS_AUTOLOCKONRESET (1 << 5)
  87. #define MSTATUS_UNSOLICITEDUPDATES (1 << 6)
  88. #define MSTATUS_UNSOLICITED (1 << 7)
  89. #define INT6KLOG_FMT_RAW 1
  90. #define INT6KLOG_FMT_XML 2
  91. #define WD_ACTION_READ (1 << 0)
  92. #define WD_ACTION_CLEAR (1 << 1)
  93. #define WD_ACTION_CUSTOM (1 << 2)
  94. /*====================================================================*
  95. * program variables;
  96. *--------------------------------------------------------------------*/
  97. static struct _code_ const formats [] =
  98. {
  99. {
  100. INT6KLOG_FMT_RAW,
  101. "raw",
  102. },
  103. {
  104. INT6KLOG_FMT_XML,
  105. "xml",
  106. }
  107. };
  108. /*====================================================================*
  109. *
  110. * static signed PrintRawWatchdogReport (struct plc * plc);
  111. *
  112. * Read the watchdog report using VS_WD_RPT and write to file in
  113. * binary format; the output can be sent to Atheros Support for
  114. * analysis;
  115. *
  116. * See the Atheros HomePlug AV Firmware Technical Reference Manual
  117. * for more information;
  118. *
  119. * VS_WD_RPT request returns an indication instead of a confirmation
  120. * message; this is unusual but so is life;
  121. *
  122. * Contributor(s):
  123. * Charles Maier <cmaier@qca.qualcomm.com>
  124. *
  125. *--------------------------------------------------------------------*/
  126. static signed PrintRawWatchdogReport (struct plc * plc)
  127. {
  128. struct channel * channel = (struct channel *) (plc->channel);
  129. struct message * message = (struct message *) (plc->message);
  130. #ifndef __GNUC__
  131. #pragma pack (push,1)
  132. #endif
  133. struct __packed vs_wd_rpt_request
  134. {
  135. struct ethernet_hdr ethernet;
  136. struct qualcomm_hdr qualcomm;
  137. uint16_t SESSIONID;
  138. uint8_t CLR;
  139. }
  140. * request = (struct vs_wd_rpt_request *) (message);
  141. struct __packed vs_wd_rpt_indicate
  142. {
  143. struct ethernet_hdr ethernet;
  144. struct qualcomm_hdr qualcomm;
  145. uint8_t MSTATUS;
  146. uint16_t SESSIONID;
  147. uint8_t NUMPARTS;
  148. uint8_t CURPART;
  149. uint16_t RDATALENGTH;
  150. uint8_t RDATAOFFSET;
  151. uint8_t RDATA [1];
  152. }
  153. * indicate = (struct vs_wd_rpt_indicate *) (message);
  154. #ifndef __GNUC__
  155. #pragma pack (pop)
  156. #endif
  157. memset (message, 0, sizeof (* message));
  158. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  159. QualcommHeader (& request->qualcomm, 0, (VS_WD_RPT | MMTYPE_REQ));
  160. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  161. request->SESSIONID = HTOLE16 (plc->cookie);
  162. request->CLR = plc->readaction;
  163. if (SendMME (plc) <= 0)
  164. {
  165. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  166. return (-1);
  167. }
  168. do
  169. {
  170. printf ("%d %d\n", LE16TOH (indicate->RDATALENGTH), indicate->RDATAOFFSET);
  171. if (ReadMME (plc, 0, (VS_WD_RPT | MMTYPE_IND)) <= 0)
  172. {
  173. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  174. return (-1);
  175. }
  176. printf ("%d %d\n", LE16TOH (indicate->RDATALENGTH), indicate->RDATAOFFSET);
  177. if (indicate->MSTATUS)
  178. {
  179. Failure (plc, PLC_WONTDOIT);
  180. return (-1);
  181. }
  182. printf ("%d %d\n", LE16TOH (indicate->RDATALENGTH), indicate->RDATAOFFSET);
  183. write (STDOUT_FILENO, indicate->RDATA + indicate->RDATAOFFSET, LE16TOH (indicate->RDATALENGTH) - indicate->RDATAOFFSET);
  184. }
  185. while (indicate->CURPART < indicate->NUMPARTS);
  186. return (0);
  187. }
  188. /*====================================================================*
  189. *
  190. * static signed PrintWatchdogReport (struct plc * plc, char const * version);
  191. *
  192. * Read the watchdog report using VS_WD_RPT and print to file in
  193. * XML format; this file may be sent to QCA for analysis;
  194. *
  195. * See the Atheros HomePlug AV Firmware Technical Reference Manual
  196. * for more information;
  197. *
  198. * VS_WD_RPT request returns an indication instead of a confirmation
  199. * message; this is unusual but so is life;
  200. *
  201. * Contributor(s):
  202. * Charles Maier <cmaier@qca.qualcomm.com>
  203. *
  204. *--------------------------------------------------------------------*/
  205. static signed PrintWatchdogReport (struct plc * plc, char const * version)
  206. {
  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_wd_rpt_request
  213. {
  214. struct ethernet_hdr ethernet;
  215. struct qualcomm_hdr qualcomm;
  216. uint16_t SESSIONID;
  217. uint8_t CLR;
  218. }
  219. * request = (struct vs_wd_rpt_request *) (message);
  220. struct __packed vs_wd_rpt_ind
  221. {
  222. struct ethernet_hdr ethernet;
  223. struct qualcomm_hdr qualcomm;
  224. uint8_t MSTATUS;
  225. uint16_t SESSIONID;
  226. uint8_t NUMPARTS;
  227. uint8_t CURPART;
  228. uint16_t RDATALENGTH;
  229. uint8_t RDATAOFFSET;
  230. uint8_t RDATA [1];
  231. }
  232. * indicate = (struct vs_wd_rpt_ind *) (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_WD_RPT | MMTYPE_REQ));
  239. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  240. request->SESSIONID = HTOLE16 (plc->cookie);
  241. request->CLR = plc->readaction;
  242. if (SendMME (plc) <= 0)
  243. {
  244. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  245. return (-1);
  246. }
  247. printf ("<WatchdogReport>");
  248. do
  249. {
  250. if (ReadMME (plc, 0, (VS_WD_RPT | MMTYPE_IND)) < 0)
  251. {
  252. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  253. return (-1);
  254. }
  255. if (indicate->MSTATUS)
  256. {
  257. Failure (plc, PLC_WONTDOIT);
  258. return (-1);
  259. }
  260. printf ("<Packet>");
  261. printf ("<Version>%s</Version>", version);
  262. printf ("<OUI>%s</OUI>", "00B052");
  263. printf ("<Status>0</Status>");
  264. printf ("<SessionId>0</SessionId>");
  265. printf ("<NumParts>%d</NumParts>", indicate->NUMPARTS);
  266. printf ("<CurPart>%d</CurPart>", indicate->CURPART);
  267. printf ("<DataLength>%d</DataLength>", LE16TOH (indicate->RDATALENGTH));
  268. printf ("<DataOffset>%d</DataOffset>", indicate->RDATAOFFSET);
  269. printf ("<Data>");
  270. b64dump (indicate->RDATA + indicate->RDATAOFFSET, LE16TOH (indicate->RDATALENGTH) - indicate->RDATAOFFSET, 0, stdout);
  271. printf ("</Data>");
  272. printf ("</Packet>");
  273. }
  274. while (indicate->CURPART < indicate->NUMPARTS);
  275. printf ("</WatchdogReport>");
  276. return (0);
  277. }
  278. /*====================================================================*
  279. *
  280. * static signed PrintCheckpointReport (struct plc * plc, char const * version);
  281. *
  282. * Read the watchdog reqport using VS_CP_RPT and write to file in
  283. * binary or XML format; this file can be sent to Atheros Support
  284. * for analysis;
  285. *
  286. * See the Atheros HomePlug AV Firmware Technical Reference Manual
  287. * for more information;
  288. *
  289. * VS_CP_RPT request returns an indication instead of a confirmation
  290. * message; this is unusual but so is life;
  291. *
  292. * Contributor(s):
  293. * Charles Maier <cmaier@qca.qualcomm.com>
  294. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  295. *
  296. *--------------------------------------------------------------------*/
  297. static signed PrintCheckpointReport (struct plc * plc, char const * version)
  298. {
  299. struct channel * channel = (struct channel *) (plc->channel);
  300. struct message * message = (struct message *) (plc->message);
  301. #ifndef __GNUC__
  302. #pragma pack (push,1)
  303. #endif
  304. struct __packed vs_cp_rpt_request
  305. {
  306. struct ethernet_hdr ethernet;
  307. struct qualcomm_hdr qualcomm;
  308. uint16_t SESSIONID;
  309. uint8_t CLR;
  310. }
  311. * request = (struct vs_cp_rpt_request *) (message);
  312. struct __packed vs_cp_rpt_ind
  313. {
  314. struct ethernet_hdr ethernet;
  315. struct qualcomm_hdr qualcomm;
  316. uint8_t MSTATUS;
  317. uint8_t MAJORVERSION;
  318. uint8_t MINORVERSION;
  319. uint8_t RESERVED [14];
  320. uint16_t SESSIONID;
  321. uint32_t TOTALBUFFERSIZE;
  322. uint32_t BLOCKOFFSET;
  323. uint32_t BYTEINDEXOFNEXTPOSITION;
  324. uint8_t NUMPARTS;
  325. uint8_t CURPART;
  326. uint16_t RDATALENGTH;
  327. uint8_t RDATAOFFSET;
  328. uint8_t RDATA [1];
  329. }
  330. * indicate = (struct vs_cp_rpt_ind *) (message);
  331. #ifndef __GNUC__
  332. #pragma pack (pop)
  333. #endif
  334. memset (message, 0, sizeof (* message));
  335. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  336. QualcommHeader (& request->qualcomm, 0, (VS_CP_RPT | MMTYPE_REQ));
  337. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  338. request->SESSIONID = HTOLE16 (plc->cookie);
  339. request->CLR = plc->readaction;
  340. if (SendMME (plc) <= 0)
  341. {
  342. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  343. return (-1);
  344. }
  345. printf ("<CheckpointReport>");
  346. do
  347. {
  348. if (ReadMME (plc, 0, (VS_CP_RPT | MMTYPE_IND)) <= 0)
  349. {
  350. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  351. return (-1);
  352. }
  353. if (_anyset (indicate->MSTATUS, MSTATUS_STATUS))
  354. {
  355. Failure (plc, PLC_WONTDOIT);
  356. return (-1);
  357. }
  358. printf ("<Packet>");
  359. printf ("<Version>%s</Version>", version);
  360. printf ("<Status>0</Status>");
  361. printf ("<MajorVersionBit>%d</MajorVersionBit>", indicate->MSTATUS & MSTATUS_MAJORVERSION? 1: 0);
  362. printf ("<BufferIsLocked>%d</BufferIsLocked>", indicate->MSTATUS & MSTATUS_BUFFERISLOCKED? 1: 0);
  363. printf ("<AutoLockOnResetIsOn>%d</AutoLockOnResetIsOn>", indicate->MSTATUS & MSTATUS_AUTOLOCKONRESET? 1: 0);
  364. printf ("<UnsolicitedUpdatesIsOn>%d</UnsolicitedUpdatesIsOn>", indicate->MSTATUS & MSTATUS_UNSOLICITEDUPDATES? 1: 0);
  365. printf ("<Unsolicited>%d</Unsolicited>", indicate->MSTATUS & MSTATUS_UNSOLICITED? 1: 0);
  366. printf ("<MajorVersion>%d</MajorVersion>", indicate->MAJORVERSION);
  367. printf ("<MinorVersion>%d</MinorVersion>", indicate->MINORVERSION);
  368. printf ("<Reserved1>0</Reserved1>");
  369. printf ("<Reserved2>0</Reserved2>");
  370. printf ("<Reserved3>0</Reserved3>");
  371. printf ("<Reserved4>0</Reserved4>");
  372. printf ("<SessionId>1</SessionId>");
  373. printf ("<TotalBufferSize>%d</TotalBufferSize>", LE32TOH (indicate->TOTALBUFFERSIZE));
  374. printf ("<BlockOffset>%d</BlockOffset>", LE32TOH (indicate->BLOCKOFFSET));
  375. printf ("<ByteIndexOfNextPos>%d</ByteIndexOfNextPos>", LE32TOH (indicate->BYTEINDEXOFNEXTPOSITION));
  376. printf ("<NumParts>%d</NumParts>", indicate->NUMPARTS);
  377. printf ("<CurPart>%d</CurPart>", indicate->CURPART);
  378. printf ("<RDataLength>%d</RDataLength>", LE16TOH (indicate->RDATALENGTH));
  379. printf ("<RDataOffset>%d</RDataOffset>", indicate->RDATAOFFSET);
  380. printf ("<RData>");
  381. b64dump (indicate->RDATA + indicate->RDATAOFFSET, LE16TOH (indicate->RDATALENGTH), 0, stdout);
  382. printf ("</RData>");
  383. printf ("</Packet>");
  384. }
  385. while (indicate->CURPART < indicate->NUMPARTS);
  386. printf ("</CheckpointReport>");
  387. return (0);
  388. }
  389. /*====================================================================*
  390. *
  391. * static signed Diagnostics (struct plc * plc)
  392. *
  393. * read the firmware version string from a device before reading
  394. * and writing the watchdog and checkpoint reports in XML format;
  395. *
  396. *--------------------------------------------------------------------*/
  397. static signed Diagnostics (struct plc * plc)
  398. {
  399. char version [PLC_VERSION_STRING];
  400. struct channel * channel = (struct channel *) (plc->channel);
  401. struct message * message = (struct message *) (plc->message);
  402. #ifndef __GNUC__
  403. #pragma pack (push,1)
  404. #endif
  405. struct __packed vs_sw_ver_request
  406. {
  407. struct ethernet_hdr ethernet;
  408. struct qualcomm_hdr qualcomm;
  409. uint8_t MSTATUS;
  410. uint8_t MDEVICEID;
  411. uint8_t MVERLENGTH;
  412. char MVERSION [PLC_VERSION_STRING];
  413. }
  414. * request = (struct vs_sw_ver_request *) (message);
  415. struct __packed vs_sw_ver_confirm
  416. {
  417. struct ethernet_hdr ethernet;
  418. struct qualcomm_hdr qualcomm;
  419. uint8_t MSTATUS;
  420. uint8_t MDEVICEID;
  421. uint8_t MVERLENGTH;
  422. char MVERSION [PLC_VERSION_STRING];
  423. }
  424. * confirm = (struct vs_sw_ver_confirm *) (message);
  425. #ifndef __GNUC__
  426. #pragma pack (pop)
  427. #endif
  428. memset (message, 0, sizeof (* message));
  429. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  430. QualcommHeader (& request->qualcomm, 0, (VS_SW_VER | MMTYPE_REQ));
  431. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  432. if (SendMME (plc) <= 0)
  433. {
  434. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  435. return (-1);
  436. }
  437. if (ReadMME (plc, 0, (VS_SW_VER | MMTYPE_CNF)) <= 0)
  438. {
  439. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  440. return (-1);
  441. }
  442. if (confirm->MSTATUS)
  443. {
  444. Failure (plc, PLC_WONTDOIT);
  445. return (-1);
  446. }
  447. memcpy (version, confirm->MVERSION, sizeof (version));
  448. if (plc->action == INT6KLOG_FMT_XML)
  449. {
  450. printf ("<?xml version='1.0' encoding='utf-8' standalone='yes'?>");
  451. printf ("<Diagnostics>");
  452. PrintWatchdogReport (plc, version);
  453. PrintCheckpointReport (plc, version);
  454. printf ("</Diagnostics>\n");
  455. return (0);
  456. }
  457. if (plc->action == INT6KLOG_FMT_RAW)
  458. {
  459. PrintRawWatchdogReport (plc);
  460. return (0);
  461. }
  462. return (0);
  463. }
  464. /*====================================================================*
  465. *
  466. * int main (int argc, char const * argv[]);
  467. *
  468. * parse command line, populate plc structure and perform selected
  469. * operations; show help summary if asked; see getoptv and putoptv
  470. * to understand command line parsing and help summary display; see
  471. * plc.h for the definition of struct plc;
  472. *
  473. * the command line accepts multiple MAC addresses and the program
  474. * performs the specified operations on each address, in turn; the
  475. * address order is significant but the option order is not; the
  476. * default address is a local broadcast that causes all devices on
  477. * the local H1 interface to respond but not those at the remote
  478. * end of the powerline;
  479. *
  480. * the default address is 00:B0:52:00:00:01; omitting the address
  481. * will automatically address the local device; some options will
  482. * cancel themselves if this makes no sense;
  483. *
  484. * the default interface is eth1 because most people use eth0 as
  485. * their principle network connection; you can specify another
  486. * interface with -i or define environment string PLC to make
  487. * that the default interface and save typing;
  488. *
  489. *--------------------------------------------------------------------*/
  490. int main (int argc, char const * argv [])
  491. {
  492. extern struct channel channel;
  493. static char const * optv [] =
  494. {
  495. "cCef:ri:qvz:",
  496. "device [device] [...] [> stdout]",
  497. "Qualcomm Atheros INT6x00 Log Retrieval Utility",
  498. "c\tclear watchdog and checkpoint reports",
  499. "C\tcustom watchdog report",
  500. "e\tredirect stderr to stdout",
  501. "f s\tprint Watchdog Report in format (s) [xml]",
  502. #if defined (WINPCAP) || defined (LIBPCAP)
  503. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  504. #else
  505. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  506. #endif
  507. "q\tquiet mode",
  508. "r\tread ",
  509. "v\tverbose mode",
  510. (char const *) (0)
  511. };
  512. #include "../plc/plc.c"
  513. signed c;
  514. if (getenv (PLCDEVICE))
  515. {
  516. #if defined (WINPCAP) || defined (LIBPCAP)
  517. channel.ifindex = atoi (getenv (PLCDEVICE));
  518. #else
  519. channel.ifname = strdup (getenv (PLCDEVICE));
  520. #endif
  521. }
  522. optind = 1;
  523. plc.readaction = 0;
  524. plc.action = INT6KLOG_FMT_XML;
  525. while (~ (c = getoptv (argc, argv, optv)))
  526. {
  527. switch (c)
  528. {
  529. case 'C':
  530. _setbits (plc.readaction, WD_ACTION_CUSTOM);
  531. break;
  532. case 'c':
  533. _setbits (plc.readaction, WD_ACTION_CLEAR);
  534. break;
  535. case 'e':
  536. dup2 (STDOUT_FILENO, STDERR_FILENO);
  537. break;
  538. case 'f':
  539. if ((c = lookup (optarg, formats, SIZEOF (formats))) == -1)
  540. {
  541. assist (optarg, "format", formats, SIZEOF (formats));
  542. }
  543. plc.action = (uint8_t) (c);
  544. break;
  545. case 'i':
  546. #if defined (WINPCAP) || defined (LIBPCAP)
  547. channel.ifindex = atoi (optarg);
  548. #else
  549. channel.ifname = optarg;
  550. #endif
  551. break;
  552. case 'q':
  553. _setbits (channel.flags, CHANNEL_SILENCE);
  554. _setbits (plc.flags, PLC_SILENCE);
  555. break;
  556. case 'r':
  557. _setbits (plc.readaction, WD_ACTION_READ);
  558. break;
  559. case 'v':
  560. _setbits (channel.flags, CHANNEL_VERBOSE);
  561. _setbits (plc.flags, PLC_VERBOSE);
  562. break;
  563. case 'z':
  564. plc.readaction = (uint8_t) (uintspec (optarg, 0, 255));
  565. break;
  566. default:
  567. break;
  568. }
  569. }
  570. argc -= optind;
  571. argv += optind;
  572. if (argc != 1)
  573. {
  574. if (plc.rpt.file != -1)
  575. {
  576. error (1, ECANCELED, PLC_NODEVICE);
  577. }
  578. }
  579. openchannel (& channel);
  580. if (! (plc.message = malloc (sizeof (* plc.message))))
  581. {
  582. error (1, errno, PLC_NOMEMORY);
  583. }
  584. #ifdef WIN32
  585. if (plc.action == INT6KLOG_FMT_RAW)
  586. {
  587. setmode (STDOUT_FILENO, O_BINARY);
  588. }
  589. #endif
  590. if (! argc)
  591. {
  592. Diagnostics (& plc);
  593. }
  594. while ((argc) && (* argv))
  595. {
  596. if (! hexencode (channel.peer, sizeof (channel.peer), synonym (* argv, devices, SIZEOF (devices))))
  597. {
  598. error (1, errno, PLC_BAD_MAC, * argv);
  599. }
  600. Diagnostics (& plc);
  601. argc--;
  602. argv++;
  603. }
  604. free (plc.message);
  605. closechannel (& channel);
  606. exit (0);
  607. }