ParseRule.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * signed ParseRule (int * argcp, char const ** argvp [], struct rule * rule, struct cspec * cspec);
  11. *
  12. * rules.h
  13. *
  14. * This module takes an argument vector and an argument count
  15. * and populates a classification rule structure that is suitable
  16. * for sending in a VS_CLASSIFICATION MME;
  17. *
  18. * This module is currently used by amprule, plcrule and pibruin;
  19. *
  20. * Contributor(s):
  21. * Charles Maier <cmaier@qca.qualcomm.com>
  22. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  23. *
  24. *--------------------------------------------------------------------*/
  25. #include <memory.h>
  26. #include <errno.h>
  27. #include "../tools/memory.h"
  28. #include "../tools/number.h"
  29. #include "../tools/error.h"
  30. #include "../ether/ether.h"
  31. #include "../plc/rules.h"
  32. signed ParseRule (int * argcp, char const ** argvp [], struct MMERule * rule, struct cspec * cspec)
  33. {
  34. int argc = * argcp;
  35. char const ** argv = * argvp;
  36. union
  37. {
  38. uint32_t wide;
  39. uint16_t word;
  40. uint8_t byte [4];
  41. }
  42. temp;
  43. signed code;
  44. struct MMEClassifier * classifier = (struct MMEClassifier *) (& rule->CLASSIFIER);
  45. if ((code = lookup (* argv++, actions, SIZEOF (actions))) == -1)
  46. {
  47. assist (* -- argv, CLASSIFIER_ACTION_NAME, actions, SIZEOF (actions));
  48. }
  49. rule->MACTION = (uint8_t) (code);
  50. argc--;
  51. if ((code = lookup (* argv++, operands, SIZEOF (operands))) == -1)
  52. {
  53. assist (* -- argv, CLASSIFIER_OPERAND_NAME, operands, SIZEOF (operands));
  54. }
  55. rule->MOPERAND = (uint8_t) (code);
  56. argc--;
  57. while ((* argv) && (lookup (* argv, controls, SIZEOF (controls)) == -1))
  58. {
  59. if ((code = lookup (* argv++, fields, SIZEOF (fields))) == -1)
  60. {
  61. assist (* -- argv, CLASSIFIER_FIELD_NAME, fields, SIZEOF (fields));
  62. }
  63. classifier->CR_PID = (uint8_t) (code);
  64. argc--;
  65. if ((code = lookup (* argv++, operators, SIZEOF (operators))) == -1)
  66. {
  67. assist (* -- argv, CLASSIFIER_OPERATOR_NAME, operators, SIZEOF (operators));
  68. }
  69. classifier->CR_OPERAND = (uint8_t) (code);
  70. argc--;
  71. if (! argc || ! * argv)
  72. {
  73. error (1, ENOTSUP, "I have %s '%s' but no value", CLASSIFIER_OPERATOR_NAME, * -- argv);
  74. }
  75. switch (classifier->CR_PID)
  76. {
  77. case FIELD_ETH_SA:
  78. case FIELD_ETH_DA:
  79. bytespec (* argv++, classifier->CR_VALUE, ETHER_ADDR_LEN);
  80. break;
  81. case FIELD_IPV4_SA:
  82. case FIELD_IPV4_DA:
  83. ipv4spec (* argv++, classifier->CR_VALUE);
  84. break;
  85. case FIELD_IPV6_SA:
  86. case FIELD_IPV6_DA:
  87. ipv6spec (* argv++, classifier->CR_VALUE);
  88. break;
  89. case FIELD_VLAN_UP:
  90. case FIELD_IPV6_TC:
  91. case FIELD_IPV4_TOS:
  92. case FIELD_IPV4_PROT:
  93. classifier->CR_VALUE [0] = (uint8_t) (basespec (* argv++, 0, sizeof (classifier->CR_VALUE [0])));
  94. break;
  95. case FIELD_VLAN_ID:
  96. case FIELD_TCP_SP:
  97. case FIELD_TCP_DP:
  98. case FIELD_UDP_SP:
  99. case FIELD_UDP_DP:
  100. case FIELD_IP_SP:
  101. case FIELD_IP_DP:
  102. temp.word = (uint16_t) (basespec (* argv++, 0, sizeof (temp.word)));
  103. temp.word = htons (temp.word);
  104. memcpy (classifier->CR_VALUE, & temp, sizeof (temp.word));
  105. break;
  106. case FIELD_ETH_TYPE:
  107. temp.word = (uint16_t) (basespec (* argv++, 0, sizeof (temp.word)));
  108. temp.word = htons (temp.word);
  109. memcpy (classifier->CR_VALUE, & temp, sizeof (temp.word));
  110. break;
  111. case FIELD_IPV6_FL:
  112. temp.wide = (uint32_t) (basespec (* argv++, 0, sizeof (temp.wide))) & 0x000FFFFF;
  113. temp.wide = htonl (temp.wide);
  114. memcpy (classifier->CR_VALUE, & temp.byte [1], 3);
  115. break;
  116. case FIELD_HPAV_MME:
  117. bytespec (* argv++, classifier->CR_VALUE, sizeof (uint16_t) + sizeof (uint8_t));
  118. temp.byte [0] = classifier->CR_VALUE [1];
  119. classifier->CR_VALUE [1] = classifier->CR_VALUE [2];
  120. classifier->CR_VALUE [2] = temp.byte [0];
  121. break;
  122. case FIELD_TCP_ACK:
  123. if ((code = lookup (* argv++, states, SIZEOF (states))) == -1)
  124. {
  125. assist (* -- argv, CLASSIFIER_STATE_NAME, states, SIZEOF (states));
  126. }
  127. memset (classifier->CR_VALUE, 0, sizeof (classifier->CR_VALUE));
  128. break;
  129. case FIELD_VLAN_TAG:
  130. if ((code = lookup (* argv++, states, SIZEOF (states))) == -1)
  131. {
  132. assist (* -- argv, CLASSIFIER_STATE_NAME, states, SIZEOF (states));
  133. }
  134. memset (classifier->CR_VALUE, 0, sizeof (classifier->CR_VALUE));
  135. classifier->CR_OPERAND ^= code;
  136. break;
  137. default:
  138. error (1, ENOTSUP, "%s", argv [- 2]);
  139. break;
  140. }
  141. rule->NUM_CLASSIFIERS++;
  142. classifier++;
  143. argc--;
  144. }
  145. memcpy (classifier, cspec, sizeof (* cspec));
  146. if ((code = lookup (* argv++, controls, SIZEOF (controls))) == -1)
  147. {
  148. assist (* -- argv, CLASSIFIER_CONTROL_NAME, controls, SIZEOF (controls));
  149. }
  150. rule->MCONTROL = (uint8_t) (code);
  151. argc--;
  152. if ((code = lookup (* argv++, volatilities, SIZEOF (volatilities))) == -1)
  153. {
  154. assist (* -- argv, CLASSIFIER_VOLATILITY_NAME, volatilities, SIZEOF (volatilities));
  155. }
  156. rule->MVOLATILITY = (uint8_t) (code);
  157. argc--;
  158. * argcp = argc;
  159. * argvp = argv;
  160. return (0);
  161. }