50ce3e7aec
Currently, there is a potential problem that calling the API function rte_eth_dev_set_vlan_offload to start VLAN hardware offloads which the driver does not support. If the PMD driver does not support certain VLAN hardware offloads and does not check for it, the hardware setting will not change, but the VLAN offloads in dev->data->dev_conf.rxmode.offloads will be turned on. It is supposed to check the hardware capabilities to decide whether the relative callback needs to be called just like the behavior in the API function named rte_eth_dev_configure. And it is also needed to cleanup duplicated checks which are done in some PMDs. Also, note that it is behaviour change for some PMDs which simply ignore (with error/warning log message) unsupported VLAN offloads, but now it will fail. Fixes: a4996bd89c42 ("ethdev: new Rx/Tx offloads API") Fixes: 0ebce6129bc6 ("net/dpaa2: support new ethdev offload APIs") Fixes: f9416bbafd98 ("net/enic: remove VLAN filter handler") Fixes: 4f7d9e383e5c ("fm10k: update vlan offload features") Fixes: fdba3bf15c7b ("net/hinic: add VLAN filter and offload") Fixes: b96fb2f0d22b ("net/i40e: handle QinQ strip") Fixes: d4a27a3b092a ("nfp: add basic features") Fixes: 56139e85abec ("net/octeontx: support VLAN filter offload") Fixes: ba1b3b081edf ("net/octeontx2: support VLAN offloads") Fixes: d87246a43759 ("net/qede: enable and disable VLAN filtering") Cc: stable@dpdk.org Signed-off-by: Chengchang Tang <tangchengchang@huawei.com> Signed-off-by: Wei Hu (Xavier) <xavier.huwei@huawei.com> Acked-by: Andrew Rybchenko <arybchenko@solarflare.com> Acked-by: Hyong Youb Kim <hyonkim@cisco.com> Acked-by: Sachin Saxena <sachin.saxena@nxp.com> Acked-by: Xiaoyun Wang <cloud.wangxiaoyun@huawei.com> Acked-by: Harman Kalra <hkalra@marvell.com> Acked-by: Jeff Guo <jia.guo@intel.com> Acked-by: Viacheslav Ovsiienko <viacheslavo@mellanox.com>
334 lines
7.7 KiB
C
334 lines
7.7 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright(C) 2020 Marvell International Ltd.
|
|
*/
|
|
|
|
#include <rte_malloc.h>
|
|
|
|
#include "octeontx_ethdev.h"
|
|
#include "octeontx_logs.h"
|
|
#include "octeontx_rxtx.h"
|
|
|
|
static int
|
|
octeontx_vlan_hw_filter(struct octeontx_nic *nic, uint8_t flag)
|
|
{
|
|
struct octeontx_vlan_info *vlan = &nic->vlan_info;
|
|
pki_port_vlan_filter_config_t fltr_conf;
|
|
int rc = 0;
|
|
|
|
if (vlan->filter_on == flag)
|
|
return rc;
|
|
|
|
fltr_conf.port_type = OCTTX_PORT_TYPE_NET;
|
|
fltr_conf.fltr_conf = flag;
|
|
|
|
rc = octeontx_pki_port_vlan_fltr_config(nic->port_id, &fltr_conf);
|
|
if (rc != 0) {
|
|
octeontx_log_err("Fail to configure vlan hw filter for port %d",
|
|
nic->port_id);
|
|
goto done;
|
|
}
|
|
|
|
vlan->filter_on = flag;
|
|
|
|
done:
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_vlan_offload_set(struct rte_eth_dev *dev, int mask)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
struct rte_eth_rxmode *rxmode;
|
|
int rc = 0;
|
|
|
|
rxmode = &dev->data->dev_conf.rxmode;
|
|
|
|
if (mask & ETH_VLAN_FILTER_MASK) {
|
|
if (rxmode->offloads & DEV_RX_OFFLOAD_VLAN_FILTER) {
|
|
rc = octeontx_vlan_hw_filter(nic, true);
|
|
if (rc)
|
|
goto done;
|
|
|
|
nic->rx_offloads |= DEV_RX_OFFLOAD_VLAN_FILTER;
|
|
nic->rx_offload_flags |= OCCTX_RX_VLAN_FLTR_F;
|
|
} else {
|
|
rc = octeontx_vlan_hw_filter(nic, false);
|
|
if (rc)
|
|
goto done;
|
|
|
|
nic->rx_offloads &= ~DEV_RX_OFFLOAD_VLAN_FILTER;
|
|
nic->rx_offload_flags &= ~OCCTX_RX_VLAN_FLTR_F;
|
|
}
|
|
}
|
|
|
|
done:
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_vlan_filter_set(struct rte_eth_dev *dev, uint16_t vlan_id, int on)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
struct octeontx_vlan_info *vlan = &nic->vlan_info;
|
|
pki_port_vlan_filter_entry_config_t fltr_entry;
|
|
struct vlan_entry *entry = NULL;
|
|
int entry_count = 0;
|
|
int rc = -EINVAL;
|
|
|
|
if (on) {
|
|
TAILQ_FOREACH(entry, &vlan->fltr_tbl, next)
|
|
if (entry->vlan_id == vlan_id) {
|
|
octeontx_log_dbg("Vlan Id is already set");
|
|
return 0;
|
|
}
|
|
} else {
|
|
TAILQ_FOREACH(entry, &vlan->fltr_tbl, next)
|
|
entry_count++;
|
|
|
|
if (!entry_count)
|
|
return 0;
|
|
}
|
|
|
|
fltr_entry.port_type = OCTTX_PORT_TYPE_NET;
|
|
fltr_entry.vlan_tpid = RTE_ETHER_TYPE_VLAN;
|
|
fltr_entry.vlan_id = vlan_id;
|
|
fltr_entry.entry_conf = on;
|
|
|
|
if (on) {
|
|
entry = rte_zmalloc("octeontx_nic_vlan_entry",
|
|
sizeof(struct vlan_entry), 0);
|
|
if (!entry) {
|
|
octeontx_log_err("Failed to allocate memory");
|
|
return -ENOMEM;
|
|
}
|
|
}
|
|
|
|
rc = octeontx_pki_port_vlan_fltr_entry_config(nic->port_id,
|
|
&fltr_entry);
|
|
if (rc != 0) {
|
|
octeontx_log_err("Fail to configure vlan filter entry "
|
|
"for port %d", nic->port_id);
|
|
if (entry)
|
|
rte_free(entry);
|
|
|
|
goto done;
|
|
}
|
|
|
|
if (on) {
|
|
entry->vlan_id = vlan_id;
|
|
TAILQ_INSERT_HEAD(&vlan->fltr_tbl, entry, next);
|
|
} else {
|
|
TAILQ_FOREACH(entry, &vlan->fltr_tbl, next) {
|
|
if (entry->vlan_id == vlan_id) {
|
|
TAILQ_REMOVE(&vlan->fltr_tbl, entry, next);
|
|
rte_free(entry);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
done:
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_vlan_offload_init(struct rte_eth_dev *dev)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
int rc;
|
|
|
|
TAILQ_INIT(&nic->vlan_info.fltr_tbl);
|
|
|
|
rc = octeontx_dev_vlan_offload_set(dev, ETH_VLAN_FILTER_MASK);
|
|
if (rc)
|
|
octeontx_log_err("Failed to set vlan offload rc=%d", rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_vlan_offload_fini(struct rte_eth_dev *dev)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
struct octeontx_vlan_info *vlan = &nic->vlan_info;
|
|
pki_port_vlan_filter_entry_config_t fltr_entry;
|
|
struct vlan_entry *entry;
|
|
int rc = 0;
|
|
|
|
TAILQ_FOREACH(entry, &vlan->fltr_tbl, next) {
|
|
fltr_entry.port_type = OCTTX_PORT_TYPE_NET;
|
|
fltr_entry.vlan_tpid = RTE_ETHER_TYPE_VLAN;
|
|
fltr_entry.vlan_id = entry->vlan_id;
|
|
fltr_entry.entry_conf = 0;
|
|
|
|
rc = octeontx_pki_port_vlan_fltr_entry_config(nic->port_id,
|
|
&fltr_entry);
|
|
if (rc != 0) {
|
|
octeontx_log_err("Fail to configure vlan filter entry "
|
|
"for port %d", nic->port_id);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_set_link_up(struct rte_eth_dev *eth_dev)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(eth_dev);
|
|
int rc, i;
|
|
|
|
rc = octeontx_bgx_port_set_link_state(nic->port_id, true);
|
|
if (rc)
|
|
goto done;
|
|
|
|
/* Start tx queues */
|
|
for (i = 0; i < eth_dev->data->nb_tx_queues; i++)
|
|
octeontx_dev_tx_queue_start(eth_dev, i);
|
|
|
|
done:
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_set_link_down(struct rte_eth_dev *eth_dev)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(eth_dev);
|
|
int i;
|
|
|
|
/* Stop tx queues */
|
|
for (i = 0; i < eth_dev->data->nb_tx_queues; i++)
|
|
octeontx_dev_tx_queue_stop(eth_dev, i);
|
|
|
|
return octeontx_bgx_port_set_link_state(nic->port_id, false);
|
|
}
|
|
|
|
int
|
|
octeontx_dev_flow_ctrl_get(struct rte_eth_dev *dev,
|
|
struct rte_eth_fc_conf *fc_conf)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
octeontx_mbox_bgx_port_fc_cfg_t conf;
|
|
int rc;
|
|
|
|
memset(&conf, 0, sizeof(octeontx_mbox_bgx_port_fc_cfg_t));
|
|
|
|
rc = octeontx_bgx_port_flow_ctrl_cfg(nic->port_id, &conf);
|
|
if (rc)
|
|
return rc;
|
|
|
|
if (conf.rx_pause && conf.tx_pause)
|
|
fc_conf->mode = RTE_FC_FULL;
|
|
else if (conf.rx_pause)
|
|
fc_conf->mode = RTE_FC_RX_PAUSE;
|
|
else if (conf.tx_pause)
|
|
fc_conf->mode = RTE_FC_TX_PAUSE;
|
|
else
|
|
fc_conf->mode = RTE_FC_NONE;
|
|
|
|
/* low_water & high_water values are in Bytes */
|
|
fc_conf->low_water = conf.low_water;
|
|
fc_conf->high_water = conf.high_water;
|
|
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_flow_ctrl_set(struct rte_eth_dev *dev,
|
|
struct rte_eth_fc_conf *fc_conf)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
struct octeontx_fc_info *fc = &nic->fc;
|
|
octeontx_mbox_bgx_port_fc_cfg_t conf;
|
|
uint8_t tx_pause, rx_pause;
|
|
uint16_t max_high_water;
|
|
int rc;
|
|
|
|
if (fc_conf->pause_time || fc_conf->mac_ctrl_frame_fwd ||
|
|
fc_conf->autoneg) {
|
|
octeontx_log_err("Below flowctrl parameters are not supported "
|
|
"pause_time, mac_ctrl_frame_fwd and autoneg");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (fc_conf->high_water == fc->high_water &&
|
|
fc_conf->low_water == fc->low_water &&
|
|
fc_conf->mode == fc->mode)
|
|
return 0;
|
|
|
|
max_high_water = fc->rx_fifosz - OCTEONTX_BGX_RSVD_RX_FIFOBYTES;
|
|
|
|
if (fc_conf->high_water > max_high_water ||
|
|
fc_conf->high_water < fc_conf->low_water) {
|
|
octeontx_log_err("Invalid high/low water values "
|
|
"High_water(in Bytes) must <= 0x%x ",
|
|
max_high_water);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (fc_conf->high_water % BIT(4) || fc_conf->low_water % BIT(4)) {
|
|
octeontx_log_err("High/low water value must be multiple of 16");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rx_pause = (fc_conf->mode == RTE_FC_FULL) ||
|
|
(fc_conf->mode == RTE_FC_RX_PAUSE);
|
|
tx_pause = (fc_conf->mode == RTE_FC_FULL) ||
|
|
(fc_conf->mode == RTE_FC_TX_PAUSE);
|
|
|
|
conf.high_water = fc_conf->high_water;
|
|
conf.low_water = fc_conf->low_water;
|
|
conf.fc_cfg = BGX_PORT_FC_CFG_SET;
|
|
conf.rx_pause = rx_pause;
|
|
conf.tx_pause = tx_pause;
|
|
|
|
rc = octeontx_bgx_port_flow_ctrl_cfg(nic->port_id, &conf);
|
|
if (rc)
|
|
return rc;
|
|
|
|
fc->high_water = fc_conf->high_water;
|
|
fc->low_water = fc_conf->low_water;
|
|
fc->mode = fc_conf->mode;
|
|
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_flow_ctrl_init(struct rte_eth_dev *dev)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
struct octeontx_fc_info *fc = &nic->fc;
|
|
struct rte_eth_fc_conf fc_conf;
|
|
int rc;
|
|
|
|
rc = octeontx_dev_flow_ctrl_get(dev, &fc_conf);
|
|
if (rc) {
|
|
octeontx_log_err("Failed to get flow control info");
|
|
return rc;
|
|
}
|
|
|
|
fc->def_highmark = fc_conf.high_water;
|
|
fc->def_lowmark = fc_conf.low_water;
|
|
fc->def_mode = fc_conf.mode;
|
|
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
octeontx_dev_flow_ctrl_fini(struct rte_eth_dev *dev)
|
|
{
|
|
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
|
|
struct octeontx_fc_info *fc = &nic->fc;
|
|
struct rte_eth_fc_conf fc_conf;
|
|
|
|
memset(&fc_conf, 0, sizeof(struct rte_eth_fc_conf));
|
|
|
|
/* Restore flow control parameters with default values */
|
|
fc_conf.high_water = fc->def_highmark;
|
|
fc_conf.low_water = fc->def_lowmark;
|
|
fc_conf.mode = fc->def_mode;
|
|
|
|
return octeontx_dev_flow_ctrl_set(dev, &fc_conf);
|
|
}
|