plchost.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * plchost.c -
  11. *
  12. * Contributor(s):
  13. * Charles Maier <cmaier@qca.qualcomm.com>
  14. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  15. *
  16. *--------------------------------------------------------------------*/
  17. /*====================================================================*
  18. * system header files;
  19. *--------------------------------------------------------------------*/
  20. #include <unistd.h>
  21. #include <stdlib.h>
  22. #include <limits.h>
  23. #include <string.h>
  24. #include <ctype.h>
  25. #include <memory.h>
  26. #include <signal.h>
  27. #include <errno.h>
  28. #include <sys/socket.h>
  29. /*====================================================================*
  30. * custom header files;
  31. *--------------------------------------------------------------------*/
  32. #include "../ether/channel.h"
  33. #include "../tools/getoptv.h"
  34. #include "../tools/putoptv.h"
  35. #include "../tools/memory.h"
  36. #include "../tools/number.h"
  37. #include "../tools/types.h"
  38. #include "../tools/flags.h"
  39. #include "../tools/files.h"
  40. #include "../tools/error.h"
  41. #include "../ram/nvram.h"
  42. #include "../ram/sdram.h"
  43. #include "../nvm/nvm.h"
  44. #include "../pib/pib.h"
  45. #include "../plc/plc.h"
  46. /*====================================================================*
  47. * custom source files;
  48. *--------------------------------------------------------------------*/
  49. #ifndef MAKEFILE
  50. #include "../plc/BootDevice2.c"
  51. #include "../plc/BootFirmware2.c"
  52. #include "../plc/BootParameters2.c"
  53. #include "../plc/chipset.c"
  54. #include "../plc/Confirm.c"
  55. #include "../plc/Devices.c"
  56. #include "../plc/Failure.c"
  57. #include "../plc/FlashDevice2.c"
  58. #include "../plc/FlashSoftloader.c"
  59. #include "../plc/FlashFirmware.c"
  60. #include "../plc/FlashParameters.c"
  61. #include "../plc/FlashNVM.c"
  62. #include "../plc/HostActionResponse.c"
  63. #include "../plc/InitDevice2.c"
  64. #include "../plc/ModuleRead.c"
  65. #include "../plc/ModuleSpec.c"
  66. #include "../plc/ModuleSession.c"
  67. #include "../plc/ModuleWrite.c"
  68. #include "../plc/ModuleCommit.c"
  69. #include "../plc/LocalDevices.c"
  70. #include "../plc/NVMSelect.c"
  71. #include "../plc/ReadFirmware2.c"
  72. #include "../plc/ReadParameters2.c"
  73. #include "../plc/Request.c"
  74. #include "../plc/ResetDevice.c"
  75. #include "../plc/ReadMME.c"
  76. #include "../plc/SendMME.c"
  77. #include "../plc/WaitForReset.c"
  78. #include "../plc/WaitForStart.c"
  79. #include "../plc/WaitForRestart.c"
  80. #include "../plc/WriteMEM.c"
  81. #include "../plc/WriteNVM.c"
  82. #include "../plc/WritePIB.c"
  83. #include "../plc/WriteExecuteFirmware2.c"
  84. #include "../plc/WriteExecuteParameters2.c"
  85. #include "../plc/WriteExecuteApplet2.c"
  86. #include "../plc/WriteFirmware2.c"
  87. #include "../plc/PLCSelect.c"
  88. #endif
  89. #ifndef MAKEFILE
  90. #include "../tools/getoptv.c"
  91. #include "../tools/putoptv.c"
  92. #include "../tools/version.c"
  93. #include "../tools/uintspec.c"
  94. #include "../tools/checkfilename.c"
  95. #include "../tools/hexdecode.c"
  96. #include "../tools/hexstring.c"
  97. #include "../tools/todigit.c"
  98. #include "../tools/hexdump.c"
  99. #include "../tools/checksum32.c"
  100. #include "../tools/fdchecksum32.c"
  101. #include "../tools/error.c"
  102. #include "../tools/strfbits.c"
  103. #include "../tools/typename.c"
  104. #endif
  105. #ifndef MAKEFILE
  106. #include "../ether/openchannel.c"
  107. #include "../ether/closechannel.c"
  108. #include "../ether/readpacket.c"
  109. #include "../ether/sendpacket.c"
  110. #include "../ether/channel.c"
  111. #endif
  112. #ifndef MAKEFILE
  113. #include "../mme/EthernetHeader.c"
  114. #include "../mme/QualcommHeader.c"
  115. #include "../mme/UnwantedMessage.c"
  116. #include "../mme/MMECode.c"
  117. #ifdef AR7x00
  118. #include "../mme/QualcommHeader1.c"
  119. #endif
  120. #endif
  121. #ifndef MAKEFILE
  122. #include "../key/keys.c"
  123. #endif
  124. #ifndef MAKEFILE
  125. #include "../nvm/panther_nvm_file.c"
  126. #endif
  127. #ifndef MAKEFILE
  128. #include "../pib/panther_pib_file.c"
  129. #include "../pib/panther_pib_peek.c"
  130. #endif
  131. /*====================================================================*
  132. * program constants;
  133. *--------------------------------------------------------------------*/
  134. #define NEWPIB "user.pib"
  135. #define NEWNVM "user.nvm"
  136. /*====================================================================*
  137. *
  138. * signed function (struct plc * plc);
  139. *
  140. * wait indefinitely for VS_HOST_ACTION messages; service the device
  141. * as directed; this function is for demonstration and experimentation
  142. * only; it will stop dead - like a bug! - on error;
  143. *
  144. * Contributor(s):
  145. * Charles Maier <cmaier@qca.qualcomm.com>
  146. *
  147. *--------------------------------------------------------------------*/
  148. signed function (struct plc * plc)
  149. {
  150. struct channel * channel = (struct channel *) (plc->channel);
  151. struct message * message = (struct message *) (plc->message);
  152. #ifndef __GNUC__
  153. #pragma pack (push,1)
  154. #endif
  155. struct __packed vs_host_action_ind
  156. {
  157. struct ethernet_hdr ethernet;
  158. struct qualcomm_hdr qualcomm;
  159. uint8_t MACTION;
  160. uint8_t MAJOR_VERSION;
  161. uint8_t MINOR_VERSION;
  162. }
  163. * indicate = (struct vs_host_action_ind *) (message);
  164. #ifndef __GNUC__
  165. #pragma pack (pop)
  166. #endif
  167. char firmware [PLC_VERSION_STRING];
  168. char const * FactoryNVM = plc->NVM.name;
  169. char const * FactoryPIB = plc->PIB.name;
  170. signed status;
  171. while (true)
  172. {
  173. status = ReadMME (plc, 0, (VS_HOST_ACTION | MMTYPE_IND));
  174. if (status < 0)
  175. {
  176. break;
  177. }
  178. if (status > 0)
  179. {
  180. signed action = indicate->MACTION;
  181. memcpy (channel->peer, indicate->ethernet.OSA, sizeof (channel->peer));
  182. if (HostActionResponse (plc))
  183. {
  184. return (-1);
  185. }
  186. if (action == 0x00)
  187. {
  188. if (BootDevice2 (plc))
  189. {
  190. return (-1);
  191. }
  192. if (WaitForStart (plc, firmware, sizeof (firmware)))
  193. {
  194. return (-1);
  195. }
  196. if (_anyset (plc->flags, PLC_FLASH_DEVICE))
  197. {
  198. FlashDevice2 (plc, (PLC_COMMIT_FORCE | PLC_COMMIT_NORESET | PLC_COMMIT_FACTPIB));
  199. }
  200. continue;
  201. }
  202. if (action == 0x01)
  203. {
  204. close (plc->NVM.file);
  205. if (ReadFirmware2 (plc))
  206. {
  207. return (-1);
  208. }
  209. if ((plc->NVM.file = open (plc->NVM.name = plc->nvm.name, O_BINARY | O_RDONLY)) == -1)
  210. {
  211. error (1, errno, "%s", plc->NVM.name);
  212. }
  213. continue;
  214. }
  215. if (action == 0x02)
  216. {
  217. close (plc->PIB.file);
  218. if (ReadParameters2 (plc))
  219. {
  220. return (-1);
  221. }
  222. if ((plc->PIB.file = open (plc->PIB.name = plc->pib.name, O_BINARY | O_RDONLY)) == -1)
  223. {
  224. error (1, errno, "%s", plc->PIB.name);
  225. }
  226. if (ResetDevice (plc))
  227. {
  228. return (-1);
  229. }
  230. continue;
  231. }
  232. if (action == 0x03)
  233. {
  234. close (plc->PIB.file);
  235. if (ReadParameters2 (plc))
  236. {
  237. return (-1);
  238. }
  239. if ((plc->PIB.file = open (plc->PIB.name = plc->pib.name, O_BINARY | O_RDONLY)) == -1)
  240. {
  241. error (1, errno, "%s", plc->PIB.name);
  242. }
  243. close (plc->NVM.file);
  244. if (ReadFirmware2 (plc))
  245. {
  246. return (-1);
  247. }
  248. if ((plc->NVM.file = open (plc->NVM.name = plc->nvm.name, O_BINARY | O_RDONLY)) == -1)
  249. {
  250. error (1, errno, "%s", plc->NVM.name);
  251. }
  252. if (ResetDevice (plc))
  253. {
  254. return (-1);
  255. }
  256. continue;
  257. }
  258. if (action == 0x04)
  259. {
  260. if (InitDevice2 (plc))
  261. {
  262. return (-1);
  263. }
  264. continue;
  265. }
  266. if (action == 0x05)
  267. {
  268. close (plc->NVM.file);
  269. if ((plc->NVM.file = open (plc->NVM.name = FactoryNVM, O_BINARY | O_RDONLY)) == -1)
  270. {
  271. error (1, errno, "%s", plc->NVM.name);
  272. }
  273. close (plc->PIB.file);
  274. if ((plc->PIB.file = open (plc->PIB.name = FactoryPIB, O_BINARY | O_RDONLY)) == -1)
  275. {
  276. error (1, errno, "%s", plc->PIB.name);
  277. }
  278. if (ResetDevice (plc))
  279. {
  280. return (-1);
  281. }
  282. continue;
  283. }
  284. if (action == 0x06)
  285. {
  286. close (plc->PIB.file);
  287. if (ReadParameters2 (plc))
  288. {
  289. return (-1);
  290. }
  291. if ((plc->PIB.file = open (plc->PIB.name = plc->pib.name, O_BINARY | O_RDONLY)) == -1)
  292. {
  293. error (1, errno, "%s", plc->PIB.name);
  294. }
  295. continue;
  296. }
  297. if (action == 0x07)
  298. {
  299. error (0, 0, "Host has rebooted.");
  300. continue;
  301. }
  302. error (0, ENOSYS, "Host Action 0x%02X", action);
  303. }
  304. }
  305. return (0);
  306. }
  307. /*====================================================================*
  308. *
  309. * int main (int argc, char const * argv[]);
  310. *
  311. * parse command line, populate plc structure and perform selected
  312. * operations; show help summary when asked; see getoptv and putoptv
  313. * to understand command line parsing and help summary display; see
  314. * plc.h for the definition of struct plc;
  315. *
  316. *--------------------------------------------------------------------*/
  317. int main (int argc, char const * argv [])
  318. {
  319. extern struct channel channel;
  320. extern void terminate (signo_t);
  321. static char const * optv [] =
  322. {
  323. "Fi:n:N:p:P:qS:t:vw:x",
  324. "-N file -P file [-S file]",
  325. "Qualcomm Atheros PLC Host Daemon",
  326. "F [F]\tflash [force] non-volatile memory",
  327. #if defined (WINPCAP) || defined (LIBPCAP)
  328. "i n\thost interface is (n) [" LITERAL (CHANNEL_ETHNUMBER) "]",
  329. #else
  330. "i s\thost interface is (s) [" LITERAL (CHANNEL_ETHDEVICE) "]",
  331. #endif
  332. "n f\tuser firmware file is (f)",
  333. "N f\tfirmware file is (f)",
  334. "p f\tuser parameter file is (f)",
  335. "P f\tparameter file is (f)",
  336. "q\tquiet mode",
  337. "S f\tsoftloader file is (f)",
  338. "t n\tread timeout is (n) milliseconds [" LITERAL (CHANNEL_TIMEOUT) "]",
  339. "v\tverbose mode",
  340. "w n\twakeup every (n) seconds [" LITERAL (PLC_LONGTIME) "]",
  341. "x\texit on error",
  342. (char const *) (0)
  343. };
  344. #include "../plc/plc.c"
  345. signed c;
  346. if (getenv (PLCDEVICE))
  347. {
  348. #if defined (WINPCAP) || defined (LIBPCAP)
  349. channel.ifindex = atoi (getenv (PLCDEVICE));
  350. #else
  351. channel.ifname = strdup (getenv (PLCDEVICE));
  352. #endif
  353. }
  354. optind = 1;
  355. plc.timer = PLC_LONGTIME;
  356. while (~ (c = getoptv (argc, argv, optv)))
  357. {
  358. switch (c)
  359. {
  360. case 'd':
  361. _setbits (plc.flags, PLC_DAEMON);
  362. break;
  363. case 'F':
  364. _setbits (plc.module, (VS_MODULE_MAC | VS_MODULE_PIB));
  365. if (_anyset (plc.flags, PLC_FLASH_DEVICE))
  366. {
  367. _setbits (plc.module, VS_MODULE_FORCE);
  368. }
  369. _setbits (plc.flags, PLC_FLASH_DEVICE);
  370. break;
  371. case 'i':
  372. #if defined (WINPCAP) || defined (LIBPCAP)
  373. channel.ifindex = atoi (optarg);
  374. #else
  375. channel.ifname = optarg;
  376. #endif
  377. break;
  378. case 'N':
  379. if (! checkfilename (optarg))
  380. {
  381. error (1, EINVAL, "%s", optarg);
  382. }
  383. if ((plc.NVM.file = open (plc.NVM.name = optarg, O_BINARY | O_RDONLY)) == -1)
  384. {
  385. error (1, errno, "%s", plc.NVM.name);
  386. }
  387. if (panther_nvm_file (& plc.NVM))
  388. {
  389. error (1, errno, "Bad panther image file: %s", plc.NVM.name);
  390. }
  391. _setbits (plc.flags, PLC_WRITE_MAC);
  392. break;
  393. case 'n':
  394. if (! checkfilename (optarg))
  395. {
  396. error (1, EINVAL, "%s", optarg);
  397. }
  398. if ((plc.nvm.file = open (plc.nvm.name = optarg, O_BINARY | O_CREAT | O_RDWR | O_TRUNC, FILE_FILEMODE)) == -1)
  399. {
  400. error (1, errno, "%s", plc.nvm.name);
  401. }
  402. break;
  403. case 'P':
  404. if (! checkfilename (optarg))
  405. {
  406. error (1, EINVAL, "%s", optarg);
  407. }
  408. if ((plc.PIB.file = open (plc.PIB.name = optarg, O_BINARY | O_RDONLY)) == -1)
  409. {
  410. error (1, errno, "%s", plc.PIB.name);
  411. }
  412. if (panther_pib_file (& plc.PIB))
  413. {
  414. error (1, errno, "Bad panther parameter file: %s", plc.PIB.name);
  415. }
  416. _setbits (plc.flags, PLC_WRITE_PIB);
  417. break;
  418. case 'p':
  419. if (! checkfilename (optarg))
  420. {
  421. error (1, EINVAL, "%s", optarg);
  422. }
  423. if ((plc.pib.file = open (plc.pib.name = optarg, O_BINARY | O_CREAT | O_RDWR | O_TRUNC, FILE_FILEMODE)) == -1)
  424. {
  425. error (1, errno, "%s", plc.pib.name);
  426. }
  427. break;
  428. case 'q':
  429. _setbits (channel.flags, CHANNEL_SILENCE);
  430. _setbits (plc.flags, PLC_SILENCE);
  431. break;
  432. case 'S':
  433. if (! checkfilename (optarg))
  434. {
  435. error (1, EINVAL, "%s", optarg);
  436. }
  437. if ((plc.SFT.file = open (plc.SFT.name = optarg, O_BINARY | O_RDONLY)) == -1)
  438. {
  439. error (1, errno, "%s", plc.SFT.name);
  440. }
  441. if (panther_nvm_file (& plc.SFT))
  442. {
  443. error (1, errno, "Bad panther image file: %s", plc.SFT.name);
  444. }
  445. break;
  446. case 't':
  447. channel.timeout = (signed) (uintspec (optarg, 0, UINT_MAX));
  448. break;
  449. case 'v':
  450. _setbits (channel.flags, CHANNEL_VERBOSE);
  451. _setbits (plc.flags, PLC_VERBOSE);
  452. break;
  453. case 'w':
  454. plc.timer = (unsigned) (uintspec (optarg, 1, 3600));
  455. break;
  456. case 'x':
  457. _setbits (plc.flags, PLC_BAILOUT);
  458. break;
  459. default:
  460. break;
  461. }
  462. }
  463. argc -= optind;
  464. argv += optind;
  465. if (argc)
  466. {
  467. error (1, ENOTSUP, ERROR_TOOMANY);
  468. }
  469. if (plc.NVM.file == -1)
  470. {
  471. error (1, ECANCELED, "No host NVM file named");
  472. }
  473. if (plc.PIB.file == -1)
  474. {
  475. error (1, ECANCELED, "No host PIB file named");
  476. }
  477. if (plc.SFT.file == -1)
  478. {
  479. if (_anyset (plc.flags, PLC_FLASH_DEVICE))
  480. {
  481. error (1, ECANCELED, "No Softloader file named");
  482. }
  483. }
  484. if (plc.nvm.file == -1)
  485. {
  486. if ((plc.nvm.file = open (plc.nvm.name = NEWNVM, O_BINARY | O_CREAT | O_RDWR | O_TRUNC, FILE_FILEMODE)) == -1)
  487. {
  488. error (1, errno, "%s", plc.nvm.name);
  489. }
  490. }
  491. if (plc.pib.file == -1)
  492. {
  493. if ((plc.pib.file = open (plc.pib.name = NEWPIB, O_BINARY | O_CREAT | O_RDWR | O_TRUNC, FILE_FILEMODE)) == -1)
  494. {
  495. error (1, errno, "%s", plc.pib.name);
  496. }
  497. }
  498. openchannel (& channel);
  499. if (! (plc.message = malloc (sizeof (* plc.message))))
  500. {
  501. error (1, errno, PLC_NOMEMORY);
  502. }
  503. function (& plc);
  504. free (plc.message);
  505. closechannel (& channel);
  506. exit (0);
  507. }