UnwantedMessage.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * bool UnwantedMessage (void const * memory, size_t extent, uint8_t MMV, uint16_t MMTYPE);
  11. *
  12. * mme.h
  13. *
  14. * return true (-1) if memory does not contains a Qualcomm Atheros
  15. * vendor specific message having the requested version and type;
  16. * otherwise, return 0;
  17. *
  18. * the message version determines the location of the OUI field;
  19. * messages with MMV = 1 have an FMI used to track multi-part
  20. * confirmation counts; out of order counts are treated as an
  21. * error;
  22. *
  23. * constant __WHYNOT__ displays reason for message rejection;
  24. *
  25. * this implementation conditionally compiles code that enforces
  26. * proper frame fragmentation but that code may be disabled due
  27. *
  28. * some problems with legacy homeplug software;
  29. *
  30. * batch is the fragment group identifier taken from the fragment
  31. * frame;
  32. *
  33. * total is the number of fragements in the group taken from the
  34. * fragment frame;
  35. *
  36. * index is the fragment identifier from the fragment frame;
  37. *
  38. * count is the number of fragments received so far;
  39. *
  40. * Contributor(s):
  41. * Charles Maier <cmaier@qca.qualcomm.com>
  42. * Matthieu Poullet <m.poullet@avm.de>
  43. *
  44. *--------------------------------------------------------------------*/
  45. #ifndef UNWANTEDMESSAGE_SOURCE
  46. #define UNWANTEDMESSAGE_SOURCE
  47. #include <stdint.h>
  48. #include <memory.h>
  49. #include "../tools/endian.h"
  50. #include "../tools/error.h"
  51. #include "../mme/mme.h"
  52. signed UnwantedMessage (void const * memory, size_t extent, uint8_t MMV, uint16_t MMTYPE)
  53. {
  54. extern const byte localcast [ETHER_ADDR_LEN];
  55. struct homeplug1 * homeplug = (struct homeplug1 *) (memory);
  56. if (! extent)
  57. {
  58. return (- 1);
  59. }
  60. if (extent < (ETHER_MIN_LEN - ETHER_CRC_LEN))
  61. {
  62. #if defined (__WHYNOT__)
  63. error (0, 0, "Wrong Ethernet Frame Size: Received " SIZE_T_SPEC " bytes but need at least %d", extent, ETHER_MIN_LEN - ETHER_CRC_LEN);
  64. #endif
  65. return (- 1);
  66. }
  67. if (extent > (ETHER_MAX_LEN))
  68. {
  69. #if defined (__WHYNOT__)
  70. error (0, 0, "Wrong Ethernet Frame Size: Received " SIZE_T_SPEC " bytes but need at most %d", extent, ETHER_MAX_LEN);
  71. #endif
  72. return (- 1);
  73. }
  74. if (ntohs (homeplug->ethernet.MTYPE) != ETH_P_HPAV)
  75. {
  76. #if defined (__WHYNOT__)
  77. error (0, 0, "Wrong Ethernet Frame Type: Received %04X while waiting for %04X", ntohs (homeplug->ethernet.MTYPE), ETH_P_HPAV);
  78. #endif
  79. return (- 1);
  80. }
  81. if (homeplug->homeplug.MMV != MMV)
  82. {
  83. #if defined (__WHYNOT__)
  84. error (0, 0, "Wrong Message Version: Received %02X but expected %02X", homeplug->homeplug.MMV, MMV);
  85. #endif
  86. return (- 1);
  87. }
  88. if (homeplug->homeplug.MMV == 0)
  89. {
  90. struct qualcomm_hdr * qualcomm = (struct qualcomm_hdr *) (& homeplug->homeplug);
  91. if (LE16TOH (qualcomm->MMTYPE) != MMTYPE)
  92. {
  93. #if defined (__WHYNOT__)
  94. error (0, 0, "Wrong Message Type: Received %04X while waiting for %04X", LE16TOH (qualcomm->MMTYPE), MMTYPE);
  95. #endif
  96. return (- 1);
  97. }
  98. if ((MMTYPE < VS_MMTYPE_MIN) || (MMTYPE > VS_MMTYPE_MAX))
  99. {
  100. }
  101. else if (memcmp (localcast, qualcomm->OUI, sizeof (qualcomm->OUI)))
  102. {
  103. #if defined (__WHYNOT__)
  104. error (0, 0, "Wrong Message Vendor");
  105. #endif
  106. return (- 1);
  107. }
  108. }
  109. if (homeplug->homeplug.MMV == 1)
  110. {
  111. struct qualcomm_fmi * qualcomm = (struct qualcomm_fmi *) (& homeplug->homeplug);
  112. #if FMI
  113. static unsigned total = 0;
  114. static unsigned index = 0;
  115. static unsigned count = 0;
  116. #endif
  117. if (LE16TOH (qualcomm->MMTYPE) != MMTYPE)
  118. {
  119. #if defined (__WHYNOT__)
  120. error (0, 0, "Wrong Message Type: Received %04X while waiting for %04X", LE16TOH (qualcomm->MMTYPE), MMTYPE);
  121. #endif
  122. return (- 1);
  123. }
  124. #if FMI
  125. index = qualcomm->FMID >> 0 & 0x0F;
  126. if (! index)
  127. {
  128. total = qualcomm->FMID >> 4 & 0x0F;
  129. count = qualcomm->FMID >> 0 & 0x0F;
  130. if (memcmp (localcast, qualcomm->OUI, sizeof (qualcomm->OUI)))
  131. {
  132. #if defined (__WHYNOT__)
  133. error (0, 0, "Wrong Message Vendor");
  134. #endif
  135. return (- 1);
  136. }
  137. }
  138. if (index != count)
  139. {
  140. #if defined (__WHYNOT__)
  141. error (0, 0, "Message Fragment Out-of-order");
  142. #endif
  143. return (- 1);
  144. }
  145. if (count > total)
  146. {
  147. #if defined (__WHYNOT__)
  148. error (0, 0, "Fragment Count Exceeds Total");
  149. #endif
  150. return (- 1);
  151. }
  152. count++;
  153. #endif
  154. }
  155. return (0);
  156. }
  157. #endif