int6ktest.c 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * plctest.c - Qualcomm Atheros Test Applet Loader;
  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. #include <limits.h>
  24. #include <errno.h>
  25. /*====================================================================*
  26. * custom header files;
  27. *--------------------------------------------------------------------*/
  28. #include "../tools/getoptv.h"
  29. #include "../tools/putoptv.h"
  30. #include "../tools/memory.h"
  31. #include "../tools/number.h"
  32. #include "../tools/types.h"
  33. #include "../tools/flags.h"
  34. #include "../tools/files.h"
  35. #include "../tools/error.h"
  36. #include "../ether/channel.h"
  37. #include "../ram/nvram.h"
  38. #include "../ram/sdram.h"
  39. #include "../nvm/nvm.h"
  40. #include "../pib/pib.h"
  41. #include "../plc/plc.h"
  42. /*====================================================================*
  43. * custom source files;
  44. *--------------------------------------------------------------------*/
  45. #ifndef MAKEFILE
  46. #include "../plc/chipset.c"
  47. #include "../plc/Confirm.c"
  48. #include "../plc/Devices.c"
  49. #include "../plc/Display.c"
  50. #include "../plc/Failure.c"
  51. #include "../plc/Request.c"
  52. #include "../plc/SendMME.c"
  53. #include "../plc/WriteFirmware1.c"
  54. #include "../plc/StartFirmware1.c"
  55. #include "../plc/WaitForStart.c"
  56. #endif
  57. #ifndef MAKEFILE
  58. #include "../tools/getoptv.c"
  59. #include "../tools/putoptv.c"
  60. #include "../tools/version.c"
  61. #include "../tools/uintspec.c"
  62. #include "../tools/checkfilename.c"
  63. #include "../tools/hexdecode.c"
  64. #include "../tools/todigit.c"
  65. #include "../tools/hexdump.c"
  66. #include "../tools/checksum32.c"
  67. #include "../tools/fdchecksum32.c"
  68. #include "../tools/error.c"
  69. #include "../tools/strfbits.c"
  70. #include "../tools/typename.c"
  71. #endif
  72. #ifndef MAKEFILE
  73. #include "../ether/openchannel.c"
  74. #include "../ether/closechannel.c"
  75. #include "../ether/readpacket.c"
  76. #include "../ether/sendpacket.c"
  77. #include "../ether/channel.c"
  78. #endif
  79. #ifndef MAKEFILE
  80. #include "../nvm/lightning_nvm_file.c"
  81. #endif
  82. #ifndef MAKEFILE
  83. #include "../mme/MMECode.c"
  84. #include "../mme/EthernetHeader.c"
  85. #include "../mme/QualcommHeader.c"
  86. #include "../mme/UnwantedMessage.c"
  87. #endif
  88. /*====================================================================*
  89. *
  90. * signed ReadMME (struct plc * plc, uint8_t MMV, uint16_t MMTYPE);
  91. *
  92. * plc.h
  93. *
  94. * this is a custom version of ReadMME that intercepts VS_ARPC
  95. * messages and print them on the console in human readable format;
  96. *
  97. * this implementation was an early attempt the does not take full
  98. * advantage of the VS_ARPC structure and so messages are printed
  99. * in a format that is incompatible with the Windows Device Manager;
  100. * program amptest corrects this problem by calling ARPCPrint;
  101. *
  102. * this function is maintained for legacy purposes only and will be
  103. * deleted at some point in the future;
  104. *
  105. * Contributor(s):
  106. * Charles Maier <cmaier@qca.qualcomm.com>
  107. *
  108. *--------------------------------------------------------------------*/
  109. signed ReadMME (struct plc * plc, uint8_t MMV, uint16_t MMTYPE)
  110. {
  111. struct channel * channel = (struct channel *) (plc->channel);
  112. struct message * message = (struct message *) (plc->message);
  113. #ifndef __GNUC__
  114. #pragma pack (push,1)
  115. #endif
  116. static qualcomm_hdr header_arpc =
  117. {
  118. 0,
  119. VS_ARPC | MMTYPE_IND,
  120. {
  121. 0x00,
  122. 0xB0,
  123. 0x52
  124. }
  125. };
  126. struct __packed vs_arpc_ind
  127. {
  128. struct ethernet_hdr ethernet;
  129. struct qualcomm_hdr qualcomm;
  130. uint16_t RDATALENGTH;
  131. uint8_t RDATAOFFSET;
  132. uint8_t RDATA [1];
  133. }
  134. * indicate = (struct vs_arpc_ind *) (plc->message);
  135. #ifndef __GNUC__
  136. #pragma pack (pop)
  137. #endif
  138. while ((plc->packetsize = readpacket (channel, message, sizeof (* message))) > 0)
  139. {
  140. #if 1
  141. /*
  142. * Normally this block of code would be replaced by function FirmwareMessage () but due to an
  143. * ARPC implmenetation error on early diagnostic applets we need to use this code instead;
  144. */
  145. header_arpc.MMTYPE = HTOLE16 (header_arpc.MMTYPE);
  146. if (! memcmp (& indicate->qualcomm, & header_arpc, sizeof (header_arpc)))
  147. {
  148. indicate->RDATA [indicate->RDATALENGTH - indicate->RDATAOFFSET] = (char) (0);
  149. Display (plc, "0x%02X %s", indicate->RDATA [0], & indicate->RDATA [5]);
  150. continue;
  151. }
  152. #endif
  153. if (UnwantedMessage (message, plc->packetsize, 0, MMTYPE))
  154. {
  155. continue;
  156. }
  157. break;
  158. }
  159. return (plc->packetsize);
  160. }
  161. /*====================================================================*
  162. *
  163. * signed sequence (struct plc * plc);
  164. *
  165. * plc.h
  166. *
  167. * sequentially download and execute all modules found in a .nvm
  168. * file regardless of module type; users must ensure that modules
  169. * are executable or results may be unpredictable;
  170. *
  171. * DO NOT USE THIS FUNCTION TO DOWNBOOT AND EXECUTE RUNTIME FIRMWARE;
  172. *
  173. * Contributor(s):
  174. * Charles Maier <cmaier@qca.qualcomm.com>
  175. *
  176. *--------------------------------------------------------------------*/
  177. static signed sequence (struct plc * plc)
  178. {
  179. struct lightning_nvm_header nvm_header;
  180. unsigned module = 0;
  181. do
  182. {
  183. if (read (plc->NVM.file, & nvm_header, sizeof (nvm_header)) != sizeof (nvm_header))
  184. {
  185. if (_allclr (plc->flags, PLC_SILENCE))
  186. {
  187. error (0, errno, NVM_HDR_CANTREAD, plc->NVM.name, module);
  188. }
  189. return (-1);
  190. }
  191. if (LE32TOH (nvm_header.HEADERVERSION) != 0x60000000)
  192. {
  193. if (_allclr (plc->flags, PLC_SILENCE))
  194. {
  195. error (0, 0, NVM_HDR_VERSION, plc->NVM.name, module);
  196. }
  197. return (-1);
  198. }
  199. if (checksum32 (& nvm_header, sizeof (nvm_header), 0))
  200. {
  201. if (_allclr (plc->flags, PLC_SILENCE))
  202. {
  203. error (0, 0, NVM_HDR_CHECKSUM, plc->NVM.name, module);
  204. }
  205. return (-1);
  206. }
  207. if (WriteFirmware1 (plc, module, & nvm_header))
  208. {
  209. return (-1);
  210. }
  211. if (StartFirmware1 (plc, module, & nvm_header))
  212. {
  213. return (-1);
  214. }
  215. if (_anyset (plc->flags, PLC_QUICK_FLASH))
  216. {
  217. break;
  218. }
  219. while (! ReadMME (plc, 0, (VS_HOST_ACTION | MMTYPE_IND)));
  220. module++;
  221. }
  222. while (nvm_header.NEXTHEADER);
  223. return (0);
  224. }
  225. /*====================================================================*
  226. *
  227. * void function (struct plc * plc, int argc, char const * argv);
  228. *
  229. * execute all applets in each file on the command line; return no
  230. * value; this function may be called repeatedly;
  231. *
  232. *--------------------------------------------------------------------*/
  233. static void function (struct plc * plc, int argc, char const * argv [])
  234. {
  235. while ((argc) && (* argv))
  236. {
  237. if ((plc->NVM.file = open (plc->NVM.name = * argv, O_BINARY | O_RDONLY)) == -1)
  238. {
  239. error (0, errno, "%s", plc->NVM.name);
  240. }
  241. else if (lightning_nvm_file (& plc->NVM))
  242. {
  243. error (0, errno, "Won't load %s", plc->NVM.name);
  244. }
  245. else if (sequence (plc))
  246. {
  247. error (0, errno, "Abandoning %s", plc->NVM.name);
  248. }
  249. close (plc->NVM.file);
  250. argc--;
  251. argv++;
  252. }
  253. return;
  254. }
  255. /*====================================================================*
  256. *
  257. * int main (int argc, char const * argv[]);
  258. *
  259. * parse command line, populate plc structure and perform selected
  260. * operations; show help summary when asked; see getoptv and putoptv
  261. * to understand command line parsing and help summary display; see
  262. * plc.h for the definition of struct plc;
  263. *
  264. * the default interface is eth1 because most people use eth0 as
  265. * their principle network connection; you can specify another
  266. * interface with -i or define environment string PLC to make
  267. * that the default interface and save typing;
  268. *
  269. *--------------------------------------------------------------------*/
  270. int main (int argc, char const * argv [])
  271. {
  272. extern struct channel channel;
  273. char firmware [PLC_VERSION_STRING];
  274. static char const * optv [] =
  275. {
  276. "c:ei:lp:qt:vx",
  277. "file [file] [...]",
  278. "Qualcomm Atheros Test Applet Loader",
  279. "e\tredirect stderr to stdout",
  280. #if defined (WINPCAP) || defined (LIBPCAP)
  281. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  282. #else
  283. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  284. #endif
  285. "l\tloop until program is terminated",
  286. "q\tquiet mode",
  287. "t n\tread timeout is (n) milliseconds [" LITERAL (CHANNEL_TIMEOUT) "]",
  288. "v\tverbose mode",
  289. "x\texit on error",
  290. (char const *) (0)
  291. };
  292. #include "../plc/plc.c"
  293. signed c;
  294. if (getenv (PLCDEVICE))
  295. {
  296. #if defined (WINPCAP) || defined (LIBPCAP)
  297. channel.ifindex = atoi (getenv (PLCDEVICE));
  298. #else
  299. channel.ifname = strdup (getenv (PLCDEVICE));
  300. #endif
  301. }
  302. optind = 1;
  303. while (~ (c = getoptv (argc, argv, optv)))
  304. {
  305. switch (c)
  306. {
  307. case 'e':
  308. dup2 (STDOUT_FILENO, STDERR_FILENO);
  309. break;
  310. case 'i':
  311. #if defined (WINPCAP) || defined (LIBPCAP)
  312. channel.ifindex = atoi (optarg);
  313. #else
  314. channel.ifname = optarg;
  315. #endif
  316. break;
  317. case 'l':
  318. _setbits (plc.flags, PLC_FOREVER);
  319. break;
  320. case 'q':
  321. _setbits (channel.flags, CHANNEL_SILENCE);
  322. _setbits (plc.flags, PLC_SILENCE);
  323. break;
  324. case 't':
  325. channel.timeout = (signed) (uintspec (optarg, 0, UINT_MAX));
  326. break;
  327. case 'v':
  328. _setbits (channel.flags, CHANNEL_VERBOSE);
  329. _setbits (plc.flags, PLC_VERBOSE);
  330. break;
  331. case 'x':
  332. _setbits (plc.flags, PLC_BAILOUT);
  333. break;
  334. default:
  335. break;
  336. }
  337. }
  338. argc -= optind;
  339. argv += optind;
  340. openchannel (& channel);
  341. if (! (plc.message = malloc (sizeof (* plc.message))))
  342. {
  343. error (1, errno, PLC_NOMEMORY);
  344. }
  345. if (WaitForStart (& plc, firmware, sizeof (firmware)))
  346. {
  347. error (1, ECANCELED, PLC_NODETECT);
  348. }
  349. if (strcmp (firmware, "BootLoader"))
  350. {
  351. error (1, ECANCELED, "BootLoader must be running");
  352. }
  353. do
  354. {
  355. function (& plc, argc, argv);
  356. }
  357. while (_anyset (plc.flags, PLC_FOREVER));
  358. free (plc.message);
  359. closechannel (& channel);
  360. return (plc.state);
  361. }