net: mscc: ocelot: create TCAM skeleton from tc filter chains
For Ocelot switches, there are 2 ingress pipelines for flow offload rules: VCAP IS1 (Ingress Classification) and IS2 (Security Enforcement). IS1 and IS2 support different sets of actions. The pipeline order for a packet on ingress is: Basic classification -> VCAP IS1 -> VCAP IS2 Furthermore, IS1 is looked up 3 times, and IS2 is looked up twice (each TCAM entry can be configured to match only on the first lookup, or only on the second, or on both etc). Because the TCAMs are completely independent in hardware, and because of the fixed pipeline, we actually have very limited options when it comes to offloading complex rules to them while still maintaining the same semantics with the software data path. This patch maps flow offload rules to ingress TCAMs according to a predefined chain index number. There is going to be a script in selftests that clarifies the usage model. There is also an egress TCAM (VCAP ES0, the Egress Rewriter), which is modeled on top of the default chain 0 of the egress qdisc, because it doesn't have multiple lookups. Suggested-by: Allan W. Nielsen <allan.nielsen@microchip.com> Co-developed-by: Xiaoliang Yang <xiaoliang.yang_1@nxp.com> Signed-off-by: Xiaoliang Yang <xiaoliang.yang_1@nxp.com> Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									319e4dd11a
								
							
						
					
					
						commit
						1397a2eb52
					
				| @ -5,49 +5,239 @@ | |||||||
| 
 | 
 | ||||||
| #include <net/pkt_cls.h> | #include <net/pkt_cls.h> | ||||||
| #include <net/tc_act/tc_gact.h> | #include <net/tc_act/tc_gact.h> | ||||||
| 
 | #include <soc/mscc/ocelot_vcap.h> | ||||||
| #include "ocelot_vcap.h" | #include "ocelot_vcap.h" | ||||||
| 
 | 
 | ||||||
| static int ocelot_flower_parse_action(struct flow_cls_offload *f, | /* Arbitrarily chosen constants for encoding the VCAP block and lookup number
 | ||||||
|  |  * into the chain number. This is UAPI. | ||||||
|  |  */ | ||||||
|  | #define VCAP_BLOCK			10000 | ||||||
|  | #define VCAP_LOOKUP			1000 | ||||||
|  | #define VCAP_IS1_NUM_LOOKUPS		3 | ||||||
|  | #define VCAP_IS2_NUM_LOOKUPS		2 | ||||||
|  | #define VCAP_IS2_NUM_PAG		256 | ||||||
|  | #define VCAP_IS1_CHAIN(lookup)		\ | ||||||
|  | 	(1 * VCAP_BLOCK + (lookup) * VCAP_LOOKUP) | ||||||
|  | #define VCAP_IS2_CHAIN(lookup, pag)	\ | ||||||
|  | 	(2 * VCAP_BLOCK + (lookup) * VCAP_LOOKUP + (pag)) | ||||||
|  | 
 | ||||||
|  | static int ocelot_chain_to_block(int chain, bool ingress) | ||||||
|  | { | ||||||
|  | 	int lookup, pag; | ||||||
|  | 
 | ||||||
|  | 	if (!ingress) { | ||||||
|  | 		if (chain == 0) | ||||||
|  | 			return VCAP_ES0; | ||||||
|  | 		return -EOPNOTSUPP; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Backwards compatibility with older, single-chain tc-flower
 | ||||||
|  | 	 * offload support in Ocelot | ||||||
|  | 	 */ | ||||||
|  | 	if (chain == 0) | ||||||
|  | 		return VCAP_IS2; | ||||||
|  | 
 | ||||||
|  | 	for (lookup = 0; lookup < VCAP_IS1_NUM_LOOKUPS; lookup++) | ||||||
|  | 		if (chain == VCAP_IS1_CHAIN(lookup)) | ||||||
|  | 			return VCAP_IS1; | ||||||
|  | 
 | ||||||
|  | 	for (lookup = 0; lookup < VCAP_IS2_NUM_LOOKUPS; lookup++) | ||||||
|  | 		for (pag = 0; pag < VCAP_IS2_NUM_PAG; pag++) | ||||||
|  | 			if (chain == VCAP_IS2_CHAIN(lookup, pag)) | ||||||
|  | 				return VCAP_IS2; | ||||||
|  | 
 | ||||||
|  | 	return -EOPNOTSUPP; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Caller must ensure this is a valid IS1 or IS2 chain first,
 | ||||||
|  |  * by calling ocelot_chain_to_block. | ||||||
|  |  */ | ||||||
|  | static int ocelot_chain_to_lookup(int chain) | ||||||
|  | { | ||||||
|  | 	return (chain / VCAP_LOOKUP) % 10; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool ocelot_is_goto_target_valid(int goto_target, int chain, | ||||||
|  | 					bool ingress) | ||||||
|  | { | ||||||
|  | 	int pag; | ||||||
|  | 
 | ||||||
|  | 	/* Can't offload GOTO in VCAP ES0 */ | ||||||
|  | 	if (!ingress) | ||||||
|  | 		return (goto_target < 0); | ||||||
|  | 
 | ||||||
|  | 	/* Non-optional GOTOs */ | ||||||
|  | 	if (chain == 0) | ||||||
|  | 		/* VCAP IS1 can be skipped, either partially or completely */ | ||||||
|  | 		return (goto_target == VCAP_IS1_CHAIN(0) || | ||||||
|  | 			goto_target == VCAP_IS1_CHAIN(1) || | ||||||
|  | 			goto_target == VCAP_IS1_CHAIN(2) || | ||||||
|  | 			goto_target == VCAP_IS2_CHAIN(0, 0) || | ||||||
|  | 			goto_target == VCAP_IS2_CHAIN(1, 0)); | ||||||
|  | 
 | ||||||
|  | 	if (chain == VCAP_IS1_CHAIN(0)) | ||||||
|  | 		return (goto_target == VCAP_IS1_CHAIN(1)); | ||||||
|  | 
 | ||||||
|  | 	if (chain == VCAP_IS1_CHAIN(1)) | ||||||
|  | 		return (goto_target == VCAP_IS1_CHAIN(2)); | ||||||
|  | 
 | ||||||
|  | 	/* Lookup 2 of VCAP IS1 can really support non-optional GOTOs,
 | ||||||
|  | 	 * using a Policy Association Group (PAG) value, which is an 8-bit | ||||||
|  | 	 * value encoding a VCAP IS2 target chain. | ||||||
|  | 	 */ | ||||||
|  | 	if (chain == VCAP_IS1_CHAIN(2)) { | ||||||
|  | 		for (pag = 0; pag < VCAP_IS2_NUM_PAG; pag++) | ||||||
|  | 			if (goto_target == VCAP_IS2_CHAIN(0, pag)) | ||||||
|  | 				return true; | ||||||
|  | 
 | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Non-optional GOTO from VCAP IS2 lookup 0 to lookup 1.
 | ||||||
|  | 	 * We cannot change the PAG at this point. | ||||||
|  | 	 */ | ||||||
|  | 	for (pag = 0; pag < VCAP_IS2_NUM_PAG; pag++) | ||||||
|  | 		if (chain == VCAP_IS2_CHAIN(0, pag)) | ||||||
|  | 			return (goto_target == VCAP_IS2_CHAIN(1, pag)); | ||||||
|  | 
 | ||||||
|  | 	/* VCAP IS2 lookup 1 cannot jump anywhere */ | ||||||
|  | 	return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct ocelot_vcap_filter * | ||||||
|  | ocelot_find_vcap_filter_that_points_at(struct ocelot *ocelot, int chain) | ||||||
|  | { | ||||||
|  | 	struct ocelot_vcap_filter *filter; | ||||||
|  | 	struct ocelot_vcap_block *block; | ||||||
|  | 	int block_id; | ||||||
|  | 
 | ||||||
|  | 	block_id = ocelot_chain_to_block(chain, true); | ||||||
|  | 	if (block_id < 0) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	if (block_id == VCAP_IS2) { | ||||||
|  | 		block = &ocelot->block[VCAP_IS1]; | ||||||
|  | 
 | ||||||
|  | 		list_for_each_entry(filter, &block->rules, list) | ||||||
|  | 			if (filter->type == OCELOT_VCAP_FILTER_PAG && | ||||||
|  | 			    filter->goto_target == chain) | ||||||
|  | 				return filter; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	list_for_each_entry(filter, &ocelot->dummy_rules, list) | ||||||
|  | 		if (filter->goto_target == chain) | ||||||
|  | 			return filter; | ||||||
|  | 
 | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ocelot_flower_parse_action(struct flow_cls_offload *f, bool ingress, | ||||||
| 				      struct ocelot_vcap_filter *filter) | 				      struct ocelot_vcap_filter *filter) | ||||||
| { | { | ||||||
|  | 	struct netlink_ext_ack *extack = f->common.extack; | ||||||
|  | 	bool allow_missing_goto_target = false; | ||||||
| 	const struct flow_action_entry *a; | 	const struct flow_action_entry *a; | ||||||
|  | 	int i, chain; | ||||||
| 	u64 rate; | 	u64 rate; | ||||||
| 	int i; |  | ||||||
| 
 | 
 | ||||||
| 	if (!flow_action_basic_hw_stats_check(&f->rule->action, | 	if (!flow_action_basic_hw_stats_check(&f->rule->action, | ||||||
| 					      f->common.extack)) | 					      f->common.extack)) | ||||||
| 		return -EOPNOTSUPP; | 		return -EOPNOTSUPP; | ||||||
| 
 | 
 | ||||||
|  | 	chain = f->common.chain_index; | ||||||
|  | 	filter->block_id = ocelot_chain_to_block(chain, ingress); | ||||||
|  | 	if (filter->block_id < 0) { | ||||||
|  | 		NL_SET_ERR_MSG_MOD(extack, "Cannot offload to this chain"); | ||||||
|  | 		return -EOPNOTSUPP; | ||||||
|  | 	} | ||||||
|  | 	if (filter->block_id == VCAP_IS1 || filter->block_id == VCAP_IS2) | ||||||
|  | 		filter->lookup = ocelot_chain_to_lookup(chain); | ||||||
|  | 
 | ||||||
|  | 	filter->goto_target = -1; | ||||||
|  | 	filter->type = OCELOT_VCAP_FILTER_DUMMY; | ||||||
|  | 
 | ||||||
| 	flow_action_for_each(i, a, &f->rule->action) { | 	flow_action_for_each(i, a, &f->rule->action) { | ||||||
| 		switch (a->id) { | 		switch (a->id) { | ||||||
| 		case FLOW_ACTION_DROP: | 		case FLOW_ACTION_DROP: | ||||||
|  | 			if (filter->block_id != VCAP_IS2) { | ||||||
|  | 				NL_SET_ERR_MSG_MOD(extack, | ||||||
|  | 						   "Drop action can only be offloaded to VCAP IS2"); | ||||||
|  | 				return -EOPNOTSUPP; | ||||||
|  | 			} | ||||||
|  | 			if (filter->goto_target != -1) { | ||||||
|  | 				NL_SET_ERR_MSG_MOD(extack, | ||||||
|  | 						   "Last action must be GOTO"); | ||||||
|  | 				return -EOPNOTSUPP; | ||||||
|  | 			} | ||||||
| 			filter->action.mask_mode = OCELOT_MASK_MODE_PERMIT_DENY; | 			filter->action.mask_mode = OCELOT_MASK_MODE_PERMIT_DENY; | ||||||
| 			filter->action.port_mask = 0; | 			filter->action.port_mask = 0; | ||||||
| 			filter->action.police_ena = true; | 			filter->action.police_ena = true; | ||||||
| 			filter->action.pol_ix = OCELOT_POLICER_DISCARD; | 			filter->action.pol_ix = OCELOT_POLICER_DISCARD; | ||||||
|  | 			filter->type = OCELOT_VCAP_FILTER_OFFLOAD; | ||||||
| 			break; | 			break; | ||||||
| 		case FLOW_ACTION_TRAP: | 		case FLOW_ACTION_TRAP: | ||||||
|  | 			if (filter->block_id != VCAP_IS2) { | ||||||
|  | 				NL_SET_ERR_MSG_MOD(extack, | ||||||
|  | 						   "Trap action can only be offloaded to VCAP IS2"); | ||||||
|  | 				return -EOPNOTSUPP; | ||||||
|  | 			} | ||||||
|  | 			if (filter->goto_target != -1) { | ||||||
|  | 				NL_SET_ERR_MSG_MOD(extack, | ||||||
|  | 						   "Last action must be GOTO"); | ||||||
|  | 				return -EOPNOTSUPP; | ||||||
|  | 			} | ||||||
| 			filter->action.mask_mode = OCELOT_MASK_MODE_PERMIT_DENY; | 			filter->action.mask_mode = OCELOT_MASK_MODE_PERMIT_DENY; | ||||||
| 			filter->action.port_mask = 0; | 			filter->action.port_mask = 0; | ||||||
| 			filter->action.cpu_copy_ena = true; | 			filter->action.cpu_copy_ena = true; | ||||||
| 			filter->action.cpu_qu_num = 0; | 			filter->action.cpu_qu_num = 0; | ||||||
|  | 			filter->type = OCELOT_VCAP_FILTER_OFFLOAD; | ||||||
| 			break; | 			break; | ||||||
| 		case FLOW_ACTION_POLICE: | 		case FLOW_ACTION_POLICE: | ||||||
|  | 			if (filter->block_id != VCAP_IS2) { | ||||||
|  | 				NL_SET_ERR_MSG_MOD(extack, | ||||||
|  | 						   "Police action can only be offloaded to VCAP IS2"); | ||||||
|  | 				return -EOPNOTSUPP; | ||||||
|  | 			} | ||||||
|  | 			if (filter->goto_target != -1) { | ||||||
|  | 				NL_SET_ERR_MSG_MOD(extack, | ||||||
|  | 						   "Last action must be GOTO"); | ||||||
|  | 				return -EOPNOTSUPP; | ||||||
|  | 			} | ||||||
| 			filter->action.police_ena = true; | 			filter->action.police_ena = true; | ||||||
| 			rate = a->police.rate_bytes_ps; | 			rate = a->police.rate_bytes_ps; | ||||||
| 			filter->action.pol.rate = div_u64(rate, 1000) * 8; | 			filter->action.pol.rate = div_u64(rate, 1000) * 8; | ||||||
| 			filter->action.pol.burst = a->police.burst; | 			filter->action.pol.burst = a->police.burst; | ||||||
|  | 			filter->type = OCELOT_VCAP_FILTER_OFFLOAD; | ||||||
|  | 			break; | ||||||
|  | 		case FLOW_ACTION_GOTO: | ||||||
|  | 			filter->goto_target = a->chain_index; | ||||||
| 			break; | 			break; | ||||||
| 		default: | 		default: | ||||||
|  | 			NL_SET_ERR_MSG_MOD(extack, "Cannot offload action"); | ||||||
| 			return -EOPNOTSUPP; | 			return -EOPNOTSUPP; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (filter->goto_target == -1) { | ||||||
|  | 		if ((filter->block_id == VCAP_IS2 && filter->lookup == 1) || | ||||||
|  | 		    chain == 0) { | ||||||
|  | 			allow_missing_goto_target = true; | ||||||
|  | 		} else { | ||||||
|  | 			NL_SET_ERR_MSG_MOD(extack, "Missing GOTO action"); | ||||||
|  | 			return -EOPNOTSUPP; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!ocelot_is_goto_target_valid(filter->goto_target, chain, ingress) && | ||||||
|  | 	    !allow_missing_goto_target) { | ||||||
|  | 		NL_SET_ERR_MSG_MOD(extack, "Cannot offload this GOTO target"); | ||||||
|  | 		return -EOPNOTSUPP; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int ocelot_flower_parse_key(struct flow_cls_offload *f, | static int ocelot_flower_parse_key(struct flow_cls_offload *f, bool ingress, | ||||||
| 				   struct ocelot_vcap_filter *filter) | 				   struct ocelot_vcap_filter *filter) | ||||||
| { | { | ||||||
| 	struct flow_rule *rule = flow_cls_offload_flow_rule(f); | 	struct flow_rule *rule = flow_cls_offload_flow_rule(f); | ||||||
| @ -185,7 +375,7 @@ finished_key_parsing: | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int ocelot_flower_parse(struct flow_cls_offload *f, | static int ocelot_flower_parse(struct flow_cls_offload *f, bool ingress, | ||||||
| 			       struct ocelot_vcap_filter *filter) | 			       struct ocelot_vcap_filter *filter) | ||||||
| { | { | ||||||
| 	int ret; | 	int ret; | ||||||
| @ -193,11 +383,11 @@ static int ocelot_flower_parse(struct flow_cls_offload *f, | |||||||
| 	filter->prio = f->common.prio; | 	filter->prio = f->common.prio; | ||||||
| 	filter->id = f->cookie; | 	filter->id = f->cookie; | ||||||
| 
 | 
 | ||||||
| 	ret = ocelot_flower_parse_action(f, filter); | 	ret = ocelot_flower_parse_action(f, ingress, filter); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		return ret; | 		return ret; | ||||||
| 
 | 
 | ||||||
| 	return ocelot_flower_parse_key(f, filter); | 	return ocelot_flower_parse_key(f, ingress, filter); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static struct ocelot_vcap_filter | static struct ocelot_vcap_filter | ||||||
| @ -214,22 +404,52 @@ static struct ocelot_vcap_filter | |||||||
| 	return filter; | 	return filter; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static int ocelot_vcap_dummy_filter_add(struct ocelot *ocelot, | ||||||
|  | 					struct ocelot_vcap_filter *filter) | ||||||
|  | { | ||||||
|  | 	list_add(&filter->list, &ocelot->dummy_rules); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ocelot_vcap_dummy_filter_del(struct ocelot *ocelot, | ||||||
|  | 					struct ocelot_vcap_filter *filter) | ||||||
|  | { | ||||||
|  | 	list_del(&filter->list); | ||||||
|  | 	kfree(filter); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| int ocelot_cls_flower_replace(struct ocelot *ocelot, int port, | int ocelot_cls_flower_replace(struct ocelot *ocelot, int port, | ||||||
| 			      struct flow_cls_offload *f, bool ingress) | 			      struct flow_cls_offload *f, bool ingress) | ||||||
| { | { | ||||||
|  | 	struct netlink_ext_ack *extack = f->common.extack; | ||||||
| 	struct ocelot_vcap_filter *filter; | 	struct ocelot_vcap_filter *filter; | ||||||
|  | 	int chain = f->common.chain_index; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
|  | 	if (chain && !ocelot_find_vcap_filter_that_points_at(ocelot, chain)) { | ||||||
|  | 		NL_SET_ERR_MSG_MOD(extack, "No default GOTO action points to this chain"); | ||||||
|  | 		return -EOPNOTSUPP; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	filter = ocelot_vcap_filter_create(ocelot, port, f); | 	filter = ocelot_vcap_filter_create(ocelot, port, f); | ||||||
| 	if (!filter) | 	if (!filter) | ||||||
| 		return -ENOMEM; | 		return -ENOMEM; | ||||||
| 
 | 
 | ||||||
| 	ret = ocelot_flower_parse(f, filter); | 	ret = ocelot_flower_parse(f, ingress, filter); | ||||||
| 	if (ret) { | 	if (ret) { | ||||||
| 		kfree(filter); | 		kfree(filter); | ||||||
| 		return ret; | 		return ret; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/* The non-optional GOTOs for the TCAM skeleton don't need
 | ||||||
|  | 	 * to be actually offloaded. | ||||||
|  | 	 */ | ||||||
|  | 	if (filter->type == OCELOT_VCAP_FILTER_DUMMY) | ||||||
|  | 		return ocelot_vcap_dummy_filter_add(ocelot, filter); | ||||||
|  | 
 | ||||||
| 	return ocelot_vcap_filter_add(ocelot, filter, f->common.extack); | 	return ocelot_vcap_filter_add(ocelot, filter, f->common.extack); | ||||||
| } | } | ||||||
| EXPORT_SYMBOL_GPL(ocelot_cls_flower_replace); | EXPORT_SYMBOL_GPL(ocelot_cls_flower_replace); | ||||||
| @ -237,13 +457,23 @@ EXPORT_SYMBOL_GPL(ocelot_cls_flower_replace); | |||||||
| int ocelot_cls_flower_destroy(struct ocelot *ocelot, int port, | int ocelot_cls_flower_destroy(struct ocelot *ocelot, int port, | ||||||
| 			      struct flow_cls_offload *f, bool ingress) | 			      struct flow_cls_offload *f, bool ingress) | ||||||
| { | { | ||||||
| 	struct ocelot_vcap_block *block = &ocelot->block; |  | ||||||
| 	struct ocelot_vcap_filter *filter; | 	struct ocelot_vcap_filter *filter; | ||||||
|  | 	struct ocelot_vcap_block *block; | ||||||
|  | 	int block_id; | ||||||
|  | 
 | ||||||
|  | 	block_id = ocelot_chain_to_block(f->common.chain_index, ingress); | ||||||
|  | 	if (block_id < 0) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	block = &ocelot->block[block_id]; | ||||||
| 
 | 
 | ||||||
| 	filter = ocelot_vcap_block_find_filter_by_id(block, f->cookie); | 	filter = ocelot_vcap_block_find_filter_by_id(block, f->cookie); | ||||||
| 	if (!filter) | 	if (!filter) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
|  | 	if (filter->type == OCELOT_VCAP_FILTER_DUMMY) | ||||||
|  | 		return ocelot_vcap_dummy_filter_del(ocelot, filter); | ||||||
|  | 
 | ||||||
| 	return ocelot_vcap_filter_del(ocelot, filter); | 	return ocelot_vcap_filter_del(ocelot, filter); | ||||||
| } | } | ||||||
| EXPORT_SYMBOL_GPL(ocelot_cls_flower_destroy); | EXPORT_SYMBOL_GPL(ocelot_cls_flower_destroy); | ||||||
| @ -251,12 +481,18 @@ EXPORT_SYMBOL_GPL(ocelot_cls_flower_destroy); | |||||||
| int ocelot_cls_flower_stats(struct ocelot *ocelot, int port, | int ocelot_cls_flower_stats(struct ocelot *ocelot, int port, | ||||||
| 			    struct flow_cls_offload *f, bool ingress) | 			    struct flow_cls_offload *f, bool ingress) | ||||||
| { | { | ||||||
| 	struct ocelot_vcap_block *block = &ocelot->block; |  | ||||||
| 	struct ocelot_vcap_filter *filter; | 	struct ocelot_vcap_filter *filter; | ||||||
| 	int ret; | 	struct ocelot_vcap_block *block; | ||||||
|  | 	int block_id, ret; | ||||||
|  | 
 | ||||||
|  | 	block_id = ocelot_chain_to_block(f->common.chain_index, ingress); | ||||||
|  | 	if (block_id < 0) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	block = &ocelot->block[block_id]; | ||||||
| 
 | 
 | ||||||
| 	filter = ocelot_vcap_block_find_filter_by_id(block, f->cookie); | 	filter = ocelot_vcap_block_find_filter_by_id(block, f->cookie); | ||||||
| 	if (!filter) | 	if (!filter || filter->type == OCELOT_VCAP_FILTER_DUMMY) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	ret = ocelot_vcap_filter_stats_update(ocelot, filter); | 	ret = ocelot_vcap_filter_stats_update(ocelot, filter); | ||||||
|  | |||||||
| @ -640,10 +640,10 @@ static void is2_entry_set(struct ocelot *ocelot, int ix, | |||||||
| 	vcap_row_cmd(ocelot, vcap, row, VCAP_CMD_WRITE, VCAP_SEL_ALL); | 	vcap_row_cmd(ocelot, vcap, row, VCAP_CMD_WRITE, VCAP_SEL_ALL); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void | static void vcap_entry_get(struct ocelot *ocelot, int ix, | ||||||
| vcap_entry_get(struct ocelot *ocelot, struct ocelot_vcap_filter *filter, int ix) | 			   struct ocelot_vcap_filter *filter) | ||||||
| { | { | ||||||
| 	const struct vcap_props *vcap = &ocelot->vcap[VCAP_IS2]; | 	const struct vcap_props *vcap = &ocelot->vcap[filter->block_id]; | ||||||
| 	struct vcap_data data; | 	struct vcap_data data; | ||||||
| 	int row, count; | 	int row, count; | ||||||
| 	u32 cnt; | 	u32 cnt; | ||||||
| @ -660,6 +660,13 @@ vcap_entry_get(struct ocelot *ocelot, struct ocelot_vcap_filter *filter, int ix) | |||||||
| 	filter->stats.pkts = cnt; | 	filter->stats.pkts = cnt; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void vcap_entry_set(struct ocelot *ocelot, int ix, | ||||||
|  | 			   struct ocelot_vcap_filter *filter) | ||||||
|  | { | ||||||
|  | 	if (filter->block_id == VCAP_IS2) | ||||||
|  | 		return is2_entry_set(ocelot, ix, filter); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int ocelot_vcap_policer_add(struct ocelot *ocelot, u32 pol_ix, | static int ocelot_vcap_policer_add(struct ocelot *ocelot, u32 pol_ix, | ||||||
| 				   struct ocelot_policer *pol) | 				   struct ocelot_policer *pol) | ||||||
| { | { | ||||||
| @ -688,7 +695,8 @@ static void ocelot_vcap_policer_del(struct ocelot *ocelot, | |||||||
| 
 | 
 | ||||||
| 	list_for_each_entry(filter, &block->rules, list) { | 	list_for_each_entry(filter, &block->rules, list) { | ||||||
| 		index++; | 		index++; | ||||||
| 		if (filter->action.police_ena && | 		if (filter->block_id == VCAP_IS2 && | ||||||
|  | 		    filter->action.police_ena && | ||||||
| 		    filter->action.pol_ix < pol_ix) { | 		    filter->action.pol_ix < pol_ix) { | ||||||
| 			filter->action.pol_ix += 1; | 			filter->action.pol_ix += 1; | ||||||
| 			ocelot_vcap_policer_add(ocelot, filter->action.pol_ix, | 			ocelot_vcap_policer_add(ocelot, filter->action.pol_ix, | ||||||
| @ -710,7 +718,7 @@ static void ocelot_vcap_filter_add_to_block(struct ocelot *ocelot, | |||||||
| 	struct ocelot_vcap_filter *tmp; | 	struct ocelot_vcap_filter *tmp; | ||||||
| 	struct list_head *pos, *n; | 	struct list_head *pos, *n; | ||||||
| 
 | 
 | ||||||
| 	if (filter->action.police_ena) { | 	if (filter->block_id == VCAP_IS2 && filter->action.police_ena) { | ||||||
| 		block->pol_lpr--; | 		block->pol_lpr--; | ||||||
| 		filter->action.pol_ix = block->pol_lpr; | 		filter->action.pol_ix = block->pol_lpr; | ||||||
| 		ocelot_vcap_policer_add(ocelot, filter->action.pol_ix, | 		ocelot_vcap_policer_add(ocelot, filter->action.pol_ix, | ||||||
| @ -848,7 +856,7 @@ static bool | |||||||
| ocelot_exclusive_mac_etype_filter_rules(struct ocelot *ocelot, | ocelot_exclusive_mac_etype_filter_rules(struct ocelot *ocelot, | ||||||
| 					struct ocelot_vcap_filter *filter) | 					struct ocelot_vcap_filter *filter) | ||||||
| { | { | ||||||
| 	struct ocelot_vcap_block *block = &ocelot->block; | 	struct ocelot_vcap_block *block = &ocelot->block[filter->block_id]; | ||||||
| 	struct ocelot_vcap_filter *tmp; | 	struct ocelot_vcap_filter *tmp; | ||||||
| 	unsigned long port; | 	unsigned long port; | ||||||
| 	int i; | 	int i; | ||||||
| @ -886,7 +894,7 @@ int ocelot_vcap_filter_add(struct ocelot *ocelot, | |||||||
| 			   struct ocelot_vcap_filter *filter, | 			   struct ocelot_vcap_filter *filter, | ||||||
| 			   struct netlink_ext_ack *extack) | 			   struct netlink_ext_ack *extack) | ||||||
| { | { | ||||||
| 	struct ocelot_vcap_block *block = &ocelot->block; | 	struct ocelot_vcap_block *block = &ocelot->block[filter->block_id]; | ||||||
| 	int i, index; | 	int i, index; | ||||||
| 
 | 
 | ||||||
| 	if (!ocelot_exclusive_mac_etype_filter_rules(ocelot, filter)) { | 	if (!ocelot_exclusive_mac_etype_filter_rules(ocelot, filter)) { | ||||||
| @ -908,11 +916,11 @@ int ocelot_vcap_filter_add(struct ocelot *ocelot, | |||||||
| 		struct ocelot_vcap_filter *tmp; | 		struct ocelot_vcap_filter *tmp; | ||||||
| 
 | 
 | ||||||
| 		tmp = ocelot_vcap_block_find_filter_by_index(block, i); | 		tmp = ocelot_vcap_block_find_filter_by_index(block, i); | ||||||
| 		is2_entry_set(ocelot, i, tmp); | 		vcap_entry_set(ocelot, i, tmp); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Now insert the new filter */ | 	/* Now insert the new filter */ | ||||||
| 	is2_entry_set(ocelot, index, filter); | 	vcap_entry_set(ocelot, index, filter); | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -926,7 +934,8 @@ static void ocelot_vcap_block_remove_filter(struct ocelot *ocelot, | |||||||
| 	list_for_each_safe(pos, q, &block->rules) { | 	list_for_each_safe(pos, q, &block->rules) { | ||||||
| 		tmp = list_entry(pos, struct ocelot_vcap_filter, list); | 		tmp = list_entry(pos, struct ocelot_vcap_filter, list); | ||||||
| 		if (tmp->id == filter->id) { | 		if (tmp->id == filter->id) { | ||||||
| 			if (tmp->action.police_ena) | 			if (tmp->block_id == VCAP_IS2 && | ||||||
|  | 			    tmp->action.police_ena) | ||||||
| 				ocelot_vcap_policer_del(ocelot, block, | 				ocelot_vcap_policer_del(ocelot, block, | ||||||
| 							tmp->action.pol_ix); | 							tmp->action.pol_ix); | ||||||
| 
 | 
 | ||||||
| @ -941,7 +950,7 @@ static void ocelot_vcap_block_remove_filter(struct ocelot *ocelot, | |||||||
| int ocelot_vcap_filter_del(struct ocelot *ocelot, | int ocelot_vcap_filter_del(struct ocelot *ocelot, | ||||||
| 			   struct ocelot_vcap_filter *filter) | 			   struct ocelot_vcap_filter *filter) | ||||||
| { | { | ||||||
| 	struct ocelot_vcap_block *block = &ocelot->block; | 	struct ocelot_vcap_block *block = &ocelot->block[filter->block_id]; | ||||||
| 	struct ocelot_vcap_filter del_filter; | 	struct ocelot_vcap_filter del_filter; | ||||||
| 	int i, index; | 	int i, index; | ||||||
| 
 | 
 | ||||||
| @ -960,11 +969,11 @@ int ocelot_vcap_filter_del(struct ocelot *ocelot, | |||||||
| 		struct ocelot_vcap_filter *tmp; | 		struct ocelot_vcap_filter *tmp; | ||||||
| 
 | 
 | ||||||
| 		tmp = ocelot_vcap_block_find_filter_by_index(block, i); | 		tmp = ocelot_vcap_block_find_filter_by_index(block, i); | ||||||
| 		is2_entry_set(ocelot, i, tmp); | 		vcap_entry_set(ocelot, i, tmp); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Now delete the last filter, because it is duplicated */ | 	/* Now delete the last filter, because it is duplicated */ | ||||||
| 	is2_entry_set(ocelot, block->count, &del_filter); | 	vcap_entry_set(ocelot, block->count, &del_filter); | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| @ -972,7 +981,7 @@ int ocelot_vcap_filter_del(struct ocelot *ocelot, | |||||||
| int ocelot_vcap_filter_stats_update(struct ocelot *ocelot, | int ocelot_vcap_filter_stats_update(struct ocelot *ocelot, | ||||||
| 				    struct ocelot_vcap_filter *filter) | 				    struct ocelot_vcap_filter *filter) | ||||||
| { | { | ||||||
| 	struct ocelot_vcap_block *block = &ocelot->block; | 	struct ocelot_vcap_block *block = &ocelot->block[filter->block_id]; | ||||||
| 	struct ocelot_vcap_filter tmp; | 	struct ocelot_vcap_filter tmp; | ||||||
| 	int index; | 	int index; | ||||||
| 
 | 
 | ||||||
| @ -980,12 +989,12 @@ int ocelot_vcap_filter_stats_update(struct ocelot *ocelot, | |||||||
| 	if (index < 0) | 	if (index < 0) | ||||||
| 		return index; | 		return index; | ||||||
| 
 | 
 | ||||||
| 	vcap_entry_get(ocelot, filter, index); | 	vcap_entry_get(ocelot, index, filter); | ||||||
| 
 | 
 | ||||||
| 	/* After we get the result we need to clear the counters */ | 	/* After we get the result we need to clear the counters */ | ||||||
| 	tmp = *filter; | 	tmp = *filter; | ||||||
| 	tmp.stats.pkts = 0; | 	tmp.stats.pkts = 0; | ||||||
| 	is2_entry_set(ocelot, index, &tmp); | 	vcap_entry_set(ocelot, index, &tmp); | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| @ -1080,7 +1089,6 @@ static void ocelot_vcap_detect_constants(struct ocelot *ocelot, | |||||||
| 
 | 
 | ||||||
| int ocelot_vcap_init(struct ocelot *ocelot) | int ocelot_vcap_init(struct ocelot *ocelot) | ||||||
| { | { | ||||||
| 	struct ocelot_vcap_block *block = &ocelot->block; |  | ||||||
| 	int i; | 	int i; | ||||||
| 
 | 
 | ||||||
| 	/* Create a policer that will drop the frames for the cpu.
 | 	/* Create a policer that will drop the frames for the cpu.
 | ||||||
| @ -1099,15 +1107,17 @@ int ocelot_vcap_init(struct ocelot *ocelot) | |||||||
| 			 OCELOT_POLICER_DISCARD); | 			 OCELOT_POLICER_DISCARD); | ||||||
| 
 | 
 | ||||||
| 	for (i = 0; i < OCELOT_NUM_VCAP_BLOCKS; i++) { | 	for (i = 0; i < OCELOT_NUM_VCAP_BLOCKS; i++) { | ||||||
|  | 		struct ocelot_vcap_block *block = &ocelot->block[i]; | ||||||
| 		struct vcap_props *vcap = &ocelot->vcap[i]; | 		struct vcap_props *vcap = &ocelot->vcap[i]; | ||||||
| 
 | 
 | ||||||
|  | 		INIT_LIST_HEAD(&block->rules); | ||||||
|  | 		block->pol_lpr = OCELOT_POLICER_DISCARD - 1; | ||||||
|  | 
 | ||||||
| 		ocelot_vcap_detect_constants(ocelot, vcap); | 		ocelot_vcap_detect_constants(ocelot, vcap); | ||||||
| 		ocelot_vcap_init_one(ocelot, vcap); | 		ocelot_vcap_init_one(ocelot, vcap); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	block->pol_lpr = OCELOT_POLICER_DISCARD - 1; | 	INIT_LIST_HEAD(&ocelot->dummy_rules); | ||||||
| 
 |  | ||||||
| 	INIT_LIST_HEAD(&block->rules); |  | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  | |||||||
| @ -204,9 +204,19 @@ struct ocelot_vcap_stats { | |||||||
| 	u64 used; | 	u64 used; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | enum ocelot_vcap_filter_type { | ||||||
|  | 	OCELOT_VCAP_FILTER_DUMMY, | ||||||
|  | 	OCELOT_VCAP_FILTER_PAG, | ||||||
|  | 	OCELOT_VCAP_FILTER_OFFLOAD, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| struct ocelot_vcap_filter { | struct ocelot_vcap_filter { | ||||||
| 	struct list_head list; | 	struct list_head list; | ||||||
| 
 | 
 | ||||||
|  | 	enum ocelot_vcap_filter_type type; | ||||||
|  | 	int block_id; | ||||||
|  | 	int goto_target; | ||||||
|  | 	int lookup; | ||||||
| 	u16 prio; | 	u16 prio; | ||||||
| 	u32 id; | 	u32 id; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -633,7 +633,8 @@ struct ocelot { | |||||||
| 
 | 
 | ||||||
| 	struct list_head		multicast; | 	struct list_head		multicast; | ||||||
| 
 | 
 | ||||||
| 	struct ocelot_vcap_block	block; | 	struct list_head		dummy_rules; | ||||||
|  | 	struct ocelot_vcap_block	block[3]; | ||||||
| 	struct vcap_props		*vcap; | 	struct vcap_props		*vcap; | ||||||
| 
 | 
 | ||||||
| 	/* Workqueue to check statistics for overflow with its lock */ | 	/* Workqueue to check statistics for overflow with its lock */ | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user