Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

net: mscc: ocelot: create a function that replaces an existing VCAP filter

VCAP (Versatile Content Aware Processor) is the TCAM-based engine behind
tc flower offload on ocelot, among other things. The ingress port mask
on which VCAP rules match is present as a bit field in the actual key of
the rule. This means that it is possible for a rule to be shared among
multiple source ports. When the rule is added one by one on each desired
port, that the ingress port mask of the key must be edited and rewritten
to hardware.

But the API in ocelot_vcap.c does not allow for this. For one thing,
ocelot_vcap_filter_add() and ocelot_vcap_filter_del() are not symmetric,
because ocelot_vcap_filter_add() works with a preallocated and
prepopulated filter and programs it to hardware, and
ocelot_vcap_filter_del() does both the job of removing the specified
filter from hardware, as well as kfreeing it. That is to say, the only
option of editing a filter in place, which is to delete it, modify the
structure and add it back, does not work because it results in
use-after-free.

This patch introduces ocelot_vcap_filter_replace, which trivially
reprograms a VCAP entry to hardware, at the exact same index at which it
existed before, without modifying any list or allocating any memory.

Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com>
Acked-by: Richard Cochran <richardcochran@gmail.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>

authored by

Vladimir Oltean and committed by
Jakub Kicinski
95706be1 8a075464

+18
+16
drivers/net/ethernet/mscc/ocelot_vcap.c
··· 1217 1217 } 1218 1218 EXPORT_SYMBOL(ocelot_vcap_filter_del); 1219 1219 1220 + int ocelot_vcap_filter_replace(struct ocelot *ocelot, 1221 + struct ocelot_vcap_filter *filter) 1222 + { 1223 + struct ocelot_vcap_block *block = &ocelot->block[filter->block_id]; 1224 + int index; 1225 + 1226 + index = ocelot_vcap_block_get_filter_index(block, filter); 1227 + if (index < 0) 1228 + return index; 1229 + 1230 + vcap_entry_set(ocelot, index, filter); 1231 + 1232 + return 0; 1233 + } 1234 + EXPORT_SYMBOL(ocelot_vcap_filter_replace); 1235 + 1220 1236 int ocelot_vcap_filter_stats_update(struct ocelot *ocelot, 1221 1237 struct ocelot_vcap_filter *filter) 1222 1238 {
+2
include/soc/mscc/ocelot_vcap.h
··· 703 703 struct netlink_ext_ack *extack); 704 704 int ocelot_vcap_filter_del(struct ocelot *ocelot, 705 705 struct ocelot_vcap_filter *rule); 706 + int ocelot_vcap_filter_replace(struct ocelot *ocelot, 707 + struct ocelot_vcap_filter *filter); 706 708 struct ocelot_vcap_filter * 707 709 ocelot_vcap_block_find_filter_by_id(struct ocelot_vcap_block *block, 708 710 unsigned long cookie, bool tc_offload);