/*====================================================================* * * Copyright (c) 2013 Qualcomm Atheros, Inc. * * All rights reserved. * *====================================================================*/ /*====================================================================* * * signed ParseRule (int * argcp, char const ** argvp [], struct rule * rule, struct cspec * cspec); * * rules.h * * This module takes an argument vector and an argument count * and populates a classification rule structure that is suitable * for sending in a VS_CLASSIFICATION MME; * * This module is currently used by amprule, plcrule and pibruin; * * Contributor(s): * Charles Maier * Nathaniel Houghton * *--------------------------------------------------------------------*/ #include #include #include "../tools/memory.h" #include "../tools/number.h" #include "../tools/error.h" #include "../ether/ether.h" #include "../plc/rules.h" signed ParseRule (int * argcp, char const ** argvp [], struct MMERule * rule, struct cspec * cspec) { int argc = * argcp; char const ** argv = * argvp; union { uint32_t wide; uint16_t word; uint8_t byte [4]; } temp; signed code; struct MMEClassifier * classifier = (struct MMEClassifier *) (& rule->CLASSIFIER); if ((code = lookup (* argv++, actions, SIZEOF (actions))) == -1) { assist (* -- argv, CLASSIFIER_ACTION_NAME, actions, SIZEOF (actions)); } rule->MACTION = (uint8_t) (code); argc--; if ((code = lookup (* argv++, operands, SIZEOF (operands))) == -1) { assist (* -- argv, CLASSIFIER_OPERAND_NAME, operands, SIZEOF (operands)); } rule->MOPERAND = (uint8_t) (code); argc--; while ((* argv) && (lookup (* argv, controls, SIZEOF (controls)) == -1)) { if ((code = lookup (* argv++, fields, SIZEOF (fields))) == -1) { assist (* -- argv, CLASSIFIER_FIELD_NAME, fields, SIZEOF (fields)); } classifier->CR_PID = (uint8_t) (code); argc--; if ((code = lookup (* argv++, operators, SIZEOF (operators))) == -1) { assist (* -- argv, CLASSIFIER_OPERATOR_NAME, operators, SIZEOF (operators)); } classifier->CR_OPERAND = (uint8_t) (code); argc--; if (! argc || ! * argv) { error (1, ENOTSUP, "I have %s '%s' but no value", CLASSIFIER_OPERATOR_NAME, * -- argv); } switch (classifier->CR_PID) { case FIELD_ETH_SA: case FIELD_ETH_DA: bytespec (* argv++, classifier->CR_VALUE, ETHER_ADDR_LEN); break; case FIELD_IPV4_SA: case FIELD_IPV4_DA: ipv4spec (* argv++, classifier->CR_VALUE); break; case FIELD_IPV6_SA: case FIELD_IPV6_DA: ipv6spec (* argv++, classifier->CR_VALUE); break; case FIELD_VLAN_UP: case FIELD_IPV6_TC: case FIELD_IPV4_TOS: case FIELD_IPV4_PROT: classifier->CR_VALUE [0] = (uint8_t) (basespec (* argv++, 0, sizeof (classifier->CR_VALUE [0]))); break; case FIELD_VLAN_ID: case FIELD_TCP_SP: case FIELD_TCP_DP: case FIELD_UDP_SP: case FIELD_UDP_DP: case FIELD_IP_SP: case FIELD_IP_DP: temp.word = (uint16_t) (basespec (* argv++, 0, sizeof (temp.word))); temp.word = htons (temp.word); memcpy (classifier->CR_VALUE, & temp, sizeof (temp.word)); break; case FIELD_ETH_TYPE: temp.word = (uint16_t) (basespec (* argv++, 0, sizeof (temp.word))); temp.word = htons (temp.word); memcpy (classifier->CR_VALUE, & temp, sizeof (temp.word)); break; case FIELD_IPV6_FL: temp.wide = (uint32_t) (basespec (* argv++, 0, sizeof (temp.wide))) & 0x000FFFFF; temp.wide = htonl (temp.wide); memcpy (classifier->CR_VALUE, & temp.byte [1], 3); break; case FIELD_HPAV_MME: bytespec (* argv++, classifier->CR_VALUE, sizeof (uint16_t) + sizeof (uint8_t)); temp.byte [0] = classifier->CR_VALUE [1]; classifier->CR_VALUE [1] = classifier->CR_VALUE [2]; classifier->CR_VALUE [2] = temp.byte [0]; break; case FIELD_TCP_ACK: if ((code = lookup (* argv++, states, SIZEOF (states))) == -1) { assist (* -- argv, CLASSIFIER_STATE_NAME, states, SIZEOF (states)); } memset (classifier->CR_VALUE, 0, sizeof (classifier->CR_VALUE)); break; case FIELD_VLAN_TAG: if ((code = lookup (* argv++, states, SIZEOF (states))) == -1) { assist (* -- argv, CLASSIFIER_STATE_NAME, states, SIZEOF (states)); } memset (classifier->CR_VALUE, 0, sizeof (classifier->CR_VALUE)); classifier->CR_OPERAND ^= code; break; default: error (1, ENOTSUP, "%s", argv [- 2]); break; } rule->NUM_CLASSIFIERS++; classifier++; argc--; } memcpy (classifier, cspec, sizeof (* cspec)); if ((code = lookup (* argv++, controls, SIZEOF (controls))) == -1) { assist (* -- argv, CLASSIFIER_CONTROL_NAME, controls, SIZEOF (controls)); } rule->MCONTROL = (uint8_t) (code); argc--; if ((code = lookup (* argv++, volatilities, SIZEOF (volatilities))) == -1) { assist (* -- argv, CLASSIFIER_VOLATILITY_NAME, volatilities, SIZEOF (volatilities)); } rule->MVOLATILITY = (uint8_t) (code); argc--; * argcp = argc; * argvp = argv; return (0); }