CPLParameters.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * CPLParameters.cpp - CPLParameters class definition;
  11. *
  12. * read/write a parameter block image from/to a device/file;
  13. *
  14. * Contributor(s):
  15. * Charles Maier <charles.maier@intellon.com>
  16. *
  17. *--------------------------------------------------------------------*/
  18. #ifndef CPLPARAMETERS_SOURCE
  19. #define CPLPARAMETERS_SOURCE
  20. /*====================================================================*
  21. * system header files;
  22. *--------------------------------------------------------------------*/
  23. #include <cstring>
  24. #include <fstream>
  25. /*====================================================================*
  26. * custom header files;
  27. *--------------------------------------------------------------------*/
  28. #include "../classes/CPLParameters.hpp"
  29. #include "../classes/ohomeplug.hpp"
  30. #include "../classes/omemory.hpp"
  31. #include "../classes/oerror.hpp"
  32. #include "../tools/endian.h"
  33. /*====================================================================*
  34. *
  35. * void * Data () const;
  36. *
  37. * return parameter block address as a void pointer; this property
  38. * can be used to perform parameter block operations that have not
  39. * been implemented by this class;
  40. *
  41. *--------------------------------------------------------------------*/
  42. void * CPLParameters::Data () const
  43. {
  44. return ((void *)(this->mbuffer));
  45. }
  46. /*====================================================================*
  47. *
  48. * size_t Size () const;
  49. *
  50. * return parameter block extent (length) in bytes;
  51. *
  52. *--------------------------------------------------------------------*/
  53. size_t CPLParameters::Size () const
  54. {
  55. return (this->mextent);
  56. }
  57. /*====================================================================*
  58. *
  59. * bool IsValid () const;
  60. *
  61. * return true of the parameter block is valid; check the length
  62. * and checksum as the minimum;
  63. *
  64. * this method may be expanded to include other parameter block
  65. * validity tests;
  66. *
  67. *--------------------------------------------------------------------*/
  68. bool CPLParameters::IsValid () const
  69. {
  70. if ((!this->mbuffer) || (!this->mextent))
  71. {
  72. oerror::error (0, ECANCELED, "No Parameter Block Available");
  73. return (false);
  74. }
  75. if (this->mextent != LE16TOH (this->mheader->PIBLENGTH))
  76. {
  77. oerror::error (0, ECANCELED, "Bad Parameter Block Length");
  78. return (false);
  79. }
  80. if (omemory::checksum32 (this->mbuffer, this->mextent, 0))
  81. {
  82. oerror::error (0, ECANCELED, "Bad Parameter Block Checksum");
  83. return (false);
  84. }
  85. return (true);
  86. }
  87. /*====================================================================*
  88. *
  89. * void CPLParameters:: SetCAPPriority (int CAPpriority);
  90. *
  91. * setCAPpriority to decide CAP for unicast traffic
  92. *
  93. *--------------------------------------------------------------------*/
  94. CPLParameters & CPLParameters::SetCAPPriority (int CAPpriority)
  95. {
  96. std::cout << "\nCurrent value of unicast CAP is : " << (int) *(mbuffer+0x020C);
  97. this->mbuffer [0x020C] = CAPpriority;
  98. return (*this);
  99. }
  100. int CPLParameters::Write_QoS_block (int selectPIB)
  101. { int QoSoffset = 0x020C;
  102. int QoSoffset2 = 0x1EB8;
  103. std::ifstream stream;
  104. switch (selectPIB)
  105. { case (0) :
  106. stream.open ("pib0.pib", std::ios::binary);
  107. break;
  108. case (1) :
  109. stream.open ("pib1.pib", std::ios::binary);
  110. break;
  111. case (2) :
  112. stream.open ("pib2.pib", std::ios::binary);
  113. break;
  114. default :
  115. oerror::error (0, ECANCELED, "Selected PIB doesn't exist");
  116. return (-1);
  117. }
  118. if (stream.good ())
  119. {
  120. // stream.seekg (0, std::ios::end);
  121. // this->mextent = 0x0990;
  122. stream.seekg (QoSoffset, std::ios::beg);
  123. // this->mbuffer = new uint8_t [this->mextent];
  124. stream.read ((char *)((this->mbuffer)+QoSoffset), (1925));
  125. stream.seekg (QoSoffset2, std::ios::beg);
  126. stream.read ((char *)((this->mbuffer)+QoSoffset2), (65));
  127. }
  128. else
  129. { oerror::error (0, ECANCELED, "Bad Stream,can't read");
  130. return (-1);
  131. }
  132. stream.close ();
  133. this->mheader = (CPLParameters::Header *) (this->mbuffer);
  134. return (0);
  135. }
  136. /*====================================================================*
  137. *
  138. * signed CPLParameters::Read (char const * filename);
  139. *
  140. * read parameter block from file;
  141. *
  142. *--------------------------------------------------------------------*/
  143. signed CPLParameters::Read (char const * filename)
  144. {
  145. if (this->mbuffer)
  146. {
  147. delete [] this->mbuffer;
  148. }
  149. this->mextent = 0;
  150. if ((filename) && (*filename))
  151. {
  152. std::ifstream stream;
  153. stream.open (filename, std::ios::binary);
  154. if (stream.good ())
  155. {
  156. stream.seekg (0, std::ios::end);
  157. this->mextent = stream.tellg ();
  158. stream.seekg (0, std::ios::beg);
  159. this->mbuffer = new uint8_t [this->mextent];
  160. stream.read ((char *)(this->mbuffer), this->mextent);
  161. }
  162. stream.close ();
  163. }
  164. this->mheader = (CPLParameters::Header *) (this->mbuffer);
  165. return (0);
  166. }
  167. /*====================================================================*
  168. *
  169. * signed Write (char const * filename);
  170. *
  171. * write parameter block to file;
  172. *
  173. *--------------------------------------------------------------------*/
  174. signed CPLParameters::Write (char const * filename)
  175. {
  176. std::ofstream stream;
  177. stream.open (filename, std::ios::binary);
  178. if (stream.good ())
  179. {
  180. stream.write ((char *)(this->mbuffer), this->mextent);
  181. }
  182. stream.close ();
  183. return (0);
  184. }
  185. /*====================================================================*
  186. *
  187. * signed Read (CPLCHannel * channel)
  188. *
  189. * read the current parameter block image from device to memory
  190. * using as many VS_RD_MOD messages as needed; the firmware must
  191. * be running for this to work;
  192. *
  193. * return 0 on success or -1 on failure;
  194. *
  195. *--------------------------------------------------------------------*/
  196. signed CPLParameters::Read (CPLChannel * channel)
  197. {
  198. ointellon intellon;
  199. uint8_t message [ETHER_MAX_LEN];
  200. #ifndef __GNUC__
  201. #pragma pack (push,1)
  202. #endif
  203. struct __packed vs_rd_mod_request
  204. {
  205. uint8_t MODULEID;
  206. uint8_t MACCESS;
  207. uint16_t MLENGTH;
  208. uint32_t MOFFSET;
  209. uint8_t MSECRET [16];
  210. }
  211. * request;
  212. struct __packed vs_rd_mod_confirm
  213. {
  214. uint8_t MSTATUS;
  215. uint8_t RESERVED1 [3];
  216. uint8_t MODULEID;
  217. uint8_t RESERVED;
  218. uint16_t MLENGTH;
  219. uint32_t MOFFSET;
  220. uint32_t MCHKSUM;
  221. uint8_t MBUFFER [oINTELLON_BLOCKSIZE];
  222. }
  223. * confirm;
  224. #ifndef __GNUC__
  225. #pragma pack (pop)
  226. #endif
  227. uint16_t extent = 0;
  228. uint16_t length = oINTELLON_BLOCKSIZE;
  229. uint32_t offset = 0;
  230. intellon.ImportPeerAddress (channel->PeerAddress ());
  231. intellon.ImportHostAddress (channel->HostAddress ());
  232. intellon.SetMessageType (VS_RD_MOD | MMTYPE_REQ);
  233. do
  234. {
  235. std::memset (message, 0, sizeof (message));
  236. request = (struct vs_rd_mod_request *)(intellon.ExportHeader (message));
  237. confirm = (struct vs_rd_mod_confirm *)(request);
  238. request->MODULEID = VS_MODULE_PIB;
  239. request->MLENGTH = HTOLE16 (length);
  240. request->MOFFSET = HTOLE32 (offset);
  241. if (channel->SendMessage (message, ETHER_MIN_LEN) <= 0)
  242. {
  243. oerror::error (0, ECANCELED, CPLCHANNEL_CANTSEND);
  244. return (-1);
  245. }
  246. if (channel->ReadMessage (message, sizeof (message)) <= 0)
  247. {
  248. oerror::error (0, ECANCELED, CPLCHANNEL_CANTREAD);
  249. return (-1);
  250. }
  251. if (confirm->MSTATUS)
  252. {
  253. oerror::error (0, ECANCELED, CPLCHANNEL_CANTDOIT,confirm->MSTATUS,CPLParameters::MessageErrorText((VS_RD_MOD | MMTYPE_CNF),confirm->MSTATUS));
  254. return (-1);
  255. }
  256. #if 1
  257. if (LE16TOH (confirm->MLENGTH) != length)
  258. {
  259. oerror::error (0, ECANCELED, oINTELLON_BAD_LENGTH);
  260. return (-1);
  261. }
  262. if (LE32TOH (confirm->MOFFSET) != offset)
  263. {
  264. oerror::error (0, ECANCELED, oINTELLON_BAD_LENGTH);
  265. return (-1);
  266. }
  267. #else
  268. if (LE16TOH (confirm->MLENGTH) != length)
  269. {
  270. oerror::error (0, EIO, oINTELLON_BAD_LENGTH);
  271. this->mlength = INT6K_BLOCKSIZE;
  272. this->moffset = 0;
  273. continue;
  274. }
  275. if (LE32TOH (confirm->MOFFSET) != this->moffset)
  276. {
  277. oerror::error (0, EIO, oINTELLON_BAD_OFFSET);
  278. this->mlength = INT6K_BLOCKSIZE;
  279. this->moffset = 0;
  280. continue;
  281. }
  282. #endif
  283. length = LE16TOH (confirm->MLENGTH);
  284. offset = LE32TOH (confirm->MOFFSET);
  285. if (omemory::checksum32 (confirm->MBUFFER, length, confirm->MCHKSUM))
  286. {
  287. oerror::error (0, ECANCELED, "Bad Message Checksum");
  288. return (-1);
  289. }
  290. if (!offset)
  291. {
  292. if (this->mbuffer)
  293. {
  294. delete [] this->mbuffer;
  295. }
  296. this->mheader = (CPLParameters::Header *) (confirm->MBUFFER);
  297. this->mextent = extent = LE16TOH (this->mheader->PIBLENGTH);
  298. this->mbuffer = new uint8_t [this->mextent];
  299. }
  300. if (length > extent)
  301. {
  302. length = extent;
  303. }
  304. std::memcpy (this->mbuffer + offset, confirm->MBUFFER, length);
  305. offset += length;
  306. extent -= length;
  307. }
  308. while (extent);
  309. this->mheader = (CPLParameters::Header *) (this->mbuffer);
  310. return (0);
  311. }
  312. /*====================================================================*
  313. *
  314. * signed Write (CPLChannel * channel);
  315. *
  316. * write the current parameter block image from memory to device
  317. * using as many VS_WR_MOD messages as needed; the firmware must
  318. * be running for this to work;
  319. *
  320. * return 0 on success or -1 on failure;
  321. *
  322. *--------------------------------------------------------------------*/
  323. signed CPLParameters::Write (CPLChannel * channel)
  324. {
  325. ointellon intellon;
  326. uint8_t message [ETHER_MAX_LEN];
  327. #ifndef __GNUC__
  328. #pragma pack (push,1)
  329. #endif
  330. struct __packed vs_wr_mod_request
  331. {
  332. uint8_t MODULEID;
  333. uint8_t RESERVED;
  334. uint16_t MLENGTH;
  335. uint32_t MOFFSET;
  336. uint32_t MCHKSUM;
  337. uint8_t MBUFFER [oINTELLON_BLOCKSIZE];
  338. }
  339. * request;
  340. struct __packed vs_wr_mod_confirm
  341. {
  342. uint8_t MSTATUS;
  343. uint8_t MODULEID;
  344. uint8_t RESERVED;
  345. uint16_t MLENGTH;
  346. uint32_t MOFFSET;
  347. }
  348. * confirm;
  349. #ifndef __GNUC__
  350. #pragma pack (pop)
  351. #endif
  352. uint16_t length = oINTELLON_BLOCKSIZE;
  353. uint16_t extent = this->mextent;
  354. uint32_t offset = 0;
  355. intellon.ImportPeerAddress (channel->PeerAddress ());
  356. intellon.ImportHostAddress (channel->HostAddress ());
  357. intellon.SetMessageType (VS_WR_MOD | MMTYPE_REQ);
  358. while (extent)
  359. {
  360. if (length > extent)
  361. {
  362. length = extent;
  363. }
  364. std::memset (message, 0, sizeof (message));
  365. request = (struct vs_wr_mod_request *) (intellon.ExportHeader (message));
  366. confirm = (struct vs_wr_mod_confirm *) (request);
  367. std::memcpy (request->MBUFFER, this->mbuffer + offset, length);
  368. request->MODULEID = VS_MODULE_PIB;
  369. request->RESERVED = 0;
  370. request->MLENGTH = HTOLE16 (length);
  371. request->MOFFSET = HTOLE32 (offset);
  372. request->MCHKSUM = omemory::checksum32 (request->MBUFFER, length, 0);
  373. if (channel->SendMessage (message, intellon.BufferLength (sizeof (struct vs_wr_mod_request))) <= 0)
  374. {
  375. oerror::error (0, ECANCELED, CPLCHANNEL_CANTSEND);
  376. return (-1);
  377. }
  378. if (channel->ReadMessage (message, sizeof (message)) <= 0)
  379. {
  380. oerror::error (0, ECANCELED, CPLCHANNEL_CANTREAD);
  381. return (-1);
  382. }
  383. if (confirm->MSTATUS)
  384. {
  385. oerror::error (0, ECANCELED, CPLCHANNEL_CANTDOIT, confirm->MSTATUS, CPLParameters::MessageErrorText((VS_WR_MOD | MMTYPE_CNF),confirm->MSTATUS));
  386. return (-1);
  387. }
  388. #if 1
  389. if (LE16TOH (confirm->MLENGTH) != length)
  390. {
  391. oerror::error (0, ECANCELED, oINTELLON_BAD_LENGTH);
  392. return (-1);
  393. }
  394. if (LE32TOH (confirm->MOFFSET) != offset)
  395. {
  396. oerror::error (0, ECANCELED, oINTELLON_BAD_OFFSET);
  397. return (-1);
  398. }
  399. #else
  400. if (LE16TOH (confirm->MLENGTH) != length)
  401. {
  402. oerror::error (0, EIO, oINTELLON_BAD_LENGTH);
  403. length = oINTELLON_BLOCKSIZE;
  404. offset = 0;
  405. continue;
  406. }
  407. if (LE32TOH (confirm->MOFFSET) != offset)
  408. {
  409. oerror::error (0, EIO, oINTELLON_BAD_OFFSET);
  410. length = oINTELLON_BLOCKSIZE;
  411. offset = 0;
  412. continue;
  413. }
  414. #endif
  415. offset += length;
  416. extent -= length;
  417. }
  418. return (0);
  419. }
  420. /*====================================================================*
  421. *
  422. * signed Flash (CPLChannel *channel);
  423. *
  424. * commit downloaded parameter block to flash memory;
  425. *
  426. *--------------------------------------------------------------------*/
  427. signed CPLParameters::Flash (CPLChannel *channel)
  428. {
  429. ointellon intellon;
  430. uint8_t message [ETHER_MAX_LEN];
  431. #ifndef __GNUC__
  432. #pragma pack (push,1)
  433. #endif
  434. struct __packed vs_wr_mod_nvm_request
  435. {
  436. uint8_t MODULEID;
  437. uint8_t RESERVED;
  438. }
  439. * request;
  440. struct __packed vs_wr_mod_nvm_confirm
  441. {
  442. uint8_t MSTATUS;
  443. uint8_t MODULEID;
  444. }
  445. * confirm;
  446. #ifndef __GNUC__
  447. #pragma pack (pop)
  448. #endif
  449. intellon.ImportPeerAddress (channel->PeerAddress ());
  450. intellon.ImportHostAddress (channel->HostAddress ());
  451. intellon.SetMessageType (VS_MOD_NVM | MMTYPE_REQ);
  452. request = (struct vs_wr_mod_nvm_request *) (intellon.ExportHeader (message));
  453. confirm = (struct vs_wr_mod_nvm_confirm *) (request);
  454. request->MODULEID = VS_MODULE_PIB;
  455. request->RESERVED = 0;
  456. if (channel->SendMessage (message, ETHER_MIN_LEN) <= 0)
  457. {
  458. oerror::error (0, ECANCELED, CPLCHANNEL_CANTSEND);
  459. return (-1);
  460. }
  461. if (channel->ReadMessage (message, sizeof (message)) <= 0)
  462. {
  463. oerror::error (0, ECANCELED, CPLCHANNEL_CANTREAD);
  464. return (-1);
  465. }
  466. if (confirm->MSTATUS)
  467. {
  468. oerror::error (0, ECANCELED, CPLCHANNEL_CANTDOIT, confirm->MSTATUS,CPLParameters::MessageErrorText((VS_MOD_NVM | MMTYPE_CNF),confirm->MSTATUS));
  469. return (-1);
  470. }
  471. return (0);
  472. }
  473. /*====================================================================*
  474. *
  475. * CPLParameters & Lock (void * memory, size_t extent);
  476. *
  477. * recompute and update the parameter block header checksum; this
  478. * conceptually locks the parameter block state;
  479. *
  480. *--------------------------------------------------------------------*/
  481. CPLParameters & CPLParameters::Lock ()
  482. {
  483. this->mheader->CHECKSUM = 0;
  484. this->mheader->CHECKSUM = omemory::checksum32 (this->mbuffer, this->mextent, 0);
  485. return (*this);
  486. }
  487. /*====================================================================*
  488. *
  489. * CPLParameters & CPLParameters::Init ();
  490. *
  491. * initialize instance variable;
  492. *
  493. *--------------------------------------------------------------------*/
  494. CPLParameters & CPLParameters::Init ()
  495. {
  496. this->mheader = (CPLParameters::Header *)(0);
  497. this->mbuffer = (uint8_t *)(0);
  498. this->mextent = 0;
  499. return (*this);
  500. }
  501. /*====================================================================*
  502. *
  503. * CPLParameters (CPLChannel * channel)
  504. *
  505. *--------------------------------------------------------------------*/
  506. CPLParameters::CPLParameters (CPLChannel * channel)
  507. {
  508. this->Init ().Read (channel);
  509. return;
  510. }
  511. /*====================================================================*
  512. *
  513. * CPLParameters (char const * filename)
  514. *
  515. *--------------------------------------------------------------------*/
  516. CPLParameters::CPLParameters (char const * filename)
  517. {
  518. this->Init ().Read (filename);
  519. return;
  520. }
  521. /*====================================================================*
  522. *
  523. * ~CPLParameters ()
  524. *
  525. *--------------------------------------------------------------------*/
  526. CPLParameters::~CPLParameters ()
  527. {
  528. delete [] this->mbuffer;
  529. return;
  530. }
  531. /*====================================================================*
  532. * end definition;
  533. *--------------------------------------------------------------------*/
  534. #endif