ReadFMI.c 2.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  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 homeplug1 * homeplug = (struct homeplug1 *) (plc->message);
  39. unsigned count = ((homeplug->homeplug.FMID >> 4) & 0x0F);
  40. unsigned index = ((homeplug->homeplug.FMID >> 0) & 0x0F);
  41. signed length = sizeof (* homeplug) + count * sizeof (homeplug->content);
  42. if ((plc->content = malloc (length)))
  43. {
  44. signed offset = plc->packetsize;
  45. memcpy (plc->content, homeplug, offset);
  46. while (index < count)
  47. {
  48. if (ReadMME (plc, MMV, MMTYPE) <= 0)
  49. {
  50. free (plc->content);
  51. plc->content = NULL;
  52. return (-1);
  53. }
  54. count = ((homeplug->homeplug.FMID >> 4) & 0x0F);
  55. index = ((homeplug->homeplug.FMID >> 0) & 0x0F);
  56. plc->packetsize -= sizeof (struct ethernet_hdr);
  57. plc->packetsize -= sizeof (struct homeplug_fmi);
  58. if ((offset + plc->packetsize) > length)
  59. {
  60. error (1, EOVERFLOW, "%s", __func__);
  61. }
  62. memcpy ((uint8_t *) (plc->content) + offset, homeplug->content, plc->packetsize);
  63. offset += plc->packetsize;
  64. }
  65. plc->packetsize = offset;
  66. }
  67. }
  68. return (plc->packetsize);
  69. }
  70. #endif