plcfwd.c 25 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * plcfwd.c - Qualcomm Atheros PLC Forward Configuration Manager;
  11. *
  12. *--------------------------------------------------------------------*/
  13. /*====================================================================*
  14. * system header files;
  15. *--------------------------------------------------------------------*/
  16. #include <unistd.h>
  17. #include <stdlib.h>
  18. #include <stdint.h>
  19. #include <limits.h>
  20. #include <ctype.h>
  21. /*====================================================================*
  22. * custom header files;
  23. *--------------------------------------------------------------------*/
  24. #include "../tools/getoptv.h"
  25. #include "../tools/putoptv.h"
  26. #include "../tools/memory.h"
  27. #include "../tools/number.h"
  28. #include "../tools/symbol.h"
  29. #include "../tools/types.h"
  30. #include "../tools/flags.h"
  31. #include "../tools/files.h"
  32. #include "../tools/error.h"
  33. #include "../plc/plc.h"
  34. /*====================================================================*
  35. * custom source files;
  36. *--------------------------------------------------------------------*/
  37. #ifndef MAKEFILE
  38. #include "../plc/Confirm.c"
  39. #include "../plc/Display.c"
  40. #include "../plc/Failure.c"
  41. #include "../plc/Request.c"
  42. #include "../plc/ReadMME.c"
  43. #include "../plc/SendMME.c"
  44. #include "../plc/Devices.c"
  45. #endif
  46. #ifndef MAKEFILE
  47. #include "../tools/getoptv.c"
  48. #include "../tools/putoptv.c"
  49. #include "../tools/version.c"
  50. #include "../tools/uintspec.c"
  51. #include "../tools/basespec.c"
  52. #include "../tools/hexdump.c"
  53. #include "../tools/hexview.c"
  54. #include "../tools/hexencode.c"
  55. #include "../tools/hexdecode.c"
  56. #include "../tools/hexout.c"
  57. #include "../tools/todigit.c"
  58. #include "../tools/synonym.c"
  59. #include "../tools/binout.c"
  60. #include "../tools/error.c"
  61. #endif
  62. #ifndef MAKEFILE
  63. #include "../ether/openchannel.c"
  64. #include "../ether/closechannel.c"
  65. #include "../ether/readpacket.c"
  66. #include "../ether/sendpacket.c"
  67. #include "../ether/channel.c"
  68. #endif
  69. #ifndef MAKEFILE
  70. #include "../mme/MMECode.c"
  71. #include "../mme/EthernetHeader.c"
  72. #include "../mme/QualcommHeader.c"
  73. #include "../mme/UnwantedMessage.c"
  74. #endif
  75. /*====================================================================*
  76. * program constants;
  77. *--------------------------------------------------------------------*/
  78. #define PLCFWD_VERBOSE (1 << 0)
  79. #define PLCFWD_SILENCE (1 << 1)
  80. #define PLCFWD_LENGTH 0
  81. #define PLCFWD_OFFSET 0
  82. #define PLCFWD_GET 0
  83. #define PLCFWD_ADD 1
  84. #define PLCFWD_REM 2
  85. #define PLCFWD_STO 3
  86. #define PLCFWD_CTL 4
  87. #define PLCFWD_SET 5
  88. #define PLCFWD_FWD 6
  89. #define PLCFWD_VER 0
  90. /*====================================================================*
  91. * program variables;
  92. *--------------------------------------------------------------------*/
  93. /*
  94. * this structure is only used in the VS_FORWARD_CONFIG message but
  95. * it is common to several variations of the message and is used in
  96. * arrays;
  97. */
  98. #ifndef __GNUC__
  99. #pragma pack (push,1)
  100. #endif
  101. typedef struct item
  102. {
  103. uint8_t MAC_ADDR [ETHER_ADDR_LEN];
  104. uint16_t NUM_VLANIDS;
  105. uint16_t VLANID [10];
  106. }
  107. item;
  108. #ifndef __GNUC__
  109. #pragma pack (pop)
  110. #endif
  111. /*
  112. * synonym table for options -M and -S;
  113. */
  114. #define STATES (sizeof (states) / sizeof (struct _term_))
  115. static const struct _term_ states [] =
  116. {
  117. {
  118. "disable",
  119. "0"
  120. },
  121. {
  122. "enable",
  123. "1"
  124. },
  125. {
  126. "off",
  127. "0"
  128. },
  129. {
  130. "on",
  131. "1"
  132. }
  133. };
  134. /*====================================================================*
  135. *
  136. * void readitem (struct item * item, char const * string);
  137. *
  138. * encode a slave structure with infomation specified by a string
  139. * specification has the following production:
  140. *
  141. * <spec> := <mac_addr>
  142. * <spec> := <spec>,<vlan_id>
  143. *
  144. * basically, encode slave->MAC_ADDR then encode slave->VLANID[]
  145. * with hexadecimal VLANID values; we allow 10 VLANID values but
  146. * only 8 are legal;
  147. *
  148. * the idea is to read multiple input strings and call this function
  149. * to initialize one or more slave structures; it is possible to fit
  150. * up to 128 slave structures in one message frame;
  151. *
  152. *--------------------------------------------------------------------*/
  153. static void readitem (struct item * item, char const * string)
  154. {
  155. register uint8_t * origin = (uint8_t *) (item->MAC_ADDR);
  156. register uint8_t * offset = (uint8_t *) (item->MAC_ADDR);
  157. size_t extent = sizeof (item->MAC_ADDR);
  158. memset (item, 0, sizeof (* item));
  159. while ((extent) && (* string))
  160. {
  161. unsigned radix = RADIX_HEX;
  162. unsigned field = sizeof (uint8_t) + sizeof (uint8_t);
  163. unsigned value = 0;
  164. unsigned digit = 0;
  165. if ((offset != origin) && (* string == HEX_EXTENDER))
  166. {
  167. string++;
  168. }
  169. while (field--)
  170. {
  171. if ((digit = todigit (* string)) < radix)
  172. {
  173. value *= radix;
  174. value += digit;
  175. string++;
  176. continue;
  177. }
  178. error (1, EINVAL, "bad MAC address: ...[%s] (1)", string);
  179. }
  180. * offset = value;
  181. offset++;
  182. extent--;
  183. }
  184. if (extent)
  185. {
  186. error (1, EINVAL, "bad MAC address: ...[%s] (2)", string);
  187. }
  188. while (isspace ((unsigned char)* string))
  189. {
  190. string++;
  191. }
  192. if ((* string) && (* string != ','))
  193. {
  194. error (1, EINVAL, "bad MAC address: ...[%s] (3)", string);
  195. }
  196. while (* string == ',')
  197. {
  198. unsigned radix = RADIX_DEC;
  199. unsigned digit = 0;
  200. unsigned value = 0;
  201. do
  202. {
  203. string++;
  204. }
  205. while (isspace ((unsigned char)* string));
  206. while ((digit = todigit (* string)) < radix)
  207. {
  208. value *= radix;
  209. value += digit;
  210. string++;
  211. }
  212. while (isspace ((unsigned char)* string))
  213. {
  214. string++;
  215. }
  216. if (item->NUM_VLANIDS < (sizeof (item->VLANID) / sizeof (uint16_t)))
  217. {
  218. item->VLANID [item->NUM_VLANIDS++] = value;
  219. }
  220. }
  221. while (isspace ((unsigned char)* string))
  222. {
  223. string++;
  224. }
  225. if (* string)
  226. {
  227. error (1, EINVAL, "bad VLAN ID: ...[%s]", string);
  228. }
  229. return;
  230. }
  231. /*====================================================================*
  232. *
  233. * unsigned readlist (struct item list [], unsigned size);
  234. *
  235. * read one or more items from stdin; discard comments; assume one
  236. * item per line; permit multiple items on one line when separated
  237. * by semicolon; items cannot straddle lines; readitem () controls
  238. * what consitutes one item;
  239. *
  240. *--------------------------------------------------------------------*/
  241. static unsigned readlist (struct item list [], unsigned size)
  242. {
  243. struct item * item = list;
  244. char string [1024];
  245. char * sp = string;
  246. signed c;
  247. for (c = getc (stdin); c != EOF; c = getc (stdin))
  248. {
  249. if (isspace (c))
  250. {
  251. continue;
  252. }
  253. if (c == '#')
  254. {
  255. while ((c != '\n') && (c != EOF))
  256. {
  257. c = getc (stdin);
  258. }
  259. continue;
  260. }
  261. sp = string;
  262. while ((c != ';') && (c != '\n') && (c != EOF))
  263. {
  264. * sp++ = (char) (c);
  265. c = getc (stdin);
  266. }
  267. * sp = (char) (0);
  268. if (size)
  269. {
  270. readitem (item++, string);
  271. size--;
  272. }
  273. }
  274. return ((unsigned) (item - list));
  275. }
  276. /*====================================================================*
  277. *
  278. * void showlist (struct item list [], unsigned items)
  279. *
  280. * print item list on stdout in a format suitable for input using
  281. * readlist (); this function may be commented out if it not used;
  282. *
  283. *--------------------------------------------------------------------*/
  284. #if 0
  285. static void showlist (struct item list [], unsigned items)
  286. {
  287. while (items--)
  288. {
  289. uint16_t fields = list->NUM_VLANIDS;
  290. uint16_t * field = list->VLANID;
  291. hexout (list->MAC_ADDR, sizeof (list->MAC_ADDR), 0, 0, stdout);
  292. while (fields--)
  293. {
  294. printf (", %d", * field);
  295. field++;
  296. }
  297. printf ("\n");
  298. list++;
  299. }
  300. return;
  301. }
  302. #endif
  303. /*====================================================================*
  304. *
  305. * signed ReadVLANIDs (struct plc * plc, uint32_t offset, uint32_t length);
  306. *
  307. *--------------------------------------------------------------------*/
  308. static signed ReadVLANIDs (struct plc * plc, uint32_t offset, uint32_t length)
  309. {
  310. struct channel * channel = (struct channel *) (plc->channel);
  311. struct message * message = (struct message *) (plc->message);
  312. #ifndef __GNUC__
  313. #pragma pack (push,1)
  314. #endif
  315. struct __packed vs_forward_config_request
  316. {
  317. struct ethernet_hdr ethernet;
  318. struct qualcomm_hdr qualcomm;
  319. uint8_t RESERVED1;
  320. uint8_t MREQUEST;
  321. uint8_t MVERSION;
  322. uint32_t RESERVED2;
  323. uint32_t DATA_LENGTH;
  324. uint32_t DATA_OFFSET;
  325. uint16_t RESERVED3;
  326. }
  327. * request = (struct vs_forward_config_request *) (message);
  328. struct __packed vs_forward_config_confirm
  329. {
  330. struct ethernet_hdr ethernet;
  331. struct qualcomm_hdr qualcomm;
  332. uint8_t RESERVED1;
  333. uint8_t RESULTCODE;
  334. uint8_t OPERATION;
  335. uint8_t MVERSION;
  336. uint32_t RESERVED2;
  337. uint32_t DATA_LENGTH;
  338. uint32_t DATA_OFFSET;
  339. uint8_t DATA [PLC_MODULE_SIZE];
  340. }
  341. * confirm = (struct vs_forward_config_confirm *) (message);
  342. #ifndef __GNUC__
  343. #pragma pack (pop)
  344. #endif
  345. memset (message, 0, sizeof (* message));
  346. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  347. QualcommHeader (& request->qualcomm, 0, (VS_FORWARD_CONFIG | MMTYPE_REQ));
  348. request->MREQUEST = PLCFWD_GET;
  349. request->MVERSION = PLCFWD_VER;
  350. request->DATA_OFFSET = HTOLE32 (offset);
  351. request->DATA_LENGTH = HTOLE32 (length);
  352. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  353. if (SendMME (plc) <= 0)
  354. {
  355. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  356. return (-1);
  357. }
  358. while (ReadMME (plc, 0, (VS_FORWARD_CONFIG | MMTYPE_CNF)) > 0)
  359. {
  360. if (confirm->RESULTCODE)
  361. {
  362. Failure (plc, PLC_WONTDOIT);
  363. continue;
  364. }
  365. hexview (confirm->DATA, LE32TOH (confirm->DATA_OFFSET), LE32TOH (confirm->DATA_LENGTH), stdout);
  366. }
  367. return (0);
  368. }
  369. /*====================================================================*
  370. *
  371. * signed AddVLANIDs (struct plc * plc, struct item list [], unsigned items);
  372. *
  373. *--------------------------------------------------------------------*/
  374. static signed AddVLANIDs (struct plc * plc, struct item list [], unsigned items)
  375. {
  376. struct channel * channel = (struct channel *) (plc->channel);
  377. struct message * message = (struct message *) (plc->message);
  378. #ifndef __GNUC__
  379. #pragma pack (push,1)
  380. #endif
  381. struct __packed vs_forward_config_request
  382. {
  383. struct ethernet_hdr ethernet;
  384. struct qualcomm_hdr qualcomm;
  385. uint8_t RESERVED1;
  386. uint8_t MREQUEST;
  387. uint8_t MVERSION;
  388. uint32_t RESERVED2;
  389. uint16_t ITEMS;
  390. struct item LIST [1];
  391. }
  392. * request = (struct vs_forward_config_request *) (message);
  393. struct __packed vs_forward_config_confirm
  394. {
  395. struct ethernet_hdr ethernet;
  396. struct qualcomm_hdr qualcomm;
  397. uint8_t RESERVED1;
  398. uint8_t RESULTCODE;
  399. uint8_t OPERATION;
  400. uint8_t MVERSION;
  401. uint32_t RESERVED2;
  402. }
  403. * confirm = (struct vs_forward_config_confirm *) (message);
  404. #ifndef __GNUC__
  405. #pragma pack (pop)
  406. #endif
  407. struct item * item = request->LIST;
  408. memset (message, 0, sizeof (* message));
  409. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  410. QualcommHeader (& request->qualcomm, 0, (VS_FORWARD_CONFIG | MMTYPE_REQ));
  411. request->MREQUEST = PLCFWD_ADD;
  412. request->MVERSION = PLCFWD_VER;
  413. request->ITEMS = HTOLE16 (items);
  414. while (items--)
  415. {
  416. unsigned count;
  417. memcpy (item->MAC_ADDR, list->MAC_ADDR, sizeof (item->MAC_ADDR));
  418. item->NUM_VLANIDS = HTOLE16 (list->NUM_VLANIDS);
  419. for (count = 0; count < list->NUM_VLANIDS; count++)
  420. {
  421. item->VLANID [count] = HTOLE16 (list->VLANID [count]);
  422. }
  423. // item++;
  424. item = (struct item *) (& item->VLANID [count]);
  425. list++;
  426. }
  427. plc->packetsize = (signed) ((uint8_t *) (item) - (uint8_t *) (request));
  428. if (SendMME (plc) <= 0)
  429. {
  430. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  431. return (-1);
  432. }
  433. while (ReadMME (plc, 0, (VS_FORWARD_CONFIG | MMTYPE_CNF)) > 0)
  434. {
  435. if (confirm->RESULTCODE)
  436. {
  437. Failure (plc, PLC_WONTDOIT);
  438. continue;
  439. }
  440. }
  441. return (0);
  442. }
  443. /*====================================================================*
  444. *
  445. * signed RemoveVLANIDs (struct plc * plc, struct item list [], unsigned items);
  446. *
  447. *--------------------------------------------------------------------*/
  448. static signed RemoveVLANIDs (struct plc * plc, struct item list [], unsigned items)
  449. {
  450. struct channel * channel = (struct channel *) (plc->channel);
  451. struct message * message = (struct message *) (plc->message);
  452. #ifndef __GNUC__
  453. #pragma pack (push,1)
  454. #endif
  455. struct __packed vs_forward_config_request
  456. {
  457. struct ethernet_hdr ethernet;
  458. struct qualcomm_hdr qualcomm;
  459. uint8_t RESERVED1;
  460. uint8_t MREQUEST;
  461. uint8_t MVERSION;
  462. uint32_t RESERVED2;
  463. uint16_t ITEMS;
  464. struct item LIST [1];
  465. }
  466. * request = (struct vs_forward_config_request *) (message);
  467. struct __packed vs_forward_config_confirm
  468. {
  469. struct ethernet_hdr ethernet;
  470. struct qualcomm_hdr qualcomm;
  471. uint8_t RESERVED1;
  472. uint8_t RESULTCODE;
  473. uint8_t OPERATION;
  474. uint8_t MVERSION;
  475. uint32_t RESERVED2;
  476. }
  477. * confirm = (struct vs_forward_config_confirm *) (message);
  478. #ifndef __GNUC__
  479. #pragma pack (pop)
  480. #endif
  481. struct item * item = request->LIST;
  482. memset (message, 0, sizeof (* message));
  483. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  484. QualcommHeader (& request->qualcomm, 0, (VS_FORWARD_CONFIG | MMTYPE_REQ));
  485. request->MREQUEST = PLCFWD_REM;
  486. request->MVERSION = PLCFWD_VER;
  487. request->ITEMS = HTOLE16 (items);
  488. while (items--)
  489. {
  490. unsigned count;
  491. memcpy (item->MAC_ADDR, list->MAC_ADDR, sizeof (item->MAC_ADDR));
  492. item->NUM_VLANIDS = HTOLE16 (list->NUM_VLANIDS);
  493. for (count = 0; count < list->NUM_VLANIDS; count++)
  494. {
  495. item->VLANID [count] = HTOLE16 (list->VLANID [count]);
  496. }
  497. // item++;
  498. item = (struct item *) (& item->VLANID [count]);
  499. list++;
  500. }
  501. plc->packetsize = (signed) ((uint8_t *) (item) - (uint8_t *) (request));
  502. if (SendMME (plc) <= 0)
  503. {
  504. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  505. return (-1);
  506. }
  507. while (ReadMME (plc, 0, (VS_FORWARD_CONFIG | MMTYPE_CNF)) > 0)
  508. {
  509. if (confirm->RESULTCODE)
  510. {
  511. Failure (plc, PLC_WONTDOIT);
  512. continue;
  513. }
  514. }
  515. return (0);
  516. }
  517. /*====================================================================*
  518. *
  519. * signed CommitVLANIDs (struct plc * plc);
  520. *
  521. *--------------------------------------------------------------------*/
  522. static signed CommitVLANIDs (struct plc * plc)
  523. {
  524. struct channel * channel = (struct channel *) (plc->channel);
  525. struct message * message = (struct message *) (plc->message);
  526. #ifndef __GNUC__
  527. #pragma pack (push,1)
  528. #endif
  529. struct __packed vs_forward_config_request
  530. {
  531. struct ethernet_hdr ethernet;
  532. struct qualcomm_hdr qualcomm;
  533. uint8_t RESERVED1;
  534. uint8_t MREQUEST;
  535. uint8_t MVERSION;
  536. uint32_t RESERVED2;
  537. }
  538. * request = (struct vs_forward_config_request *) (message);
  539. struct __packed vs_forward_config_confirm
  540. {
  541. struct ethernet_hdr ethernet;
  542. struct qualcomm_hdr qualcomm;
  543. uint8_t RESERVED1;
  544. uint8_t RESULTCODE;
  545. uint8_t OPERATION;
  546. uint8_t MVERSION;
  547. uint32_t RESERVED2;
  548. }
  549. * confirm = (struct vs_forward_config_confirm *) (message);
  550. #ifndef __GNUC__
  551. #pragma pack (pop)
  552. #endif
  553. memset (message, 0, sizeof (* message));
  554. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  555. QualcommHeader (& request->qualcomm, 0, (VS_FORWARD_CONFIG | MMTYPE_REQ));
  556. request->MREQUEST = PLCFWD_STO;
  557. request->MVERSION = PLCFWD_VER;
  558. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  559. if (SendMME (plc) <= 0)
  560. {
  561. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  562. return (-1);
  563. }
  564. while (ReadMME (plc, 0, (VS_FORWARD_CONFIG | MMTYPE_CNF)) > 0)
  565. {
  566. if (confirm->RESULTCODE)
  567. {
  568. Failure (plc, PLC_WONTDOIT);
  569. continue;
  570. }
  571. }
  572. return (0);
  573. }
  574. /*====================================================================*
  575. *
  576. * signed ControlVLANIDs (struct plc * plc);
  577. *
  578. *--------------------------------------------------------------------*/
  579. static signed ControlVLANIDs (struct plc * plc)
  580. {
  581. struct channel * channel = (struct channel *) (plc->channel);
  582. struct message * message = (struct message *) (plc->message);
  583. #ifndef __GNUC__
  584. #pragma pack (push,1)
  585. #endif
  586. struct __packed vs_forward_config_request
  587. {
  588. struct ethernet_hdr ethernet;
  589. struct qualcomm_hdr qualcomm;
  590. uint8_t RESERVED1;
  591. uint8_t MREQUEST;
  592. uint8_t MVERSION;
  593. uint32_t RESERVED2;
  594. uint8_t ENABLE;
  595. uint8_t UPSTREAMCHECK;
  596. uint8_t RESERVED3;
  597. }
  598. * request = (struct vs_forward_config_request *) (message);
  599. struct __packed vs_forward_config_confirm
  600. {
  601. struct ethernet_hdr ethernet;
  602. struct qualcomm_hdr qualcomm;
  603. uint8_t RESERVED1;
  604. uint8_t RESULTCODE;
  605. uint8_t OPERATION;
  606. uint8_t MVERSION;
  607. uint32_t RESERVED2;
  608. }
  609. * confirm = (struct vs_forward_config_confirm *) (message);
  610. #ifndef __GNUC__
  611. #pragma pack (pop)
  612. #endif
  613. memset (message, 0, sizeof (* message));
  614. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  615. QualcommHeader (& request->qualcomm, 0, (VS_FORWARD_CONFIG | MMTYPE_REQ));
  616. request->MREQUEST = PLCFWD_CTL;
  617. request->MVERSION = PLCFWD_VER;
  618. request->ENABLE = plc->module;
  619. request->UPSTREAMCHECK = plc->pushbutton;
  620. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  621. if (SendMME (plc) <= 0)
  622. {
  623. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  624. return (-1);
  625. }
  626. while (ReadMME (plc, 0, (VS_FORWARD_CONFIG | MMTYPE_CNF)) > 0)
  627. {
  628. if (confirm->RESULTCODE)
  629. {
  630. Failure (plc, PLC_WONTDOIT);
  631. continue;
  632. }
  633. }
  634. return (0);
  635. }
  636. /*====================================================================*
  637. *
  638. * signed DefaultVLANIDs (struct plc * plc);
  639. *
  640. *--------------------------------------------------------------------*/
  641. static signed DefaultVLANIDs (struct plc * plc, struct item list [], unsigned items)
  642. {
  643. struct channel * channel = (struct channel *) (plc->channel);
  644. struct message * message = (struct message *) (plc->message);
  645. #ifndef __GNUC__
  646. #pragma pack (push,1)
  647. #endif
  648. struct __packed vs_forward_config_request
  649. {
  650. struct ethernet_hdr ethernet;
  651. struct qualcomm_hdr qualcomm;
  652. uint8_t RESERVED1;
  653. uint8_t MREQUEST;
  654. uint8_t MVERSION;
  655. uint32_t RESERVED2;
  656. uint16_t VLANID;
  657. uint16_t RESERVED3;
  658. }
  659. * request = (struct vs_forward_config_request *) (message);
  660. struct __packed vs_forward_config_confirm
  661. {
  662. struct ethernet_hdr ethernet;
  663. struct qualcomm_hdr qualcomm;
  664. uint8_t RESERVED1;
  665. uint8_t RESULTCODE;
  666. uint8_t OPERATION;
  667. uint8_t MVERSION;
  668. uint32_t RESERVED2;
  669. }
  670. * confirm = (struct vs_forward_config_confirm *) (message);
  671. #ifndef __GNUC__
  672. #pragma pack (pop)
  673. #endif
  674. memset (message, 0, sizeof (* message));
  675. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  676. QualcommHeader (& request->qualcomm, 0, (VS_FORWARD_CONFIG | MMTYPE_REQ));
  677. request->MREQUEST = PLCFWD_SET;
  678. request->MVERSION = PLCFWD_VER;
  679. request->VLANID = HTOLE16 (list [0].VLANID [0]);
  680. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  681. if (SendMME (plc) <= 0)
  682. {
  683. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  684. return (-1);
  685. }
  686. while (ReadMME (plc, 0, (VS_FORWARD_CONFIG | MMTYPE_CNF)) > 0)
  687. {
  688. if (confirm->RESULTCODE)
  689. {
  690. Failure (plc, PLC_WONTDOIT);
  691. continue;
  692. }
  693. }
  694. return (0);
  695. }
  696. /*====================================================================*
  697. *
  698. * signed ForwardVLANIDs (struct plc * plc);
  699. *
  700. *--------------------------------------------------------------------*/
  701. static signed ForwardVLANIDs (struct plc * plc)
  702. {
  703. struct channel * channel = (struct channel *) (plc->channel);
  704. struct message * message = (struct message *) (plc->message);
  705. #ifndef __GNUC__
  706. #pragma pack (push,1)
  707. #endif
  708. struct __packed vs_forward_config_request
  709. {
  710. struct ethernet_hdr ethernet;
  711. struct qualcomm_hdr qualcomm;
  712. uint8_t RESERVED1;
  713. uint8_t MREQUEST;
  714. uint8_t MVERSION;
  715. uint32_t RESERVED2;
  716. uint8_t ENABLED;
  717. uint16_t VLANID;
  718. uint16_t RESERVED3;
  719. struct item ITEM;
  720. }
  721. * request = (struct vs_forward_config_request *) (message);
  722. struct __packed vs_forward_config_confirm
  723. {
  724. struct ethernet_hdr ethernet;
  725. struct qualcomm_hdr qualcomm;
  726. uint8_t RESERVED1;
  727. uint8_t RESULTCODE;
  728. uint8_t OPERATION;
  729. uint8_t MVERSION;
  730. uint32_t RESERVED2;
  731. }
  732. * confirm = (struct vs_forward_config_confirm *) (message);
  733. #ifndef __GNUC__
  734. #pragma pack (pop)
  735. #endif
  736. memset (message, 0, sizeof (* message));
  737. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  738. QualcommHeader (& request->qualcomm, 0, (VS_FORWARD_CONFIG | MMTYPE_REQ));
  739. request->MREQUEST = PLCFWD_FWD;
  740. request->MVERSION = PLCFWD_VER;
  741. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  742. if (SendMME (plc) <= 0)
  743. {
  744. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  745. return (-1);
  746. }
  747. while (ReadMME (plc, 0, (VS_FORWARD_CONFIG | MMTYPE_CNF)) > 0)
  748. {
  749. if (confirm->RESULTCODE)
  750. {
  751. Failure (plc, PLC_WONTDOIT);
  752. continue;
  753. }
  754. }
  755. return (0);
  756. }
  757. /*====================================================================*
  758. *
  759. * void function (struct plc * plc, struct item list [], unsigned items);
  760. *
  761. * perform the VLANID action specified by the action member
  762. * in struct plc as set in the main program; only one action
  763. * is performed;
  764. *
  765. *--------------------------------------------------------------------*/
  766. static void function (struct plc * plc, uint32_t offset, uint32_t length, struct item * list, unsigned items)
  767. {
  768. if (plc->action == PLCFWD_GET)
  769. {
  770. ReadVLANIDs (plc, offset, length);
  771. return;
  772. }
  773. if (plc->action == PLCFWD_ADD)
  774. {
  775. AddVLANIDs (plc, list, items);
  776. return;
  777. }
  778. if (plc->action == PLCFWD_REM)
  779. {
  780. RemoveVLANIDs (plc, list, items);
  781. return;
  782. }
  783. if (plc->action == PLCFWD_STO)
  784. {
  785. CommitVLANIDs (plc);
  786. return;
  787. }
  788. if (plc->action == PLCFWD_CTL)
  789. {
  790. ControlVLANIDs (plc);
  791. return;
  792. }
  793. if (plc->action == PLCFWD_SET)
  794. {
  795. DefaultVLANIDs (plc, list, items);
  796. return;
  797. }
  798. if (plc->action == PLCFWD_FWD)
  799. {
  800. ForwardVLANIDs (plc);
  801. return;
  802. }
  803. return;
  804. }
  805. /*====================================================================*
  806. *
  807. * int main (int argc, char const * argv[]);
  808. *
  809. *--------------------------------------------------------------------*/
  810. int main (int argc, char const * argv [])
  811. {
  812. extern struct channel channel;
  813. static char const * optv [] =
  814. {
  815. "ACD:ef:i:l:M:o:qRS:t:vxz:",
  816. "device [device] [...] [> stdout]",
  817. "Qualcomm Atheros VLANID Forward Configuration Manager",
  818. "A\tadd VLAN ID of multiple slaves to memory",
  819. "C\tcommit configuration to flash memory",
  820. "D x\tset default VLAN ID",
  821. "e\tredirect stderr to stdout",
  822. "f s\tread VLANIDS from file (s)",
  823. #if defined (WINPCAP) || defined (LIBPCAP)
  824. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  825. #else
  826. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  827. #endif
  828. "l n\tdata length in bytes [" LITERAL (PLCFWD_LENGTH) "]",
  829. "M n\tenable VLANID forwarding on the master",
  830. "o x\tdata offset in bytes [" LITERAL (PLCFWD_OFFSET) "]",
  831. "q\tquiet mode",
  832. "R\tremove VLAN ID of multiple slaves from memory",
  833. "S n\tenable VLANID forwarding on all slaves",
  834. "t n\ttimeout is (n) millisecond [" LITERAL (CHANNEL_TIMEOUT) "]",
  835. "v\tverbose mode",
  836. "x\texit on error",
  837. "z s\tslavespec",
  838. (char const *) (0)
  839. };
  840. #include "../plc/plc.c"
  841. struct item list [128];
  842. unsigned size = sizeof (list) / sizeof (struct item);
  843. unsigned items = 0;
  844. uint32_t offset = 0;
  845. uint32_t length = 0;
  846. signed c;
  847. memset (& list, 0, sizeof (list));
  848. if (getenv (PLCDEVICE))
  849. {
  850. channel.ifname = strdup (getenv (PLCDEVICE));
  851. }
  852. optind = 1;
  853. while (~ (c = getoptv (argc, argv, optv)))
  854. {
  855. switch (c)
  856. {
  857. case 'A':
  858. plc.action = PLCFWD_ADD;
  859. break;
  860. case 'C':
  861. plc.action = PLCFWD_STO;
  862. break;
  863. case 'D':
  864. plc.action = PLCFWD_SET;
  865. list [0].VLANID [0] = (uint16_t) (basespec (optarg, 10, sizeof (uint16_t)));
  866. break;
  867. case 'e':
  868. dup2 (STDOUT_FILENO, STDERR_FILENO);
  869. break;
  870. case 'f':
  871. if (! freopen (optarg, "rb", stdin))
  872. {
  873. error (1, errno, "%s", optarg);
  874. }
  875. items += readlist (& list [items], size - items);
  876. break;
  877. case 'i':
  878. #if defined (WINPCAP) || defined (LIBPCAP)
  879. channel.ifindex = atoi (optarg);
  880. #else
  881. channel.ifname = optarg;
  882. #endif
  883. break;
  884. case 'M':
  885. plc.action = PLCFWD_CTL;
  886. plc.module = (uint8_t) (uintspec (synonym (optarg, states, STATES), 0, UCHAR_MAX));
  887. break;
  888. case 'l':
  889. length = (uint32_t) (basespec (optarg, 10, sizeof (length)));
  890. break;
  891. case 'o':
  892. offset = (uint32_t) (basespec (optarg, 10, sizeof (offset)));
  893. break;
  894. case 'q':
  895. _setbits (channel.flags, CHANNEL_SILENCE);
  896. _setbits (plc.flags, PLC_SILENCE);
  897. break;
  898. case 'R':
  899. plc.action = PLCFWD_REM;
  900. break;
  901. case 'S':
  902. plc.action = PLCFWD_CTL;
  903. plc.pushbutton = (uint8_t) (uintspec (synonym (optarg, states, STATES), 0, UCHAR_MAX));
  904. break;
  905. case 't':
  906. channel.timeout = (signed) (uintspec (optarg, 0, UINT_MAX));
  907. break;
  908. case 'v':
  909. _setbits (channel.flags, CHANNEL_VERBOSE);
  910. _setbits (plc.flags, PLC_VERBOSE);
  911. break;
  912. case 'x':
  913. _setbits (plc.flags, PLC_BAILOUT);
  914. break;
  915. case 'z':
  916. readitem (& list [items++], optarg);
  917. break;
  918. default:
  919. break;
  920. }
  921. }
  922. argc -= optind;
  923. argv += optind;
  924. #if 0
  925. showlist (list, items);
  926. #endif
  927. openchannel (& channel);
  928. if (! (plc.message = malloc (sizeof (* plc.message))))
  929. {
  930. error (1, errno, PLC_NOMEMORY);
  931. }
  932. if (! argc)
  933. {
  934. function (& plc, offset, length, list, items);
  935. }
  936. while ((argc) && (* argv))
  937. {
  938. if (! hexencode (channel.peer, sizeof (channel.peer), synonym (* argv, devices, SIZEOF (devices))))
  939. {
  940. error (1, errno, PLC_BAD_MAC, * argv);
  941. }
  942. function (& plc, offset, length, list, items);
  943. argv++;
  944. argc--;
  945. }
  946. free (plc.message);
  947. closechannel (& channel);
  948. return (0);
  949. }