chknvm.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. * system header files;
  10. *--------------------------------------------------------------------*/
  11. #include <stdio.h>
  12. #include <stdlib.h>
  13. #include <unistd.h>
  14. #include <memory.h>
  15. #include <fcntl.h>
  16. #include <errno.h>
  17. /*====================================================================*
  18. * custom header files;
  19. *--------------------------------------------------------------------*/
  20. #include "../tools/getoptv.h"
  21. #include "../tools/memory.h"
  22. #include "../tools/flags.h"
  23. #include "../tools/error.h"
  24. #include "../tools/files.h"
  25. #include "../ram/sdram.h"
  26. #include "../nvm/nvm.h"
  27. /*====================================================================*
  28. * custom source files;
  29. *--------------------------------------------------------------------*/
  30. #ifndef MAKEFILE
  31. #include "../tools/getoptv.c"
  32. #include "../tools/putoptv.c"
  33. #include "../tools/version.c"
  34. #include "../tools/checksum32.c"
  35. #include "../tools/fdchecksum32.c"
  36. #include "../tools/strfbits.c"
  37. #include "../tools/error.c"
  38. #endif
  39. #ifndef MAKEFILE
  40. #include "../ram/sdrampeek.c"
  41. #endif
  42. #ifndef MAKEFILE
  43. #include "../nvm/nvm.c"
  44. #include "../nvm/nvmpeek.c"
  45. #include "../nvm/lightning_nvm_peek.c"
  46. #include "../nvm/panther_nvm_peek.c"
  47. #include "../nvm/panther_nvm_manifest.c"
  48. #include "../nvm/panther_nvm_revision.c"
  49. #include "../nvm/fdmanifest.c"
  50. #include "../nvm/fdrevision.c"
  51. #endif
  52. /*====================================================================*
  53. * program constants;
  54. *--------------------------------------------------------------------*/
  55. #define HARDWARE 0
  56. #define SOFTWARE 1
  57. #define VER 2
  58. #define REV 3
  59. #define BUILD 6
  60. #define DATE 7
  61. /*====================================================================*
  62. *
  63. * unsigned string2vector (char ** vector, length, char * string, char c);
  64. *
  65. * convert string to a vector and return vector count; split string
  66. * on characer (c);
  67. *
  68. * Contributor(s):
  69. * Charles Maier <cmaier@qca.qualcomm.com>
  70. *
  71. *--------------------------------------------------------------------*/
  72. static unsigned string2vector (char ** vector, char * string, char c)
  73. {
  74. char ** origin = vector;
  75. for (* vector++ = string; * string; string++)
  76. {
  77. if (* string == c)
  78. {
  79. * string++ = (char) (0);
  80. * vector++ = string--;
  81. }
  82. }
  83. * vector = (char *) (0);
  84. return ((unsigned) (vector - origin));
  85. }
  86. /*====================================================================*
  87. *
  88. * void lightning_nvm_revision (signed fd, char const * filename, unsigned module, unsigned offset, flag_t flags);
  89. *
  90. * Contributor(s):
  91. * Charles Maier <cmaier@qca.qualcomm.com>
  92. *
  93. *--------------------------------------------------------------------*/
  94. static void lightning_nvm_revision (signed fd, char const * filename, unsigned module, unsigned offset, flag_t flags)
  95. {
  96. char memory [512];
  97. read (fd, memory, sizeof (memory));
  98. lseek (fd, (off_t) (0) - sizeof (memory), SEEK_CUR);
  99. if (_anyset (flags, NVM_REVISION))
  100. {
  101. if (_anyset (flags, NVM_VERBOSE))
  102. {
  103. printf ("%s (%d) ", filename, module);
  104. }
  105. if ((* memory > 0x20) && (* memory < 0x7F))
  106. {
  107. printf ("%s\n", memory);
  108. }
  109. else
  110. {
  111. printf ("%s\n", memory + offset);
  112. }
  113. }
  114. if (_anyset (flags, NVM_IDENTITY))
  115. {
  116. char * vector [16];
  117. char buffer [256];
  118. if ((* memory > 0x20) && (* memory < 0x7f))
  119. {
  120. strncpy (buffer, memory, sizeof (buffer));
  121. }
  122. else
  123. {
  124. strncpy (buffer, memory + offset, sizeof (buffer));
  125. }
  126. string2vector (vector, buffer, '-');
  127. printf ("%s ", vector [HARDWARE]);
  128. printf ("%04d ", atoi (vector [BUILD]));
  129. printf ("%s ", vector [DATE]);
  130. printf ("%s.", vector [VER]);
  131. printf ("%s ", vector [REV]);
  132. printf ("%s (%d)\n", filename, module);
  133. return;
  134. }
  135. return;
  136. }
  137. /*====================================================================*
  138. *
  139. * signed lightning_nvm_chain (signed fd, char const * filename, flag_t flags);
  140. *
  141. * validate a legcy PLC firmware file; return 0 on success or -1 on
  142. * error;
  143. *
  144. * keep as little information as possible in memory; this slows
  145. * validation but minimizes the resources needed at runtime so
  146. * that resource requirements do not grow with file size;
  147. *
  148. * the checksum of the entire header, including header checksum, is
  149. * always 0 for valid headers; similarly, the checksum of the module
  150. * and module checksum is always 0 for valid modules;
  151. *
  152. *
  153. * Contributor(s):
  154. * Charles Maier <cmaier@qca.qualcomm.com>
  155. *
  156. *--------------------------------------------------------------------*/
  157. static signed lightning_nvm_chain (signed fd, char const * filename, flag_t flags)
  158. {
  159. struct lightning_nvm_header header;
  160. struct config_ram config_ram;
  161. unsigned module = 0;
  162. do
  163. {
  164. if (read (fd, & header, sizeof (header)) != sizeof (header))
  165. {
  166. if (_allclr (flags, NVM_SILENCE))
  167. {
  168. error (0, errno, NVM_HDR_CANTREAD, filename, module);
  169. }
  170. return (- 1);
  171. }
  172. if (LE32TOH (header.HEADERVERSION) != 0x60000000)
  173. {
  174. if (_allclr (flags, NVM_SILENCE))
  175. {
  176. error (0, errno, NVM_HDR_VERSION, filename, module);
  177. }
  178. return (- 1);
  179. }
  180. if (checksum32 (& header, sizeof (header), 0))
  181. {
  182. if (_allclr (flags, NVM_SILENCE))
  183. {
  184. error (0, errno, NVM_HDR_CHECKSUM, filename, module);
  185. }
  186. return (- 1);
  187. }
  188. if (_anyset (flags, NVM_VERBOSE))
  189. {
  190. printf ("------- %s (%d) -------\n", filename, module);
  191. lightning_nvm_peek (& header);
  192. }
  193. if (header.HEADERMINORVERSION)
  194. {
  195. if (LE32TOH (header.IMAGETYPE) == NVM_IMAGE_CONFIG_SYNOPSIS)
  196. {
  197. if (_anyset (flags, NVM_SDRAM))
  198. {
  199. printf ("------- %s (%d) -------\n", filename, module);
  200. read (fd, & config_ram, sizeof (config_ram));
  201. lseek (fd, (off_t) (0) - sizeof (config_ram), SEEK_CUR);
  202. sdrampeek (& config_ram);
  203. }
  204. }
  205. else if (LE32TOH (header.IMAGETYPE) == NVM_IMAGE_FIRMWARE)
  206. {
  207. lightning_nvm_revision (fd, filename, module, 0x70, flags);
  208. }
  209. }
  210. else if (! module)
  211. {
  212. if (_anyset (flags, NVM_SDRAM))
  213. {
  214. printf ("------- %s (%d) -------\n", filename, module);
  215. read (fd, & config_ram, sizeof (config_ram));
  216. lseek (fd, (off_t) (0) - sizeof (config_ram), SEEK_CUR);
  217. sdrampeek (& config_ram);
  218. }
  219. }
  220. else if (! header.NEXTHEADER)
  221. {
  222. lightning_nvm_revision (fd, filename, module, 0x00, flags);
  223. }
  224. if (fdchecksum32 (fd, LE32TOH (header.IMAGELENGTH), header.IMAGECHECKSUM))
  225. {
  226. if (_allclr (flags, NVM_SILENCE))
  227. {
  228. error (0, errno, NVM_IMG_CHECKSUM, filename, module);
  229. }
  230. return (- 1);
  231. }
  232. module++;
  233. }
  234. while (header.NEXTHEADER);
  235. if (lseek (fd, 0, SEEK_CUR) != lseek (fd, 0, SEEK_END))
  236. {
  237. if (_allclr (flags, NVM_SILENCE))
  238. {
  239. error (0, errno, NVM_HDR_LINK, filename, module);
  240. }
  241. return (- 1);
  242. }
  243. return (0);
  244. }
  245. /*====================================================================*
  246. *
  247. * signed panther_nvm_chain (signed fd, char const * filename, flag_t flags);
  248. *
  249. * validate a PLC image file; return 0 on success or -1 on error;
  250. *
  251. * keep as little information as possible in memory; this slows
  252. * validation but minimizes the resources needed at runtime so
  253. * that resource requirements do not grow with file size;
  254. *
  255. * the checksum of the entire header, including header checksum, is
  256. * always 0 for valid headers; similarly, the checksum of the module
  257. * and module checksum is always 0 for valid modules;
  258. *
  259. *
  260. * Contributor(s):
  261. * Charles Maier <cmaier@qca.qualcomm.com>
  262. *
  263. *--------------------------------------------------------------------*/
  264. static signed panther_nvm_chain (signed fd, char const * filename, flag_t flags)
  265. {
  266. struct panther_nvm_header header;
  267. unsigned module = 0;
  268. uint32_t origin = ~ 0;
  269. uint32_t offset = 0;
  270. do
  271. {
  272. if (read (fd, & header, sizeof (header)) != sizeof (header))
  273. {
  274. if (_allclr (flags, NVM_SILENCE))
  275. {
  276. error (0, errno, NVM_HDR_CANTREAD, filename, module);
  277. }
  278. return (- 1);
  279. }
  280. if (LE16TOH (header.MajorVersion) != 1)
  281. {
  282. if (_allclr (flags, NVM_SILENCE))
  283. {
  284. error (0, errno, NVM_HDR_VERSION, filename, module);
  285. }
  286. return (- 1);
  287. }
  288. if (LE16TOH (header.MinorVersion) != 1)
  289. {
  290. if (_allclr (flags, NVM_SILENCE))
  291. {
  292. error (0, errno, NVM_HDR_VERSION, filename, module);
  293. }
  294. return (- 1);
  295. }
  296. if (checksum32 (& header, sizeof (header), 0))
  297. {
  298. if (_allclr (flags, NVM_SILENCE))
  299. {
  300. error (0, errno, NVM_HDR_CHECKSUM, filename, module);
  301. }
  302. return (- 1);
  303. }
  304. if (LE32TOH (header.PrevHeader) != origin)
  305. {
  306. if (_allclr (flags, NVM_SILENCE))
  307. {
  308. error (0, errno, NVM_HDR_LINK, filename, module);
  309. }
  310. return (- 1);
  311. }
  312. if (_anyset (flags, NVM_VERBOSE))
  313. {
  314. printf ("------- %s (%d) -------\n", filename, module);
  315. panther_nvm_peek (& header);
  316. }
  317. if (LE32TOH (header.ImageType) == NVM_IMAGE_MANIFEST)
  318. {
  319. if (_anyset (flags, NVM_MANIFEST))
  320. {
  321. printf ("------- %s (%d) -------\n", filename, module);
  322. fdmanifest (fd, filename, & header);
  323. return (0);
  324. }
  325. if (_anyset (flags, NVM_REVISION))
  326. {
  327. fdrevision (fd, filename, & header);
  328. return (0);
  329. }
  330. }
  331. if (fdchecksum32 (fd, LE32TOH (header.ImageLength), header.ImageChecksum))
  332. {
  333. if (_allclr (flags, NVM_SILENCE))
  334. {
  335. error (0, errno, NVM_IMG_CHECKSUM, filename, module);
  336. }
  337. return (- 1);
  338. }
  339. origin = offset;
  340. offset = LE32TOH (header.NextHeader);
  341. module++;
  342. }
  343. while (~ header.NextHeader);
  344. #if 0
  345. if (lseek (fd, 0, SEEK_CUR) != lseek (fd, 0, SEEK_END))
  346. {
  347. if (_allclr (flags, NVM_SILENCE))
  348. {
  349. error (0, errno, NVM_HDR_LINK, filename, module);
  350. }
  351. return (- 1);
  352. }
  353. #endif
  354. return (0);
  355. }
  356. /*====================================================================*
  357. *
  358. * signed chknvm (char const * filename, flag_t flags);
  359. *
  360. * validate a PLC firmware module file; keep as little information
  361. * as possible in memory; this slows validation but minimizes the
  362. * resources needed at runtime; resource requirements do not grow
  363. * with file size;
  364. *
  365. * determine file format based on module header version then rewind
  366. * the file and call the appropriate chknvm;
  367. *
  368. *
  369. * Contributor(s):
  370. * Charles Maier <cmaier@qca.qualcomm.com>
  371. *
  372. *--------------------------------------------------------------------*/
  373. static signed chknvm (char const * filename, flag_t flags)
  374. {
  375. uint32_t version;
  376. signed status;
  377. signed fd;
  378. if ((fd = open (filename, O_BINARY | O_RDONLY)) == - 1)
  379. {
  380. if (_allclr (flags, NVM_SILENCE))
  381. {
  382. error (0, errno, FILE_CANTOPEN, filename);
  383. }
  384. return (- 1);
  385. }
  386. if (read (fd, & version, sizeof (version)) != sizeof (version))
  387. {
  388. if (_allclr (flags, NVM_SILENCE))
  389. {
  390. error (0, errno, FILE_CANTREAD, filename);
  391. }
  392. return (- 1);
  393. }
  394. if (lseek (fd, 0, SEEK_SET))
  395. {
  396. if (_allclr (flags, NVM_SILENCE))
  397. {
  398. error (0, errno, FILE_CANTHOME, filename);
  399. }
  400. return (- 1);
  401. }
  402. if (LE32TOH (version) == 0x60000000)
  403. {
  404. status = lightning_nvm_chain (fd, filename, flags);
  405. }
  406. else
  407. {
  408. status = panther_nvm_chain (fd, filename, flags);
  409. }
  410. close (fd);
  411. return (status);
  412. }
  413. /*====================================================================*
  414. *
  415. * int main (int argc, char const * argv []);
  416. *
  417. * validate a PLC firmware image file; keep as little information
  418. * as possible in memory; this slows validation but minimizes the
  419. * resources needed at runtime; resource requirements do not grow
  420. * with file size;
  421. *
  422. *
  423. * Contributor(s):
  424. * Charles Maier <cmaier@qca.qualcomm.com>
  425. *
  426. *--------------------------------------------------------------------*/
  427. int main (int argc, char const * argv [])
  428. {
  429. static char const * optv [] =
  430. {
  431. "imqrsv",
  432. "file [file] [...]",
  433. "Qualcomm Atheros PLC Image File Validator",
  434. "i\tprint firmware identity string",
  435. "m\tdisplay manifest",
  436. "q\tsuppress messages",
  437. "r\tprint firmware revision string",
  438. "s\tprint SDRAM configuration blocks",
  439. "v\tverbose messages",
  440. (char const *) (0)
  441. };
  442. signed state = 0;
  443. flag_t flags = (flag_t) (0);
  444. signed c;
  445. optind = 1;
  446. while (~ (c = getoptv (argc, argv, optv)))
  447. {
  448. switch ((char) (c))
  449. {
  450. case 'i':
  451. _setbits (flags, NVM_IDENTITY);
  452. break;
  453. case 'r':
  454. _setbits (flags, NVM_REVISION);
  455. break;
  456. case 'm':
  457. _setbits (flags, NVM_MANIFEST);
  458. break;
  459. case 's':
  460. _setbits (flags, NVM_SDRAM);
  461. break;
  462. case 'q':
  463. _setbits (flags, NVM_SILENCE);
  464. break;
  465. case 'v':
  466. _setbits (flags, NVM_VERBOSE);
  467. break;
  468. default:
  469. break;
  470. }
  471. }
  472. argc -= optind;
  473. argv += optind;
  474. while ((argc) && (* argv))
  475. {
  476. if (chknvm (* argv, flags))
  477. {
  478. state = 1;
  479. }
  480. else if (_allclr (flags, (NVM_VERBOSE | NVM_SILENCE | NVM_MANIFEST | NVM_REVISION | NVM_IDENTITY | NVM_SDRAM)))
  481. {
  482. printf ("file %s looks good\n", * argv);
  483. }
  484. argc--;
  485. argv++;
  486. }
  487. return (state);
  488. }