ReadParameterBlock.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * signed PLCReadParameterBlock (struct plc * plc, void * memory, size_t extent);
  11. *
  12. * plc.h
  13. *
  14. * Read the PIB Image from an INT6x00 using as many VS_RD_MOD
  15. * messages as needed; Write image blocks to file as they are
  16. * 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 actual; we do
  20. * not know the PIB actual until we have read the header contained
  21. * in the first block read from 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. *
  33. *--------------------------------------------------------------------*/
  34. #ifndef READPARAMETERBLOCK_SOURCE
  35. #define READPARAMETERBLOCK_SOURCE
  36. #include <stdint.h>
  37. #include <unistd.h>
  38. #include <memory.h>
  39. #include "../tools/error.h"
  40. #include "../tools/files.h"
  41. #include "../plc/plc.h"
  42. #include "../pib/pib.h"
  43. signed ReadParameterBlock (struct plc * plc, void * memory, size_t extent)
  44. {
  45. uint8_t * buffer = (uint8_t *) (memory);
  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. uint16_t length = 0;
  80. uint32_t offset = 0;
  81. signed actual = PLC_RECORD_SIZE;
  82. do
  83. {
  84. memset (message, 0, sizeof (* message));
  85. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  86. QualcommHeader (& request->qualcomm, 0, (VS_RD_MOD | MMTYPE_REQ));
  87. request->MODULEID = VS_MODULE_PIB;
  88. request->MLENGTH = HTOLE16 (actual);
  89. request->MOFFSET = HTOLE32 (offset);
  90. plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
  91. if (SendMME (plc) <= 0)
  92. {
  93. error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
  94. return (-1);
  95. }
  96. if (ReadMME (plc, 0, (VS_RD_MOD | MMTYPE_CNF)) <= 0)
  97. {
  98. error (PLC_EXIT (plc), errno, CHANNEL_CANTREAD);
  99. return (-1);
  100. }
  101. if (confirm->MSTATUS)
  102. {
  103. error (PLC_EXIT (plc), ECANCELED, PLC_WONTDOIT);
  104. return (-1);
  105. }
  106. if (LE16TOH (confirm->MLENGTH) != actual)
  107. {
  108. Failure (plc, PLC_ERR_LENGTH);
  109. return (-1);
  110. }
  111. if (LE32TOH (confirm->MOFFSET) != offset)
  112. {
  113. Failure (plc, PLC_ERR_OFFSET);
  114. return (-1);
  115. }
  116. actual = LE16TOH (confirm->MLENGTH);
  117. offset = LE32TOH (confirm->MOFFSET);
  118. if (checksum32 (confirm->BUFFER, actual, confirm->CHKSUM))
  119. {
  120. error (PLC_EXIT (plc), ECANCELED, "Bad Packet Checksum");
  121. return (-1);
  122. }
  123. if (offset == length)
  124. {
  125. struct pib_header * pib_header = (struct pib_header *) (confirm->BUFFER);
  126. length = LE16TOH (pib_header->PIBLENGTH);
  127. }
  128. if ((offset + actual) > length)
  129. {
  130. actual = length - offset;
  131. }
  132. memcpy (buffer + offset, confirm->BUFFER, actual);
  133. offset += actual;
  134. extent -= actual;
  135. }
  136. while (offset < length);
  137. return (offset);
  138. }
  139. #endif