ReadParameters1.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * signed ReadParameters1 (struct plc * plc);
  11. *
  12. * plc.h
  13. *
  14. * Read the PIB Image from an INT6x00 or AR7x00 using as many
  15. * VS_RD_MOD messages as needed; Write image blocks to file as
  16. * they are read;
  17. *
  18. * the objective here is to read the PIB module in 1024 byte blocks
  19. * until the module offset equals or exceeds the PIB length; we do
  20. * not know the PIB length until we have read the header contained
  21. * in the first block at OFFSET 0;
  22. *
  23. * MOFFSET and MLENGTH fields occupy different offsets in REQ and CNF
  24. * messages; we exploit that by initializing them using CNF message
  25. * offsets then copying values into REQ message offsets before each
  26. * read; this works because confirmation messages always return all
  27. * the data we request or nothing at all;
  28. *
  29. * Contributor(s):
  30. * Charles Maier <cmaier@qca.qualcomm.com>
  31. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  32. * Lluis Paulet Gimbert <dimat.com>
  33. *
  34. *--------------------------------------------------------------------*/
  35. #ifndef READPARAMETERS1_SOURCE
  36. #define READPARAMETERS1_SOURCE
  37. #include <stdint.h>
  38. #include <unistd.h>
  39. #include <memory.h>
  40. #include "../tools/error.h"
  41. #include "../tools/files.h"
  42. #include "../plc/plc.h"
  43. #include "../pib/pib.h"
  44. signed ReadParameters1 (struct plc * plc)
  45. {
  46. struct channel * channel = (struct channel *) (plc->channel);
  47. struct message * message = (struct message *) (plc->message);
  48. #ifndef __GNUC__
  49. #pragma pack (push,1)
  50. #endif
  51. struct __packed vs_rd_mod_request
  52. {
  53. struct ethernet_hdr ethernet;
  54. struct qualcomm_hdr qualcomm;
  55. uint8_t MODULEID;
  56. uint8_t MACCESS;
  57. uint16_t MLENGTH;
  58. uint32_t MOFFSET;
  59. uint8_t MSECRET [16];
  60. }
  61. * request = (struct vs_rd_mod_request *) (message);
  62. struct __packed vs_rd_mod_confirm
  63. {
  64. struct ethernet_hdr ethernet;
  65. struct qualcomm_hdr qualcomm;
  66. uint8_t MSTATUS;
  67. uint8_t RESERVED1 [3];
  68. uint8_t MODULEID;
  69. uint8_t RESERVED;
  70. uint16_t MLENGTH;
  71. uint32_t MOFFSET;
  72. uint32_t CHKSUM;
  73. uint8_t BUFFER [PLC_RECORD_SIZE];
  74. }
  75. * confirm = (struct vs_rd_mod_confirm *) (message);
  76. #ifndef __GNUC__
  77. #pragma pack (pop)
  78. #endif
  79. uint32_t extent = 0;
  80. uint32_t offset = 0;
  81. uint16_t length = PLC_RECORD_SIZE;
  82. Request (plc, "Read Parameters from Device");
  83. if (lseek (plc->pib.file, 0, SEEK_SET))
  84. {
  85. error (PLC_EXIT (plc), errno, FILE_CANTHOME, plc->pib.name);
  86. return (1);
  87. }
  88. do
  89. {
  90. memset (message, 0, sizeof (* message));
  91. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  92. QualcommHeader (& request->qualcomm, 0, (VS_RD_MOD | MMTYPE_REQ));
  93. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  94. request->MODULEID = VS_MODULE_PIB;
  95. request->MLENGTH = HTOLE16 (length);
  96. request->MOFFSET = HTOLE32 (offset);
  97. if (SendMME (plc) <= 0)
  98. {
  99. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  100. return (-1);
  101. }
  102. if (ReadMME (plc, 0, (VS_RD_MOD | MMTYPE_CNF)) <= 0)
  103. {
  104. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  105. return (-1);
  106. }
  107. if (confirm->MSTATUS)
  108. {
  109. Failure (plc, PLC_WONTDOIT);
  110. return (-1);
  111. }
  112. if (LE16TOH (confirm->MLENGTH) != length)
  113. {
  114. error (PLC_EXIT (plc), 0, PLC_ERR_LENGTH);
  115. return (-1);
  116. }
  117. if (LE32TOH (confirm->MOFFSET) != offset)
  118. {
  119. error (PLC_EXIT (plc), 0, PLC_ERR_OFFSET);
  120. return (-1);
  121. }
  122. length = LE16TOH (confirm->MLENGTH);
  123. offset = LE32TOH (confirm->MOFFSET);
  124. if (checksum32 (confirm->BUFFER, length, confirm->CHKSUM))
  125. {
  126. error (PLC_EXIT (plc), ECANCELED, "Bad Packet Checksum");
  127. return (-1);
  128. }
  129. if (offset == extent)
  130. {
  131. struct pib_header * pib_header = (struct pib_header *) (confirm->BUFFER);
  132. extent = LE16TOH (pib_header->PIBLENGTH);
  133. }
  134. if ((offset + length) > extent)
  135. {
  136. length = extent - offset;
  137. }
  138. if (lseek (plc->pib.file, offset, SEEK_SET) != (signed) (offset))
  139. {
  140. error (PLC_EXIT (plc), errno, FILE_CANTSEEK, plc->pib.name);
  141. return (-1);
  142. }
  143. if (write (plc->pib.file, confirm->BUFFER, length) != (signed) (length))
  144. {
  145. error (PLC_EXIT (plc), errno, FILE_CANTSAVE, plc->pib.name);
  146. return (-1);
  147. }
  148. offset += length;
  149. }
  150. while (offset < extent);
  151. Confirm (plc, "Read %s", plc->pib.name);
  152. return (0);
  153. }
  154. #endif