plclog.c 20 KB

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