PLCReadParameterBlock.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * signed PLCReadParameterBlock (struct channel * channel, struct message * message, void * memory, size_t extent);
  11. *
  12. * plc.h
  13. *
  14. * Read the PIB Image from an INT6000 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 "../pib/pib.h"
  42. #include "../plc/plc.h"
  43. signed PLCReadParameterBlock (struct channel * channel, struct message * message, void * memory, size_t extent)
  44. {
  45. uint8_t * buffer = (uint8_t *) (memory);
  46. #ifndef __GNUC__
  47. #pragma pack (push,1)
  48. #endif
  49. struct __packed vs_rd_mod_request
  50. {
  51. struct ethernet_hdr ethernet;
  52. struct qualcomm_hdr qualcomm;
  53. uint8_t MODULEID;
  54. uint8_t MACCESS;
  55. uint16_t MLENGTH;
  56. uint32_t MOFFSET;
  57. uint8_t MSECRET [16];
  58. }
  59. * request = (struct vs_rd_mod_request *) (message);
  60. struct __packed vs_rd_mod_confirm
  61. {
  62. struct ethernet_hdr ethernet;
  63. struct qualcomm_hdr qualcomm;
  64. uint8_t MSTATUS;
  65. uint8_t RESERVED1 [3];
  66. uint8_t MODULEID;
  67. uint8_t RESERVED;
  68. uint16_t MLENGTH;
  69. uint32_t MOFFSET;
  70. uint32_t CHKSUM;
  71. uint8_t BUFFER [PLC_RECORD_SIZE];
  72. }
  73. * confirm = (struct vs_rd_mod_confirm *) (message);
  74. #ifndef __GNUC__
  75. #pragma pack (pop)
  76. #endif
  77. uint16_t length = 0;
  78. uint32_t offset = 0;
  79. signed actual = PLC_RECORD_SIZE;
  80. do
  81. {
  82. memset (message, 0, sizeof (* message));
  83. EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
  84. QualcommHeader (& request->qualcomm, 0, (VS_RD_MOD | MMTYPE_REQ));
  85. request->MODULEID = VS_MODULE_PIB;
  86. request->MLENGTH = HTOLE16 (actual);
  87. request->MOFFSET = HTOLE32 (offset);
  88. if (sendpacket (channel, message, (ETHER_MIN_LEN - ETHER_CRC_LEN)) <= 0)
  89. {
  90. error (1, errno, CHANNEL_CANTSEND);
  91. }
  92. if (readpacket (channel, message, sizeof (* message)) <= 0)
  93. {
  94. error (1, errno, CHANNEL_CANTREAD);
  95. }
  96. if (UnwantedMessage (message, sizeof (* confirm), 0, (VS_RD_MOD | MMTYPE_CNF)))
  97. {
  98. actual = PLC_RECORD_SIZE;
  99. offset = 0;
  100. continue;
  101. }
  102. if (confirm->MSTATUS)
  103. {
  104. error (1, ECANCELED, PLC_WONTDOIT);
  105. }
  106. if (LE16TOH (confirm->MLENGTH) != actual)
  107. {
  108. error (1, ECANCELED, PLC_ERR_LENGTH);
  109. }
  110. if (LE32TOH (confirm->MOFFSET) != offset)
  111. {
  112. error (1, ECANCELED, PLC_ERR_OFFSET);
  113. }
  114. actual = LE16TOH (confirm->MLENGTH);
  115. offset = LE32TOH (confirm->MOFFSET);
  116. if (checksum32 (confirm->BUFFER, actual, confirm->CHKSUM))
  117. {
  118. error (1, ECANCELED, "Bad Packet Checksum");
  119. }
  120. if (offset == length)
  121. {
  122. struct pib_header * pib_header = (struct pib_header *) (confirm->BUFFER);
  123. length = LE16TOH (pib_header->PIBLENGTH);
  124. }
  125. if ((offset + actual) > length)
  126. {
  127. actual = length - offset;
  128. }
  129. memcpy (buffer + offset, confirm->BUFFER, actual);
  130. offset += actual;
  131. extent -= actual;
  132. }
  133. while (offset < length);
  134. return (offset);
  135. }
  136. #endif