PLCTopology.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * PLCTopology.c
  11. *
  12. * plc.h
  13. *
  14. * Contributor(s):
  15. * Charles Maier <cmaier@qca.qualcomm.com>
  16. * Matthieu Poullet <m.poullet@avm.de>
  17. *
  18. *--------------------------------------------------------------------*/
  19. #ifndef PLCTOPOLOGY_SOURCE
  20. #define PLCTOPOLOGY_SOURCE
  21. #include <memory.h>
  22. #include <errno.h>
  23. #include "../tools/memory.h"
  24. #include "../tools/symbol.h"
  25. #include "../tools/error.h"
  26. #include "../tools/flags.h"
  27. #include "../plc/plc.h"
  28. /*====================================================================*
  29. *
  30. * signed PLCPlatform (struct channel * channel, struct plcstation * station);
  31. *
  32. * populate plcstation structure with hardware and firmware version
  33. * string using a VS_SW_VER message;
  34. *
  35. * extern char const * chipset [] contains chipset name strings in
  36. * order of chipset code but function chipset() must be called to
  37. * insert the true code into the confirmation message because some
  38. * chipsets return the wrong code; alien technology and voodoo are
  39. * needed;
  40. *
  41. *--------------------------------------------------------------------*/
  42. static signed PLCPlatform (struct channel * channel, struct plcstation * plcstation)
  43. {
  44. struct message message;
  45. ssize_t packetsize;
  46. #ifndef __GNUC__
  47. #pragma pack (push,1)
  48. #endif
  49. struct __packed vs_sw_ver_request
  50. {
  51. struct ethernet_hdr ethernet;
  52. struct qualcomm_hdr qualcomm;
  53. }
  54. * request = (struct vs_sw_ver_request *) (& message);
  55. struct __packed vs_sw_ver_confirm
  56. {
  57. struct ethernet_hdr ethernet;
  58. struct qualcomm_hdr qualcomm;
  59. uint8_t MSTATUS;
  60. uint8_t MDEVICE;
  61. uint8_t MLENGTH;
  62. char MSTRING [0x80];
  63. }
  64. * confirm = (struct vs_sw_ver_confirm *) (& message);
  65. #ifndef __GNUC__
  66. #pragma pack (pop)
  67. #endif
  68. memset (& message, 0, sizeof (message));
  69. EthernetHeader (& request->ethernet, plcstation->MAC, channel->host, channel->type);
  70. QualcommHeader (& request->qualcomm, 0, (VS_SW_VER | MMTYPE_REQ));
  71. if (sendpacket (channel, & message, (ETHER_MIN_LEN - ETHER_CRC_LEN)) > 0)
  72. {
  73. while ((packetsize = readpacket (channel, & message, sizeof (message))) > 0)
  74. {
  75. if (! UnwantedMessage (& message, packetsize, 0, (VS_SW_VER | MMTYPE_CNF)))
  76. {
  77. chipset (confirm);
  78. strncpy (plcstation->hardware, chipsetname (confirm->MDEVICE), sizeof (plcstation->hardware));
  79. strncpy (plcstation->firmware, confirm->MSTRING, sizeof (plcstation->firmware));
  80. return (0);
  81. }
  82. }
  83. }
  84. return (-1);
  85. }
  86. /*====================================================================*
  87. *
  88. * signed PLCIdentity (struct channel * channel, struct plcstation * station);
  89. *
  90. * populate plcstation structure with the device identity using a
  91. * VS_MFG_STRING message;
  92. *
  93. *--------------------------------------------------------------------*/
  94. static signed PLCIdentity (struct channel * channel, struct plcstation * plcstation)
  95. {
  96. struct message message;
  97. ssize_t packetsize;
  98. #ifndef __GNUC__
  99. #pragma pack (push,1)
  100. #endif
  101. struct __packed vs_mfg_string_request
  102. {
  103. struct ethernet_hdr ethernet;
  104. struct qualcomm_hdr qualcomm;
  105. }
  106. * request = (struct vs_mfg_string_request *) (& message);
  107. struct __packed vs_mfg_string_confirm
  108. {
  109. struct ethernet_hdr ethernet;
  110. struct qualcomm_hdr qualcomm;
  111. uint8_t MSTATUS;
  112. uint8_t MLENGTH;
  113. char MSTRING [0x40];
  114. }
  115. * confirm = (struct vs_mfg_string_confirm *) (& message);
  116. #ifndef __GNUC__
  117. #pragma pack (pop)
  118. #endif
  119. memset (& message, 0, sizeof (message));
  120. EthernetHeader (& request->ethernet, plcstation->MAC, channel->host, channel->type);
  121. QualcommHeader (& request->qualcomm, 0, (VS_MFG_STRING | MMTYPE_REQ));
  122. if (sendpacket (channel, & message, (ETHER_MIN_LEN - ETHER_CRC_LEN)) > 0)
  123. {
  124. while ((packetsize = readpacket (channel, & message, sizeof (message))) > 0)
  125. {
  126. if (! UnwantedMessage (& message, packetsize, 0, (VS_MFG_STRING | MMTYPE_CNF)))
  127. {
  128. strncpy (plcstation->identity, confirm->MSTRING, sizeof (plcstation->identity));
  129. return (0);
  130. }
  131. }
  132. }
  133. return (-1);
  134. }
  135. /*====================================================================*
  136. *
  137. * signed PLCTopology (struct channel * channel, struct message * message, struct plctopology * plctopology)
  138. *
  139. * populate a plctopology structure with network information; the
  140. * logic this is elusive due to the way the VS_NW_INFO message is
  141. * structured;
  142. *
  143. * the host can have many interfaces and each interface can have
  144. * many powerline devices connected to it; each powerline device
  145. * can bridge to an independent powerline network having a unique
  146. * set of member devices; alternately, some powerline devices can
  147. * be members of the same powerline network;
  148. *
  149. * INT6x00 chipsets have an 8-bit PHY rate while AR7x00 chipsets
  150. * have a 16-bit PHY rate; this means that AR7x00 PHY rates need
  151. * endian conversion;
  152. *
  153. *--------------------------------------------------------------------*/
  154. signed PLCTopology (struct channel * channel, struct message * message, struct plctopology * plctopology)
  155. {
  156. ssize_t packetsize;
  157. #if defined (INT6x00)
  158. struct __packed vs_nw_info_request
  159. {
  160. struct ethernet_hdr ethernet;
  161. struct qualcomm_hdr qualcomm;
  162. }
  163. * request = (struct vs_nw_info_request *) (message);
  164. struct __packed vs_nw_info_confirm
  165. {
  166. struct ethernet_hdr ethernet;
  167. struct qualcomm_hdr qualcomm;
  168. uint8_t data [1];
  169. }
  170. * confirm = (struct vs_nw_info_confirm *) (message);
  171. struct __packed station
  172. {
  173. uint8_t MAC [ETHER_ADDR_LEN];
  174. uint8_t TEI;
  175. uint8_t BDA [ETHER_ADDR_LEN];
  176. uint8_t AVGTX;
  177. uint8_t AVGRX;
  178. }
  179. * station;
  180. struct __packed network
  181. {
  182. uint8_t NID [7];
  183. uint8_t SNID;
  184. uint8_t TEI;
  185. uint8_t ROLE;
  186. uint8_t CCO_MAC [ETHER_ADDR_LEN];
  187. uint8_t CCO_TEI;
  188. uint8_t NUMSTAS;
  189. struct station stations [1];
  190. }
  191. * network;
  192. struct __packed networks
  193. {
  194. uint8_t NUMAVLNS;
  195. struct network networks [1];
  196. }
  197. * networks = (struct networks *) (confirm->data);
  198. #elif defined (AR7x00)
  199. struct __packed vs_nw_info_request
  200. {
  201. struct ethernet_hdr ethernet;
  202. struct qualcomm_fmi qualcomm;
  203. }
  204. * request = (struct vs_nw_info_request *) (message);
  205. struct __packed vs_nw_info_confirm
  206. {
  207. struct ethernet_hdr ethernet;
  208. struct qualcomm_fmi qualcomm;
  209. uint8_t SUB_VERSION;
  210. uint8_t Reserved;
  211. uint16_t DATA_LEN;
  212. uint8_t DATA [1];
  213. }
  214. * confirm = (struct vs_nw_info_confirm *) (message);
  215. struct __packed station
  216. {
  217. uint8_t MAC [ETHER_ADDR_LEN];
  218. uint8_t TEI;
  219. uint8_t Reserved [3];
  220. uint8_t BDA [ETHER_ADDR_LEN];
  221. uint16_t AVGTX;
  222. uint8_t COUPLING;
  223. uint8_t Reserved3;
  224. uint16_t AVGRX;
  225. uint16_t Reserved4;
  226. }
  227. * station;
  228. struct __packed network
  229. {
  230. uint8_t NID [7];
  231. uint8_t Reserved1 [2];
  232. uint8_t SNID;
  233. uint8_t TEI;
  234. uint8_t Reserved2 [4];
  235. uint8_t ROLE;
  236. uint8_t CCO_MAC [ETHER_ADDR_LEN];
  237. uint8_t CCO_TEI;
  238. uint8_t Reserved3 [3];
  239. uint8_t NUMSTAS;
  240. uint8_t Reserved4 [5];
  241. struct station stations [1];
  242. }
  243. * network;
  244. struct __packed networks
  245. {
  246. uint8_t Reserved;
  247. uint8_t NUMAVLNS;
  248. struct network networks [1];
  249. }
  250. * networks = (struct networks *) (confirm->DATA);
  251. #else
  252. #error "Unspecified chipset"
  253. #endif
  254. struct plcnetwork * plcnetwork = (struct plcnetwork *) (& plctopology->plcnetwork);
  255. struct plcstation * plcstation;
  256. byte bridges [255] [ETHER_ADDR_LEN];
  257. unsigned bridge = LocalDevices (channel, message, bridges, (size_t) (sizeof (bridges)));
  258. while (bridge--)
  259. {
  260. memset (message, 0, sizeof (* message));
  261. memcpy (channel->peer, bridges [bridge], sizeof (channel->peer));
  262. #if defined (INT6x00)
  263. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  264. QualcommHeader (& request->qualcomm, 0, (VS_NW_INFO | MMTYPE_REQ));
  265. #elif defined (AR7x00)
  266. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  267. QualcommHeader1 (& request->qualcomm, 1, (VS_NW_INFO | MMTYPE_REQ));
  268. #else
  269. #error "Unspecified chipset"
  270. #endif
  271. if (sendpacket (channel, message, (ETHER_MIN_LEN - ETHER_CRC_LEN)) <= 0)
  272. {
  273. error (1, errno, CHANNEL_CANTSEND);
  274. }
  275. while ((packetsize = readpacket (channel, message, sizeof (* message))) > 0)
  276. {
  277. #if defined (INT6x00)
  278. if (UnwantedMessage (confirm, packetsize, 0, (VS_NW_INFO | MMTYPE_CNF)))
  279. #elif defined (AR7x00)
  280. if (UnwantedMessage (confirm, packetsize, 1, (VS_NW_INFO | MMTYPE_CNF)))
  281. #else
  282. #error "Unspecified chipset"
  283. #endif
  284. {
  285. continue;
  286. }
  287. network = (struct network *) (& networks->networks);
  288. plcstation = (struct plcstation *) (& plcnetwork->plcstation);
  289. memset (plcnetwork, 0, sizeof (* plcnetwork));
  290. memset (plcstation, 0, sizeof (* plcstation));
  291. plcstation->LOC = ! memcmp (confirm->ethernet.OSA, bridges [bridge], sizeof (confirm->ethernet.OSA));
  292. plcstation->CCO = ! memcmp (confirm->ethernet.OSA, network->CCO_MAC, sizeof (confirm->ethernet.OSA));
  293. plcstation->TEI = network->TEI;
  294. memcpy (plcstation->MAC, confirm->ethernet.OSA, sizeof (plcstation->MAC));
  295. memcpy (plcstation->BDA, confirm->ethernet.ODA, sizeof (plcstation->BDA));
  296. PLCPlatform (channel, plcstation);
  297. plcnetwork->plcstations++;
  298. plcstation++;
  299. while (networks->NUMAVLNS--)
  300. {
  301. station = (struct station *) (& network->stations);
  302. while (network->NUMSTAS--)
  303. {
  304. memset (plcstation, 0, sizeof (* plcstation));
  305. plcstation->LOC = ! memcmp (station->BDA, channel->host, sizeof (station->BDA));
  306. plcstation->CCO = ! memcmp (station->MAC, network->CCO_MAC, sizeof (station->MAC));
  307. plcstation->TEI = station->TEI;
  308. memcpy (plcstation->MAC, station->MAC, sizeof (plcstation->MAC));
  309. memcpy (plcstation->BDA, station->BDA, sizeof (plcstation->BDA));
  310. #if defined (INT6x00)
  311. plcstation->TX = station->AVGTX;
  312. plcstation->RX = station->AVGRX;
  313. #elif defined (AR7x00)
  314. plcstation->TX = LE16TOH (station->AVGTX);
  315. plcstation->RX = LE16TOH (station->AVGRX);
  316. #else
  317. #error "Unspecified chipset"
  318. #endif
  319. PLCPlatform (channel, plcstation);
  320. PLCIdentity (channel, plcstation);
  321. plcnetwork->plcstations++;
  322. plcstation++;
  323. station++;
  324. }
  325. plctopology->plcnetworks++;
  326. plcnetwork = (struct plcnetwork *) (plcstation);
  327. network = (struct network *) (station);
  328. }
  329. }
  330. }
  331. return (0);
  332. }
  333. /*====================================================================*
  334. *
  335. *--------------------------------------------------------------------*/
  336. #endif