net/bnxt: fix VLAN strip
HWRM_VNIC_CFG command to configure VNIC dynamically with
traffic running is not working. Driver has to free and
recreate the VNIC and then reconfigure the VNIC filters.
Fixes: 7fe5668d2e
("net/bnxt: support VLAN filter and strip")
Cc: stable@dpdk.org
Signed-off-by: Kalesh AP <kalesh-anakkur.purayil@broadcom.com>
Reviewed-by: Ajit Khaparde <ajit.khaparde@broadcom.com>
Reviewed-by: Somnath Kotur <somnath.kotur@broadcom.com>
This commit is contained in:
parent
39b88344e3
commit
cfadfee41e
@ -132,6 +132,7 @@ static int bnxt_dev_uninit(struct rte_eth_dev *eth_dev);
|
|||||||
static int bnxt_init_resources(struct bnxt *bp, bool reconfig_dev);
|
static int bnxt_init_resources(struct bnxt *bp, bool reconfig_dev);
|
||||||
static int bnxt_uninit_resources(struct bnxt *bp, bool reconfig_dev);
|
static int bnxt_uninit_resources(struct bnxt *bp, bool reconfig_dev);
|
||||||
static void bnxt_cancel_fw_health_check(struct bnxt *bp);
|
static void bnxt_cancel_fw_health_check(struct bnxt *bp);
|
||||||
|
static int bnxt_restore_vlan_filters(struct bnxt *bp);
|
||||||
|
|
||||||
int is_bnxt_in_error(struct bnxt *bp)
|
int is_bnxt_in_error(struct bnxt *bp)
|
||||||
{
|
{
|
||||||
@ -228,14 +229,97 @@ static int bnxt_alloc_mem(struct bnxt *bp, bool reconfig)
|
|||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int bnxt_setup_one_vnic(struct bnxt *bp, uint16_t vnic_id)
|
||||||
|
{
|
||||||
|
struct rte_eth_conf *dev_conf = &bp->eth_dev->data->dev_conf;
|
||||||
|
struct bnxt_vnic_info *vnic = &bp->vnic_info[vnic_id];
|
||||||
|
uint64_t rx_offloads = dev_conf->rxmode.offloads;
|
||||||
|
struct bnxt_rx_queue *rxq;
|
||||||
|
unsigned int j;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = bnxt_vnic_grp_alloc(bp, vnic);
|
||||||
|
if (rc)
|
||||||
|
goto err_out;
|
||||||
|
|
||||||
|
PMD_DRV_LOG(DEBUG, "vnic[%d] = %p vnic->fw_grp_ids = %p\n",
|
||||||
|
vnic_id, vnic, vnic->fw_grp_ids);
|
||||||
|
|
||||||
|
rc = bnxt_hwrm_vnic_alloc(bp, vnic);
|
||||||
|
if (rc)
|
||||||
|
goto err_out;
|
||||||
|
|
||||||
|
/* Alloc RSS context only if RSS mode is enabled */
|
||||||
|
if (dev_conf->rxmode.mq_mode & ETH_MQ_RX_RSS) {
|
||||||
|
int j, nr_ctxs = bnxt_rss_ctxts(bp);
|
||||||
|
|
||||||
|
rc = 0;
|
||||||
|
for (j = 0; j < nr_ctxs; j++) {
|
||||||
|
rc = bnxt_hwrm_vnic_ctx_alloc(bp, vnic, j);
|
||||||
|
if (rc)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (rc) {
|
||||||
|
PMD_DRV_LOG(ERR,
|
||||||
|
"HWRM vnic %d ctx %d alloc failure rc: %x\n",
|
||||||
|
vnic_id, j, rc);
|
||||||
|
goto err_out;
|
||||||
|
}
|
||||||
|
vnic->num_lb_ctxts = nr_ctxs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Firmware sets pf pair in default vnic cfg. If the VLAN strip
|
||||||
|
* setting is not available at this time, it will not be
|
||||||
|
* configured correctly in the CFA.
|
||||||
|
*/
|
||||||
|
if (rx_offloads & DEV_RX_OFFLOAD_VLAN_STRIP)
|
||||||
|
vnic->vlan_strip = true;
|
||||||
|
else
|
||||||
|
vnic->vlan_strip = false;
|
||||||
|
|
||||||
|
rc = bnxt_hwrm_vnic_cfg(bp, vnic);
|
||||||
|
if (rc)
|
||||||
|
goto err_out;
|
||||||
|
|
||||||
|
rc = bnxt_set_hwrm_vnic_filters(bp, vnic);
|
||||||
|
if (rc)
|
||||||
|
goto err_out;
|
||||||
|
|
||||||
|
for (j = 0; j < bp->rx_num_qs_per_vnic; j++) {
|
||||||
|
rxq = bp->eth_dev->data->rx_queues[j];
|
||||||
|
|
||||||
|
PMD_DRV_LOG(DEBUG,
|
||||||
|
"rxq[%d]->vnic=%p vnic->fw_grp_ids=%p\n",
|
||||||
|
j, rxq->vnic, rxq->vnic->fw_grp_ids);
|
||||||
|
|
||||||
|
if (BNXT_HAS_RING_GRPS(bp) && rxq->rx_deferred_start)
|
||||||
|
rxq->vnic->fw_grp_ids[j] = INVALID_HW_RING_ID;
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = bnxt_vnic_rss_configure(bp, vnic);
|
||||||
|
if (rc)
|
||||||
|
goto err_out;
|
||||||
|
|
||||||
|
bnxt_hwrm_vnic_plcmode_cfg(bp, vnic);
|
||||||
|
|
||||||
|
if (rx_offloads & DEV_RX_OFFLOAD_TCP_LRO)
|
||||||
|
bnxt_hwrm_vnic_tpa_cfg(bp, vnic, 1);
|
||||||
|
else
|
||||||
|
bnxt_hwrm_vnic_tpa_cfg(bp, vnic, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
err_out:
|
||||||
|
PMD_DRV_LOG(ERR, "HWRM vnic %d cfg failure rc: %x\n",
|
||||||
|
vnic_id, rc);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
static int bnxt_init_chip(struct bnxt *bp)
|
static int bnxt_init_chip(struct bnxt *bp)
|
||||||
{
|
{
|
||||||
struct bnxt_rx_queue *rxq;
|
|
||||||
struct rte_eth_link new;
|
struct rte_eth_link new;
|
||||||
struct rte_pci_device *pci_dev = RTE_ETH_DEV_TO_PCI(bp->eth_dev);
|
struct rte_pci_device *pci_dev = RTE_ETH_DEV_TO_PCI(bp->eth_dev);
|
||||||
struct rte_eth_conf *dev_conf = &bp->eth_dev->data->dev_conf;
|
|
||||||
struct rte_intr_handle *intr_handle = &pci_dev->intr_handle;
|
struct rte_intr_handle *intr_handle = &pci_dev->intr_handle;
|
||||||
uint64_t rx_offloads = dev_conf->rxmode.offloads;
|
|
||||||
uint32_t intr_vector = 0;
|
uint32_t intr_vector = 0;
|
||||||
uint32_t queue_id, base = BNXT_MISC_VEC_ID;
|
uint32_t queue_id, base = BNXT_MISC_VEC_ID;
|
||||||
uint32_t vec = BNXT_MISC_VEC_ID;
|
uint32_t vec = BNXT_MISC_VEC_ID;
|
||||||
@ -303,93 +387,11 @@ static int bnxt_init_chip(struct bnxt *bp)
|
|||||||
|
|
||||||
/* VNIC configuration */
|
/* VNIC configuration */
|
||||||
for (i = 0; i < bp->nr_vnics; i++) {
|
for (i = 0; i < bp->nr_vnics; i++) {
|
||||||
struct rte_eth_conf *dev_conf = &bp->eth_dev->data->dev_conf;
|
rc = bnxt_setup_one_vnic(bp, i);
|
||||||
struct bnxt_vnic_info *vnic = &bp->vnic_info[i];
|
|
||||||
|
|
||||||
rc = bnxt_vnic_grp_alloc(bp, vnic);
|
|
||||||
if (rc)
|
if (rc)
|
||||||
goto err_out;
|
goto err_out;
|
||||||
|
|
||||||
PMD_DRV_LOG(DEBUG, "vnic[%d] = %p vnic->fw_grp_ids = %p\n",
|
|
||||||
i, vnic, vnic->fw_grp_ids);
|
|
||||||
|
|
||||||
rc = bnxt_hwrm_vnic_alloc(bp, vnic);
|
|
||||||
if (rc) {
|
|
||||||
PMD_DRV_LOG(ERR, "HWRM vnic %d alloc failure rc: %x\n",
|
|
||||||
i, rc);
|
|
||||||
goto err_out;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Alloc RSS context only if RSS mode is enabled */
|
|
||||||
if (dev_conf->rxmode.mq_mode & ETH_MQ_RX_RSS) {
|
|
||||||
int j, nr_ctxs = bnxt_rss_ctxts(bp);
|
|
||||||
|
|
||||||
rc = 0;
|
|
||||||
for (j = 0; j < nr_ctxs; j++) {
|
|
||||||
rc = bnxt_hwrm_vnic_ctx_alloc(bp, vnic, j);
|
|
||||||
if (rc)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (rc) {
|
|
||||||
PMD_DRV_LOG(ERR,
|
|
||||||
"HWRM vnic %d ctx %d alloc failure rc: %x\n",
|
|
||||||
i, j, rc);
|
|
||||||
goto err_out;
|
|
||||||
}
|
|
||||||
vnic->num_lb_ctxts = nr_ctxs;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Firmware sets pf pair in default vnic cfg. If the VLAN strip
|
|
||||||
* setting is not available at this time, it will not be
|
|
||||||
* configured correctly in the CFA.
|
|
||||||
*/
|
|
||||||
if (rx_offloads & DEV_RX_OFFLOAD_VLAN_STRIP)
|
|
||||||
vnic->vlan_strip = true;
|
|
||||||
else
|
|
||||||
vnic->vlan_strip = false;
|
|
||||||
|
|
||||||
rc = bnxt_hwrm_vnic_cfg(bp, vnic);
|
|
||||||
if (rc) {
|
|
||||||
PMD_DRV_LOG(ERR, "HWRM vnic %d cfg failure rc: %x\n",
|
|
||||||
i, rc);
|
|
||||||
goto err_out;
|
|
||||||
}
|
|
||||||
|
|
||||||
rc = bnxt_set_hwrm_vnic_filters(bp, vnic);
|
|
||||||
if (rc) {
|
|
||||||
PMD_DRV_LOG(ERR,
|
|
||||||
"HWRM vnic %d filter failure rc: %x\n",
|
|
||||||
i, rc);
|
|
||||||
goto err_out;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (j = 0; j < bp->rx_num_qs_per_vnic; j++) {
|
|
||||||
rxq = bp->eth_dev->data->rx_queues[j];
|
|
||||||
|
|
||||||
PMD_DRV_LOG(DEBUG,
|
|
||||||
"rxq[%d]->vnic=%p vnic->fw_grp_ids=%p\n",
|
|
||||||
j, rxq->vnic, rxq->vnic->fw_grp_ids);
|
|
||||||
|
|
||||||
if (BNXT_HAS_RING_GRPS(bp) && rxq->rx_deferred_start)
|
|
||||||
rxq->vnic->fw_grp_ids[j] = INVALID_HW_RING_ID;
|
|
||||||
}
|
|
||||||
|
|
||||||
rc = bnxt_vnic_rss_configure(bp, vnic);
|
|
||||||
if (rc) {
|
|
||||||
PMD_DRV_LOG(ERR,
|
|
||||||
"HWRM vnic set RSS failure rc: %x\n", rc);
|
|
||||||
goto err_out;
|
|
||||||
}
|
|
||||||
|
|
||||||
bnxt_hwrm_vnic_plcmode_cfg(bp, vnic);
|
|
||||||
|
|
||||||
if (bp->eth_dev->data->dev_conf.rxmode.offloads &
|
|
||||||
DEV_RX_OFFLOAD_TCP_LRO)
|
|
||||||
bnxt_hwrm_vnic_tpa_cfg(bp, vnic, 1);
|
|
||||||
else
|
|
||||||
bnxt_hwrm_vnic_tpa_cfg(bp, vnic, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = bnxt_hwrm_cfa_l2_set_rx_mask(bp, &bp->vnic_info[0], 0, NULL);
|
rc = bnxt_hwrm_cfa_l2_set_rx_mask(bp, &bp->vnic_info[0], 0, NULL);
|
||||||
if (rc) {
|
if (rc) {
|
||||||
PMD_DRV_LOG(ERR,
|
PMD_DRV_LOG(ERR,
|
||||||
@ -1892,12 +1894,69 @@ bnxt_config_vlan_hw_filter(struct bnxt *bp, uint64_t rx_offloads)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int bnxt_free_one_vnic(struct bnxt *bp, uint16_t vnic_id)
|
||||||
|
{
|
||||||
|
struct bnxt_vnic_info *vnic = &bp->vnic_info[vnic_id];
|
||||||
|
unsigned int i;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
/* Destroy vnic filters and vnic */
|
||||||
|
if (bp->eth_dev->data->dev_conf.rxmode.offloads &
|
||||||
|
DEV_RX_OFFLOAD_VLAN_FILTER) {
|
||||||
|
for (i = 0; i < RTE_ETHER_MAX_VLAN_ID; i++)
|
||||||
|
bnxt_del_vlan_filter(bp, i);
|
||||||
|
}
|
||||||
|
bnxt_del_dflt_mac_filter(bp, vnic);
|
||||||
|
|
||||||
|
rc = bnxt_hwrm_vnic_free(bp, vnic);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
rte_free(vnic->fw_grp_ids);
|
||||||
|
vnic->fw_grp_ids = NULL;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
bnxt_config_vlan_hw_stripping(struct bnxt *bp, uint64_t rx_offloads)
|
||||||
|
{
|
||||||
|
struct bnxt_vnic_info *vnic = BNXT_GET_DEFAULT_VNIC(bp);
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
/* Destroy, recreate and reconfigure the default vnic */
|
||||||
|
rc = bnxt_free_one_vnic(bp, 0);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
/* default vnic 0 */
|
||||||
|
rc = bnxt_setup_one_vnic(bp, 0);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
if (bp->eth_dev->data->dev_conf.rxmode.offloads &
|
||||||
|
DEV_RX_OFFLOAD_VLAN_FILTER) {
|
||||||
|
rc = bnxt_add_vlan_filter(bp, 0);
|
||||||
|
bnxt_restore_vlan_filters(bp);
|
||||||
|
} else {
|
||||||
|
rc = bnxt_add_mac_filter(bp, vnic, NULL, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = bnxt_hwrm_cfa_l2_set_rx_mask(bp, vnic, 0, NULL);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
PMD_DRV_LOG(DEBUG, "VLAN Strip Offload: %d\n",
|
||||||
|
!!(rx_offloads & DEV_RX_OFFLOAD_VLAN_STRIP));
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
bnxt_vlan_offload_set_op(struct rte_eth_dev *dev, int mask)
|
bnxt_vlan_offload_set_op(struct rte_eth_dev *dev, int mask)
|
||||||
{
|
{
|
||||||
uint64_t rx_offloads = dev->data->dev_conf.rxmode.offloads;
|
uint64_t rx_offloads = dev->data->dev_conf.rxmode.offloads;
|
||||||
struct bnxt *bp = dev->data->dev_private;
|
struct bnxt *bp = dev->data->dev_private;
|
||||||
unsigned int i;
|
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
rc = is_bnxt_in_error(bp);
|
rc = is_bnxt_in_error(bp);
|
||||||
@ -1913,16 +1972,9 @@ bnxt_vlan_offload_set_op(struct rte_eth_dev *dev, int mask)
|
|||||||
|
|
||||||
if (mask & ETH_VLAN_STRIP_MASK) {
|
if (mask & ETH_VLAN_STRIP_MASK) {
|
||||||
/* Enable or disable VLAN stripping */
|
/* Enable or disable VLAN stripping */
|
||||||
for (i = 0; i < bp->nr_vnics; i++) {
|
rc = bnxt_config_vlan_hw_stripping(bp, rx_offloads);
|
||||||
struct bnxt_vnic_info *vnic = &bp->vnic_info[i];
|
if (rc)
|
||||||
if (rx_offloads & DEV_RX_OFFLOAD_VLAN_STRIP)
|
return rc;
|
||||||
vnic->vlan_strip = true;
|
|
||||||
else
|
|
||||||
vnic->vlan_strip = false;
|
|
||||||
bnxt_hwrm_vnic_cfg(bp, vnic);
|
|
||||||
}
|
|
||||||
PMD_DRV_LOG(DEBUG, "VLAN Strip Offload: %d\n",
|
|
||||||
!!(rx_offloads & DEV_RX_OFFLOAD_VLAN_STRIP));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask & ETH_VLAN_EXTEND_MASK) {
|
if (mask & ETH_VLAN_EXTEND_MASK) {
|
||||||
|
Loading…
Reference in New Issue
Block a user