]> Git Repo - J-linux.git/blob - drivers/net/ethernet/intel/ice/ice_vsi_vlan_lib.c
Merge tag 'block-6.5-2023-07-03' of git://git.kernel.dk/linux
[J-linux.git] / drivers / net / ethernet / intel / ice / ice_vsi_vlan_lib.c
1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright (C) 2019-2021, Intel Corporation. */
3
4 #include "ice_vsi_vlan_lib.h"
5 #include "ice_lib.h"
6 #include "ice_fltr.h"
7 #include "ice.h"
8
9 static void print_invalid_tpid(struct ice_vsi *vsi, u16 tpid)
10 {
11         dev_err(ice_pf_to_dev(vsi->back), "%s %d specified invalid VLAN tpid 0x%04x\n",
12                 ice_vsi_type_str(vsi->type), vsi->idx, tpid);
13 }
14
15 /**
16  * validate_vlan - check if the ice_vlan passed in is valid
17  * @vsi: VSI used for printing error message
18  * @vlan: ice_vlan structure to validate
19  *
20  * Return true if the VLAN TPID is valid or if the VLAN TPID is 0 and the VLAN
21  * VID is 0, which allows for non-zero VLAN filters with the specified VLAN TPID
22  * and untagged VLAN 0 filters to be added to the prune list respectively.
23  */
24 static bool validate_vlan(struct ice_vsi *vsi, struct ice_vlan *vlan)
25 {
26         if (vlan->tpid != ETH_P_8021Q && vlan->tpid != ETH_P_8021AD &&
27             vlan->tpid != ETH_P_QINQ1 && (vlan->tpid || vlan->vid)) {
28                 print_invalid_tpid(vsi, vlan->tpid);
29                 return false;
30         }
31
32         return true;
33 }
34
35 /**
36  * ice_vsi_add_vlan - default add VLAN implementation for all VSI types
37  * @vsi: VSI being configured
38  * @vlan: VLAN filter to add
39  */
40 int ice_vsi_add_vlan(struct ice_vsi *vsi, struct ice_vlan *vlan)
41 {
42         int err;
43
44         if (!validate_vlan(vsi, vlan))
45                 return -EINVAL;
46
47         err = ice_fltr_add_vlan(vsi, vlan);
48         if (err && err != -EEXIST) {
49                 dev_err(ice_pf_to_dev(vsi->back), "Failure Adding VLAN %d on VSI %i, status %d\n",
50                         vlan->vid, vsi->vsi_num, err);
51                 return err;
52         }
53
54         vsi->num_vlan++;
55         return 0;
56 }
57
58 /**
59  * ice_vsi_del_vlan - default del VLAN implementation for all VSI types
60  * @vsi: VSI being configured
61  * @vlan: VLAN filter to delete
62  */
63 int ice_vsi_del_vlan(struct ice_vsi *vsi, struct ice_vlan *vlan)
64 {
65         struct ice_pf *pf = vsi->back;
66         struct device *dev;
67         int err;
68
69         if (!validate_vlan(vsi, vlan))
70                 return -EINVAL;
71
72         dev = ice_pf_to_dev(pf);
73
74         err = ice_fltr_remove_vlan(vsi, vlan);
75         if (!err)
76                 vsi->num_vlan--;
77         else if (err == -ENOENT || err == -EBUSY)
78                 err = 0;
79         else
80                 dev_err(dev, "Error removing VLAN %d on VSI %i error: %d\n",
81                         vlan->vid, vsi->vsi_num, err);
82
83         return err;
84 }
85
86 /**
87  * ice_vsi_manage_vlan_insertion - Manage VLAN insertion for the VSI for Tx
88  * @vsi: the VSI being changed
89  */
90 static int ice_vsi_manage_vlan_insertion(struct ice_vsi *vsi)
91 {
92         struct ice_hw *hw = &vsi->back->hw;
93         struct ice_vsi_ctx *ctxt;
94         int err;
95
96         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
97         if (!ctxt)
98                 return -ENOMEM;
99
100         /* Here we are configuring the VSI to let the driver add VLAN tags by
101          * setting inner_vlan_flags to ICE_AQ_VSI_INNER_VLAN_TX_MODE_ALL. The actual VLAN tag
102          * insertion happens in the Tx hot path, in ice_tx_map.
103          */
104         ctxt->info.inner_vlan_flags = ICE_AQ_VSI_INNER_VLAN_TX_MODE_ALL;
105
106         /* Preserve existing VLAN strip setting */
107         ctxt->info.inner_vlan_flags |= (vsi->info.inner_vlan_flags &
108                                         ICE_AQ_VSI_INNER_VLAN_EMODE_M);
109
110         ctxt->info.valid_sections = cpu_to_le16(ICE_AQ_VSI_PROP_VLAN_VALID);
111
112         err = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
113         if (err) {
114                 dev_err(ice_pf_to_dev(vsi->back), "update VSI for VLAN insert failed, err %d aq_err %s\n",
115                         err, ice_aq_str(hw->adminq.sq_last_status));
116                 goto out;
117         }
118
119         vsi->info.inner_vlan_flags = ctxt->info.inner_vlan_flags;
120 out:
121         kfree(ctxt);
122         return err;
123 }
124
125 /**
126  * ice_vsi_manage_vlan_stripping - Manage VLAN stripping for the VSI for Rx
127  * @vsi: the VSI being changed
128  * @ena: boolean value indicating if this is a enable or disable request
129  */
130 static int ice_vsi_manage_vlan_stripping(struct ice_vsi *vsi, bool ena)
131 {
132         struct ice_hw *hw = &vsi->back->hw;
133         struct ice_vsi_ctx *ctxt;
134         int err;
135
136         /* do not allow modifying VLAN stripping when a port VLAN is configured
137          * on this VSI
138          */
139         if (vsi->info.port_based_inner_vlan)
140                 return 0;
141
142         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
143         if (!ctxt)
144                 return -ENOMEM;
145
146         /* Here we are configuring what the VSI should do with the VLAN tag in
147          * the Rx packet. We can either leave the tag in the packet or put it in
148          * the Rx descriptor.
149          */
150         if (ena)
151                 /* Strip VLAN tag from Rx packet and put it in the desc */
152                 ctxt->info.inner_vlan_flags = ICE_AQ_VSI_INNER_VLAN_EMODE_STR_BOTH;
153         else
154                 /* Disable stripping. Leave tag in packet */
155                 ctxt->info.inner_vlan_flags = ICE_AQ_VSI_INNER_VLAN_EMODE_NOTHING;
156
157         /* Allow all packets untagged/tagged */
158         ctxt->info.inner_vlan_flags |= ICE_AQ_VSI_INNER_VLAN_TX_MODE_ALL;
159
160         ctxt->info.valid_sections = cpu_to_le16(ICE_AQ_VSI_PROP_VLAN_VALID);
161
162         err = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
163         if (err) {
164                 dev_err(ice_pf_to_dev(vsi->back), "update VSI for VLAN strip failed, ena = %d err %d aq_err %s\n",
165                         ena, err, ice_aq_str(hw->adminq.sq_last_status));
166                 goto out;
167         }
168
169         vsi->info.inner_vlan_flags = ctxt->info.inner_vlan_flags;
170 out:
171         kfree(ctxt);
172         return err;
173 }
174
175 int ice_vsi_ena_inner_stripping(struct ice_vsi *vsi, const u16 tpid)
176 {
177         if (tpid != ETH_P_8021Q) {
178                 print_invalid_tpid(vsi, tpid);
179                 return -EINVAL;
180         }
181
182         return ice_vsi_manage_vlan_stripping(vsi, true);
183 }
184
185 int ice_vsi_dis_inner_stripping(struct ice_vsi *vsi)
186 {
187         return ice_vsi_manage_vlan_stripping(vsi, false);
188 }
189
190 int ice_vsi_ena_inner_insertion(struct ice_vsi *vsi, const u16 tpid)
191 {
192         if (tpid != ETH_P_8021Q) {
193                 print_invalid_tpid(vsi, tpid);
194                 return -EINVAL;
195         }
196
197         return ice_vsi_manage_vlan_insertion(vsi);
198 }
199
200 int ice_vsi_dis_inner_insertion(struct ice_vsi *vsi)
201 {
202         return ice_vsi_manage_vlan_insertion(vsi);
203 }
204
205 /**
206  * __ice_vsi_set_inner_port_vlan - set port VLAN VSI context settings to enable a port VLAN
207  * @vsi: the VSI to update
208  * @pvid_info: VLAN ID and QoS used to set the PVID VSI context field
209  */
210 static int __ice_vsi_set_inner_port_vlan(struct ice_vsi *vsi, u16 pvid_info)
211 {
212         struct ice_hw *hw = &vsi->back->hw;
213         struct ice_aqc_vsi_props *info;
214         struct ice_vsi_ctx *ctxt;
215         int ret;
216
217         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
218         if (!ctxt)
219                 return -ENOMEM;
220
221         ctxt->info = vsi->info;
222         info = &ctxt->info;
223         info->inner_vlan_flags = ICE_AQ_VSI_INNER_VLAN_TX_MODE_ACCEPTUNTAGGED |
224                 ICE_AQ_VSI_INNER_VLAN_INSERT_PVID |
225                 ICE_AQ_VSI_INNER_VLAN_EMODE_STR;
226         info->sw_flags2 |= ICE_AQ_VSI_SW_FLAG_RX_VLAN_PRUNE_ENA;
227
228         info->port_based_inner_vlan = cpu_to_le16(pvid_info);
229         info->valid_sections = cpu_to_le16(ICE_AQ_VSI_PROP_VLAN_VALID |
230                                            ICE_AQ_VSI_PROP_SW_VALID);
231
232         ret = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
233         if (ret) {
234                 dev_info(ice_hw_to_dev(hw), "update VSI for port VLAN failed, err %d aq_err %s\n",
235                          ret, ice_aq_str(hw->adminq.sq_last_status));
236                 goto out;
237         }
238
239         vsi->info.inner_vlan_flags = info->inner_vlan_flags;
240         vsi->info.sw_flags2 = info->sw_flags2;
241         vsi->info.port_based_inner_vlan = info->port_based_inner_vlan;
242 out:
243         kfree(ctxt);
244         return ret;
245 }
246
247 int ice_vsi_set_inner_port_vlan(struct ice_vsi *vsi, struct ice_vlan *vlan)
248 {
249         u16 port_vlan_info;
250
251         if (vlan->tpid != ETH_P_8021Q)
252                 return -EINVAL;
253
254         if (vlan->prio > 7)
255                 return -EINVAL;
256
257         port_vlan_info = vlan->vid | (vlan->prio << VLAN_PRIO_SHIFT);
258
259         return __ice_vsi_set_inner_port_vlan(vsi, port_vlan_info);
260 }
261
262 /**
263  * ice_cfg_vlan_pruning - enable or disable VLAN pruning on the VSI
264  * @vsi: VSI to enable or disable VLAN pruning on
265  * @ena: set to true to enable VLAN pruning and false to disable it
266  *
267  * returns 0 if VSI is updated, negative otherwise
268  */
269 static int ice_cfg_vlan_pruning(struct ice_vsi *vsi, bool ena)
270 {
271         struct ice_vsi_ctx *ctxt;
272         struct ice_pf *pf;
273         int status;
274
275         if (!vsi)
276                 return -EINVAL;
277
278         /* Don't enable VLAN pruning if the netdev is currently in promiscuous
279          * mode. VLAN pruning will be enabled when the interface exits
280          * promiscuous mode if any VLAN filters are active.
281          */
282         if (vsi->netdev && vsi->netdev->flags & IFF_PROMISC && ena)
283                 return 0;
284
285         pf = vsi->back;
286         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
287         if (!ctxt)
288                 return -ENOMEM;
289
290         ctxt->info = vsi->info;
291
292         if (ena)
293                 ctxt->info.sw_flags2 |= ICE_AQ_VSI_SW_FLAG_RX_VLAN_PRUNE_ENA;
294         else
295                 ctxt->info.sw_flags2 &= ~ICE_AQ_VSI_SW_FLAG_RX_VLAN_PRUNE_ENA;
296
297         ctxt->info.valid_sections = cpu_to_le16(ICE_AQ_VSI_PROP_SW_VALID);
298
299         status = ice_update_vsi(&pf->hw, vsi->idx, ctxt, NULL);
300         if (status) {
301                 netdev_err(vsi->netdev, "%sabling VLAN pruning on VSI handle: %d, VSI HW ID: %d failed, err = %d, aq_err = %s\n",
302                            ena ? "En" : "Dis", vsi->idx, vsi->vsi_num, status,
303                            ice_aq_str(pf->hw.adminq.sq_last_status));
304                 goto err_out;
305         }
306
307         vsi->info.sw_flags2 = ctxt->info.sw_flags2;
308
309         kfree(ctxt);
310         return 0;
311
312 err_out:
313         kfree(ctxt);
314         return status;
315 }
316
317 int ice_vsi_ena_rx_vlan_filtering(struct ice_vsi *vsi)
318 {
319         return ice_cfg_vlan_pruning(vsi, true);
320 }
321
322 int ice_vsi_dis_rx_vlan_filtering(struct ice_vsi *vsi)
323 {
324         return ice_cfg_vlan_pruning(vsi, false);
325 }
326
327 static int ice_cfg_vlan_antispoof(struct ice_vsi *vsi, bool enable)
328 {
329         struct ice_vsi_ctx *ctx;
330         int err;
331
332         ctx = kzalloc(sizeof(*ctx), GFP_KERNEL);
333         if (!ctx)
334                 return -ENOMEM;
335
336         ctx->info.sec_flags = vsi->info.sec_flags;
337         ctx->info.valid_sections = cpu_to_le16(ICE_AQ_VSI_PROP_SECURITY_VALID);
338
339         if (enable)
340                 ctx->info.sec_flags |= ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA <<
341                         ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S;
342         else
343                 ctx->info.sec_flags &= ~(ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA <<
344                                          ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S);
345
346         err = ice_update_vsi(&vsi->back->hw, vsi->idx, ctx, NULL);
347         if (err)
348                 dev_err(ice_pf_to_dev(vsi->back), "Failed to configure Tx VLAN anti-spoof %s for VSI %d, error %d\n",
349                         enable ? "ON" : "OFF", vsi->vsi_num, err);
350         else
351                 vsi->info.sec_flags = ctx->info.sec_flags;
352
353         kfree(ctx);
354
355         return err;
356 }
357
358 int ice_vsi_ena_tx_vlan_filtering(struct ice_vsi *vsi)
359 {
360         return ice_cfg_vlan_antispoof(vsi, true);
361 }
362
363 int ice_vsi_dis_tx_vlan_filtering(struct ice_vsi *vsi)
364 {
365         return ice_cfg_vlan_antispoof(vsi, false);
366 }
367
368 /**
369  * tpid_to_vsi_outer_vlan_type - convert from TPID to VSI context based tag_type
370  * @tpid: tpid used to translate into VSI context based tag_type
371  * @tag_type: output variable to hold the VSI context based tag type
372  */
373 static int tpid_to_vsi_outer_vlan_type(u16 tpid, u8 *tag_type)
374 {
375         switch (tpid) {
376         case ETH_P_8021Q:
377                 *tag_type = ICE_AQ_VSI_OUTER_TAG_VLAN_8100;
378                 break;
379         case ETH_P_8021AD:
380                 *tag_type = ICE_AQ_VSI_OUTER_TAG_STAG;
381                 break;
382         case ETH_P_QINQ1:
383                 *tag_type = ICE_AQ_VSI_OUTER_TAG_VLAN_9100;
384                 break;
385         default:
386                 *tag_type = 0;
387                 return -EINVAL;
388         }
389
390         return 0;
391 }
392
393 /**
394  * ice_vsi_ena_outer_stripping - enable outer VLAN stripping
395  * @vsi: VSI to configure
396  * @tpid: TPID to enable outer VLAN stripping for
397  *
398  * Enable outer VLAN stripping via VSI context. This function should only be
399  * used if DVM is supported. Also, this function should never be called directly
400  * as it should be part of ice_vsi_vlan_ops if it's needed.
401  *
402  * Since the VSI context only supports a single TPID for insertion and
403  * stripping, setting the TPID for stripping will affect the TPID for insertion.
404  * Callers need to be aware of this limitation.
405  *
406  * Only modify outer VLAN stripping settings and the VLAN TPID. Outer VLAN
407  * insertion settings are unmodified.
408  *
409  * This enables hardware to strip a VLAN tag with the specified TPID to be
410  * stripped from the packet and placed in the receive descriptor.
411  */
412 int ice_vsi_ena_outer_stripping(struct ice_vsi *vsi, u16 tpid)
413 {
414         struct ice_hw *hw = &vsi->back->hw;
415         struct ice_vsi_ctx *ctxt;
416         u8 tag_type;
417         int err;
418
419         /* do not allow modifying VLAN stripping when a port VLAN is configured
420          * on this VSI
421          */
422         if (vsi->info.port_based_outer_vlan)
423                 return 0;
424
425         if (tpid_to_vsi_outer_vlan_type(tpid, &tag_type))
426                 return -EINVAL;
427
428         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
429         if (!ctxt)
430                 return -ENOMEM;
431
432         ctxt->info.valid_sections =
433                 cpu_to_le16(ICE_AQ_VSI_PROP_OUTER_TAG_VALID);
434         /* clear current outer VLAN strip settings */
435         ctxt->info.outer_vlan_flags = vsi->info.outer_vlan_flags &
436                 ~(ICE_AQ_VSI_OUTER_VLAN_EMODE_M | ICE_AQ_VSI_OUTER_TAG_TYPE_M);
437         ctxt->info.outer_vlan_flags |=
438                 ((ICE_AQ_VSI_OUTER_VLAN_EMODE_SHOW_BOTH <<
439                   ICE_AQ_VSI_OUTER_VLAN_EMODE_S) |
440                  ((tag_type << ICE_AQ_VSI_OUTER_TAG_TYPE_S) &
441                   ICE_AQ_VSI_OUTER_TAG_TYPE_M));
442
443         err = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
444         if (err)
445                 dev_err(ice_pf_to_dev(vsi->back), "update VSI for enabling outer VLAN stripping failed, err %d aq_err %s\n",
446                         err, ice_aq_str(hw->adminq.sq_last_status));
447         else
448                 vsi->info.outer_vlan_flags = ctxt->info.outer_vlan_flags;
449
450         kfree(ctxt);
451         return err;
452 }
453
454 /**
455  * ice_vsi_dis_outer_stripping - disable outer VLAN stripping
456  * @vsi: VSI to configure
457  *
458  * Disable outer VLAN stripping via VSI context. This function should only be
459  * used if DVM is supported. Also, this function should never be called directly
460  * as it should be part of ice_vsi_vlan_ops if it's needed.
461  *
462  * Only modify the outer VLAN stripping settings. The VLAN TPID and outer VLAN
463  * insertion settings are unmodified.
464  *
465  * This tells the hardware to not strip any VLAN tagged packets, thus leaving
466  * them in the packet. This enables software offloaded VLAN stripping and
467  * disables hardware offloaded VLAN stripping.
468  */
469 int ice_vsi_dis_outer_stripping(struct ice_vsi *vsi)
470 {
471         struct ice_hw *hw = &vsi->back->hw;
472         struct ice_vsi_ctx *ctxt;
473         int err;
474
475         if (vsi->info.port_based_outer_vlan)
476                 return 0;
477
478         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
479         if (!ctxt)
480                 return -ENOMEM;
481
482         ctxt->info.valid_sections =
483                 cpu_to_le16(ICE_AQ_VSI_PROP_OUTER_TAG_VALID);
484         /* clear current outer VLAN strip settings */
485         ctxt->info.outer_vlan_flags = vsi->info.outer_vlan_flags &
486                 ~ICE_AQ_VSI_OUTER_VLAN_EMODE_M;
487         ctxt->info.outer_vlan_flags |= ICE_AQ_VSI_OUTER_VLAN_EMODE_NOTHING <<
488                 ICE_AQ_VSI_OUTER_VLAN_EMODE_S;
489
490         err = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
491         if (err)
492                 dev_err(ice_pf_to_dev(vsi->back), "update VSI for disabling outer VLAN stripping failed, err %d aq_err %s\n",
493                         err, ice_aq_str(hw->adminq.sq_last_status));
494         else
495                 vsi->info.outer_vlan_flags = ctxt->info.outer_vlan_flags;
496
497         kfree(ctxt);
498         return err;
499 }
500
501 /**
502  * ice_vsi_ena_outer_insertion - enable outer VLAN insertion
503  * @vsi: VSI to configure
504  * @tpid: TPID to enable outer VLAN insertion for
505  *
506  * Enable outer VLAN insertion via VSI context. This function should only be
507  * used if DVM is supported. Also, this function should never be called directly
508  * as it should be part of ice_vsi_vlan_ops if it's needed.
509  *
510  * Since the VSI context only supports a single TPID for insertion and
511  * stripping, setting the TPID for insertion will affect the TPID for stripping.
512  * Callers need to be aware of this limitation.
513  *
514  * Only modify outer VLAN insertion settings and the VLAN TPID. Outer VLAN
515  * stripping settings are unmodified.
516  *
517  * This allows a VLAN tag with the specified TPID to be inserted in the transmit
518  * descriptor.
519  */
520 int ice_vsi_ena_outer_insertion(struct ice_vsi *vsi, u16 tpid)
521 {
522         struct ice_hw *hw = &vsi->back->hw;
523         struct ice_vsi_ctx *ctxt;
524         u8 tag_type;
525         int err;
526
527         if (vsi->info.port_based_outer_vlan)
528                 return 0;
529
530         if (tpid_to_vsi_outer_vlan_type(tpid, &tag_type))
531                 return -EINVAL;
532
533         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
534         if (!ctxt)
535                 return -ENOMEM;
536
537         ctxt->info.valid_sections =
538                 cpu_to_le16(ICE_AQ_VSI_PROP_OUTER_TAG_VALID);
539         /* clear current outer VLAN insertion settings */
540         ctxt->info.outer_vlan_flags = vsi->info.outer_vlan_flags &
541                 ~(ICE_AQ_VSI_OUTER_VLAN_PORT_BASED_INSERT |
542                   ICE_AQ_VSI_OUTER_VLAN_BLOCK_TX_DESC |
543                   ICE_AQ_VSI_OUTER_VLAN_TX_MODE_M |
544                   ICE_AQ_VSI_OUTER_TAG_TYPE_M);
545         ctxt->info.outer_vlan_flags |=
546                 ((ICE_AQ_VSI_OUTER_VLAN_TX_MODE_ALL <<
547                   ICE_AQ_VSI_OUTER_VLAN_TX_MODE_S) &
548                  ICE_AQ_VSI_OUTER_VLAN_TX_MODE_M) |
549                 ((tag_type << ICE_AQ_VSI_OUTER_TAG_TYPE_S) &
550                  ICE_AQ_VSI_OUTER_TAG_TYPE_M);
551
552         err = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
553         if (err)
554                 dev_err(ice_pf_to_dev(vsi->back), "update VSI for enabling outer VLAN insertion failed, err %d aq_err %s\n",
555                         err, ice_aq_str(hw->adminq.sq_last_status));
556         else
557                 vsi->info.outer_vlan_flags = ctxt->info.outer_vlan_flags;
558
559         kfree(ctxt);
560         return err;
561 }
562
563 /**
564  * ice_vsi_dis_outer_insertion - disable outer VLAN insertion
565  * @vsi: VSI to configure
566  *
567  * Disable outer VLAN insertion via VSI context. This function should only be
568  * used if DVM is supported. Also, this function should never be called directly
569  * as it should be part of ice_vsi_vlan_ops if it's needed.
570  *
571  * Only modify the outer VLAN insertion settings. The VLAN TPID and outer VLAN
572  * settings are unmodified.
573  *
574  * This tells the hardware to not allow any VLAN tagged packets in the transmit
575  * descriptor. This enables software offloaded VLAN insertion and disables
576  * hardware offloaded VLAN insertion.
577  */
578 int ice_vsi_dis_outer_insertion(struct ice_vsi *vsi)
579 {
580         struct ice_hw *hw = &vsi->back->hw;
581         struct ice_vsi_ctx *ctxt;
582         int err;
583
584         if (vsi->info.port_based_outer_vlan)
585                 return 0;
586
587         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
588         if (!ctxt)
589                 return -ENOMEM;
590
591         ctxt->info.valid_sections =
592                 cpu_to_le16(ICE_AQ_VSI_PROP_OUTER_TAG_VALID);
593         /* clear current outer VLAN insertion settings */
594         ctxt->info.outer_vlan_flags = vsi->info.outer_vlan_flags &
595                 ~(ICE_AQ_VSI_OUTER_VLAN_PORT_BASED_INSERT |
596                   ICE_AQ_VSI_OUTER_VLAN_TX_MODE_M);
597         ctxt->info.outer_vlan_flags |=
598                 ICE_AQ_VSI_OUTER_VLAN_BLOCK_TX_DESC |
599                 ((ICE_AQ_VSI_OUTER_VLAN_TX_MODE_ALL <<
600                   ICE_AQ_VSI_OUTER_VLAN_TX_MODE_S) &
601                  ICE_AQ_VSI_OUTER_VLAN_TX_MODE_M);
602
603         err = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
604         if (err)
605                 dev_err(ice_pf_to_dev(vsi->back), "update VSI for disabling outer VLAN insertion failed, err %d aq_err %s\n",
606                         err, ice_aq_str(hw->adminq.sq_last_status));
607         else
608                 vsi->info.outer_vlan_flags = ctxt->info.outer_vlan_flags;
609
610         kfree(ctxt);
611         return err;
612 }
613
614 /**
615  * __ice_vsi_set_outer_port_vlan - set the outer port VLAN and related settings
616  * @vsi: VSI to configure
617  * @vlan_info: packed u16 that contains the VLAN prio and ID
618  * @tpid: TPID of the port VLAN
619  *
620  * Set the port VLAN prio, ID, and TPID.
621  *
622  * Enable VLAN pruning so the VSI doesn't receive any traffic that doesn't match
623  * a VLAN prune rule. The caller should take care to add a VLAN prune rule that
624  * matches the port VLAN ID and TPID.
625  *
626  * Tell hardware to strip outer VLAN tagged packets on receive and don't put
627  * them in the receive descriptor. VSI(s) in port VLANs should not be aware of
628  * the port VLAN ID or TPID they are assigned to.
629  *
630  * Tell hardware to prevent outer VLAN tag insertion on transmit and only allow
631  * untagged outer packets from the transmit descriptor.
632  *
633  * Also, tell the hardware to insert the port VLAN on transmit.
634  */
635 static int
636 __ice_vsi_set_outer_port_vlan(struct ice_vsi *vsi, u16 vlan_info, u16 tpid)
637 {
638         struct ice_hw *hw = &vsi->back->hw;
639         struct ice_vsi_ctx *ctxt;
640         u8 tag_type;
641         int err;
642
643         if (tpid_to_vsi_outer_vlan_type(tpid, &tag_type))
644                 return -EINVAL;
645
646         ctxt = kzalloc(sizeof(*ctxt), GFP_KERNEL);
647         if (!ctxt)
648                 return -ENOMEM;
649
650         ctxt->info = vsi->info;
651
652         ctxt->info.sw_flags2 |= ICE_AQ_VSI_SW_FLAG_RX_VLAN_PRUNE_ENA;
653
654         ctxt->info.port_based_outer_vlan = cpu_to_le16(vlan_info);
655         ctxt->info.outer_vlan_flags =
656                 (ICE_AQ_VSI_OUTER_VLAN_EMODE_SHOW <<
657                  ICE_AQ_VSI_OUTER_VLAN_EMODE_S) |
658                 ((tag_type << ICE_AQ_VSI_OUTER_TAG_TYPE_S) &
659                  ICE_AQ_VSI_OUTER_TAG_TYPE_M) |
660                 ICE_AQ_VSI_OUTER_VLAN_BLOCK_TX_DESC |
661                 (ICE_AQ_VSI_OUTER_VLAN_TX_MODE_ACCEPTUNTAGGED <<
662                  ICE_AQ_VSI_OUTER_VLAN_TX_MODE_S) |
663                 ICE_AQ_VSI_OUTER_VLAN_PORT_BASED_INSERT;
664
665         ctxt->info.valid_sections =
666                 cpu_to_le16(ICE_AQ_VSI_PROP_OUTER_TAG_VALID |
667                             ICE_AQ_VSI_PROP_SW_VALID);
668
669         err = ice_update_vsi(hw, vsi->idx, ctxt, NULL);
670         if (err) {
671                 dev_err(ice_pf_to_dev(vsi->back), "update VSI for setting outer port based VLAN failed, err %d aq_err %s\n",
672                         err, ice_aq_str(hw->adminq.sq_last_status));
673         } else {
674                 vsi->info.port_based_outer_vlan = ctxt->info.port_based_outer_vlan;
675                 vsi->info.outer_vlan_flags = ctxt->info.outer_vlan_flags;
676                 vsi->info.sw_flags2 = ctxt->info.sw_flags2;
677         }
678
679         kfree(ctxt);
680         return err;
681 }
682
683 /**
684  * ice_vsi_set_outer_port_vlan - public version of __ice_vsi_set_outer_port_vlan
685  * @vsi: VSI to configure
686  * @vlan: ice_vlan structure used to set the port VLAN
687  *
688  * Set the outer port VLAN via VSI context. This function should only be
689  * used if DVM is supported. Also, this function should never be called directly
690  * as it should be part of ice_vsi_vlan_ops if it's needed.
691  *
692  * This function does not support clearing the port VLAN as there is currently
693  * no use case for this.
694  *
695  * Use the ice_vlan structure passed in to set this VSI in a port VLAN.
696  */
697 int ice_vsi_set_outer_port_vlan(struct ice_vsi *vsi, struct ice_vlan *vlan)
698 {
699         u16 port_vlan_info;
700
701         if (vlan->prio > (VLAN_PRIO_MASK >> VLAN_PRIO_SHIFT))
702                 return -EINVAL;
703
704         port_vlan_info = vlan->vid | (vlan->prio << VLAN_PRIO_SHIFT);
705
706         return __ice_vsi_set_outer_port_vlan(vsi, port_vlan_info, vlan->tpid);
707 }
This page took 0.069054 seconds and 4 git commands to generate.