Identity2.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. /*====================================================================*
  2. Copyright (c) 2013,2021 Qualcomm Technologies, Inc.
  3. All Rights Reserved.
  4. Confidential and Proprietary - Qualcomm Technologies, Inc.
  5. ******************************************************************
  6. 2013 Qualcomm Atheros, Inc.
  7. *--------------------------------------------------------------------*/
  8. #ifndef IDENTITY2_SOURCE
  9. #define IDENTITY2_SOURCE
  10. #include <stdint.h>
  11. #include <unistd.h>
  12. #include <memory.h>
  13. #include "../tools/error.h"
  14. #include "../tools/files.h"
  15. #include "../tools/endian.h"
  16. #include "../tools/memory.h"
  17. #include "../tools/flags.h"
  18. #include "../nvm/nvm.h"
  19. #include "../plc/plc.h"
  20. /*====================================================================*
  21. *
  22. * signed pibchain2 (void const * memory);
  23. *
  24. * search panther/lynx image chain for the next PIB image; print
  25. * information on stdout and return;
  26. *
  27. * Contributor(s):
  28. * Charles Maier <cmaier@qca.qualcomm.com>
  29. *
  30. *--------------------------------------------------------------------*/
  31. static signed pibchain2 (void const * memory, char const * filename, flag_t flags)
  32. {
  33. struct panther_nvm_header * nvm_header;
  34. uint32_t origin = ~ 0;
  35. uint32_t offset = 0;
  36. signed module = 0;
  37. do
  38. {
  39. nvm_header = (struct panther_nvm_header *) ((char *) (memory) + offset);
  40. if (LE16TOH (nvm_header->MajorVersion) != 1)
  41. {
  42. if (_allclr (flags, NVM_SILENCE))
  43. {
  44. error (0, errno, NVM_HDR_VERSION, filename, module);
  45. }
  46. return (-1);
  47. }
  48. if (LE16TOH (nvm_header->MinorVersion) != 1)
  49. {
  50. if (_allclr (flags, NVM_SILENCE))
  51. {
  52. error (0, errno, NVM_HDR_VERSION, filename, module);
  53. }
  54. return (-1);
  55. }
  56. if (LE32TOH (nvm_header->PrevHeader) != origin)
  57. {
  58. if (_allclr (flags, NVM_SILENCE))
  59. {
  60. error (0, errno, NVM_HDR_LINK, filename, module);
  61. }
  62. return (-1);
  63. }
  64. if (checksum32 (nvm_header, sizeof (* nvm_header), 0))
  65. {
  66. error (0, 0, NVM_HDR_CHECKSUM, filename, module);
  67. return (-1);
  68. }
  69. origin = offset;
  70. offset += sizeof (* nvm_header);
  71. if (LE32TOH (nvm_header->ImageType) == NVM_IMAGE_PIB)
  72. {
  73. struct pib_header * pib_header = (struct pib_header *) ((char *) (memory) + offset);
  74. pib_header->PIBLENGTH = HTOLE16 ((uint16_t) (LE32TOH (nvm_header->ImageLength)));
  75. panther_pib_peek ((char *) (memory) + offset);
  76. pib_header->PIBLENGTH = 0;
  77. break;
  78. }
  79. if (checksum32 ((char *) (memory) + offset, LE32TOH (nvm_header->ImageLength), nvm_header->ImageChecksum))
  80. {
  81. if (_allclr (flags, NVM_SILENCE))
  82. {
  83. error (0, errno, NVM_IMG_CHECKSUM, filename, module);
  84. }
  85. return (-1);
  86. }
  87. offset += LE32TOH (nvm_header->ImageLength);
  88. module++;
  89. }
  90. while (~ nvm_header->NextHeader);
  91. return (0);
  92. }
  93. static signed pibchain2_no_pib_peek(void const* memory, char const* filename, flag_t flags, uint32_t* pib_offset)
  94. {
  95. struct panther_nvm_header* nvm_header;
  96. uint32_t origin = ~0;
  97. uint32_t offset = 0;
  98. signed module = 0;
  99. do
  100. {
  101. nvm_header = (struct panther_nvm_header*)((char*)(memory)+offset);
  102. if (LE16TOH(nvm_header->MajorVersion) != 1)
  103. {
  104. if (_allclr(flags, NVM_SILENCE))
  105. {
  106. error(0, errno, NVM_HDR_VERSION, filename, module);
  107. }
  108. return (-1);
  109. }
  110. if (LE16TOH(nvm_header->MinorVersion) != 1)
  111. {
  112. if (_allclr(flags, NVM_SILENCE))
  113. {
  114. error(0, errno, NVM_HDR_VERSION, filename, module);
  115. }
  116. return (-1);
  117. }
  118. if (LE32TOH(nvm_header->PrevHeader) != origin)
  119. {
  120. if (_allclr(flags, NVM_SILENCE))
  121. {
  122. error(0, errno, NVM_HDR_LINK, filename, module);
  123. }
  124. return (-1);
  125. }
  126. if (checksum32(nvm_header, sizeof(*nvm_header), 0))
  127. {
  128. error(0, 0, NVM_HDR_CHECKSUM, filename, module);
  129. return (-1);
  130. }
  131. origin = offset;
  132. offset += sizeof(*nvm_header);
  133. if (LE32TOH(nvm_header->ImageType) == NVM_IMAGE_PIB)
  134. {
  135. struct pib_header* pib_header = (struct pib_header*)((char*)(memory)+offset);
  136. pib_header->PIBLENGTH = HTOLE16((uint16_t)(LE32TOH(nvm_header->ImageLength)));
  137. pib_header->PIBLENGTH = 0;
  138. *pib_offset = offset;
  139. break;
  140. }
  141. if (checksum32((char*)(memory)+offset, LE32TOH(nvm_header->ImageLength), nvm_header->ImageChecksum))
  142. {
  143. if (_allclr(flags, NVM_SILENCE))
  144. {
  145. error(0, errno, NVM_IMG_CHECKSUM, filename, module);
  146. }
  147. return (-1);
  148. }
  149. offset += LE32TOH(nvm_header->ImageLength);
  150. module++;
  151. } while (~nvm_header->NextHeader);
  152. return (0);
  153. }
  154. /*====================================================================*
  155. *
  156. * signed Identity2 (struct plc * plc);
  157. *
  158. * plc.h
  159. *
  160. * read start of parameter chain from flash memory using single
  161. * VS_MODULE_OPERATION message and print identity information on
  162. * stdout;
  163. *
  164. * Contributor(s):
  165. * Charles Maier <cmaier@qca.qualcomm.com>
  166. *
  167. *--------------------------------------------------------------------*/
  168. signed Identity2 (struct plc * plc)
  169. {
  170. struct channel * channel = (struct channel *) (plc->channel);
  171. struct message * message = (struct message *) (plc->message);
  172. #ifndef __GNUC__
  173. #pragma pack (push,1)
  174. #endif
  175. struct __packed vs_module_operation_read_request
  176. {
  177. struct ethernet_hdr ethernet;
  178. struct qualcomm_hdr qualcomm;
  179. uint32_t RESERVED;
  180. uint8_t NUM_OP_DATA;
  181. struct __packed
  182. {
  183. uint16_t MOD_OP;
  184. uint16_t MOD_OP_DATA_LEN;
  185. uint32_t MOD_OP_RSVD;
  186. uint16_t MODULE_ID;
  187. uint16_t MODULE_SUB_ID;
  188. uint16_t MODULE_LENGTH;
  189. uint32_t MODULE_OFFSET;
  190. }
  191. MODULE_SPEC;
  192. }
  193. * request = (struct vs_module_operation_read_request *) (message);
  194. struct __packed vs_module_operation_read_confirm
  195. {
  196. struct ethernet_hdr ethernet;
  197. struct qualcomm_hdr qualcomm;
  198. uint16_t MSTATUS;
  199. uint16_t ERR_REC_CODE;
  200. uint32_t RESERVED;
  201. uint8_t NUM_OP_DATA;
  202. struct __packed
  203. {
  204. uint16_t MOD_OP;
  205. uint16_t MOD_OP_DATA_LEN;
  206. uint32_t MOD_OP_RSVD;
  207. uint16_t MODULE_ID;
  208. uint16_t MODULE_SUB_ID;
  209. uint16_t MODULE_LENGTH;
  210. uint32_t MODULE_OFFSET;
  211. }
  212. MODULE_SPEC;
  213. uint8_t MODULE_DATA [PLC_MODULE_SIZE];
  214. }
  215. * confirm = (struct vs_module_operation_read_confirm *) (message);
  216. #ifndef __GNUC__
  217. #pragma pack (pop)
  218. #endif
  219. Request (plc, "Device Identity");
  220. memset (message, 0, sizeof (* message));
  221. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  222. QualcommHeader (& request->qualcomm, 0, (VS_MODULE_OPERATION | MMTYPE_REQ));
  223. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  224. request->NUM_OP_DATA = 1;
  225. request->MODULE_SPEC.MOD_OP = HTOLE16 (0);
  226. request->MODULE_SPEC.MOD_OP_DATA_LEN = HTOLE16 (sizeof (request->MODULE_SPEC));
  227. request->MODULE_SPEC.MOD_OP_RSVD = HTOLE32 (0);
  228. request->MODULE_SPEC.MODULE_ID = HTOLE16 (PLC_MODULEID_PARAMETERS);
  229. request->MODULE_SPEC.MODULE_SUB_ID = HTOLE16 (0);
  230. request->MODULE_SPEC.MODULE_LENGTH = HTOLE16 (PLC_MODULE_SIZE);
  231. request->MODULE_SPEC.MODULE_OFFSET = HTOLE32 (0);
  232. if (SendMME (plc) <= 0)
  233. {
  234. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  235. return (-1);
  236. }
  237. while (ReadMME (plc, 0, (VS_MODULE_OPERATION | MMTYPE_CNF)) > 0)
  238. {
  239. if (confirm->MSTATUS)
  240. {
  241. Failure (plc, PLC_WONTDOIT);
  242. continue;
  243. }
  244. Confirm (plc, "-------");
  245. pibchain2 (confirm->MODULE_DATA, "device", plc->flags);
  246. }
  247. return (0);
  248. }
  249. signed get_nid_nmk_req(struct plc* plc)
  250. {
  251. struct channel* channel = (struct channel*)(plc->channel);
  252. struct message* message = (struct message*)(plc->message);
  253. #ifndef __GNUC__
  254. #pragma pack (push,1)
  255. #endif
  256. struct __packed vs_module_operation_read_request
  257. {
  258. struct ethernet_hdr ethernet;
  259. struct qualcomm_hdr qualcomm;
  260. uint32_t RESERVED;
  261. uint8_t NUM_OP_DATA;
  262. struct __packed
  263. {
  264. uint16_t MOD_OP;
  265. uint16_t MOD_OP_DATA_LEN;
  266. uint32_t MOD_OP_RSVD;
  267. uint16_t MODULE_ID;
  268. uint16_t MODULE_SUB_ID;
  269. uint16_t MODULE_LENGTH;
  270. uint32_t MODULE_OFFSET;
  271. }
  272. MODULE_SPEC;
  273. }
  274. *request = (struct vs_module_operation_read_request*)(message);
  275. #ifndef __GNUC__
  276. #pragma pack (pop)
  277. #endif
  278. debug(0, __func__, "Send read request to get NMK NID from pev");
  279. memset(message, 0, sizeof(*message));
  280. EthernetHeader(&request->ethernet, channel->peer, channel->host, channel->type);
  281. QualcommHeader(&request->qualcomm, 0, (VS_MODULE_OPERATION | MMTYPE_REQ));
  282. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  283. request->NUM_OP_DATA = 1;
  284. request->MODULE_SPEC.MOD_OP = HTOLE16(0);
  285. request->MODULE_SPEC.MOD_OP_DATA_LEN = HTOLE16(sizeof(request->MODULE_SPEC));
  286. request->MODULE_SPEC.MOD_OP_RSVD = HTOLE32(0);
  287. request->MODULE_SPEC.MODULE_ID = HTOLE16(PLC_MODULEID_PARAMETERS);
  288. request->MODULE_SPEC.MODULE_SUB_ID = HTOLE16(0);
  289. request->MODULE_SPEC.MODULE_LENGTH = HTOLE16(PLC_MODULE_SIZE);
  290. request->MODULE_SPEC.MODULE_OFFSET = HTOLE32(0);
  291. if (SendMME(plc) <= 0)
  292. {
  293. error(PLC_EXIT(plc), errno, CHANNEL_CANTSEND);
  294. return (-1);
  295. }
  296. return (0);
  297. }
  298. signed process_get_nid_cnf(struct plc* plc, void* NID, void* NMK)
  299. {
  300. struct message* message = (struct message*)(plc->message);
  301. #ifndef __GNUC__
  302. #pragma pack (push,1)
  303. #endif
  304. struct __packed vs_module_operation_read_confirm
  305. {
  306. struct ethernet_hdr ethernet;
  307. struct qualcomm_hdr qualcomm;
  308. uint16_t MSTATUS;
  309. uint16_t ERR_REC_CODE;
  310. uint32_t RESERVED;
  311. uint8_t NUM_OP_DATA;
  312. struct __packed
  313. {
  314. uint16_t MOD_OP;
  315. uint16_t MOD_OP_DATA_LEN;
  316. uint32_t MOD_OP_RSVD;
  317. uint16_t MODULE_ID;
  318. uint16_t MODULE_SUB_ID;
  319. uint16_t MODULE_LENGTH;
  320. uint32_t MODULE_OFFSET;
  321. }
  322. MODULE_SPEC;
  323. uint8_t MODULE_DATA[PLC_MODULE_SIZE];
  324. }
  325. *confirm = (struct vs_module_operation_read_confirm*)(message);
  326. #ifndef __GNUC__
  327. #pragma pack (pop)
  328. #endif
  329. if (confirm->MSTATUS)
  330. {
  331. Failure(plc, PLC_WONTDOIT);
  332. return(-1);
  333. }
  334. debug(0, __func__, "NMK NID received from pev");
  335. uint32_t pib_data_offset = 0;
  336. if (!pibchain2_no_pib_peek(confirm->MODULE_DATA, "device", plc->flags, &pib_data_offset))
  337. {
  338. const char* memory = (const char*)(confirm->MODULE_DATA + pib_data_offset);
  339. struct PIB3_0* PIB = (struct PIB3_0*)(memory);
  340. char buffer[100];
  341. if (_anyset(plc->flags, PLC_VERBOSE))
  342. {
  343. printf("pib offset %0x", pib_data_offset);
  344. printf("NMK %s\n", hexstring(buffer, sizeof(buffer), PIB->LocalDeviceConfig.NMK, sizeof(PIB->LocalDeviceConfig.NMK)));
  345. printf("NID %s\n", hexstring(buffer, sizeof(buffer), PIB->LocalDeviceConfig.PreferredNID, sizeof(PIB->LocalDeviceConfig.PreferredNID)));
  346. }
  347. memcpy((byte*)NMK, PIB->LocalDeviceConfig.NMK, sizeof(PIB->LocalDeviceConfig.NMK));
  348. memcpy((byte*)NID, PIB->LocalDeviceConfig.PreferredNID, sizeof(PIB->LocalDeviceConfig.PreferredNID));
  349. }
  350. return 0;
  351. }
  352. #endif