1 // SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
2 /* Copyright (c) 2021 Mellanox Technologies. */
3
4 #include <linux/build_bug.h>
5 #include <linux/list.h>
6 #include <linux/notifier.h>
7 #include <net/netevent.h>
8 #include <net/switchdev.h>
9 #include "lib/devcom.h"
10 #include "bridge.h"
11 #include "eswitch.h"
12 #include "bridge_priv.h"
13 #define CREATE_TRACE_POINTS
14 #include "diag/bridge_tracepoint.h"
15
16 static const struct rhashtable_params fdb_ht_params = {
17 .key_offset = offsetof(struct mlx5_esw_bridge_fdb_entry, key),
18 .key_len = sizeof(struct mlx5_esw_bridge_fdb_key),
19 .head_offset = offsetof(struct mlx5_esw_bridge_fdb_entry, ht_node),
20 .automatic_shrinking = true,
21 };
22
23 static void
mlx5_esw_bridge_fdb_offload_notify(struct net_device * dev,const unsigned char * addr,u16 vid,unsigned long val)24 mlx5_esw_bridge_fdb_offload_notify(struct net_device *dev, const unsigned char *addr, u16 vid,
25 unsigned long val)
26 {
27 struct switchdev_notifier_fdb_info send_info = {};
28
29 send_info.addr = addr;
30 send_info.vid = vid;
31 send_info.offloaded = true;
32 call_switchdev_notifiers(val, dev, &send_info.info, NULL);
33 }
34
35 static void
mlx5_esw_bridge_fdb_del_notify(struct mlx5_esw_bridge_fdb_entry * entry)36 mlx5_esw_bridge_fdb_del_notify(struct mlx5_esw_bridge_fdb_entry *entry)
37 {
38 if (!(entry->flags & (MLX5_ESW_BRIDGE_FLAG_ADDED_BY_USER | MLX5_ESW_BRIDGE_FLAG_PEER)))
39 mlx5_esw_bridge_fdb_offload_notify(entry->dev, entry->key.addr,
40 entry->key.vid,
41 SWITCHDEV_FDB_DEL_TO_BRIDGE);
42 }
43
mlx5_esw_bridge_pkt_reformat_vlan_pop_supported(struct mlx5_eswitch * esw)44 static bool mlx5_esw_bridge_pkt_reformat_vlan_pop_supported(struct mlx5_eswitch *esw)
45 {
46 return BIT(MLX5_CAP_ESW_FLOWTABLE_FDB(esw->dev, reformat_remove)) &&
47 MLX5_CAP_GEN_2(esw->dev, max_reformat_remove_size) >= sizeof(struct vlan_hdr) &&
48 MLX5_CAP_GEN_2(esw->dev, max_reformat_remove_offset) >=
49 offsetof(struct vlan_ethhdr, h_vlan_proto);
50 }
51
52 static struct mlx5_pkt_reformat *
mlx5_esw_bridge_pkt_reformat_vlan_pop_create(struct mlx5_eswitch * esw)53 mlx5_esw_bridge_pkt_reformat_vlan_pop_create(struct mlx5_eswitch *esw)
54 {
55 struct mlx5_pkt_reformat_params reformat_params = {};
56
57 reformat_params.type = MLX5_REFORMAT_TYPE_REMOVE_HDR;
58 reformat_params.param_0 = MLX5_REFORMAT_CONTEXT_ANCHOR_MAC_START;
59 reformat_params.param_1 = offsetof(struct vlan_ethhdr, h_vlan_proto);
60 reformat_params.size = sizeof(struct vlan_hdr);
61 return mlx5_packet_reformat_alloc(esw->dev, &reformat_params, MLX5_FLOW_NAMESPACE_FDB);
62 }
63
64 struct mlx5_flow_table *
mlx5_esw_bridge_table_create(int max_fte,u32 level,struct mlx5_eswitch * esw)65 mlx5_esw_bridge_table_create(int max_fte, u32 level, struct mlx5_eswitch *esw)
66 {
67 struct mlx5_flow_table_attr ft_attr = {};
68 struct mlx5_core_dev *dev = esw->dev;
69 struct mlx5_flow_namespace *ns;
70 struct mlx5_flow_table *fdb;
71
72 ns = mlx5_get_flow_namespace(dev, MLX5_FLOW_NAMESPACE_FDB);
73 if (!ns) {
74 esw_warn(dev, "Failed to get FDB namespace\n");
75 return ERR_PTR(-ENOENT);
76 }
77
78 ft_attr.flags = MLX5_FLOW_TABLE_TUNNEL_EN_REFORMAT;
79 ft_attr.max_fte = max_fte;
80 ft_attr.level = level;
81 ft_attr.prio = FDB_BR_OFFLOAD;
82 fdb = mlx5_create_flow_table(ns, &ft_attr);
83 if (IS_ERR(fdb))
84 esw_warn(dev, "Failed to create bridge FDB Table (err=%ld)\n", PTR_ERR(fdb));
85
86 return fdb;
87 }
88
89 static struct mlx5_flow_group *
mlx5_esw_bridge_ingress_vlan_proto_fg_create(unsigned int from,unsigned int to,u16 vlan_proto,struct mlx5_eswitch * esw,struct mlx5_flow_table * ingress_ft)90 mlx5_esw_bridge_ingress_vlan_proto_fg_create(unsigned int from, unsigned int to, u16 vlan_proto,
91 struct mlx5_eswitch *esw,
92 struct mlx5_flow_table *ingress_ft)
93 {
94 int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
95 struct mlx5_flow_group *fg;
96 u32 *in, *match;
97
98 in = kvzalloc(inlen, GFP_KERNEL);
99 if (!in)
100 return ERR_PTR(-ENOMEM);
101
102 MLX5_SET(create_flow_group_in, in, match_criteria_enable,
103 MLX5_MATCH_OUTER_HEADERS | MLX5_MATCH_MISC_PARAMETERS_2);
104 match = MLX5_ADDR_OF(create_flow_group_in, in, match_criteria);
105
106 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.smac_47_16);
107 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.smac_15_0);
108 if (vlan_proto == ETH_P_8021Q)
109 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.cvlan_tag);
110 else if (vlan_proto == ETH_P_8021AD)
111 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.svlan_tag);
112 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.first_vid);
113
114 MLX5_SET(fte_match_param, match, misc_parameters_2.metadata_reg_c_0,
115 mlx5_eswitch_get_vport_metadata_mask());
116
117 MLX5_SET(create_flow_group_in, in, start_flow_index, from);
118 MLX5_SET(create_flow_group_in, in, end_flow_index, to);
119
120 fg = mlx5_create_flow_group(ingress_ft, in);
121 kvfree(in);
122 if (IS_ERR(fg))
123 esw_warn(esw->dev,
124 "Failed to create VLAN(proto=%x) flow group for bridge ingress table (err=%ld)\n",
125 vlan_proto, PTR_ERR(fg));
126
127 return fg;
128 }
129
130 static struct mlx5_flow_group *
mlx5_esw_bridge_ingress_vlan_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * ingress_ft)131 mlx5_esw_bridge_ingress_vlan_fg_create(struct mlx5_eswitch *esw,
132 struct mlx5_flow_table *ingress_ft)
133 {
134 unsigned int from = MLX5_ESW_BRIDGE_INGRESS_TABLE_VLAN_GRP_IDX_FROM;
135 unsigned int to = MLX5_ESW_BRIDGE_INGRESS_TABLE_VLAN_GRP_IDX_TO;
136
137 return mlx5_esw_bridge_ingress_vlan_proto_fg_create(from, to, ETH_P_8021Q, esw, ingress_ft);
138 }
139
140 static struct mlx5_flow_group *
mlx5_esw_bridge_ingress_qinq_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * ingress_ft)141 mlx5_esw_bridge_ingress_qinq_fg_create(struct mlx5_eswitch *esw,
142 struct mlx5_flow_table *ingress_ft)
143 {
144 unsigned int from = MLX5_ESW_BRIDGE_INGRESS_TABLE_QINQ_GRP_IDX_FROM;
145 unsigned int to = MLX5_ESW_BRIDGE_INGRESS_TABLE_QINQ_GRP_IDX_TO;
146
147 return mlx5_esw_bridge_ingress_vlan_proto_fg_create(from, to, ETH_P_8021AD, esw,
148 ingress_ft);
149 }
150
151 static struct mlx5_flow_group *
mlx5_esw_bridge_ingress_vlan_proto_filter_fg_create(unsigned int from,unsigned int to,u16 vlan_proto,struct mlx5_eswitch * esw,struct mlx5_flow_table * ingress_ft)152 mlx5_esw_bridge_ingress_vlan_proto_filter_fg_create(unsigned int from, unsigned int to,
153 u16 vlan_proto, struct mlx5_eswitch *esw,
154 struct mlx5_flow_table *ingress_ft)
155 {
156 int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
157 struct mlx5_flow_group *fg;
158 u32 *in, *match;
159
160 in = kvzalloc(inlen, GFP_KERNEL);
161 if (!in)
162 return ERR_PTR(-ENOMEM);
163
164 MLX5_SET(create_flow_group_in, in, match_criteria_enable,
165 MLX5_MATCH_OUTER_HEADERS | MLX5_MATCH_MISC_PARAMETERS_2);
166 match = MLX5_ADDR_OF(create_flow_group_in, in, match_criteria);
167
168 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.smac_47_16);
169 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.smac_15_0);
170 if (vlan_proto == ETH_P_8021Q)
171 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.cvlan_tag);
172 else if (vlan_proto == ETH_P_8021AD)
173 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.svlan_tag);
174 MLX5_SET(fte_match_param, match, misc_parameters_2.metadata_reg_c_0,
175 mlx5_eswitch_get_vport_metadata_mask());
176
177 MLX5_SET(create_flow_group_in, in, start_flow_index, from);
178 MLX5_SET(create_flow_group_in, in, end_flow_index, to);
179
180 fg = mlx5_create_flow_group(ingress_ft, in);
181 if (IS_ERR(fg))
182 esw_warn(esw->dev,
183 "Failed to create bridge ingress table VLAN filter flow group (err=%ld)\n",
184 PTR_ERR(fg));
185 kvfree(in);
186 return fg;
187 }
188
189 static struct mlx5_flow_group *
mlx5_esw_bridge_ingress_vlan_filter_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * ingress_ft)190 mlx5_esw_bridge_ingress_vlan_filter_fg_create(struct mlx5_eswitch *esw,
191 struct mlx5_flow_table *ingress_ft)
192 {
193 unsigned int from = MLX5_ESW_BRIDGE_INGRESS_TABLE_VLAN_FILTER_GRP_IDX_FROM;
194 unsigned int to = MLX5_ESW_BRIDGE_INGRESS_TABLE_VLAN_FILTER_GRP_IDX_TO;
195
196 return mlx5_esw_bridge_ingress_vlan_proto_filter_fg_create(from, to, ETH_P_8021Q, esw,
197 ingress_ft);
198 }
199
200 static struct mlx5_flow_group *
mlx5_esw_bridge_ingress_qinq_filter_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * ingress_ft)201 mlx5_esw_bridge_ingress_qinq_filter_fg_create(struct mlx5_eswitch *esw,
202 struct mlx5_flow_table *ingress_ft)
203 {
204 unsigned int from = MLX5_ESW_BRIDGE_INGRESS_TABLE_QINQ_FILTER_GRP_IDX_FROM;
205 unsigned int to = MLX5_ESW_BRIDGE_INGRESS_TABLE_QINQ_FILTER_GRP_IDX_TO;
206
207 return mlx5_esw_bridge_ingress_vlan_proto_filter_fg_create(from, to, ETH_P_8021AD, esw,
208 ingress_ft);
209 }
210
211 static struct mlx5_flow_group *
mlx5_esw_bridge_ingress_mac_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * ingress_ft)212 mlx5_esw_bridge_ingress_mac_fg_create(struct mlx5_eswitch *esw, struct mlx5_flow_table *ingress_ft)
213 {
214 int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
215 struct mlx5_flow_group *fg;
216 u32 *in, *match;
217
218 in = kvzalloc(inlen, GFP_KERNEL);
219 if (!in)
220 return ERR_PTR(-ENOMEM);
221
222 MLX5_SET(create_flow_group_in, in, match_criteria_enable,
223 MLX5_MATCH_OUTER_HEADERS | MLX5_MATCH_MISC_PARAMETERS_2);
224 match = MLX5_ADDR_OF(create_flow_group_in, in, match_criteria);
225
226 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.smac_47_16);
227 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.smac_15_0);
228
229 MLX5_SET(fte_match_param, match, misc_parameters_2.metadata_reg_c_0,
230 mlx5_eswitch_get_vport_metadata_mask());
231
232 MLX5_SET(create_flow_group_in, in, start_flow_index,
233 MLX5_ESW_BRIDGE_INGRESS_TABLE_MAC_GRP_IDX_FROM);
234 MLX5_SET(create_flow_group_in, in, end_flow_index,
235 MLX5_ESW_BRIDGE_INGRESS_TABLE_MAC_GRP_IDX_TO);
236
237 fg = mlx5_create_flow_group(ingress_ft, in);
238 if (IS_ERR(fg))
239 esw_warn(esw->dev,
240 "Failed to create MAC flow group for bridge ingress table (err=%ld)\n",
241 PTR_ERR(fg));
242
243 kvfree(in);
244 return fg;
245 }
246
247 static struct mlx5_flow_group *
mlx5_esw_bridge_egress_vlan_proto_fg_create(unsigned int from,unsigned int to,u16 vlan_proto,struct mlx5_eswitch * esw,struct mlx5_flow_table * egress_ft)248 mlx5_esw_bridge_egress_vlan_proto_fg_create(unsigned int from, unsigned int to, u16 vlan_proto,
249 struct mlx5_eswitch *esw,
250 struct mlx5_flow_table *egress_ft)
251 {
252 int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
253 struct mlx5_flow_group *fg;
254 u32 *in, *match;
255
256 in = kvzalloc(inlen, GFP_KERNEL);
257 if (!in)
258 return ERR_PTR(-ENOMEM);
259
260 MLX5_SET(create_flow_group_in, in, match_criteria_enable, MLX5_MATCH_OUTER_HEADERS);
261 match = MLX5_ADDR_OF(create_flow_group_in, in, match_criteria);
262
263 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.dmac_47_16);
264 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.dmac_15_0);
265 if (vlan_proto == ETH_P_8021Q)
266 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.cvlan_tag);
267 else if (vlan_proto == ETH_P_8021AD)
268 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.svlan_tag);
269 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.first_vid);
270
271 MLX5_SET(create_flow_group_in, in, start_flow_index, from);
272 MLX5_SET(create_flow_group_in, in, end_flow_index, to);
273
274 fg = mlx5_create_flow_group(egress_ft, in);
275 if (IS_ERR(fg))
276 esw_warn(esw->dev,
277 "Failed to create VLAN flow group for bridge egress table (err=%ld)\n",
278 PTR_ERR(fg));
279 kvfree(in);
280 return fg;
281 }
282
283 static struct mlx5_flow_group *
mlx5_esw_bridge_egress_vlan_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * egress_ft)284 mlx5_esw_bridge_egress_vlan_fg_create(struct mlx5_eswitch *esw, struct mlx5_flow_table *egress_ft)
285 {
286 unsigned int from = MLX5_ESW_BRIDGE_EGRESS_TABLE_VLAN_GRP_IDX_FROM;
287 unsigned int to = MLX5_ESW_BRIDGE_EGRESS_TABLE_VLAN_GRP_IDX_TO;
288
289 return mlx5_esw_bridge_egress_vlan_proto_fg_create(from, to, ETH_P_8021Q, esw, egress_ft);
290 }
291
292 static struct mlx5_flow_group *
mlx5_esw_bridge_egress_qinq_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * egress_ft)293 mlx5_esw_bridge_egress_qinq_fg_create(struct mlx5_eswitch *esw,
294 struct mlx5_flow_table *egress_ft)
295 {
296 unsigned int from = MLX5_ESW_BRIDGE_EGRESS_TABLE_QINQ_GRP_IDX_FROM;
297 unsigned int to = MLX5_ESW_BRIDGE_EGRESS_TABLE_QINQ_GRP_IDX_TO;
298
299 return mlx5_esw_bridge_egress_vlan_proto_fg_create(from, to, ETH_P_8021AD, esw, egress_ft);
300 }
301
302 static struct mlx5_flow_group *
mlx5_esw_bridge_egress_mac_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * egress_ft)303 mlx5_esw_bridge_egress_mac_fg_create(struct mlx5_eswitch *esw, struct mlx5_flow_table *egress_ft)
304 {
305 int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
306 struct mlx5_flow_group *fg;
307 u32 *in, *match;
308
309 in = kvzalloc(inlen, GFP_KERNEL);
310 if (!in)
311 return ERR_PTR(-ENOMEM);
312
313 MLX5_SET(create_flow_group_in, in, match_criteria_enable, MLX5_MATCH_OUTER_HEADERS);
314 match = MLX5_ADDR_OF(create_flow_group_in, in, match_criteria);
315
316 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.dmac_47_16);
317 MLX5_SET_TO_ONES(fte_match_param, match, outer_headers.dmac_15_0);
318
319 MLX5_SET(create_flow_group_in, in, start_flow_index,
320 MLX5_ESW_BRIDGE_EGRESS_TABLE_MAC_GRP_IDX_FROM);
321 MLX5_SET(create_flow_group_in, in, end_flow_index,
322 MLX5_ESW_BRIDGE_EGRESS_TABLE_MAC_GRP_IDX_TO);
323
324 fg = mlx5_create_flow_group(egress_ft, in);
325 if (IS_ERR(fg))
326 esw_warn(esw->dev,
327 "Failed to create bridge egress table MAC flow group (err=%ld)\n",
328 PTR_ERR(fg));
329 kvfree(in);
330 return fg;
331 }
332
333 static struct mlx5_flow_group *
mlx5_esw_bridge_egress_miss_fg_create(struct mlx5_eswitch * esw,struct mlx5_flow_table * egress_ft)334 mlx5_esw_bridge_egress_miss_fg_create(struct mlx5_eswitch *esw, struct mlx5_flow_table *egress_ft)
335 {
336 int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
337 struct mlx5_flow_group *fg;
338 u32 *in, *match;
339
340 in = kvzalloc(inlen, GFP_KERNEL);
341 if (!in)
342 return ERR_PTR(-ENOMEM);
343
344 MLX5_SET(create_flow_group_in, in, match_criteria_enable, MLX5_MATCH_MISC_PARAMETERS_2);
345 match = MLX5_ADDR_OF(create_flow_group_in, in, match_criteria);
346
347 MLX5_SET(fte_match_param, match, misc_parameters_2.metadata_reg_c_1, ESW_TUN_MASK);
348
349 MLX5_SET(create_flow_group_in, in, start_flow_index,
350 MLX5_ESW_BRIDGE_EGRESS_TABLE_MISS_GRP_IDX_FROM);
351 MLX5_SET(create_flow_group_in, in, end_flow_index,
352 MLX5_ESW_BRIDGE_EGRESS_TABLE_MISS_GRP_IDX_TO);
353
354 fg = mlx5_create_flow_group(egress_ft, in);
355 if (IS_ERR(fg))
356 esw_warn(esw->dev,
357 "Failed to create bridge egress table miss flow group (err=%ld)\n",
358 PTR_ERR(fg));
359 kvfree(in);
360 return fg;
361 }
362
363 static int
mlx5_esw_bridge_ingress_table_init(struct mlx5_esw_bridge_offloads * br_offloads)364 mlx5_esw_bridge_ingress_table_init(struct mlx5_esw_bridge_offloads *br_offloads)
365 {
366 struct mlx5_flow_group *mac_fg, *qinq_filter_fg, *qinq_fg, *vlan_filter_fg, *vlan_fg;
367 struct mlx5_flow_table *ingress_ft, *skip_ft;
368 struct mlx5_eswitch *esw = br_offloads->esw;
369 int err;
370
371 if (!mlx5_eswitch_vport_match_metadata_enabled(esw))
372 return -EOPNOTSUPP;
373
374 ingress_ft = mlx5_esw_bridge_table_create(MLX5_ESW_BRIDGE_INGRESS_TABLE_SIZE,
375 MLX5_ESW_BRIDGE_LEVEL_INGRESS_TABLE,
376 esw);
377 if (IS_ERR(ingress_ft))
378 return PTR_ERR(ingress_ft);
379
380 skip_ft = mlx5_esw_bridge_table_create(MLX5_ESW_BRIDGE_SKIP_TABLE_SIZE,
381 MLX5_ESW_BRIDGE_LEVEL_SKIP_TABLE,
382 esw);
383 if (IS_ERR(skip_ft)) {
384 err = PTR_ERR(skip_ft);
385 goto err_skip_tbl;
386 }
387
388 vlan_fg = mlx5_esw_bridge_ingress_vlan_fg_create(esw, ingress_ft);
389 if (IS_ERR(vlan_fg)) {
390 err = PTR_ERR(vlan_fg);
391 goto err_vlan_fg;
392 }
393
394 vlan_filter_fg = mlx5_esw_bridge_ingress_vlan_filter_fg_create(esw, ingress_ft);
395 if (IS_ERR(vlan_filter_fg)) {
396 err = PTR_ERR(vlan_filter_fg);
397 goto err_vlan_filter_fg;
398 }
399
400 qinq_fg = mlx5_esw_bridge_ingress_qinq_fg_create(esw, ingress_ft);
401 if (IS_ERR(qinq_fg)) {
402 err = PTR_ERR(qinq_fg);
403 goto err_qinq_fg;
404 }
405
406 qinq_filter_fg = mlx5_esw_bridge_ingress_qinq_filter_fg_create(esw, ingress_ft);
407 if (IS_ERR(qinq_filter_fg)) {
408 err = PTR_ERR(qinq_filter_fg);
409 goto err_qinq_filter_fg;
410 }
411
412 mac_fg = mlx5_esw_bridge_ingress_mac_fg_create(esw, ingress_ft);
413 if (IS_ERR(mac_fg)) {
414 err = PTR_ERR(mac_fg);
415 goto err_mac_fg;
416 }
417
418 br_offloads->ingress_ft = ingress_ft;
419 br_offloads->skip_ft = skip_ft;
420 br_offloads->ingress_vlan_fg = vlan_fg;
421 br_offloads->ingress_vlan_filter_fg = vlan_filter_fg;
422 br_offloads->ingress_qinq_fg = qinq_fg;
423 br_offloads->ingress_qinq_filter_fg = qinq_filter_fg;
424 br_offloads->ingress_mac_fg = mac_fg;
425 return 0;
426
427 err_mac_fg:
428 mlx5_destroy_flow_group(qinq_filter_fg);
429 err_qinq_filter_fg:
430 mlx5_destroy_flow_group(qinq_fg);
431 err_qinq_fg:
432 mlx5_destroy_flow_group(vlan_filter_fg);
433 err_vlan_filter_fg:
434 mlx5_destroy_flow_group(vlan_fg);
435 err_vlan_fg:
436 mlx5_destroy_flow_table(skip_ft);
437 err_skip_tbl:
438 mlx5_destroy_flow_table(ingress_ft);
439 return err;
440 }
441
442 static void
mlx5_esw_bridge_ingress_table_cleanup(struct mlx5_esw_bridge_offloads * br_offloads)443 mlx5_esw_bridge_ingress_table_cleanup(struct mlx5_esw_bridge_offloads *br_offloads)
444 {
445 mlx5_destroy_flow_group(br_offloads->ingress_mac_fg);
446 br_offloads->ingress_mac_fg = NULL;
447 mlx5_destroy_flow_group(br_offloads->ingress_qinq_filter_fg);
448 br_offloads->ingress_qinq_filter_fg = NULL;
449 mlx5_destroy_flow_group(br_offloads->ingress_qinq_fg);
450 br_offloads->ingress_qinq_fg = NULL;
451 mlx5_destroy_flow_group(br_offloads->ingress_vlan_filter_fg);
452 br_offloads->ingress_vlan_filter_fg = NULL;
453 mlx5_destroy_flow_group(br_offloads->ingress_vlan_fg);
454 br_offloads->ingress_vlan_fg = NULL;
455 mlx5_destroy_flow_table(br_offloads->skip_ft);
456 br_offloads->skip_ft = NULL;
457 mlx5_destroy_flow_table(br_offloads->ingress_ft);
458 br_offloads->ingress_ft = NULL;
459 }
460
461 static struct mlx5_flow_handle *
462 mlx5_esw_bridge_egress_miss_flow_create(struct mlx5_flow_table *egress_ft,
463 struct mlx5_flow_table *skip_ft,
464 struct mlx5_pkt_reformat *pkt_reformat);
465
466 static int
mlx5_esw_bridge_egress_table_init(struct mlx5_esw_bridge_offloads * br_offloads,struct mlx5_esw_bridge * bridge)467 mlx5_esw_bridge_egress_table_init(struct mlx5_esw_bridge_offloads *br_offloads,
468 struct mlx5_esw_bridge *bridge)
469 {
470 struct mlx5_flow_group *miss_fg = NULL, *mac_fg, *vlan_fg, *qinq_fg;
471 struct mlx5_pkt_reformat *miss_pkt_reformat = NULL;
472 struct mlx5_flow_handle *miss_handle = NULL;
473 struct mlx5_eswitch *esw = br_offloads->esw;
474 struct mlx5_flow_table *egress_ft;
475 int err;
476
477 egress_ft = mlx5_esw_bridge_table_create(MLX5_ESW_BRIDGE_EGRESS_TABLE_SIZE,
478 MLX5_ESW_BRIDGE_LEVEL_EGRESS_TABLE,
479 esw);
480 if (IS_ERR(egress_ft))
481 return PTR_ERR(egress_ft);
482
483 vlan_fg = mlx5_esw_bridge_egress_vlan_fg_create(esw, egress_ft);
484 if (IS_ERR(vlan_fg)) {
485 err = PTR_ERR(vlan_fg);
486 goto err_vlan_fg;
487 }
488
489 qinq_fg = mlx5_esw_bridge_egress_qinq_fg_create(esw, egress_ft);
490 if (IS_ERR(qinq_fg)) {
491 err = PTR_ERR(qinq_fg);
492 goto err_qinq_fg;
493 }
494
495 mac_fg = mlx5_esw_bridge_egress_mac_fg_create(esw, egress_ft);
496 if (IS_ERR(mac_fg)) {
497 err = PTR_ERR(mac_fg);
498 goto err_mac_fg;
499 }
500
501 if (mlx5_esw_bridge_pkt_reformat_vlan_pop_supported(esw)) {
502 miss_fg = mlx5_esw_bridge_egress_miss_fg_create(esw, egress_ft);
503 if (IS_ERR(miss_fg)) {
504 esw_warn(esw->dev, "Failed to create miss flow group (err=%ld)\n",
505 PTR_ERR(miss_fg));
506 miss_fg = NULL;
507 goto skip_miss_flow;
508 }
509
510 miss_pkt_reformat = mlx5_esw_bridge_pkt_reformat_vlan_pop_create(esw);
511 if (IS_ERR(miss_pkt_reformat)) {
512 esw_warn(esw->dev,
513 "Failed to alloc packet reformat REMOVE_HEADER (err=%ld)\n",
514 PTR_ERR(miss_pkt_reformat));
515 miss_pkt_reformat = NULL;
516 mlx5_destroy_flow_group(miss_fg);
517 miss_fg = NULL;
518 goto skip_miss_flow;
519 }
520
521 miss_handle = mlx5_esw_bridge_egress_miss_flow_create(egress_ft,
522 br_offloads->skip_ft,
523 miss_pkt_reformat);
524 if (IS_ERR(miss_handle)) {
525 esw_warn(esw->dev, "Failed to create miss flow (err=%ld)\n",
526 PTR_ERR(miss_handle));
527 miss_handle = NULL;
528 mlx5_packet_reformat_dealloc(esw->dev, miss_pkt_reformat);
529 miss_pkt_reformat = NULL;
530 mlx5_destroy_flow_group(miss_fg);
531 miss_fg = NULL;
532 goto skip_miss_flow;
533 }
534 }
535 skip_miss_flow:
536
537 bridge->egress_ft = egress_ft;
538 bridge->egress_vlan_fg = vlan_fg;
539 bridge->egress_qinq_fg = qinq_fg;
540 bridge->egress_mac_fg = mac_fg;
541 bridge->egress_miss_fg = miss_fg;
542 bridge->egress_miss_pkt_reformat = miss_pkt_reformat;
543 bridge->egress_miss_handle = miss_handle;
544 return 0;
545
546 err_mac_fg:
547 mlx5_destroy_flow_group(qinq_fg);
548 err_qinq_fg:
549 mlx5_destroy_flow_group(vlan_fg);
550 err_vlan_fg:
551 mlx5_destroy_flow_table(egress_ft);
552 return err;
553 }
554
555 static void
mlx5_esw_bridge_egress_table_cleanup(struct mlx5_esw_bridge * bridge)556 mlx5_esw_bridge_egress_table_cleanup(struct mlx5_esw_bridge *bridge)
557 {
558 if (bridge->egress_miss_handle)
559 mlx5_del_flow_rules(bridge->egress_miss_handle);
560 if (bridge->egress_miss_pkt_reformat)
561 mlx5_packet_reformat_dealloc(bridge->br_offloads->esw->dev,
562 bridge->egress_miss_pkt_reformat);
563 if (bridge->egress_miss_fg)
564 mlx5_destroy_flow_group(bridge->egress_miss_fg);
565 mlx5_destroy_flow_group(bridge->egress_mac_fg);
566 mlx5_destroy_flow_group(bridge->egress_qinq_fg);
567 mlx5_destroy_flow_group(bridge->egress_vlan_fg);
568 mlx5_destroy_flow_table(bridge->egress_ft);
569 }
570
571 static struct mlx5_flow_handle *
mlx5_esw_bridge_ingress_flow_with_esw_create(u16 vport_num,const unsigned char * addr,struct mlx5_esw_bridge_vlan * vlan,u32 counter_id,struct mlx5_esw_bridge * bridge,struct mlx5_eswitch * esw)572 mlx5_esw_bridge_ingress_flow_with_esw_create(u16 vport_num, const unsigned char *addr,
573 struct mlx5_esw_bridge_vlan *vlan, u32 counter_id,
574 struct mlx5_esw_bridge *bridge,
575 struct mlx5_eswitch *esw)
576 {
577 struct mlx5_esw_bridge_offloads *br_offloads = bridge->br_offloads;
578 struct mlx5_flow_act flow_act = {
579 .action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST | MLX5_FLOW_CONTEXT_ACTION_COUNT,
580 .flags = FLOW_ACT_NO_APPEND,
581 };
582 struct mlx5_flow_destination dests[2] = {};
583 struct mlx5_flow_spec *rule_spec;
584 struct mlx5_flow_handle *handle;
585 u8 *smac_v, *smac_c;
586
587 rule_spec = kvzalloc(sizeof(*rule_spec), GFP_KERNEL);
588 if (!rule_spec)
589 return ERR_PTR(-ENOMEM);
590
591 rule_spec->match_criteria_enable = MLX5_MATCH_OUTER_HEADERS | MLX5_MATCH_MISC_PARAMETERS_2;
592
593 smac_v = MLX5_ADDR_OF(fte_match_param, rule_spec->match_value,
594 outer_headers.smac_47_16);
595 ether_addr_copy(smac_v, addr);
596 smac_c = MLX5_ADDR_OF(fte_match_param, rule_spec->match_criteria,
597 outer_headers.smac_47_16);
598 eth_broadcast_addr(smac_c);
599
600 MLX5_SET(fte_match_param, rule_spec->match_criteria,
601 misc_parameters_2.metadata_reg_c_0, mlx5_eswitch_get_vport_metadata_mask());
602 MLX5_SET(fte_match_param, rule_spec->match_value, misc_parameters_2.metadata_reg_c_0,
603 mlx5_eswitch_get_vport_metadata_for_match(esw, vport_num));
604
605 if (vlan && vlan->pkt_reformat_push) {
606 flow_act.action |= MLX5_FLOW_CONTEXT_ACTION_PACKET_REFORMAT |
607 MLX5_FLOW_CONTEXT_ACTION_MOD_HDR;
608 flow_act.pkt_reformat = vlan->pkt_reformat_push;
609 flow_act.modify_hdr = vlan->pkt_mod_hdr_push_mark;
610 } else if (vlan) {
611 if (bridge->vlan_proto == ETH_P_8021Q) {
612 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
613 outer_headers.cvlan_tag);
614 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_value,
615 outer_headers.cvlan_tag);
616 } else if (bridge->vlan_proto == ETH_P_8021AD) {
617 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
618 outer_headers.svlan_tag);
619 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_value,
620 outer_headers.svlan_tag);
621 }
622 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
623 outer_headers.first_vid);
624 MLX5_SET(fte_match_param, rule_spec->match_value, outer_headers.first_vid,
625 vlan->vid);
626 }
627
628 dests[0].type = MLX5_FLOW_DESTINATION_TYPE_FLOW_TABLE;
629 dests[0].ft = bridge->egress_ft;
630 dests[1].type = MLX5_FLOW_DESTINATION_TYPE_COUNTER;
631 dests[1].counter_id = counter_id;
632
633 handle = mlx5_add_flow_rules(br_offloads->ingress_ft, rule_spec, &flow_act, dests,
634 ARRAY_SIZE(dests));
635
636 kvfree(rule_spec);
637 return handle;
638 }
639
640 static struct mlx5_flow_handle *
mlx5_esw_bridge_ingress_flow_create(u16 vport_num,const unsigned char * addr,struct mlx5_esw_bridge_vlan * vlan,u32 counter_id,struct mlx5_esw_bridge * bridge)641 mlx5_esw_bridge_ingress_flow_create(u16 vport_num, const unsigned char *addr,
642 struct mlx5_esw_bridge_vlan *vlan, u32 counter_id,
643 struct mlx5_esw_bridge *bridge)
644 {
645 return mlx5_esw_bridge_ingress_flow_with_esw_create(vport_num, addr, vlan, counter_id,
646 bridge, bridge->br_offloads->esw);
647 }
648
649 static struct mlx5_flow_handle *
mlx5_esw_bridge_ingress_flow_peer_create(u16 vport_num,u16 esw_owner_vhca_id,const unsigned char * addr,struct mlx5_esw_bridge_vlan * vlan,u32 counter_id,struct mlx5_esw_bridge * bridge)650 mlx5_esw_bridge_ingress_flow_peer_create(u16 vport_num, u16 esw_owner_vhca_id,
651 const unsigned char *addr,
652 struct mlx5_esw_bridge_vlan *vlan, u32 counter_id,
653 struct mlx5_esw_bridge *bridge)
654 {
655 struct mlx5_devcom_comp_dev *devcom = bridge->br_offloads->esw->devcom, *pos;
656 struct mlx5_eswitch *tmp, *peer_esw = NULL;
657 static struct mlx5_flow_handle *handle;
658
659 if (!mlx5_devcom_for_each_peer_begin(devcom))
660 return ERR_PTR(-ENODEV);
661
662 mlx5_devcom_for_each_peer_entry(devcom, tmp, pos) {
663 if (mlx5_esw_is_owner(tmp, vport_num, esw_owner_vhca_id)) {
664 peer_esw = tmp;
665 break;
666 }
667 }
668
669 if (!peer_esw) {
670 handle = ERR_PTR(-ENODEV);
671 goto out;
672 }
673
674 handle = mlx5_esw_bridge_ingress_flow_with_esw_create(vport_num, addr, vlan, counter_id,
675 bridge, peer_esw);
676
677 out:
678 mlx5_devcom_for_each_peer_end(devcom);
679 return handle;
680 }
681
682 static struct mlx5_flow_handle *
mlx5_esw_bridge_ingress_filter_flow_create(u16 vport_num,const unsigned char * addr,struct mlx5_esw_bridge * bridge)683 mlx5_esw_bridge_ingress_filter_flow_create(u16 vport_num, const unsigned char *addr,
684 struct mlx5_esw_bridge *bridge)
685 {
686 struct mlx5_esw_bridge_offloads *br_offloads = bridge->br_offloads;
687 struct mlx5_flow_destination dest = {
688 .type = MLX5_FLOW_DESTINATION_TYPE_FLOW_TABLE,
689 .ft = br_offloads->skip_ft,
690 };
691 struct mlx5_flow_act flow_act = {
692 .action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST,
693 .flags = FLOW_ACT_NO_APPEND,
694 };
695 struct mlx5_flow_spec *rule_spec;
696 struct mlx5_flow_handle *handle;
697 u8 *smac_v, *smac_c;
698
699 rule_spec = kvzalloc(sizeof(*rule_spec), GFP_KERNEL);
700 if (!rule_spec)
701 return ERR_PTR(-ENOMEM);
702
703 rule_spec->match_criteria_enable = MLX5_MATCH_OUTER_HEADERS | MLX5_MATCH_MISC_PARAMETERS_2;
704
705 smac_v = MLX5_ADDR_OF(fte_match_param, rule_spec->match_value,
706 outer_headers.smac_47_16);
707 ether_addr_copy(smac_v, addr);
708 smac_c = MLX5_ADDR_OF(fte_match_param, rule_spec->match_criteria,
709 outer_headers.smac_47_16);
710 eth_broadcast_addr(smac_c);
711
712 MLX5_SET(fte_match_param, rule_spec->match_criteria,
713 misc_parameters_2.metadata_reg_c_0, mlx5_eswitch_get_vport_metadata_mask());
714 MLX5_SET(fte_match_param, rule_spec->match_value, misc_parameters_2.metadata_reg_c_0,
715 mlx5_eswitch_get_vport_metadata_for_match(br_offloads->esw, vport_num));
716
717 if (bridge->vlan_proto == ETH_P_8021Q) {
718 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
719 outer_headers.cvlan_tag);
720 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_value,
721 outer_headers.cvlan_tag);
722 } else if (bridge->vlan_proto == ETH_P_8021AD) {
723 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
724 outer_headers.svlan_tag);
725 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_value,
726 outer_headers.svlan_tag);
727 }
728
729 handle = mlx5_add_flow_rules(br_offloads->ingress_ft, rule_spec, &flow_act, &dest, 1);
730
731 kvfree(rule_spec);
732 return handle;
733 }
734
735 static struct mlx5_flow_handle *
mlx5_esw_bridge_egress_flow_create(u16 vport_num,u16 esw_owner_vhca_id,const unsigned char * addr,struct mlx5_esw_bridge_vlan * vlan,struct mlx5_esw_bridge * bridge)736 mlx5_esw_bridge_egress_flow_create(u16 vport_num, u16 esw_owner_vhca_id, const unsigned char *addr,
737 struct mlx5_esw_bridge_vlan *vlan,
738 struct mlx5_esw_bridge *bridge)
739 {
740 struct mlx5_flow_destination dest = {
741 .type = MLX5_FLOW_DESTINATION_TYPE_VPORT,
742 .vport.num = vport_num,
743 };
744 struct mlx5_flow_act flow_act = {
745 .action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST,
746 .flags = FLOW_ACT_NO_APPEND,
747 };
748 struct mlx5_flow_spec *rule_spec;
749 struct mlx5_flow_handle *handle;
750 u8 *dmac_v, *dmac_c;
751
752 rule_spec = kvzalloc(sizeof(*rule_spec), GFP_KERNEL);
753 if (!rule_spec)
754 return ERR_PTR(-ENOMEM);
755
756 if (MLX5_CAP_ESW_FLOWTABLE(bridge->br_offloads->esw->dev, flow_source) &&
757 vport_num == MLX5_VPORT_UPLINK)
758 rule_spec->flow_context.flow_source =
759 MLX5_FLOW_CONTEXT_FLOW_SOURCE_LOCAL_VPORT;
760 rule_spec->match_criteria_enable = MLX5_MATCH_OUTER_HEADERS;
761
762 dmac_v = MLX5_ADDR_OF(fte_match_param, rule_spec->match_value,
763 outer_headers.dmac_47_16);
764 ether_addr_copy(dmac_v, addr);
765 dmac_c = MLX5_ADDR_OF(fte_match_param, rule_spec->match_criteria,
766 outer_headers.dmac_47_16);
767 eth_broadcast_addr(dmac_c);
768
769 if (vlan) {
770 if (vlan->pkt_reformat_pop) {
771 flow_act.action |= MLX5_FLOW_CONTEXT_ACTION_PACKET_REFORMAT;
772 flow_act.pkt_reformat = vlan->pkt_reformat_pop;
773 }
774
775 if (bridge->vlan_proto == ETH_P_8021Q) {
776 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
777 outer_headers.cvlan_tag);
778 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_value,
779 outer_headers.cvlan_tag);
780 } else if (bridge->vlan_proto == ETH_P_8021AD) {
781 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
782 outer_headers.svlan_tag);
783 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_value,
784 outer_headers.svlan_tag);
785 }
786 MLX5_SET_TO_ONES(fte_match_param, rule_spec->match_criteria,
787 outer_headers.first_vid);
788 MLX5_SET(fte_match_param, rule_spec->match_value, outer_headers.first_vid,
789 vlan->vid);
790 }
791
792 if (MLX5_CAP_ESW(bridge->br_offloads->esw->dev, merged_eswitch)) {
793 dest.vport.flags = MLX5_FLOW_DEST_VPORT_VHCA_ID;
794 dest.vport.vhca_id = esw_owner_vhca_id;
795 }
796 handle = mlx5_add_flow_rules(bridge->egress_ft, rule_spec, &flow_act, &dest, 1);
797
798 kvfree(rule_spec);
799 return handle;
800 }
801
802 static struct mlx5_flow_handle *
mlx5_esw_bridge_egress_miss_flow_create(struct mlx5_flow_table * egress_ft,struct mlx5_flow_table * skip_ft,struct mlx5_pkt_reformat * pkt_reformat)803 mlx5_esw_bridge_egress_miss_flow_create(struct mlx5_flow_table *egress_ft,
804 struct mlx5_flow_table *skip_ft,
805 struct mlx5_pkt_reformat *pkt_reformat)
806 {
807 struct mlx5_flow_destination dest = {
808 .type = MLX5_FLOW_DESTINATION_TYPE_FLOW_TABLE,
809 .ft = skip_ft,
810 };
811 struct mlx5_flow_act flow_act = {
812 .action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST |
813 MLX5_FLOW_CONTEXT_ACTION_PACKET_REFORMAT,
814 .flags = FLOW_ACT_NO_APPEND,
815 .pkt_reformat = pkt_reformat,
816 };
817 struct mlx5_flow_spec *rule_spec;
818 struct mlx5_flow_handle *handle;
819
820 rule_spec = kvzalloc(sizeof(*rule_spec), GFP_KERNEL);
821 if (!rule_spec)
822 return ERR_PTR(-ENOMEM);
823
824 rule_spec->match_criteria_enable = MLX5_MATCH_MISC_PARAMETERS_2;
825
826 MLX5_SET(fte_match_param, rule_spec->match_criteria,
827 misc_parameters_2.metadata_reg_c_1, ESW_TUN_MASK);
828 MLX5_SET(fte_match_param, rule_spec->match_value, misc_parameters_2.metadata_reg_c_1,
829 ESW_TUN_BRIDGE_INGRESS_PUSH_VLAN_MARK);
830
831 handle = mlx5_add_flow_rules(egress_ft, rule_spec, &flow_act, &dest, 1);
832
833 kvfree(rule_spec);
834 return handle;
835 }
836
mlx5_esw_bridge_create(struct net_device * br_netdev,struct mlx5_esw_bridge_offloads * br_offloads)837 static struct mlx5_esw_bridge *mlx5_esw_bridge_create(struct net_device *br_netdev,
838 struct mlx5_esw_bridge_offloads *br_offloads)
839 {
840 struct mlx5_esw_bridge *bridge;
841 int err;
842
843 bridge = kvzalloc(sizeof(*bridge), GFP_KERNEL);
844 if (!bridge)
845 return ERR_PTR(-ENOMEM);
846
847 bridge->br_offloads = br_offloads;
848 err = mlx5_esw_bridge_egress_table_init(br_offloads, bridge);
849 if (err)
850 goto err_egress_tbl;
851
852 err = rhashtable_init(&bridge->fdb_ht, &fdb_ht_params);
853 if (err)
854 goto err_fdb_ht;
855
856 err = mlx5_esw_bridge_mdb_init(bridge);
857 if (err)
858 goto err_mdb_ht;
859
860 INIT_LIST_HEAD(&bridge->fdb_list);
861 bridge->ifindex = br_netdev->ifindex;
862 bridge->refcnt = 1;
863 bridge->ageing_time = clock_t_to_jiffies(BR_DEFAULT_AGEING_TIME);
864 bridge->vlan_proto = ETH_P_8021Q;
865 list_add(&bridge->list, &br_offloads->bridges);
866 mlx5_esw_bridge_debugfs_init(br_netdev, bridge);
867
868 return bridge;
869
870 err_mdb_ht:
871 rhashtable_destroy(&bridge->fdb_ht);
872 err_fdb_ht:
873 mlx5_esw_bridge_egress_table_cleanup(bridge);
874 err_egress_tbl:
875 kvfree(bridge);
876 return ERR_PTR(err);
877 }
878
mlx5_esw_bridge_get(struct mlx5_esw_bridge * bridge)879 static void mlx5_esw_bridge_get(struct mlx5_esw_bridge *bridge)
880 {
881 bridge->refcnt++;
882 }
883
mlx5_esw_bridge_put(struct mlx5_esw_bridge_offloads * br_offloads,struct mlx5_esw_bridge * bridge)884 static void mlx5_esw_bridge_put(struct mlx5_esw_bridge_offloads *br_offloads,
885 struct mlx5_esw_bridge *bridge)
886 {
887 if (--bridge->refcnt)
888 return;
889
890 mlx5_esw_bridge_debugfs_cleanup(bridge);
891 mlx5_esw_bridge_egress_table_cleanup(bridge);
892 mlx5_esw_bridge_mcast_disable(bridge);
893 list_del(&bridge->list);
894 mlx5_esw_bridge_mdb_cleanup(bridge);
895 rhashtable_destroy(&bridge->fdb_ht);
896 kvfree(bridge);
897
898 if (list_empty(&br_offloads->bridges))
899 mlx5_esw_bridge_ingress_table_cleanup(br_offloads);
900 }
901
902 static struct mlx5_esw_bridge *
mlx5_esw_bridge_lookup(struct net_device * br_netdev,struct mlx5_esw_bridge_offloads * br_offloads)903 mlx5_esw_bridge_lookup(struct net_device *br_netdev, struct mlx5_esw_bridge_offloads *br_offloads)
904 {
905 struct mlx5_esw_bridge *bridge;
906
907 ASSERT_RTNL();
908
909 list_for_each_entry(bridge, &br_offloads->bridges, list) {
910 if (bridge->ifindex == br_netdev->ifindex) {
911 mlx5_esw_bridge_get(bridge);
912 return bridge;
913 }
914 }
915
916 if (!br_offloads->ingress_ft) {
917 int err = mlx5_esw_bridge_ingress_table_init(br_offloads);
918
919 if (err)
920 return ERR_PTR(err);
921 }
922
923 bridge = mlx5_esw_bridge_create(br_netdev, br_offloads);
924 if (IS_ERR(bridge) && list_empty(&br_offloads->bridges))
925 mlx5_esw_bridge_ingress_table_cleanup(br_offloads);
926 return bridge;
927 }
928
mlx5_esw_bridge_port_key_from_data(u16 vport_num,u16 esw_owner_vhca_id)929 static unsigned long mlx5_esw_bridge_port_key_from_data(u16 vport_num, u16 esw_owner_vhca_id)
930 {
931 return vport_num | (unsigned long)esw_owner_vhca_id << sizeof(vport_num) * BITS_PER_BYTE;
932 }
933
mlx5_esw_bridge_port_key(struct mlx5_esw_bridge_port * port)934 unsigned long mlx5_esw_bridge_port_key(struct mlx5_esw_bridge_port *port)
935 {
936 return mlx5_esw_bridge_port_key_from_data(port->vport_num, port->esw_owner_vhca_id);
937 }
938
mlx5_esw_bridge_port_insert(struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge_offloads * br_offloads)939 static int mlx5_esw_bridge_port_insert(struct mlx5_esw_bridge_port *port,
940 struct mlx5_esw_bridge_offloads *br_offloads)
941 {
942 return xa_insert(&br_offloads->ports, mlx5_esw_bridge_port_key(port), port, GFP_KERNEL);
943 }
944
945 static struct mlx5_esw_bridge_port *
mlx5_esw_bridge_port_lookup(u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads)946 mlx5_esw_bridge_port_lookup(u16 vport_num, u16 esw_owner_vhca_id,
947 struct mlx5_esw_bridge_offloads *br_offloads)
948 {
949 return xa_load(&br_offloads->ports, mlx5_esw_bridge_port_key_from_data(vport_num,
950 esw_owner_vhca_id));
951 }
952
mlx5_esw_bridge_port_erase(struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge_offloads * br_offloads)953 static void mlx5_esw_bridge_port_erase(struct mlx5_esw_bridge_port *port,
954 struct mlx5_esw_bridge_offloads *br_offloads)
955 {
956 xa_erase(&br_offloads->ports, mlx5_esw_bridge_port_key(port));
957 }
958
959 static struct mlx5_esw_bridge *
mlx5_esw_bridge_from_port_lookup(u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads)960 mlx5_esw_bridge_from_port_lookup(u16 vport_num, u16 esw_owner_vhca_id,
961 struct mlx5_esw_bridge_offloads *br_offloads)
962 {
963 struct mlx5_esw_bridge_port *port;
964
965 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
966 if (!port)
967 return NULL;
968
969 return port->bridge;
970 }
971
mlx5_esw_bridge_fdb_entry_refresh(struct mlx5_esw_bridge_fdb_entry * entry)972 static void mlx5_esw_bridge_fdb_entry_refresh(struct mlx5_esw_bridge_fdb_entry *entry)
973 {
974 trace_mlx5_esw_bridge_fdb_entry_refresh(entry);
975
976 mlx5_esw_bridge_fdb_offload_notify(entry->dev, entry->key.addr,
977 entry->key.vid,
978 SWITCHDEV_FDB_ADD_TO_BRIDGE);
979 }
980
981 static void
mlx5_esw_bridge_fdb_entry_cleanup(struct mlx5_esw_bridge_fdb_entry * entry,struct mlx5_esw_bridge * bridge)982 mlx5_esw_bridge_fdb_entry_cleanup(struct mlx5_esw_bridge_fdb_entry *entry,
983 struct mlx5_esw_bridge *bridge)
984 {
985 trace_mlx5_esw_bridge_fdb_entry_cleanup(entry);
986
987 rhashtable_remove_fast(&bridge->fdb_ht, &entry->ht_node, fdb_ht_params);
988 mlx5_del_flow_rules(entry->egress_handle);
989 if (entry->filter_handle)
990 mlx5_del_flow_rules(entry->filter_handle);
991 mlx5_del_flow_rules(entry->ingress_handle);
992 mlx5_fc_destroy(bridge->br_offloads->esw->dev, entry->ingress_counter);
993 list_del(&entry->vlan_list);
994 list_del(&entry->list);
995 kvfree(entry);
996 }
997
998 static void
mlx5_esw_bridge_fdb_entry_notify_and_cleanup(struct mlx5_esw_bridge_fdb_entry * entry,struct mlx5_esw_bridge * bridge)999 mlx5_esw_bridge_fdb_entry_notify_and_cleanup(struct mlx5_esw_bridge_fdb_entry *entry,
1000 struct mlx5_esw_bridge *bridge)
1001 {
1002 mlx5_esw_bridge_fdb_del_notify(entry);
1003 mlx5_esw_bridge_fdb_entry_cleanup(entry, bridge);
1004 }
1005
mlx5_esw_bridge_fdb_flush(struct mlx5_esw_bridge * bridge)1006 static void mlx5_esw_bridge_fdb_flush(struct mlx5_esw_bridge *bridge)
1007 {
1008 struct mlx5_esw_bridge_fdb_entry *entry, *tmp;
1009
1010 list_for_each_entry_safe(entry, tmp, &bridge->fdb_list, list)
1011 mlx5_esw_bridge_fdb_entry_notify_and_cleanup(entry, bridge);
1012 }
1013
1014 static struct mlx5_esw_bridge_vlan *
mlx5_esw_bridge_vlan_lookup(u16 vid,struct mlx5_esw_bridge_port * port)1015 mlx5_esw_bridge_vlan_lookup(u16 vid, struct mlx5_esw_bridge_port *port)
1016 {
1017 return xa_load(&port->vlans, vid);
1018 }
1019
1020 static int
mlx5_esw_bridge_vlan_push_create(u16 vlan_proto,struct mlx5_esw_bridge_vlan * vlan,struct mlx5_eswitch * esw)1021 mlx5_esw_bridge_vlan_push_create(u16 vlan_proto, struct mlx5_esw_bridge_vlan *vlan,
1022 struct mlx5_eswitch *esw)
1023 {
1024 struct {
1025 __be16 h_vlan_proto;
1026 __be16 h_vlan_TCI;
1027 } vlan_hdr = { htons(vlan_proto), htons(vlan->vid) };
1028 struct mlx5_pkt_reformat_params reformat_params = {};
1029 struct mlx5_pkt_reformat *pkt_reformat;
1030
1031 if (!BIT(MLX5_CAP_ESW_FLOWTABLE_FDB(esw->dev, reformat_insert)) ||
1032 MLX5_CAP_GEN_2(esw->dev, max_reformat_insert_size) < sizeof(vlan_hdr) ||
1033 MLX5_CAP_GEN_2(esw->dev, max_reformat_insert_offset) <
1034 offsetof(struct vlan_ethhdr, h_vlan_proto)) {
1035 esw_warn(esw->dev, "Packet reformat INSERT_HEADER is not supported\n");
1036 return -EOPNOTSUPP;
1037 }
1038
1039 reformat_params.type = MLX5_REFORMAT_TYPE_INSERT_HDR;
1040 reformat_params.param_0 = MLX5_REFORMAT_CONTEXT_ANCHOR_MAC_START;
1041 reformat_params.param_1 = offsetof(struct vlan_ethhdr, h_vlan_proto);
1042 reformat_params.size = sizeof(vlan_hdr);
1043 reformat_params.data = &vlan_hdr;
1044 pkt_reformat = mlx5_packet_reformat_alloc(esw->dev,
1045 &reformat_params,
1046 MLX5_FLOW_NAMESPACE_FDB);
1047 if (IS_ERR(pkt_reformat)) {
1048 esw_warn(esw->dev, "Failed to alloc packet reformat INSERT_HEADER (err=%ld)\n",
1049 PTR_ERR(pkt_reformat));
1050 return PTR_ERR(pkt_reformat);
1051 }
1052
1053 vlan->pkt_reformat_push = pkt_reformat;
1054 return 0;
1055 }
1056
1057 static void
mlx5_esw_bridge_vlan_push_cleanup(struct mlx5_esw_bridge_vlan * vlan,struct mlx5_eswitch * esw)1058 mlx5_esw_bridge_vlan_push_cleanup(struct mlx5_esw_bridge_vlan *vlan, struct mlx5_eswitch *esw)
1059 {
1060 mlx5_packet_reformat_dealloc(esw->dev, vlan->pkt_reformat_push);
1061 vlan->pkt_reformat_push = NULL;
1062 }
1063
1064 static int
mlx5_esw_bridge_vlan_pop_create(struct mlx5_esw_bridge_vlan * vlan,struct mlx5_eswitch * esw)1065 mlx5_esw_bridge_vlan_pop_create(struct mlx5_esw_bridge_vlan *vlan, struct mlx5_eswitch *esw)
1066 {
1067 struct mlx5_pkt_reformat *pkt_reformat;
1068
1069 if (!mlx5_esw_bridge_pkt_reformat_vlan_pop_supported(esw)) {
1070 esw_warn(esw->dev, "Packet reformat REMOVE_HEADER is not supported\n");
1071 return -EOPNOTSUPP;
1072 }
1073
1074 pkt_reformat = mlx5_esw_bridge_pkt_reformat_vlan_pop_create(esw);
1075 if (IS_ERR(pkt_reformat)) {
1076 esw_warn(esw->dev, "Failed to alloc packet reformat REMOVE_HEADER (err=%ld)\n",
1077 PTR_ERR(pkt_reformat));
1078 return PTR_ERR(pkt_reformat);
1079 }
1080
1081 vlan->pkt_reformat_pop = pkt_reformat;
1082 return 0;
1083 }
1084
1085 static void
mlx5_esw_bridge_vlan_pop_cleanup(struct mlx5_esw_bridge_vlan * vlan,struct mlx5_eswitch * esw)1086 mlx5_esw_bridge_vlan_pop_cleanup(struct mlx5_esw_bridge_vlan *vlan, struct mlx5_eswitch *esw)
1087 {
1088 mlx5_packet_reformat_dealloc(esw->dev, vlan->pkt_reformat_pop);
1089 vlan->pkt_reformat_pop = NULL;
1090 }
1091
1092 static int
mlx5_esw_bridge_vlan_push_mark_create(struct mlx5_esw_bridge_vlan * vlan,struct mlx5_eswitch * esw)1093 mlx5_esw_bridge_vlan_push_mark_create(struct mlx5_esw_bridge_vlan *vlan, struct mlx5_eswitch *esw)
1094 {
1095 u8 action[MLX5_UN_SZ_BYTES(set_add_copy_action_in_auto)] = {};
1096 struct mlx5_modify_hdr *pkt_mod_hdr;
1097
1098 MLX5_SET(set_action_in, action, action_type, MLX5_ACTION_TYPE_SET);
1099 MLX5_SET(set_action_in, action, field, MLX5_ACTION_IN_FIELD_METADATA_REG_C_1);
1100 MLX5_SET(set_action_in, action, offset, 8);
1101 MLX5_SET(set_action_in, action, length, ESW_TUN_OPTS_BITS + ESW_TUN_ID_BITS);
1102 MLX5_SET(set_action_in, action, data, ESW_TUN_BRIDGE_INGRESS_PUSH_VLAN);
1103
1104 pkt_mod_hdr = mlx5_modify_header_alloc(esw->dev, MLX5_FLOW_NAMESPACE_FDB, 1, action);
1105 if (IS_ERR(pkt_mod_hdr))
1106 return PTR_ERR(pkt_mod_hdr);
1107
1108 vlan->pkt_mod_hdr_push_mark = pkt_mod_hdr;
1109 return 0;
1110 }
1111
1112 static void
mlx5_esw_bridge_vlan_push_mark_cleanup(struct mlx5_esw_bridge_vlan * vlan,struct mlx5_eswitch * esw)1113 mlx5_esw_bridge_vlan_push_mark_cleanup(struct mlx5_esw_bridge_vlan *vlan, struct mlx5_eswitch *esw)
1114 {
1115 mlx5_modify_header_dealloc(esw->dev, vlan->pkt_mod_hdr_push_mark);
1116 vlan->pkt_mod_hdr_push_mark = NULL;
1117 }
1118
1119 static int
mlx5_esw_bridge_vlan_push_pop_fhs_create(u16 vlan_proto,struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge_vlan * vlan)1120 mlx5_esw_bridge_vlan_push_pop_fhs_create(u16 vlan_proto, struct mlx5_esw_bridge_port *port,
1121 struct mlx5_esw_bridge_vlan *vlan)
1122 {
1123 return mlx5_esw_bridge_vlan_mcast_init(vlan_proto, port, vlan);
1124 }
1125
1126 static void
mlx5_esw_bridge_vlan_push_pop_fhs_cleanup(struct mlx5_esw_bridge_vlan * vlan)1127 mlx5_esw_bridge_vlan_push_pop_fhs_cleanup(struct mlx5_esw_bridge_vlan *vlan)
1128 {
1129 mlx5_esw_bridge_vlan_mcast_cleanup(vlan);
1130 }
1131
1132 static int
mlx5_esw_bridge_vlan_push_pop_create(u16 vlan_proto,u16 flags,struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge_vlan * vlan,struct mlx5_eswitch * esw)1133 mlx5_esw_bridge_vlan_push_pop_create(u16 vlan_proto, u16 flags, struct mlx5_esw_bridge_port *port,
1134 struct mlx5_esw_bridge_vlan *vlan, struct mlx5_eswitch *esw)
1135 {
1136 int err;
1137
1138 if (flags & BRIDGE_VLAN_INFO_PVID) {
1139 err = mlx5_esw_bridge_vlan_push_create(vlan_proto, vlan, esw);
1140 if (err)
1141 return err;
1142
1143 err = mlx5_esw_bridge_vlan_push_mark_create(vlan, esw);
1144 if (err)
1145 goto err_vlan_push_mark;
1146 }
1147
1148 if (flags & BRIDGE_VLAN_INFO_UNTAGGED) {
1149 err = mlx5_esw_bridge_vlan_pop_create(vlan, esw);
1150 if (err)
1151 goto err_vlan_pop;
1152
1153 err = mlx5_esw_bridge_vlan_push_pop_fhs_create(vlan_proto, port, vlan);
1154 if (err)
1155 goto err_vlan_pop_fhs;
1156 }
1157
1158 return 0;
1159
1160 err_vlan_pop_fhs:
1161 mlx5_esw_bridge_vlan_pop_cleanup(vlan, esw);
1162 err_vlan_pop:
1163 if (vlan->pkt_mod_hdr_push_mark)
1164 mlx5_esw_bridge_vlan_push_mark_cleanup(vlan, esw);
1165 err_vlan_push_mark:
1166 if (vlan->pkt_reformat_push)
1167 mlx5_esw_bridge_vlan_push_cleanup(vlan, esw);
1168 return err;
1169 }
1170
1171 static struct mlx5_esw_bridge_vlan *
mlx5_esw_bridge_vlan_create(u16 vlan_proto,u16 vid,u16 flags,struct mlx5_esw_bridge_port * port,struct mlx5_eswitch * esw)1172 mlx5_esw_bridge_vlan_create(u16 vlan_proto, u16 vid, u16 flags, struct mlx5_esw_bridge_port *port,
1173 struct mlx5_eswitch *esw)
1174 {
1175 struct mlx5_esw_bridge_vlan *vlan;
1176 int err;
1177
1178 vlan = kvzalloc(sizeof(*vlan), GFP_KERNEL);
1179 if (!vlan)
1180 return ERR_PTR(-ENOMEM);
1181
1182 vlan->vid = vid;
1183 vlan->flags = flags;
1184 INIT_LIST_HEAD(&vlan->fdb_list);
1185
1186 err = mlx5_esw_bridge_vlan_push_pop_create(vlan_proto, flags, port, vlan, esw);
1187 if (err)
1188 goto err_vlan_push_pop;
1189
1190 err = xa_insert(&port->vlans, vid, vlan, GFP_KERNEL);
1191 if (err)
1192 goto err_xa_insert;
1193
1194 trace_mlx5_esw_bridge_vlan_create(vlan);
1195 return vlan;
1196
1197 err_xa_insert:
1198 if (vlan->mcast_handle)
1199 mlx5_esw_bridge_vlan_push_pop_fhs_cleanup(vlan);
1200 if (vlan->pkt_reformat_pop)
1201 mlx5_esw_bridge_vlan_pop_cleanup(vlan, esw);
1202 if (vlan->pkt_mod_hdr_push_mark)
1203 mlx5_esw_bridge_vlan_push_mark_cleanup(vlan, esw);
1204 if (vlan->pkt_reformat_push)
1205 mlx5_esw_bridge_vlan_push_cleanup(vlan, esw);
1206 err_vlan_push_pop:
1207 kvfree(vlan);
1208 return ERR_PTR(err);
1209 }
1210
mlx5_esw_bridge_vlan_erase(struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge_vlan * vlan)1211 static void mlx5_esw_bridge_vlan_erase(struct mlx5_esw_bridge_port *port,
1212 struct mlx5_esw_bridge_vlan *vlan)
1213 {
1214 xa_erase(&port->vlans, vlan->vid);
1215 }
1216
mlx5_esw_bridge_vlan_flush(struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge_vlan * vlan,struct mlx5_esw_bridge * bridge)1217 static void mlx5_esw_bridge_vlan_flush(struct mlx5_esw_bridge_port *port,
1218 struct mlx5_esw_bridge_vlan *vlan,
1219 struct mlx5_esw_bridge *bridge)
1220 {
1221 struct mlx5_eswitch *esw = bridge->br_offloads->esw;
1222 struct mlx5_esw_bridge_fdb_entry *entry, *tmp;
1223
1224 list_for_each_entry_safe(entry, tmp, &vlan->fdb_list, vlan_list)
1225 mlx5_esw_bridge_fdb_entry_notify_and_cleanup(entry, bridge);
1226 mlx5_esw_bridge_port_mdb_vlan_flush(port, vlan);
1227
1228 if (vlan->mcast_handle)
1229 mlx5_esw_bridge_vlan_push_pop_fhs_cleanup(vlan);
1230 if (vlan->pkt_reformat_pop)
1231 mlx5_esw_bridge_vlan_pop_cleanup(vlan, esw);
1232 if (vlan->pkt_mod_hdr_push_mark)
1233 mlx5_esw_bridge_vlan_push_mark_cleanup(vlan, esw);
1234 if (vlan->pkt_reformat_push)
1235 mlx5_esw_bridge_vlan_push_cleanup(vlan, esw);
1236 }
1237
mlx5_esw_bridge_vlan_cleanup(struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge_vlan * vlan,struct mlx5_esw_bridge * bridge)1238 static void mlx5_esw_bridge_vlan_cleanup(struct mlx5_esw_bridge_port *port,
1239 struct mlx5_esw_bridge_vlan *vlan,
1240 struct mlx5_esw_bridge *bridge)
1241 {
1242 trace_mlx5_esw_bridge_vlan_cleanup(vlan);
1243 mlx5_esw_bridge_vlan_flush(port, vlan, bridge);
1244 mlx5_esw_bridge_vlan_erase(port, vlan);
1245 kvfree(vlan);
1246 }
1247
mlx5_esw_bridge_port_vlans_flush(struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge * bridge)1248 static void mlx5_esw_bridge_port_vlans_flush(struct mlx5_esw_bridge_port *port,
1249 struct mlx5_esw_bridge *bridge)
1250 {
1251 struct mlx5_esw_bridge_vlan *vlan;
1252 unsigned long index;
1253
1254 xa_for_each(&port->vlans, index, vlan)
1255 mlx5_esw_bridge_vlan_cleanup(port, vlan, bridge);
1256 }
1257
mlx5_esw_bridge_port_vlans_recreate(struct mlx5_esw_bridge_port * port,struct mlx5_esw_bridge * bridge)1258 static int mlx5_esw_bridge_port_vlans_recreate(struct mlx5_esw_bridge_port *port,
1259 struct mlx5_esw_bridge *bridge)
1260 {
1261 struct mlx5_esw_bridge_offloads *br_offloads = bridge->br_offloads;
1262 struct mlx5_esw_bridge_vlan *vlan;
1263 unsigned long i;
1264 int err;
1265
1266 xa_for_each(&port->vlans, i, vlan) {
1267 mlx5_esw_bridge_vlan_flush(port, vlan, bridge);
1268 err = mlx5_esw_bridge_vlan_push_pop_create(bridge->vlan_proto, vlan->flags, port,
1269 vlan, br_offloads->esw);
1270 if (err) {
1271 esw_warn(br_offloads->esw->dev,
1272 "Failed to create VLAN=%u(proto=%x) push/pop actions (vport=%u,err=%d)\n",
1273 vlan->vid, bridge->vlan_proto, port->vport_num,
1274 err);
1275 return err;
1276 }
1277 }
1278
1279 return 0;
1280 }
1281
1282 static int
mlx5_esw_bridge_vlans_recreate(struct mlx5_esw_bridge * bridge)1283 mlx5_esw_bridge_vlans_recreate(struct mlx5_esw_bridge *bridge)
1284 {
1285 struct mlx5_esw_bridge_offloads *br_offloads = bridge->br_offloads;
1286 struct mlx5_esw_bridge_port *port;
1287 unsigned long i;
1288 int err;
1289
1290 xa_for_each(&br_offloads->ports, i, port) {
1291 if (port->bridge != bridge)
1292 continue;
1293
1294 err = mlx5_esw_bridge_port_vlans_recreate(port, bridge);
1295 if (err)
1296 return err;
1297 }
1298
1299 return 0;
1300 }
1301
1302 static struct mlx5_esw_bridge_vlan *
mlx5_esw_bridge_port_vlan_lookup(u16 vid,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge * bridge,struct mlx5_eswitch * esw)1303 mlx5_esw_bridge_port_vlan_lookup(u16 vid, u16 vport_num, u16 esw_owner_vhca_id,
1304 struct mlx5_esw_bridge *bridge, struct mlx5_eswitch *esw)
1305 {
1306 struct mlx5_esw_bridge_port *port;
1307 struct mlx5_esw_bridge_vlan *vlan;
1308
1309 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, bridge->br_offloads);
1310 if (!port) {
1311 /* FDB is added asynchronously on wq while port might have been deleted
1312 * concurrently. Report on 'info' logging level and skip the FDB offload.
1313 */
1314 esw_info(esw->dev, "Failed to lookup bridge port (vport=%u)\n", vport_num);
1315 return ERR_PTR(-EINVAL);
1316 }
1317
1318 vlan = mlx5_esw_bridge_vlan_lookup(vid, port);
1319 if (!vlan) {
1320 /* FDB is added asynchronously on wq while vlan might have been deleted
1321 * concurrently. Report on 'info' logging level and skip the FDB offload.
1322 */
1323 esw_info(esw->dev, "Failed to lookup bridge port vlan metadata (vport=%u)\n",
1324 vport_num);
1325 return ERR_PTR(-EINVAL);
1326 }
1327
1328 return vlan;
1329 }
1330
1331 static struct mlx5_esw_bridge_fdb_entry *
mlx5_esw_bridge_fdb_lookup(struct mlx5_esw_bridge * bridge,const unsigned char * addr,u16 vid)1332 mlx5_esw_bridge_fdb_lookup(struct mlx5_esw_bridge *bridge,
1333 const unsigned char *addr, u16 vid)
1334 {
1335 struct mlx5_esw_bridge_fdb_key key = {};
1336
1337 ether_addr_copy(key.addr, addr);
1338 key.vid = vid;
1339 return rhashtable_lookup_fast(&bridge->fdb_ht, &key, fdb_ht_params);
1340 }
1341
1342 static struct mlx5_esw_bridge_fdb_entry *
mlx5_esw_bridge_fdb_entry_init(struct net_device * dev,u16 vport_num,u16 esw_owner_vhca_id,const unsigned char * addr,u16 vid,bool added_by_user,bool peer,struct mlx5_eswitch * esw,struct mlx5_esw_bridge * bridge)1343 mlx5_esw_bridge_fdb_entry_init(struct net_device *dev, u16 vport_num, u16 esw_owner_vhca_id,
1344 const unsigned char *addr, u16 vid, bool added_by_user, bool peer,
1345 struct mlx5_eswitch *esw, struct mlx5_esw_bridge *bridge)
1346 {
1347 struct mlx5_esw_bridge_vlan *vlan = NULL;
1348 struct mlx5_esw_bridge_fdb_entry *entry;
1349 struct mlx5_flow_handle *handle;
1350 struct mlx5_fc *counter;
1351 int err;
1352
1353 if (bridge->flags & MLX5_ESW_BRIDGE_VLAN_FILTERING_FLAG && vid) {
1354 vlan = mlx5_esw_bridge_port_vlan_lookup(vid, vport_num, esw_owner_vhca_id, bridge,
1355 esw);
1356 if (IS_ERR(vlan))
1357 return ERR_CAST(vlan);
1358 }
1359
1360 entry = mlx5_esw_bridge_fdb_lookup(bridge, addr, vid);
1361 if (entry)
1362 mlx5_esw_bridge_fdb_entry_notify_and_cleanup(entry, bridge);
1363
1364 entry = kvzalloc(sizeof(*entry), GFP_KERNEL);
1365 if (!entry)
1366 return ERR_PTR(-ENOMEM);
1367
1368 ether_addr_copy(entry->key.addr, addr);
1369 entry->key.vid = vid;
1370 entry->dev = dev;
1371 entry->vport_num = vport_num;
1372 entry->esw_owner_vhca_id = esw_owner_vhca_id;
1373 entry->lastuse = jiffies;
1374 if (added_by_user)
1375 entry->flags |= MLX5_ESW_BRIDGE_FLAG_ADDED_BY_USER;
1376 if (peer)
1377 entry->flags |= MLX5_ESW_BRIDGE_FLAG_PEER;
1378
1379 counter = mlx5_fc_create(esw->dev, true);
1380 if (IS_ERR(counter)) {
1381 err = PTR_ERR(counter);
1382 goto err_ingress_fc_create;
1383 }
1384 entry->ingress_counter = counter;
1385
1386 handle = peer ?
1387 mlx5_esw_bridge_ingress_flow_peer_create(vport_num, esw_owner_vhca_id,
1388 addr, vlan, mlx5_fc_id(counter),
1389 bridge) :
1390 mlx5_esw_bridge_ingress_flow_create(vport_num, addr, vlan,
1391 mlx5_fc_id(counter), bridge);
1392 if (IS_ERR(handle)) {
1393 err = PTR_ERR(handle);
1394 esw_warn(esw->dev, "Failed to create ingress flow(vport=%u,err=%d,peer=%d)\n",
1395 vport_num, err, peer);
1396 goto err_ingress_flow_create;
1397 }
1398 entry->ingress_handle = handle;
1399
1400 if (bridge->flags & MLX5_ESW_BRIDGE_VLAN_FILTERING_FLAG) {
1401 handle = mlx5_esw_bridge_ingress_filter_flow_create(vport_num, addr, bridge);
1402 if (IS_ERR(handle)) {
1403 err = PTR_ERR(handle);
1404 esw_warn(esw->dev, "Failed to create ingress filter(vport=%u,err=%d)\n",
1405 vport_num, err);
1406 goto err_ingress_filter_flow_create;
1407 }
1408 entry->filter_handle = handle;
1409 }
1410
1411 handle = mlx5_esw_bridge_egress_flow_create(vport_num, esw_owner_vhca_id, addr, vlan,
1412 bridge);
1413 if (IS_ERR(handle)) {
1414 err = PTR_ERR(handle);
1415 esw_warn(esw->dev, "Failed to create egress flow(vport=%u,err=%d)\n",
1416 vport_num, err);
1417 goto err_egress_flow_create;
1418 }
1419 entry->egress_handle = handle;
1420
1421 err = rhashtable_insert_fast(&bridge->fdb_ht, &entry->ht_node, fdb_ht_params);
1422 if (err) {
1423 esw_warn(esw->dev, "Failed to insert FDB flow(vport=%u,err=%d)\n", vport_num, err);
1424 goto err_ht_init;
1425 }
1426
1427 if (vlan)
1428 list_add(&entry->vlan_list, &vlan->fdb_list);
1429 else
1430 INIT_LIST_HEAD(&entry->vlan_list);
1431 list_add(&entry->list, &bridge->fdb_list);
1432
1433 trace_mlx5_esw_bridge_fdb_entry_init(entry);
1434 return entry;
1435
1436 err_ht_init:
1437 mlx5_del_flow_rules(entry->egress_handle);
1438 err_egress_flow_create:
1439 if (entry->filter_handle)
1440 mlx5_del_flow_rules(entry->filter_handle);
1441 err_ingress_filter_flow_create:
1442 mlx5_del_flow_rules(entry->ingress_handle);
1443 err_ingress_flow_create:
1444 mlx5_fc_destroy(esw->dev, entry->ingress_counter);
1445 err_ingress_fc_create:
1446 kvfree(entry);
1447 return ERR_PTR(err);
1448 }
1449
mlx5_esw_bridge_ageing_time_set(u16 vport_num,u16 esw_owner_vhca_id,unsigned long ageing_time,struct mlx5_esw_bridge_offloads * br_offloads)1450 int mlx5_esw_bridge_ageing_time_set(u16 vport_num, u16 esw_owner_vhca_id, unsigned long ageing_time,
1451 struct mlx5_esw_bridge_offloads *br_offloads)
1452 {
1453 struct mlx5_esw_bridge *bridge;
1454
1455 bridge = mlx5_esw_bridge_from_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1456 if (!bridge)
1457 return -EINVAL;
1458
1459 bridge->ageing_time = clock_t_to_jiffies(ageing_time);
1460 return 0;
1461 }
1462
mlx5_esw_bridge_vlan_filtering_set(u16 vport_num,u16 esw_owner_vhca_id,bool enable,struct mlx5_esw_bridge_offloads * br_offloads)1463 int mlx5_esw_bridge_vlan_filtering_set(u16 vport_num, u16 esw_owner_vhca_id, bool enable,
1464 struct mlx5_esw_bridge_offloads *br_offloads)
1465 {
1466 struct mlx5_esw_bridge *bridge;
1467 bool filtering;
1468
1469 bridge = mlx5_esw_bridge_from_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1470 if (!bridge)
1471 return -EINVAL;
1472
1473 filtering = bridge->flags & MLX5_ESW_BRIDGE_VLAN_FILTERING_FLAG;
1474 if (filtering == enable)
1475 return 0;
1476
1477 mlx5_esw_bridge_fdb_flush(bridge);
1478 mlx5_esw_bridge_mdb_flush(bridge);
1479 if (enable)
1480 bridge->flags |= MLX5_ESW_BRIDGE_VLAN_FILTERING_FLAG;
1481 else
1482 bridge->flags &= ~MLX5_ESW_BRIDGE_VLAN_FILTERING_FLAG;
1483
1484 return 0;
1485 }
1486
mlx5_esw_bridge_vlan_proto_set(u16 vport_num,u16 esw_owner_vhca_id,u16 proto,struct mlx5_esw_bridge_offloads * br_offloads)1487 int mlx5_esw_bridge_vlan_proto_set(u16 vport_num, u16 esw_owner_vhca_id, u16 proto,
1488 struct mlx5_esw_bridge_offloads *br_offloads)
1489 {
1490 struct mlx5_esw_bridge *bridge;
1491
1492 bridge = mlx5_esw_bridge_from_port_lookup(vport_num, esw_owner_vhca_id,
1493 br_offloads);
1494 if (!bridge)
1495 return -EINVAL;
1496
1497 if (bridge->vlan_proto == proto)
1498 return 0;
1499 if (proto != ETH_P_8021Q && proto != ETH_P_8021AD) {
1500 esw_warn(br_offloads->esw->dev, "Can't set unsupported VLAN protocol %x", proto);
1501 return -EOPNOTSUPP;
1502 }
1503
1504 mlx5_esw_bridge_fdb_flush(bridge);
1505 mlx5_esw_bridge_mdb_flush(bridge);
1506 bridge->vlan_proto = proto;
1507 mlx5_esw_bridge_vlans_recreate(bridge);
1508
1509 return 0;
1510 }
1511
mlx5_esw_bridge_mcast_set(u16 vport_num,u16 esw_owner_vhca_id,bool enable,struct mlx5_esw_bridge_offloads * br_offloads)1512 int mlx5_esw_bridge_mcast_set(u16 vport_num, u16 esw_owner_vhca_id, bool enable,
1513 struct mlx5_esw_bridge_offloads *br_offloads)
1514 {
1515 struct mlx5_eswitch *esw = br_offloads->esw;
1516 struct mlx5_esw_bridge *bridge;
1517 int err = 0;
1518 bool mcast;
1519
1520 if (!(MLX5_CAP_ESW_FLOWTABLE((esw)->dev, fdb_multi_path_any_table) ||
1521 MLX5_CAP_ESW_FLOWTABLE((esw)->dev, fdb_multi_path_any_table_limit_regc)) ||
1522 !MLX5_CAP_ESW_FLOWTABLE((esw)->dev, fdb_uplink_hairpin) ||
1523 !MLX5_CAP_ESW_FLOWTABLE_FDB((esw)->dev, ignore_flow_level))
1524 return -EOPNOTSUPP;
1525
1526 bridge = mlx5_esw_bridge_from_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1527 if (!bridge)
1528 return -EINVAL;
1529
1530 mcast = bridge->flags & MLX5_ESW_BRIDGE_MCAST_FLAG;
1531 if (mcast == enable)
1532 return 0;
1533
1534 if (enable)
1535 err = mlx5_esw_bridge_mcast_enable(bridge);
1536 else
1537 mlx5_esw_bridge_mcast_disable(bridge);
1538
1539 return err;
1540 }
1541
mlx5_esw_bridge_vport_init(u16 vport_num,u16 esw_owner_vhca_id,u16 flags,struct mlx5_esw_bridge_offloads * br_offloads,struct mlx5_esw_bridge * bridge)1542 static int mlx5_esw_bridge_vport_init(u16 vport_num, u16 esw_owner_vhca_id, u16 flags,
1543 struct mlx5_esw_bridge_offloads *br_offloads,
1544 struct mlx5_esw_bridge *bridge)
1545 {
1546 struct mlx5_eswitch *esw = br_offloads->esw;
1547 struct mlx5_esw_bridge_port *port;
1548 int err;
1549
1550 port = kvzalloc(sizeof(*port), GFP_KERNEL);
1551 if (!port)
1552 return -ENOMEM;
1553
1554 port->vport_num = vport_num;
1555 port->esw_owner_vhca_id = esw_owner_vhca_id;
1556 port->bridge = bridge;
1557 port->flags |= flags;
1558 xa_init(&port->vlans);
1559
1560 err = mlx5_esw_bridge_port_mcast_init(port);
1561 if (err) {
1562 esw_warn(esw->dev,
1563 "Failed to initialize port multicast (vport=%u,esw_owner_vhca_id=%u,err=%d)\n",
1564 port->vport_num, port->esw_owner_vhca_id, err);
1565 goto err_port_mcast;
1566 }
1567
1568 err = mlx5_esw_bridge_port_insert(port, br_offloads);
1569 if (err) {
1570 esw_warn(esw->dev,
1571 "Failed to insert port metadata (vport=%u,esw_owner_vhca_id=%u,err=%d)\n",
1572 port->vport_num, port->esw_owner_vhca_id, err);
1573 goto err_port_insert;
1574 }
1575 trace_mlx5_esw_bridge_vport_init(port);
1576
1577 return 0;
1578
1579 err_port_insert:
1580 mlx5_esw_bridge_port_mcast_cleanup(port);
1581 err_port_mcast:
1582 kvfree(port);
1583 return err;
1584 }
1585
mlx5_esw_bridge_vport_cleanup(struct mlx5_esw_bridge_offloads * br_offloads,struct mlx5_esw_bridge_port * port)1586 static int mlx5_esw_bridge_vport_cleanup(struct mlx5_esw_bridge_offloads *br_offloads,
1587 struct mlx5_esw_bridge_port *port)
1588 {
1589 u16 vport_num = port->vport_num, esw_owner_vhca_id = port->esw_owner_vhca_id;
1590 struct mlx5_esw_bridge *bridge = port->bridge;
1591 struct mlx5_esw_bridge_fdb_entry *entry, *tmp;
1592
1593 list_for_each_entry_safe(entry, tmp, &bridge->fdb_list, list)
1594 if (entry->vport_num == vport_num && entry->esw_owner_vhca_id == esw_owner_vhca_id)
1595 mlx5_esw_bridge_fdb_entry_cleanup(entry, bridge);
1596
1597 trace_mlx5_esw_bridge_vport_cleanup(port);
1598 mlx5_esw_bridge_port_vlans_flush(port, bridge);
1599 mlx5_esw_bridge_port_mcast_cleanup(port);
1600 mlx5_esw_bridge_port_erase(port, br_offloads);
1601 kvfree(port);
1602 mlx5_esw_bridge_put(br_offloads, bridge);
1603 return 0;
1604 }
1605
mlx5_esw_bridge_vport_link_with_flags(struct net_device * br_netdev,u16 vport_num,u16 esw_owner_vhca_id,u16 flags,struct mlx5_esw_bridge_offloads * br_offloads,struct netlink_ext_ack * extack)1606 static int mlx5_esw_bridge_vport_link_with_flags(struct net_device *br_netdev, u16 vport_num,
1607 u16 esw_owner_vhca_id, u16 flags,
1608 struct mlx5_esw_bridge_offloads *br_offloads,
1609 struct netlink_ext_ack *extack)
1610 {
1611 struct mlx5_esw_bridge *bridge;
1612 int err;
1613
1614 bridge = mlx5_esw_bridge_lookup(br_netdev, br_offloads);
1615 if (IS_ERR(bridge)) {
1616 NL_SET_ERR_MSG_MOD(extack, "Error checking for existing bridge with same ifindex");
1617 return PTR_ERR(bridge);
1618 }
1619
1620 err = mlx5_esw_bridge_vport_init(vport_num, esw_owner_vhca_id, flags, br_offloads, bridge);
1621 if (err) {
1622 NL_SET_ERR_MSG_MOD(extack, "Error initializing port");
1623 goto err_vport;
1624 }
1625 return 0;
1626
1627 err_vport:
1628 mlx5_esw_bridge_put(br_offloads, bridge);
1629 return err;
1630 }
1631
mlx5_esw_bridge_vport_link(struct net_device * br_netdev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct netlink_ext_ack * extack)1632 int mlx5_esw_bridge_vport_link(struct net_device *br_netdev, u16 vport_num, u16 esw_owner_vhca_id,
1633 struct mlx5_esw_bridge_offloads *br_offloads,
1634 struct netlink_ext_ack *extack)
1635 {
1636 return mlx5_esw_bridge_vport_link_with_flags(br_netdev, vport_num, esw_owner_vhca_id, 0,
1637 br_offloads, extack);
1638 }
1639
mlx5_esw_bridge_vport_unlink(struct net_device * br_netdev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct netlink_ext_ack * extack)1640 int mlx5_esw_bridge_vport_unlink(struct net_device *br_netdev, u16 vport_num,
1641 u16 esw_owner_vhca_id,
1642 struct mlx5_esw_bridge_offloads *br_offloads,
1643 struct netlink_ext_ack *extack)
1644 {
1645 struct mlx5_esw_bridge_port *port;
1646 int err;
1647
1648 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1649 if (!port) {
1650 NL_SET_ERR_MSG_MOD(extack, "Port is not attached to any bridge");
1651 return -EINVAL;
1652 }
1653 if (port->bridge->ifindex != br_netdev->ifindex) {
1654 NL_SET_ERR_MSG_MOD(extack, "Port is attached to another bridge");
1655 return -EINVAL;
1656 }
1657
1658 err = mlx5_esw_bridge_vport_cleanup(br_offloads, port);
1659 if (err)
1660 NL_SET_ERR_MSG_MOD(extack, "Port cleanup failed");
1661 return err;
1662 }
1663
mlx5_esw_bridge_vport_peer_link(struct net_device * br_netdev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct netlink_ext_ack * extack)1664 int mlx5_esw_bridge_vport_peer_link(struct net_device *br_netdev, u16 vport_num,
1665 u16 esw_owner_vhca_id,
1666 struct mlx5_esw_bridge_offloads *br_offloads,
1667 struct netlink_ext_ack *extack)
1668 {
1669 if (!MLX5_CAP_ESW(br_offloads->esw->dev, merged_eswitch))
1670 return 0;
1671
1672 return mlx5_esw_bridge_vport_link_with_flags(br_netdev, vport_num, esw_owner_vhca_id,
1673 MLX5_ESW_BRIDGE_PORT_FLAG_PEER,
1674 br_offloads, extack);
1675 }
1676
mlx5_esw_bridge_vport_peer_unlink(struct net_device * br_netdev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct netlink_ext_ack * extack)1677 int mlx5_esw_bridge_vport_peer_unlink(struct net_device *br_netdev, u16 vport_num,
1678 u16 esw_owner_vhca_id,
1679 struct mlx5_esw_bridge_offloads *br_offloads,
1680 struct netlink_ext_ack *extack)
1681 {
1682 return mlx5_esw_bridge_vport_unlink(br_netdev, vport_num, esw_owner_vhca_id, br_offloads,
1683 extack);
1684 }
1685
mlx5_esw_bridge_port_vlan_add(u16 vport_num,u16 esw_owner_vhca_id,u16 vid,u16 flags,struct mlx5_esw_bridge_offloads * br_offloads,struct netlink_ext_ack * extack)1686 int mlx5_esw_bridge_port_vlan_add(u16 vport_num, u16 esw_owner_vhca_id, u16 vid, u16 flags,
1687 struct mlx5_esw_bridge_offloads *br_offloads,
1688 struct netlink_ext_ack *extack)
1689 {
1690 struct mlx5_esw_bridge_port *port;
1691 struct mlx5_esw_bridge_vlan *vlan;
1692
1693 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1694 if (!port)
1695 return -EINVAL;
1696
1697 vlan = mlx5_esw_bridge_vlan_lookup(vid, port);
1698 if (vlan) {
1699 if (vlan->flags == flags)
1700 return 0;
1701 mlx5_esw_bridge_vlan_cleanup(port, vlan, port->bridge);
1702 }
1703
1704 vlan = mlx5_esw_bridge_vlan_create(port->bridge->vlan_proto, vid, flags, port,
1705 br_offloads->esw);
1706 if (IS_ERR(vlan)) {
1707 NL_SET_ERR_MSG_MOD(extack, "Failed to create VLAN entry");
1708 return PTR_ERR(vlan);
1709 }
1710 return 0;
1711 }
1712
mlx5_esw_bridge_port_vlan_del(u16 vport_num,u16 esw_owner_vhca_id,u16 vid,struct mlx5_esw_bridge_offloads * br_offloads)1713 void mlx5_esw_bridge_port_vlan_del(u16 vport_num, u16 esw_owner_vhca_id, u16 vid,
1714 struct mlx5_esw_bridge_offloads *br_offloads)
1715 {
1716 struct mlx5_esw_bridge_port *port;
1717 struct mlx5_esw_bridge_vlan *vlan;
1718
1719 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1720 if (!port)
1721 return;
1722
1723 vlan = mlx5_esw_bridge_vlan_lookup(vid, port);
1724 if (!vlan)
1725 return;
1726 mlx5_esw_bridge_vlan_cleanup(port, vlan, port->bridge);
1727 }
1728
mlx5_esw_bridge_fdb_update_used(struct net_device * dev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct switchdev_notifier_fdb_info * fdb_info)1729 void mlx5_esw_bridge_fdb_update_used(struct net_device *dev, u16 vport_num, u16 esw_owner_vhca_id,
1730 struct mlx5_esw_bridge_offloads *br_offloads,
1731 struct switchdev_notifier_fdb_info *fdb_info)
1732 {
1733 struct mlx5_esw_bridge_fdb_entry *entry;
1734 struct mlx5_esw_bridge *bridge;
1735
1736 bridge = mlx5_esw_bridge_from_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1737 if (!bridge)
1738 return;
1739
1740 entry = mlx5_esw_bridge_fdb_lookup(bridge, fdb_info->addr, fdb_info->vid);
1741 if (!entry) {
1742 esw_debug(br_offloads->esw->dev,
1743 "FDB update entry with specified key not found (MAC=%pM,vid=%u,vport=%u)\n",
1744 fdb_info->addr, fdb_info->vid, vport_num);
1745 return;
1746 }
1747
1748 entry->lastuse = jiffies;
1749 }
1750
mlx5_esw_bridge_fdb_mark_deleted(struct net_device * dev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct switchdev_notifier_fdb_info * fdb_info)1751 void mlx5_esw_bridge_fdb_mark_deleted(struct net_device *dev, u16 vport_num, u16 esw_owner_vhca_id,
1752 struct mlx5_esw_bridge_offloads *br_offloads,
1753 struct switchdev_notifier_fdb_info *fdb_info)
1754 {
1755 struct mlx5_esw_bridge_fdb_entry *entry;
1756 struct mlx5_esw_bridge *bridge;
1757
1758 bridge = mlx5_esw_bridge_from_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1759 if (!bridge)
1760 return;
1761
1762 entry = mlx5_esw_bridge_fdb_lookup(bridge, fdb_info->addr, fdb_info->vid);
1763 if (!entry) {
1764 esw_debug(br_offloads->esw->dev,
1765 "FDB mark deleted entry with specified key not found (MAC=%pM,vid=%u,vport=%u)\n",
1766 fdb_info->addr, fdb_info->vid, vport_num);
1767 return;
1768 }
1769
1770 entry->flags |= MLX5_ESW_BRIDGE_FLAG_DELETED;
1771 }
1772
mlx5_esw_bridge_fdb_create(struct net_device * dev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct switchdev_notifier_fdb_info * fdb_info)1773 void mlx5_esw_bridge_fdb_create(struct net_device *dev, u16 vport_num, u16 esw_owner_vhca_id,
1774 struct mlx5_esw_bridge_offloads *br_offloads,
1775 struct switchdev_notifier_fdb_info *fdb_info)
1776 {
1777 struct mlx5_esw_bridge_fdb_entry *entry;
1778 struct mlx5_esw_bridge_port *port;
1779 struct mlx5_esw_bridge *bridge;
1780
1781 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1782 if (!port)
1783 return;
1784
1785 bridge = port->bridge;
1786 entry = mlx5_esw_bridge_fdb_entry_init(dev, vport_num, esw_owner_vhca_id, fdb_info->addr,
1787 fdb_info->vid, fdb_info->added_by_user,
1788 port->flags & MLX5_ESW_BRIDGE_PORT_FLAG_PEER,
1789 br_offloads->esw, bridge);
1790 if (IS_ERR(entry))
1791 return;
1792
1793 if (entry->flags & MLX5_ESW_BRIDGE_FLAG_ADDED_BY_USER)
1794 mlx5_esw_bridge_fdb_offload_notify(dev, entry->key.addr, entry->key.vid,
1795 SWITCHDEV_FDB_OFFLOADED);
1796 else if (!(entry->flags & MLX5_ESW_BRIDGE_FLAG_PEER))
1797 /* Take over dynamic entries to prevent kernel bridge from aging them out. */
1798 mlx5_esw_bridge_fdb_offload_notify(dev, entry->key.addr, entry->key.vid,
1799 SWITCHDEV_FDB_ADD_TO_BRIDGE);
1800 }
1801
mlx5_esw_bridge_fdb_remove(struct net_device * dev,u16 vport_num,u16 esw_owner_vhca_id,struct mlx5_esw_bridge_offloads * br_offloads,struct switchdev_notifier_fdb_info * fdb_info)1802 void mlx5_esw_bridge_fdb_remove(struct net_device *dev, u16 vport_num, u16 esw_owner_vhca_id,
1803 struct mlx5_esw_bridge_offloads *br_offloads,
1804 struct switchdev_notifier_fdb_info *fdb_info)
1805 {
1806 struct mlx5_eswitch *esw = br_offloads->esw;
1807 struct mlx5_esw_bridge_fdb_entry *entry;
1808 struct mlx5_esw_bridge *bridge;
1809
1810 bridge = mlx5_esw_bridge_from_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1811 if (!bridge)
1812 return;
1813
1814 entry = mlx5_esw_bridge_fdb_lookup(bridge, fdb_info->addr, fdb_info->vid);
1815 if (!entry) {
1816 esw_debug(esw->dev,
1817 "FDB remove entry with specified key not found (MAC=%pM,vid=%u,vport=%u)\n",
1818 fdb_info->addr, fdb_info->vid, vport_num);
1819 return;
1820 }
1821
1822 mlx5_esw_bridge_fdb_entry_notify_and_cleanup(entry, bridge);
1823 }
1824
mlx5_esw_bridge_update(struct mlx5_esw_bridge_offloads * br_offloads)1825 void mlx5_esw_bridge_update(struct mlx5_esw_bridge_offloads *br_offloads)
1826 {
1827 struct mlx5_esw_bridge_fdb_entry *entry, *tmp;
1828 struct mlx5_esw_bridge *bridge;
1829
1830 list_for_each_entry(bridge, &br_offloads->bridges, list) {
1831 list_for_each_entry_safe(entry, tmp, &bridge->fdb_list, list) {
1832 unsigned long lastuse =
1833 (unsigned long)mlx5_fc_query_lastuse(entry->ingress_counter);
1834
1835 if (entry->flags & (MLX5_ESW_BRIDGE_FLAG_ADDED_BY_USER |
1836 MLX5_ESW_BRIDGE_FLAG_DELETED))
1837 continue;
1838
1839 if (time_after(lastuse, entry->lastuse))
1840 mlx5_esw_bridge_fdb_entry_refresh(entry);
1841 else if (!(entry->flags & MLX5_ESW_BRIDGE_FLAG_PEER) &&
1842 time_is_before_jiffies(entry->lastuse + bridge->ageing_time))
1843 mlx5_esw_bridge_fdb_entry_notify_and_cleanup(entry, bridge);
1844 }
1845 }
1846 }
1847
mlx5_esw_bridge_port_mdb_add(struct net_device * dev,u16 vport_num,u16 esw_owner_vhca_id,const unsigned char * addr,u16 vid,struct mlx5_esw_bridge_offloads * br_offloads,struct netlink_ext_ack * extack)1848 int mlx5_esw_bridge_port_mdb_add(struct net_device *dev, u16 vport_num, u16 esw_owner_vhca_id,
1849 const unsigned char *addr, u16 vid,
1850 struct mlx5_esw_bridge_offloads *br_offloads,
1851 struct netlink_ext_ack *extack)
1852 {
1853 struct mlx5_esw_bridge_vlan *vlan;
1854 struct mlx5_esw_bridge_port *port;
1855 struct mlx5_esw_bridge *bridge;
1856 int err;
1857
1858 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1859 if (!port) {
1860 esw_warn(br_offloads->esw->dev,
1861 "Failed to lookup bridge port to add MDB (MAC=%pM,vport=%u)\n",
1862 addr, vport_num);
1863 NL_SET_ERR_MSG_FMT_MOD(extack,
1864 "Failed to lookup bridge port to add MDB (MAC=%pM,vport=%u)\n",
1865 addr, vport_num);
1866 return -EINVAL;
1867 }
1868
1869 bridge = port->bridge;
1870 if (bridge->flags & MLX5_ESW_BRIDGE_VLAN_FILTERING_FLAG && vid) {
1871 vlan = mlx5_esw_bridge_vlan_lookup(vid, port);
1872 if (!vlan) {
1873 esw_warn(br_offloads->esw->dev,
1874 "Failed to lookup bridge port vlan metadata to create MDB (MAC=%pM,vid=%u,vport=%u)\n",
1875 addr, vid, vport_num);
1876 NL_SET_ERR_MSG_FMT_MOD(extack,
1877 "Failed to lookup vlan metadata for MDB (MAC=%pM,vid=%u,vport=%u)\n",
1878 addr, vid, vport_num);
1879 return -EINVAL;
1880 }
1881 }
1882
1883 err = mlx5_esw_bridge_port_mdb_attach(dev, port, addr, vid);
1884 if (err) {
1885 NL_SET_ERR_MSG_FMT_MOD(extack, "Failed to add MDB (MAC=%pM,vid=%u,vport=%u)\n",
1886 addr, vid, vport_num);
1887 return err;
1888 }
1889
1890 return 0;
1891 }
1892
mlx5_esw_bridge_port_mdb_del(struct net_device * dev,u16 vport_num,u16 esw_owner_vhca_id,const unsigned char * addr,u16 vid,struct mlx5_esw_bridge_offloads * br_offloads)1893 void mlx5_esw_bridge_port_mdb_del(struct net_device *dev, u16 vport_num, u16 esw_owner_vhca_id,
1894 const unsigned char *addr, u16 vid,
1895 struct mlx5_esw_bridge_offloads *br_offloads)
1896 {
1897 struct mlx5_esw_bridge_port *port;
1898
1899 port = mlx5_esw_bridge_port_lookup(vport_num, esw_owner_vhca_id, br_offloads);
1900 if (!port)
1901 return;
1902
1903 mlx5_esw_bridge_port_mdb_detach(dev, port, addr, vid);
1904 }
1905
mlx5_esw_bridge_flush(struct mlx5_esw_bridge_offloads * br_offloads)1906 static void mlx5_esw_bridge_flush(struct mlx5_esw_bridge_offloads *br_offloads)
1907 {
1908 struct mlx5_esw_bridge_port *port;
1909 unsigned long i;
1910
1911 xa_for_each(&br_offloads->ports, i, port)
1912 mlx5_esw_bridge_vport_cleanup(br_offloads, port);
1913
1914 WARN_ONCE(!list_empty(&br_offloads->bridges),
1915 "Cleaning up bridge offloads while still having bridges attached\n");
1916 }
1917
mlx5_esw_bridge_init(struct mlx5_eswitch * esw)1918 struct mlx5_esw_bridge_offloads *mlx5_esw_bridge_init(struct mlx5_eswitch *esw)
1919 {
1920 struct mlx5_esw_bridge_offloads *br_offloads;
1921
1922 ASSERT_RTNL();
1923
1924 br_offloads = kvzalloc(sizeof(*br_offloads), GFP_KERNEL);
1925 if (!br_offloads)
1926 return ERR_PTR(-ENOMEM);
1927
1928 INIT_LIST_HEAD(&br_offloads->bridges);
1929 xa_init(&br_offloads->ports);
1930 br_offloads->esw = esw;
1931 esw->br_offloads = br_offloads;
1932 mlx5_esw_bridge_debugfs_offloads_init(br_offloads);
1933
1934 return br_offloads;
1935 }
1936
mlx5_esw_bridge_cleanup(struct mlx5_eswitch * esw)1937 void mlx5_esw_bridge_cleanup(struct mlx5_eswitch *esw)
1938 {
1939 struct mlx5_esw_bridge_offloads *br_offloads = esw->br_offloads;
1940
1941 ASSERT_RTNL();
1942
1943 if (!br_offloads)
1944 return;
1945
1946 mlx5_esw_bridge_flush(br_offloads);
1947 WARN_ON(!xa_empty(&br_offloads->ports));
1948 mlx5_esw_bridge_debugfs_offloads_cleanup(br_offloads);
1949
1950 esw->br_offloads = NULL;
1951 kvfree(br_offloads);
1952 }
1953