int6keth.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * int6keth.c - Qualcomm Atheros Ethernet PHY Settings;
  11. *
  12. * Contributor(s):
  13. * Charles Maier <cmaier@qca.qualcomm.com>
  14. * Matthieu Poullet <m.poullet@avm.de>
  15. *
  16. *--------------------------------------------------------------------*/
  17. /*====================================================================*
  18. * system header files;
  19. *--------------------------------------------------------------------*/
  20. #include <unistd.h>
  21. #include <stdlib.h>
  22. #include <stdint.h>
  23. /*====================================================================*
  24. * custom header files;
  25. *--------------------------------------------------------------------*/
  26. #include "../tools/getoptv.h"
  27. #include "../tools/putoptv.h"
  28. #include "../tools/memory.h"
  29. #include "../tools/number.h"
  30. #include "../tools/symbol.h"
  31. #include "../tools/types.h"
  32. #include "../tools/flags.h"
  33. #include "../tools/files.h"
  34. #include "../tools/error.h"
  35. #include "../plc/plc.h"
  36. #include "../ether/channel.h"
  37. #include "../mme/mme.h"
  38. /*====================================================================*
  39. * custom source files;
  40. *--------------------------------------------------------------------*/
  41. #ifndef MAKEFILE
  42. #include "../mme/UnwantedMessage.c"
  43. #include "../plc/Devices.c"
  44. #endif
  45. #ifndef MAKEFILE
  46. #include "../tools/assist.c"
  47. #include "../tools/error.c"
  48. #include "../tools/codelist.c"
  49. #include "../tools/getoptv.c"
  50. #include "../tools/putoptv.c"
  51. #include "../tools/version.c"
  52. #include "../tools/uintspec.c"
  53. #include "../tools/hexdump.c"
  54. #include "../tools/hexencode.c"
  55. #include "../tools/hexdecode.c"
  56. #include "../tools/hexstring.c"
  57. #include "../tools/todigit.c"
  58. #include "../tools/checkfilename.c"
  59. #include "../tools/checksum32.c"
  60. #include "../tools/fdchecksum32.c"
  61. #include "../tools/strfbits.c"
  62. #include "../tools/typename.c"
  63. #include "../tools/lookup.c"
  64. #include "../tools/synonym.c"
  65. #include "../tools/uintspec.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. #endif
  79. /*====================================================================*
  80. * program constants;
  81. *--------------------------------------------------------------------*/
  82. #define MCONTROL_READ 0x00
  83. #define MCONTROL_WRITE 0x01
  84. #define ETH_PORT 0
  85. static struct _code_ const negotiate [] =
  86. {
  87. {
  88. 0,
  89. "Off"
  90. },
  91. {
  92. 1,
  93. "On"
  94. }
  95. };
  96. static struct _code_ const speeds [] =
  97. {
  98. {
  99. 0,
  100. "10"
  101. },
  102. {
  103. 1,
  104. "100"
  105. },
  106. {
  107. 2,
  108. "1000"
  109. }
  110. };
  111. static struct _code_ const duplex [] =
  112. {
  113. {
  114. 0,
  115. "Half"
  116. },
  117. {
  118. 1,
  119. "Full"
  120. }
  121. };
  122. static struct _code_ const control [] =
  123. {
  124. {
  125. 0,
  126. "Off"
  127. },
  128. {
  129. 1,
  130. "Tx"
  131. },
  132. {
  133. 2,
  134. "Rx"
  135. },
  136. {
  137. 3,
  138. "On"
  139. }
  140. };
  141. static struct _code_ const advcap [] =
  142. {
  143. {
  144. (1 << 0),
  145. "100Full"
  146. },
  147. {
  148. (1 << 1),
  149. "100Half"
  150. },
  151. {
  152. (1 << 2),
  153. "10Full"
  154. },
  155. {
  156. (1 << 3),
  157. "10Half"
  158. },
  159. {
  160. (1 << 4),
  161. "1000Full"
  162. },
  163. {
  164. 0x1F,
  165. "All"
  166. }
  167. };
  168. static char const * rates [] =
  169. {
  170. "10",
  171. "100",
  172. "1000"
  173. };
  174. static char const * modes [] =
  175. {
  176. "Half",
  177. "Full"
  178. };
  179. static char const * links [] =
  180. {
  181. "Unknown",
  182. "Off",
  183. "On"
  184. };
  185. static char const * flows [] =
  186. {
  187. "Off",
  188. "Tx",
  189. "Rx",
  190. "On"
  191. };
  192. /*====================================================================*
  193. * program variables;
  194. *--------------------------------------------------------------------*/
  195. #ifndef __GNUC__
  196. #pragma pack (push,1)
  197. #endif
  198. typedef struct __packed phy_settings
  199. {
  200. uint8_t MCONTROL;
  201. uint8_t AUTONEGOTIATE;
  202. uint8_t ADVCAPS;
  203. uint8_t ESPEED;
  204. uint8_t EDUPLEX;
  205. uint8_t EFLOWCONTROL;
  206. }
  207. phy_settings;
  208. typedef struct __packed phy_readings
  209. {
  210. uint8_t MSTATUS;
  211. uint8_t ESPEED;
  212. uint8_t EDUPLEX;
  213. uint8_t ELINKSTATUS;
  214. uint8_t EFLOWCONTROL;
  215. }
  216. phy_readings;
  217. #ifndef __GNUC__
  218. #pragma pack (pop)
  219. #endif
  220. /*====================================================================*
  221. *
  222. * signed PHYSettings (struct channel * channel, struct phy_settings * settings, flag_t flags);
  223. *
  224. * plc.h
  225. *
  226. * read and display Ethernet PHY settings or write then read and
  227. * display settings;
  228. *
  229. * Contributor(s):
  230. * Charles Maier <cmaier@qca.qualcomm.com>
  231. *
  232. *--------------------------------------------------------------------*/
  233. signed PHYSettings (struct channel * channel, struct phy_settings * settings, flag_t flags)
  234. {
  235. struct message message;
  236. signed packetsize;
  237. #ifndef __GNUC__
  238. #pragma pack (push,1)
  239. #endif
  240. struct __packed vs_enet_settings_request
  241. {
  242. struct ethernet_hdr ethernet;
  243. struct qualcomm_hdr qualcomm;
  244. uint8_t MCONTROL;
  245. uint8_t AUTONEGOTIATE;
  246. uint8_t ADVCAPS;
  247. uint8_t ESPEED;
  248. uint8_t EDUPLEX;
  249. uint8_t EFLOWCONTROL;
  250. }
  251. * request = (struct vs_enet_settings_request *) (& message);
  252. struct __packed vs_enet_settings_confirm
  253. {
  254. struct ethernet_hdr ethernet;
  255. struct qualcomm_hdr qualcomm;
  256. uint8_t MSTATUS;
  257. uint8_t ESPEED;
  258. uint8_t EDUPLEX;
  259. uint8_t ELINKSTATUS;
  260. uint8_t EFLOWCONTROL;
  261. }
  262. * confirm = (struct vs_enet_settings_confirm *) (& message);
  263. #ifndef __GNUC__
  264. #pragma pack (pop)
  265. #endif
  266. char address [ETHER_ADDR_LEN * 3];
  267. memset (& message, 0, sizeof (message));
  268. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  269. QualcommHeader (& request->qualcomm, 0, (VS_ENET_SETTINGS | MMTYPE_REQ));
  270. request->MCONTROL = settings->MCONTROL;
  271. request->AUTONEGOTIATE = settings->AUTONEGOTIATE;
  272. request->ADVCAPS = settings->ADVCAPS;
  273. request->ESPEED = settings->ESPEED;
  274. request->EDUPLEX = settings->EDUPLEX;
  275. request->EFLOWCONTROL = settings->EFLOWCONTROL;
  276. if (sendpacket (channel, & message, (ETHER_MIN_LEN - ETHER_CRC_LEN)) < 0)
  277. {
  278. error (1, errno, CHANNEL_CANTSEND);
  279. }
  280. while ((packetsize = readpacket (channel, & message, sizeof (message))) > 0)
  281. {
  282. if (UnwantedMessage (& message, packetsize, 0, (VS_ENET_SETTINGS | MMTYPE_CNF)))
  283. {
  284. continue;
  285. }
  286. if ((confirm->MSTATUS == 1) || (confirm->MSTATUS == 3))
  287. {
  288. error (0, 0, "%s: %s (0x%0X): ", PLC_WONTDOIT, MMECode (confirm->qualcomm.MMTYPE, confirm->MSTATUS), confirm->MSTATUS);
  289. continue;
  290. }
  291. if (_anyset (flags, PLC_ANALYSE))
  292. {
  293. printf ("Bits Mode Link Flow\n");
  294. printf ("%4d ", confirm->ESPEED);
  295. printf ("%4d ", confirm->EDUPLEX);
  296. printf ("%4d ", confirm->ELINKSTATUS);
  297. printf ("%4d\n", confirm->EFLOWCONTROL);
  298. }
  299. else
  300. {
  301. printf ("%s %s ", channel->ifname, hexstring (address, sizeof (address), channel->host, sizeof (channel->host)));
  302. printf ("Speed=%s ", rates [confirm->ESPEED]);
  303. printf ("Duplex=%s ", modes [confirm->EDUPLEX]);
  304. printf ("LinkStatus=%s ", links [confirm->ELINKSTATUS]);
  305. printf ("FlowControl=%s\n", flows [confirm->EFLOWCONTROL]);
  306. }
  307. }
  308. if (packetsize < 0)
  309. {
  310. error (1, errno, CHANNEL_CANTREAD);
  311. }
  312. return (0);
  313. }
  314. /*====================================================================*
  315. *
  316. * int main (int argc, char const * argv[]);
  317. *
  318. * parse command line, populate plc structure and perform selected
  319. * operations; show help summary if asked; see getoptv and putoptv
  320. * to understand command line parsing and help summary display; see
  321. * plc.h for the definition of struct plc;
  322. *
  323. * the command line accepts multiple MAC addresses and the program
  324. * performs the specified operations on each address, in turn; the
  325. * address order is significant but the option order is not; the
  326. * default address is a local broadcast that causes all devices on
  327. * the local H1 interface to respond but not those at the remote
  328. * end of the powerline;
  329. *
  330. * the default address is 00:B0:52:00:00:01; omitting the address
  331. * will automatically address the local device; some options will
  332. * cancel themselves if this makes no sense;
  333. *
  334. * the default interface is eth1 because most people use eth0 as
  335. * their principle network connection; you can specify another
  336. * interface with -i or define environment string PLC to make
  337. * that the default interface and save typing;
  338. *
  339. *--------------------------------------------------------------------*/
  340. int main (int argc, char const * argv [])
  341. {
  342. extern struct channel channel;
  343. static char const * optv [] =
  344. {
  345. "a:d:ef:i:n:p:qrs:tvw",
  346. "device [device] [...] [> stdout]",
  347. "Qualcomm Atheros Ethernet PHY Settings",
  348. "a s\tadvertise capabilities as (s) ['1000Full'|'100Full'|'100Half'|10Full'|'10Half'|'All']",
  349. "d s\tduplex setting is (s) ['half'|'full']",
  350. "e\tredirect stderr to stdout",
  351. "f s\tflow control is (s) ['on'|'tx'|'rx'|'off']",
  352. #if defined (WINPCAP) || defined (LIBPCAP)
  353. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  354. #else
  355. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  356. #endif
  357. "n s\tauto-negotiate mode is (s) ['on'|'off']",
  358. "p n\tport number is (n) [" LITERAL (ETH_PORT) "]",
  359. "q\tquiet mode",
  360. "r\tread settings instead of write settings",
  361. "s s\ttransmission speed in mbps is (s) ['10'|'100'|'1000']",
  362. "v\tverbose mode",
  363. "w\twrite settings instead of read settings",
  364. (char const *) (0)
  365. };
  366. struct phy_settings settings =
  367. {
  368. 0,
  369. 1,
  370. 0,
  371. 0,
  372. 0,
  373. 0
  374. };
  375. flag_t flags = (flag_t) (0);
  376. signed c;
  377. if (getenv (PLCDEVICE))
  378. {
  379. #if defined (WINPCAP) || defined (LIBPCAP)
  380. channel.ifindex = atoi (getenv (PLCDEVICE));
  381. #else
  382. channel.ifname = strdup (getenv (PLCDEVICE));
  383. #endif
  384. }
  385. optind = 1;
  386. while (~ (c = getoptv (argc, argv, optv)))
  387. {
  388. switch (c)
  389. {
  390. case 'a':
  391. if ((c = lookup (optarg, advcap, SIZEOF (advcap))) == -1)
  392. {
  393. assist (optarg, "capability", advcap, SIZEOF (advcap));
  394. }
  395. settings.ADVCAPS |= (uint8_t) (c);
  396. break;
  397. case 'd':
  398. if ((c = lookup (optarg, duplex, SIZEOF (duplex))) == -1)
  399. {
  400. assist (optarg, "duplex", duplex, SIZEOF (duplex));
  401. }
  402. settings.EDUPLEX = (uint8_t) (c);
  403. break;
  404. case 'e':
  405. dup2 (STDOUT_FILENO, STDERR_FILENO);
  406. break;
  407. case 'f':
  408. if ((c = lookup (optarg, control, SIZEOF (control))) == -1)
  409. {
  410. assist (optarg, "control", control, SIZEOF (control));
  411. }
  412. settings.EFLOWCONTROL = (uint8_t) (c);
  413. break;
  414. case 'n':
  415. if ((c = lookup (optarg, negotiate, SIZEOF (negotiate))) == -1)
  416. {
  417. assist (optarg, "auto-negotiate", negotiate, SIZEOF (negotiate));
  418. }
  419. settings.AUTONEGOTIATE = (uint8_t) (c);
  420. break;
  421. case 's':
  422. if ((c = lookup (optarg, speeds, SIZEOF (speeds))) == -1)
  423. {
  424. assist (optarg, "speed", speeds, SIZEOF (speeds));
  425. }
  426. settings.ESPEED = (uint8_t) (c);
  427. break;
  428. case 't':
  429. _setbits (flags, PLC_ANALYSE);
  430. break;
  431. case 'i':
  432. #if defined (WINPCAP) || defined (LIBPCAP)
  433. channel.ifindex = atoi (optarg);
  434. #else
  435. channel.ifname = optarg;
  436. #endif
  437. break;
  438. case 'p':
  439. settings.MCONTROL &= 0x0F;
  440. settings.MCONTROL |= (unsigned) (uintspec (optarg, 0, 7)) << 4;
  441. break;
  442. case 'q':
  443. _setbits (channel.flags, CHANNEL_SILENCE);
  444. break;
  445. case 'r':
  446. settings.MCONTROL &= 0xF0;
  447. settings.MCONTROL |= 0x00;
  448. break;
  449. case 'v':
  450. _setbits (channel.flags, CHANNEL_VERBOSE);
  451. break;
  452. case 'w':
  453. settings.MCONTROL &= 0xF0;
  454. settings.MCONTROL |= 0x01;
  455. break;
  456. default:
  457. break;
  458. }
  459. }
  460. argc -= optind;
  461. argv += optind;
  462. openchannel (& channel);
  463. if (! argc)
  464. {
  465. PHYSettings (& channel, & settings, flags);
  466. }
  467. while ((argc) && (* argv))
  468. {
  469. if (! hexencode (channel.peer, sizeof (channel.peer), synonym (* argv, devices, SIZEOF (devices))))
  470. {
  471. error (1, errno, PLC_BAD_MAC, * argv);
  472. }
  473. PHYSettings (& channel, & settings, flags);
  474. argc--;
  475. argv++;
  476. }
  477. closechannel (& channel);
  478. exit (0);
  479. }