int6kf.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * int6kf.c - Qualcomm Atheros Powerline Device Flash Utility;
  11. *
  12. * this program sends and receives raw ethernet frames and so needs
  13. * root priviledges; if you install it using "chmod 555" and "chown
  14. * root:root" then you must login as root to run it; otherwise, you
  15. * can install it using "chmod 4555" and "chown root:root" so that
  16. * anyone can run it; the program will refuse to run until you get
  17. * things right;
  18. *
  19. * Contributor(s):
  20. * Charles Maier <cmaier@qca.qualcomm.com>
  21. *
  22. *--------------------------------------------------------------------*/
  23. /*====================================================================*
  24. * system header files;
  25. *--------------------------------------------------------------------*/
  26. #include <unistd.h>
  27. #include <stdlib.h>
  28. #include <limits.h>
  29. #include <string.h>
  30. #include <ctype.h>
  31. /*====================================================================*
  32. * custom header files;
  33. *--------------------------------------------------------------------*/
  34. #include "../tools/getoptv.h"
  35. #include "../tools/putoptv.h"
  36. #include "../tools/memory.h"
  37. #include "../tools/number.h"
  38. #include "../tools/types.h"
  39. #include "../tools/flags.h"
  40. #include "../tools/files.h"
  41. #include "../tools/error.h"
  42. #include "../ram/nvram.h"
  43. #include "../ram/sdram.h"
  44. #include "../nvm/nvm.h"
  45. #include "../plc/plc.h"
  46. #include "../pib/pib.h"
  47. /*====================================================================*
  48. * custom source files;
  49. *--------------------------------------------------------------------*/
  50. #ifndef MAKEFILE
  51. #include "../plc/chipset.c"
  52. #include "../plc/Confirm.c"
  53. #include "../plc/Devices.c"
  54. #include "../plc/Display.c"
  55. #include "../plc/Failure.c"
  56. #include "../plc/FlashNVM.c"
  57. #include "../plc/ReadMME.c"
  58. #include "../plc/Request.c"
  59. #include "../plc/SendMME.c"
  60. #include "../plc/StartDevice1.c"
  61. #include "../plc/FlashDevice1.c"
  62. #include "../plc/StartFirmware1.c"
  63. #include "../plc/WriteFirmware1.c"
  64. #include "../plc/WriteCFG.c"
  65. #include "../plc/WriteMEM.c"
  66. #include "../plc/WriteNVM.c"
  67. #include "../plc/WritePIB.c"
  68. #include "../plc/WaitForReset.c"
  69. #include "../plc/WaitForRestart.c"
  70. #include "../plc/WaitForStart.c"
  71. #endif
  72. #ifndef MAKEFILE
  73. #include "../tools/getoptv.c"
  74. #include "../tools/putoptv.c"
  75. #include "../tools/version.c"
  76. #include "../tools/uintspec.c"
  77. #include "../tools/checkfilename.c"
  78. #include "../tools/hexdecode.c"
  79. #include "../tools/hexstring.c"
  80. #include "../tools/todigit.c"
  81. #include "../tools/hexdump.c"
  82. #include "../tools/checksum32.c"
  83. #include "../tools/fdchecksum32.c"
  84. #include "../tools/typename.c"
  85. #include "../tools/strfbits.c"
  86. #include "../tools/error.c"
  87. #endif
  88. #ifndef MAKEFILE
  89. #include "../ram/sdramfile.c"
  90. #include "../ram/sdrampeek.c"
  91. #endif
  92. #ifndef MAKEFILE
  93. #include "../nvm/nvm.c"
  94. #include "../nvm/lightning_nvm_file.c"
  95. #include "../nvm/nvmpeek1.c"
  96. #endif
  97. #ifndef MAKEFILE
  98. #include "../pib/lightning_pib_file.c"
  99. #include "../pib/lightning_pib_peek.c"
  100. #endif
  101. #ifndef MAKEFILE
  102. #include "../ether/openchannel.c"
  103. #include "../ether/closechannel.c"
  104. #include "../ether/readpacket.c"
  105. #include "../ether/sendpacket.c"
  106. #include "../ether/channel.c"
  107. #endif
  108. #ifndef MAKEFILE
  109. #include "../mme/EthernetHeader.c"
  110. #include "../mme/HomePlugHeader.c"
  111. #include "../mme/QualcommHeader.c"
  112. #include "../mme/UnwantedMessage.c"
  113. #include "../mme/MMECode.c"
  114. #endif
  115. #ifndef MAKEFILE
  116. #include "../key/keys.c"
  117. #endif
  118. /*====================================================================*
  119. *
  120. * int main (int argc, const char * argv[]);
  121. *
  122. * parse command line, populate int6k structure and perform selected
  123. * operations; show help summary when asked; see getoptv and putoptv
  124. * to understand command line parsing and help summary display; see
  125. * int6k.h for the definition of struct int6k;
  126. *
  127. * the default interface is eth1 because most people use eth0 as
  128. * their principle network connection; you can specify another
  129. * interface with -i or define environment string PLC to make
  130. * that the default interface and save typing;
  131. *
  132. *--------------------------------------------------------------------*/
  133. int main (int argc, const char * argv [])
  134. {
  135. extern struct channel channel;
  136. static const char * optv [] =
  137. {
  138. "C:i:eFN:p:P:qt:vx",
  139. "-C file -P file -N file",
  140. "Qualcomm Atheros Powerline Device Flash Utility for INT6300",
  141. "C f\twrite CFG file to device using VS_SET_SDRAM",
  142. "e\tredirect stderr messages to stdout",
  143. #if defined (WINPCAP) || defined (LIBPCAP)
  144. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  145. #else
  146. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  147. #endif
  148. "F[F]\tflash [force] NVRAM after firmware start using VS_MOD_NVM",
  149. "N f\twrite NVM file to device using VS_WR_MEM",
  150. "P f\twrite PIB file to device using VS_WR_MEM",
  151. "q\tquiet mode",
  152. #if defined (WINPCAP) || defined (LIBPCAP)
  153. "t n\tread capture time is (n) milliseconds [50]",
  154. #else
  155. "t n\tread timeout is (n) milliseconds [50]",
  156. #endif
  157. "v\tverbose mode",
  158. "x\texit on error",
  159. (const char *) (0)
  160. };
  161. #include "../plc/plc.c"
  162. char firmware [PLC_VERSION_STRING];
  163. signed c;
  164. if (getenv (PLCDEVICE))
  165. {
  166. #if defined (WINPCAP) || defined (LIBPCAP)
  167. channel.ifindex = atoi (getenv (PLCDEVICE));
  168. #else
  169. channel.ifname = strdup (getenv (PLCDEVICE));
  170. #endif
  171. }
  172. optind = 1;
  173. opterr = 1;
  174. while (~ (c = getoptv (argc, argv, optv)))
  175. {
  176. switch ((char) (c))
  177. {
  178. case 'C':
  179. if (! checkfilename (optarg))
  180. {
  181. error (1, EINVAL, "%s", optarg);
  182. }
  183. if ((plc.CFG.file = open (optarg, O_BINARY | O_RDONLY)) == -1)
  184. {
  185. error (1, errno, "%s", optarg);
  186. }
  187. if (sdramfile (plc.CFG.file, optarg, plc.flags))
  188. {
  189. error (1, ECANCELED, "CFG file %s is corrupt", optarg);
  190. }
  191. _setbits (plc.flags, PLC_SDRAM_CONFIG);
  192. plc.CFG.name = optarg;
  193. break;
  194. case 'e':
  195. dup2 (STDOUT_FILENO, STDERR_FILENO);
  196. break;
  197. case 'i':
  198. #if defined (WINPCAP) || defined (LIBPCAP)
  199. channel.ifindex = atoi (optarg);
  200. #else
  201. channel.ifname = optarg;
  202. #endif
  203. break;
  204. case 'F':
  205. _setbits (plc.module, PLC_MODULE_NVM_PIB);
  206. if (_anyset (plc.flags, PLC_FLASH_DEVICE))
  207. {
  208. _setbits (plc.module, VS_MODULE_FORCE);
  209. }
  210. _setbits (plc.flags, PLC_FLASH_DEVICE);
  211. break;
  212. case 'N':
  213. if (! checkfilename (optarg))
  214. {
  215. error (1, EINVAL, "%s", optarg);
  216. }
  217. if ((plc.NVM.file = open (optarg, O_BINARY | O_RDONLY)) == -1)
  218. {
  219. error (1, errno, "%s", optarg);
  220. }
  221. plc.NVM.name = optarg;
  222. if (lightning_nvm_file (& plc.NVM))
  223. {
  224. error (1, errno, "Bad NVM file: %s", plc.NVM.name);
  225. }
  226. _setbits (plc.flags, PLC_WRITE_MAC);
  227. break;
  228. case 'P':
  229. if (! checkfilename (optarg))
  230. {
  231. error (1, EINVAL, "%s", optarg);
  232. }
  233. if ((plc.PIB.file = open (optarg, O_BINARY | O_RDONLY)) == -1)
  234. {
  235. error (1, errno, "%s", optarg);
  236. }
  237. plc.PIB.name = optarg;
  238. if (lightning_pib_file (& plc.PIB))
  239. {
  240. error (1, errno, "Bad PIB file: %s", plc.PIB.name);
  241. }
  242. _setbits (plc.flags, PLC_WRITE_PIB);
  243. break;
  244. case 'q':
  245. _setbits (channel.flags, CHANNEL_SILENCE);
  246. _setbits (plc.flags, PLC_SILENCE);
  247. break;
  248. case 't':
  249. channel.timeout = (signed) (uintspec (optarg, 0, UINT_MAX));
  250. break;
  251. case 'v':
  252. _setbits (channel.flags, CHANNEL_VERBOSE);
  253. _setbits (plc.flags, PLC_VERBOSE);
  254. break;
  255. case 'x':
  256. _setbits (plc.flags, PLC_BAILOUT);
  257. break;
  258. default:
  259. break;
  260. }
  261. }
  262. argc -= optind;
  263. argv += optind;
  264. if (argc)
  265. {
  266. error (1, ECANCELED, "Too many arguments");
  267. }
  268. if (plc.CFG.file == -1)
  269. {
  270. error (1, ECANCELED, "No CFG file specified");
  271. }
  272. if (plc.PIB.file == -1)
  273. {
  274. error (1, ECANCELED, "No PIB file specified");
  275. }
  276. if (plc.NVM.file == -1)
  277. {
  278. error (1, ECANCELED, "No NVM file specified");
  279. }
  280. openchannel (& channel);
  281. if (! (plc.message = malloc (sizeof (struct message))))
  282. {
  283. error (1, errno, PLC_NOMEMORY);
  284. }
  285. if (WaitForStart (& plc, firmware, sizeof (firmware)))
  286. {
  287. Failure (& plc, PLC_NODETECT);
  288. return (-1);
  289. }
  290. if (plc.hardwareID > CHIPSET_INT6300)
  291. {
  292. Failure (& plc, "Device must be %s or earlier; try using int6kboot.", chipsetname (CHIPSET_INT6300));
  293. return (-1);
  294. }
  295. if (strcmp (firmware, "BootLoader"))
  296. {
  297. Failure (& plc, "Bootloader must be running");
  298. return (-1);
  299. }
  300. if (! StartDevice1 (& plc))
  301. {
  302. if (_anyset (plc.flags, PLC_FLASH_DEVICE))
  303. {
  304. FlashDevice1 (& plc);
  305. }
  306. }
  307. free (plc.message);
  308. closechannel (& channel);
  309. exit (0);
  310. }