ReadFMI.keepme.c 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * signed ReadFMI (struct plc * plc, uint8_t MMV, uint16_t MMTYPE);
  11. *
  12. * plc.h
  13. *
  14. * read a fragmented message and return a pointer to a buffer that
  15. * contains the concatenated message fragments; the buffer address
  16. * is returned in plc->content; the calling function must free the
  17. * buffer when done; buffer length is computed from the number of
  18. * fragments returned in the FMI field of the first fragment;
  19. *
  20. * Contributor(s):
  21. * Charles Maier <cmaier@qca.qualcomm.com>
  22. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  23. *
  24. *--------------------------------------------------------------------*/
  25. #ifndef READFMI_SOURCE
  26. #define READFMI_SOURCE
  27. #include <stdint.h>
  28. #include <stdlib.h>
  29. #include <memory.h>
  30. #include "../tools/memory.h"
  31. #include "../tools/number.h"
  32. #include "../tools/error.h"
  33. #include "../plc/plc.h"
  34. signed ReadFMI (struct plc * plc, uint8_t MMV, uint16_t MMTYPE)
  35. {
  36. if (ReadMME (plc, MMV, MMTYPE) > 0)
  37. {
  38. struct homeplug * homeplug = (struct homeplug *) (plc->message);
  39. if (homeplug->homeplug.FMSN || homeplug->homeplug.FMID)
  40. {
  41. unsigned count = ((homeplug->homeplug.FMID >> 0) & 0x0F);
  42. unsigned extra = ((homeplug->homeplug.FMID >> 4) & 0x0F);
  43. unsigned length = sizeof (* homeplug) + extra * sizeof (homeplug->content);
  44. if ((plc->content = malloc (length)))
  45. {
  46. signed offset = plc->packetsize;
  47. memcpy (plc->content, homeplug, offset);
  48. while (count < extra)
  49. {
  50. if (ReadMME (plc, MMV, MMTYPE) <= 0)
  51. {
  52. free (plc->content);
  53. plc->content = NULL;
  54. return (-1);
  55. }
  56. count = ((homeplug->homeplug.FMID >> 0) & 0x0F);
  57. extra = ((homeplug->homeplug.FMID >> 4) & 0x0F);
  58. plc->packetsize -= sizeof (struct ethernet_hdr);
  59. plc->packetsize -= sizeof (struct homeplug_fmi);
  60. memcpy ((byte *) (plc->content) + offset, homeplug->content, plc->packetsize);
  61. offset += plc->packetsize;
  62. }
  63. plc->packetsize = offset;
  64. }
  65. }
  66. else
  67. {
  68. struct fragment_hdr * fragment = (struct fragment_hdr *) (homeplug->content);
  69. signed length = sizeof (struct ethernet_hdr) + sizeof (struct qualcomm_fmi) + LE16TOH (fragment->MME_LENGTH);
  70. error (0, 0, "fragment->MME_LENGTH %d", fragment->MME_LENGTH);
  71. if ((plc->content = malloc (length)))
  72. {
  73. signed offset = plc->packetsize;
  74. memcpy (plc->content, homeplug, offset);
  75. while (offset < length)
  76. {
  77. if (ReadMME (plc, MMV, MMTYPE) <= 0)
  78. {
  79. free (plc->content);
  80. plc->content = NULL;
  81. return (-1);
  82. }
  83. plc->packetsize -= sizeof (struct ethernet_hdr);
  84. plc->packetsize -= sizeof (struct qualcomm_fmi);
  85. memcpy ((byte *) (plc->content) + offset, homeplug->content, plc->packetsize);
  86. offset += plc->packetsize;
  87. }
  88. plc->packetsize = offset;
  89. }
  90. }
  91. }
  92. error (0, 0, "plc->packetsize %d", plc->packetsize);
  93. return (plc->packetsize);
  94. }
  95. #endif