2014-11-25 22:56:43 +05:30
|
|
|
/*
|
|
|
|
* Copyright 2008-2014 Cisco Systems, Inc. All rights reserved.
|
|
|
|
* Copyright 2007 Nuova Systems, Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2014, Cisco Systems, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
*
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in
|
|
|
|
* the documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
* COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include <rte_dev.h>
|
|
|
|
#include <rte_pci.h>
|
|
|
|
#include <rte_ethdev.h>
|
|
|
|
#include <rte_string_fns.h>
|
|
|
|
|
|
|
|
#include "vnic_intr.h"
|
|
|
|
#include "vnic_cq.h"
|
|
|
|
#include "vnic_wq.h"
|
|
|
|
#include "vnic_rq.h"
|
|
|
|
#include "vnic_enet.h"
|
|
|
|
#include "enic.h"
|
|
|
|
|
2015-04-07 19:17:06 +02:00
|
|
|
#ifdef RTE_LIBRTE_ENIC_DEBUG
|
2014-11-25 22:56:43 +05:30
|
|
|
#define ENICPMD_FUNC_TRACE() \
|
|
|
|
RTE_LOG(DEBUG, PMD, "ENICPMD trace: %s\n", __func__)
|
2015-04-07 19:17:06 +02:00
|
|
|
#else
|
|
|
|
#define ENICPMD_FUNC_TRACE() (void)0
|
|
|
|
#endif
|
2014-11-25 22:56:43 +05:30
|
|
|
|
|
|
|
/*
|
|
|
|
* The set of PCI devices this driver supports
|
|
|
|
*/
|
2016-07-11 16:40:42 +02:00
|
|
|
#define CISCO_PCI_VENDOR_ID 0x1137
|
2015-04-16 16:23:39 -07:00
|
|
|
static const struct rte_pci_id pci_id_enic_map[] = {
|
2016-07-11 16:40:42 +02:00
|
|
|
{ RTE_PCI_DEVICE(CISCO_PCI_VENDOR_ID, PCI_DEVICE_ID_CISCO_VIC_ENET) },
|
|
|
|
{ RTE_PCI_DEVICE(CISCO_PCI_VENDOR_ID, PCI_DEVICE_ID_CISCO_VIC_ENET_VF) },
|
|
|
|
{.vendor_id = 0, /* sentinel */},
|
2014-11-25 22:56:43 +05:30
|
|
|
};
|
|
|
|
|
2015-04-09 14:59:32 +05:30
|
|
|
static int
|
|
|
|
enicpmd_fdir_ctrl_func(struct rte_eth_dev *eth_dev,
|
|
|
|
enum rte_filter_op filter_op, void *arg)
|
2014-11-25 22:56:43 +05:30
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
2015-04-09 14:59:32 +05:30
|
|
|
int ret = 0;
|
2014-11-25 22:56:43 +05:30
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
2015-04-09 14:59:32 +05:30
|
|
|
if (filter_op == RTE_ETH_FILTER_NOP)
|
|
|
|
return 0;
|
2014-11-25 22:56:43 +05:30
|
|
|
|
2015-04-09 14:59:32 +05:30
|
|
|
if (arg == NULL && filter_op != RTE_ETH_FILTER_FLUSH)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
switch (filter_op) {
|
|
|
|
case RTE_ETH_FILTER_ADD:
|
|
|
|
case RTE_ETH_FILTER_UPDATE:
|
|
|
|
ret = enic_fdir_add_fltr(enic,
|
|
|
|
(struct rte_eth_fdir_filter *)arg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RTE_ETH_FILTER_DELETE:
|
|
|
|
ret = enic_fdir_del_fltr(enic,
|
|
|
|
(struct rte_eth_fdir_filter *)arg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RTE_ETH_FILTER_STATS:
|
|
|
|
enic_fdir_stats_get(enic, (struct rte_eth_fdir_stats *)arg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RTE_ETH_FILTER_FLUSH:
|
|
|
|
case RTE_ETH_FILTER_INFO:
|
|
|
|
dev_warning(enic, "unsupported operation %u", filter_op);
|
|
|
|
ret = -ENOTSUP;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
dev_err(enic, "unknown operation %u", filter_op);
|
|
|
|
ret = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return ret;
|
2014-11-25 22:56:43 +05:30
|
|
|
}
|
|
|
|
|
2015-04-09 14:59:32 +05:30
|
|
|
static int
|
|
|
|
enicpmd_dev_filter_ctrl(struct rte_eth_dev *dev,
|
|
|
|
enum rte_filter_type filter_type,
|
|
|
|
enum rte_filter_op filter_op,
|
|
|
|
void *arg)
|
2014-11-25 22:56:43 +05:30
|
|
|
{
|
2015-04-09 14:59:32 +05:30
|
|
|
int ret = -EINVAL;
|
2014-11-25 22:56:43 +05:30
|
|
|
|
2015-04-09 14:59:32 +05:30
|
|
|
if (RTE_ETH_FILTER_FDIR == filter_type)
|
|
|
|
ret = enicpmd_fdir_ctrl_func(dev, filter_op, arg);
|
|
|
|
else
|
|
|
|
dev_warning(enic, "Filter type (%d) not supported",
|
|
|
|
filter_type);
|
|
|
|
|
|
|
|
return ret;
|
2014-11-25 22:56:43 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_tx_queue_release(void *txq)
|
|
|
|
{
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_free_wq(txq);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_setup_intr(struct enic *enic)
|
|
|
|
{
|
|
|
|
int ret;
|
2014-12-02 14:38:31 +01:00
|
|
|
unsigned int index;
|
2014-11-25 22:56:43 +05:30
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
|
|
|
|
/* Are we done with the init of all the queues? */
|
|
|
|
for (index = 0; index < enic->cq_count; index++) {
|
|
|
|
if (!enic->cq[index].ctrl)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (enic->cq_count != index)
|
|
|
|
return 0;
|
2016-07-10 14:15:30 -07:00
|
|
|
for (index = 0; index < enic->wq_count; index++) {
|
|
|
|
if (!enic->wq[index].ctrl)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (enic->wq_count != index)
|
|
|
|
return 0;
|
|
|
|
/* check start of packet (SOP) RQs only in case scatter is disabled. */
|
|
|
|
for (index = 0; index < enic->rq_count; index++) {
|
|
|
|
if (!enic->rq[enic_sop_rq(index)].ctrl)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (enic->rq_count != index)
|
|
|
|
return 0;
|
2014-11-25 22:56:43 +05:30
|
|
|
|
|
|
|
ret = enic_alloc_intr_resources(enic);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(enic, "alloc intr failed\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
enic_init_vnic_resources(enic);
|
|
|
|
|
|
|
|
ret = enic_setup_finish(enic);
|
|
|
|
if (ret)
|
|
|
|
dev_err(enic, "setup could not be finished\n");
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_tx_queue_setup(struct rte_eth_dev *eth_dev,
|
|
|
|
uint16_t queue_idx,
|
|
|
|
uint16_t nb_desc,
|
|
|
|
unsigned int socket_id,
|
2014-12-02 14:38:31 +01:00
|
|
|
__rte_unused const struct rte_eth_txconf *tx_conf)
|
2014-11-25 22:56:43 +05:30
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
2016-03-17 15:49:58 -07:00
|
|
|
if (queue_idx >= ENIC_WQ_MAX) {
|
|
|
|
dev_err(enic,
|
|
|
|
"Max number of TX queues exceeded. Max is %d\n",
|
|
|
|
ENIC_WQ_MAX);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2014-11-25 22:56:43 +05:30
|
|
|
eth_dev->data->tx_queues[queue_idx] = (void *)&enic->wq[queue_idx];
|
|
|
|
|
|
|
|
ret = enic_alloc_wq(enic, queue_idx, socket_id, nb_desc);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(enic, "error in allocating wq\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return enicpmd_dev_setup_intr(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_tx_queue_start(struct rte_eth_dev *eth_dev,
|
|
|
|
uint16_t queue_idx)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
|
|
|
|
enic_start_wq(enic, queue_idx);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_tx_queue_stop(struct rte_eth_dev *eth_dev,
|
|
|
|
uint16_t queue_idx)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
|
|
|
|
ret = enic_stop_wq(enic, queue_idx);
|
|
|
|
if (ret)
|
|
|
|
dev_err(enic, "error in stopping wq %d\n", queue_idx);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_rx_queue_start(struct rte_eth_dev *eth_dev,
|
|
|
|
uint16_t queue_idx)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
|
|
|
|
enic_start_rq(enic, queue_idx);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_rx_queue_stop(struct rte_eth_dev *eth_dev,
|
|
|
|
uint16_t queue_idx)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
|
|
|
|
ret = enic_stop_rq(enic, queue_idx);
|
|
|
|
if (ret)
|
|
|
|
dev_err(enic, "error in stopping rq %d\n", queue_idx);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_rx_queue_release(void *rxq)
|
|
|
|
{
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_free_rq(rxq);
|
|
|
|
}
|
|
|
|
|
2016-09-19 11:50:29 -07:00
|
|
|
static uint32_t enicpmd_dev_rx_queue_count(struct rte_eth_dev *dev,
|
|
|
|
uint16_t rx_queue_id)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(dev);
|
|
|
|
uint32_t queue_count = 0;
|
|
|
|
struct vnic_cq *cq;
|
|
|
|
uint32_t cq_tail;
|
|
|
|
uint16_t cq_idx;
|
|
|
|
int rq_num;
|
|
|
|
|
|
|
|
if (rx_queue_id >= dev->data->nb_rx_queues) {
|
|
|
|
dev_err(enic, "Invalid RX queue id=%d", rx_queue_id);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
rq_num = enic_sop_rq(rx_queue_id);
|
|
|
|
cq = &enic->cq[enic_cq_rq(enic, rq_num)];
|
|
|
|
cq_idx = cq->to_clean;
|
|
|
|
|
|
|
|
cq_tail = ioread32(&cq->ctrl->cq_tail);
|
|
|
|
|
|
|
|
if (cq_tail < cq_idx)
|
|
|
|
cq_tail += cq->ring.desc_count;
|
|
|
|
|
|
|
|
queue_count = cq_tail - cq_idx;
|
|
|
|
|
|
|
|
return queue_count;
|
|
|
|
}
|
|
|
|
|
2014-11-25 22:56:43 +05:30
|
|
|
static int enicpmd_dev_rx_queue_setup(struct rte_eth_dev *eth_dev,
|
|
|
|
uint16_t queue_idx,
|
|
|
|
uint16_t nb_desc,
|
|
|
|
unsigned int socket_id,
|
2016-03-04 13:09:00 -08:00
|
|
|
const struct rte_eth_rxconf *rx_conf,
|
2014-11-25 22:56:43 +05:30
|
|
|
struct rte_mempool *mp)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
2016-06-16 12:19:05 -07:00
|
|
|
/* With Rx scatter support, two RQs are now used on VIC per RQ used
|
|
|
|
* by the application.
|
|
|
|
*/
|
|
|
|
if (queue_idx * 2 >= ENIC_RQ_MAX) {
|
2016-03-17 15:49:58 -07:00
|
|
|
dev_err(enic,
|
2016-06-16 12:19:05 -07:00
|
|
|
"Max number of RX queues exceeded. Max is %d. This PMD uses 2 RQs on VIC per RQ used by DPDK.\n",
|
2016-03-17 15:49:58 -07:00
|
|
|
ENIC_RQ_MAX);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2016-06-16 12:19:05 -07:00
|
|
|
eth_dev->data->rx_queues[queue_idx] =
|
|
|
|
(void *)&enic->rq[enic_sop_rq(queue_idx)];
|
2014-11-25 22:56:43 +05:30
|
|
|
|
|
|
|
ret = enic_alloc_rq(enic, queue_idx, socket_id, mp, nb_desc);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(enic, "error in allocating rq\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-03-04 13:09:00 -08:00
|
|
|
enic->rq[queue_idx].rx_free_thresh = rx_conf->rx_free_thresh;
|
|
|
|
dev_debug(enic, "Set queue_id:%u free thresh:%u\n", queue_idx,
|
|
|
|
enic->rq[queue_idx].rx_free_thresh);
|
|
|
|
|
2014-11-25 22:56:43 +05:30
|
|
|
return enicpmd_dev_setup_intr(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_vlan_filter_set(struct rte_eth_dev *eth_dev,
|
|
|
|
uint16_t vlan_id, int on)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
2015-10-30 17:13:23 +01:00
|
|
|
int err;
|
2014-11-25 22:56:43 +05:30
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
if (on)
|
2015-10-30 17:13:23 +01:00
|
|
|
err = enic_add_vlan(enic, vlan_id);
|
2014-11-25 22:56:43 +05:30
|
|
|
else
|
2015-10-30 17:13:23 +01:00
|
|
|
err = enic_del_vlan(enic, vlan_id);
|
|
|
|
return err;
|
2014-11-25 22:56:43 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_vlan_offload_set(struct rte_eth_dev *eth_dev, int mask)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
|
|
|
|
if (mask & ETH_VLAN_STRIP_MASK) {
|
|
|
|
if (eth_dev->data->dev_conf.rxmode.hw_vlan_strip)
|
|
|
|
enic->ig_vlan_strip_en = 1;
|
|
|
|
else
|
|
|
|
enic->ig_vlan_strip_en = 0;
|
|
|
|
}
|
|
|
|
enic_set_rss_nic_cfg(enic);
|
|
|
|
|
|
|
|
|
|
|
|
if (mask & ETH_VLAN_FILTER_MASK) {
|
|
|
|
dev_warning(enic,
|
|
|
|
"Configuration of VLAN filter is not supported\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & ETH_VLAN_EXTEND_MASK) {
|
|
|
|
dev_warning(enic,
|
|
|
|
"Configuration of extended VLAN is not supported\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_configure(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
ret = enic_set_vnic_res(enic);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(enic, "Set vNIC resource num failed, aborting\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eth_dev->data->dev_conf.rxmode.split_hdr_size &&
|
|
|
|
eth_dev->data->dev_conf.rxmode.header_split) {
|
|
|
|
/* Enable header-data-split */
|
|
|
|
enic_set_hdr_split_size(enic,
|
|
|
|
eth_dev->data->dev_conf.rxmode.split_hdr_size);
|
|
|
|
}
|
|
|
|
|
2016-07-19 15:43:45 -07:00
|
|
|
enicpmd_vlan_offload_set(eth_dev, ETH_VLAN_STRIP_MASK);
|
2014-11-25 22:56:43 +05:30
|
|
|
enic->hw_ip_checksum = eth_dev->data->dev_conf.rxmode.hw_ip_checksum;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Start the device.
|
|
|
|
* It returns 0 on success.
|
|
|
|
*/
|
|
|
|
static int enicpmd_dev_start(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
return enic_enable(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Stop device: disable rx and tx functions to allow for reconfiguring.
|
|
|
|
*/
|
|
|
|
static void enicpmd_dev_stop(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct rte_eth_link link;
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_disable(enic);
|
|
|
|
memset(&link, 0, sizeof(link));
|
|
|
|
rte_atomic64_cmpset((uint64_t *)ð_dev->data->dev_link,
|
|
|
|
*(uint64_t *)ð_dev->data->dev_link,
|
|
|
|
*(uint64_t *)&link);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Stop device.
|
|
|
|
*/
|
|
|
|
static void enicpmd_dev_close(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_remove(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enicpmd_dev_link_update(struct rte_eth_dev *eth_dev,
|
2014-12-02 14:38:31 +01:00
|
|
|
__rte_unused int wait_to_complete)
|
2014-11-25 22:56:43 +05:30
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
int ret;
|
|
|
|
int link_status = 0;
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
link_status = enic_get_link_status(enic);
|
|
|
|
ret = (link_status == enic->link_status);
|
|
|
|
enic->link_status = link_status;
|
|
|
|
eth_dev->data->dev_link.link_status = link_status;
|
|
|
|
eth_dev->data->dev_link.link_duplex = ETH_LINK_FULL_DUPLEX;
|
|
|
|
eth_dev->data->dev_link.link_speed = vnic_dev_port_speed(enic->vdev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_stats_get(struct rte_eth_dev *eth_dev,
|
|
|
|
struct rte_eth_stats *stats)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_dev_stats_get(enic, stats);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_stats_reset(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_dev_stats_clear(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_info_get(struct rte_eth_dev *eth_dev,
|
|
|
|
struct rte_eth_dev_info *device_info)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
net/enic: fix resource check failures when bonding devices
The enic PMD was using the same variables in the enic structure to
track two different things. Initially rq_count, wq_count, cq_count,
and intr_count were set to the values obtained from the VIC adapters
as the maximum resources allocated on the VIC, then in
enic_set_vnic_res(), they were set to the counts of resources actually
used, discarding the initial values. The checks in enic_set_vnic_res()
were technically incorrect if it is called more than once on a port,
which happens when using bonding, but were harmless in practice as the
checks couldn't fail on the second call.
The enic rx-scatter patch misunderstood the subtleties of
enic_set_vnic_res(), and naively added a multiply by two to the
rq_count check. This resulted in the rq_count check failing when
enic_set_vnic_res() was called a second time, ie when using bonding.
This patch adds new variables to the enic structure to track the
maximum resources the VIC is configured to provide so that the
information isn't later lost and calls to enic_set_vnic_res() do
the expected thing.
Fixes: 856d7ba7ed22 ("net/enic: support scattered Rx")
Signed-off-by: Nelson Escobar <neescoba@cisco.com>
2016-07-06 16:21:59 -07:00
|
|
|
/* Scattered Rx uses two receive queues per rx queue exposed to dpdk */
|
|
|
|
device_info->max_rx_queues = enic->conf_rq_count / 2;
|
|
|
|
device_info->max_tx_queues = enic->conf_wq_count;
|
2014-11-25 22:56:43 +05:30
|
|
|
device_info->min_rx_bufsize = ENIC_MIN_MTU;
|
2016-06-24 15:29:27 -07:00
|
|
|
device_info->max_rx_pktlen = enic->rte_dev->data->mtu
|
|
|
|
+ ETHER_HDR_LEN + 4;
|
2014-11-25 22:56:43 +05:30
|
|
|
device_info->max_mac_addrs = 1;
|
|
|
|
device_info->rx_offload_capa =
|
|
|
|
DEV_RX_OFFLOAD_VLAN_STRIP |
|
|
|
|
DEV_RX_OFFLOAD_IPV4_CKSUM |
|
|
|
|
DEV_RX_OFFLOAD_UDP_CKSUM |
|
|
|
|
DEV_RX_OFFLOAD_TCP_CKSUM;
|
|
|
|
device_info->tx_offload_capa =
|
|
|
|
DEV_TX_OFFLOAD_VLAN_INSERT |
|
|
|
|
DEV_TX_OFFLOAD_IPV4_CKSUM |
|
|
|
|
DEV_TX_OFFLOAD_UDP_CKSUM |
|
|
|
|
DEV_TX_OFFLOAD_TCP_CKSUM;
|
2016-03-04 13:09:00 -08:00
|
|
|
device_info->default_rxconf = (struct rte_eth_rxconf) {
|
|
|
|
.rx_free_thresh = ENIC_DEFAULT_RX_FREE_THRESH
|
|
|
|
};
|
2014-11-25 22:56:43 +05:30
|
|
|
}
|
|
|
|
|
2016-03-15 04:50:50 +08:00
|
|
|
static const uint32_t *enicpmd_dev_supported_ptypes_get(struct rte_eth_dev *dev)
|
|
|
|
{
|
|
|
|
static const uint32_t ptypes[] = {
|
2016-06-14 16:54:05 -07:00
|
|
|
RTE_PTYPE_L3_IPV4_EXT_UNKNOWN,
|
|
|
|
RTE_PTYPE_L3_IPV6_EXT_UNKNOWN,
|
|
|
|
RTE_PTYPE_L4_TCP,
|
|
|
|
RTE_PTYPE_L4_UDP,
|
|
|
|
RTE_PTYPE_L4_FRAG,
|
|
|
|
RTE_PTYPE_L4_NONFRAG,
|
2016-03-15 04:50:50 +08:00
|
|
|
RTE_PTYPE_UNKNOWN
|
|
|
|
};
|
|
|
|
|
|
|
|
if (dev->rx_pkt_burst == enic_recv_pkts)
|
|
|
|
return ptypes;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-11-25 22:56:43 +05:30
|
|
|
static void enicpmd_dev_promiscuous_enable(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic->promisc = 1;
|
|
|
|
enic_add_packet_filter(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_promiscuous_disable(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic->promisc = 0;
|
|
|
|
enic_add_packet_filter(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_allmulticast_enable(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic->allmulti = 1;
|
|
|
|
enic_add_packet_filter(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_dev_allmulticast_disable(struct rte_eth_dev *eth_dev)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic->allmulti = 0;
|
|
|
|
enic_add_packet_filter(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enicpmd_add_mac_addr(struct rte_eth_dev *eth_dev,
|
|
|
|
struct ether_addr *mac_addr,
|
2014-12-02 14:38:31 +01:00
|
|
|
__rte_unused uint32_t index, __rte_unused uint32_t pool)
|
2014-11-25 22:56:43 +05:30
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_set_mac_address(enic, mac_addr->addr_bytes);
|
|
|
|
}
|
|
|
|
|
2014-12-02 14:38:31 +01:00
|
|
|
static void enicpmd_remove_mac_addr(struct rte_eth_dev *eth_dev, __rte_unused uint32_t index)
|
2014-11-25 22:56:43 +05:30
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
enic_del_mac_address(enic);
|
|
|
|
}
|
|
|
|
|
2016-06-24 15:29:28 -07:00
|
|
|
static int enicpmd_mtu_set(struct rte_eth_dev *eth_dev, uint16_t mtu)
|
|
|
|
{
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
return enic_set_mtu(enic, mtu);
|
|
|
|
}
|
|
|
|
|
2015-04-07 14:21:03 -07:00
|
|
|
static const struct eth_dev_ops enicpmd_eth_dev_ops = {
|
2014-11-25 22:56:43 +05:30
|
|
|
.dev_configure = enicpmd_dev_configure,
|
|
|
|
.dev_start = enicpmd_dev_start,
|
|
|
|
.dev_stop = enicpmd_dev_stop,
|
|
|
|
.dev_set_link_up = NULL,
|
|
|
|
.dev_set_link_down = NULL,
|
|
|
|
.dev_close = enicpmd_dev_close,
|
|
|
|
.promiscuous_enable = enicpmd_dev_promiscuous_enable,
|
|
|
|
.promiscuous_disable = enicpmd_dev_promiscuous_disable,
|
|
|
|
.allmulticast_enable = enicpmd_dev_allmulticast_enable,
|
|
|
|
.allmulticast_disable = enicpmd_dev_allmulticast_disable,
|
|
|
|
.link_update = enicpmd_dev_link_update,
|
|
|
|
.stats_get = enicpmd_dev_stats_get,
|
|
|
|
.stats_reset = enicpmd_dev_stats_reset,
|
|
|
|
.queue_stats_mapping_set = NULL,
|
|
|
|
.dev_infos_get = enicpmd_dev_info_get,
|
2016-03-15 04:50:50 +08:00
|
|
|
.dev_supported_ptypes_get = enicpmd_dev_supported_ptypes_get,
|
2016-06-24 15:29:28 -07:00
|
|
|
.mtu_set = enicpmd_mtu_set,
|
2014-11-25 22:56:43 +05:30
|
|
|
.vlan_filter_set = enicpmd_vlan_filter_set,
|
|
|
|
.vlan_tpid_set = NULL,
|
|
|
|
.vlan_offload_set = enicpmd_vlan_offload_set,
|
|
|
|
.vlan_strip_queue_set = NULL,
|
|
|
|
.rx_queue_start = enicpmd_dev_rx_queue_start,
|
|
|
|
.rx_queue_stop = enicpmd_dev_rx_queue_stop,
|
|
|
|
.tx_queue_start = enicpmd_dev_tx_queue_start,
|
|
|
|
.tx_queue_stop = enicpmd_dev_tx_queue_stop,
|
|
|
|
.rx_queue_setup = enicpmd_dev_rx_queue_setup,
|
|
|
|
.rx_queue_release = enicpmd_dev_rx_queue_release,
|
2016-09-19 11:50:29 -07:00
|
|
|
.rx_queue_count = enicpmd_dev_rx_queue_count,
|
2014-11-25 22:56:43 +05:30
|
|
|
.rx_descriptor_done = NULL,
|
|
|
|
.tx_queue_setup = enicpmd_dev_tx_queue_setup,
|
|
|
|
.tx_queue_release = enicpmd_dev_tx_queue_release,
|
|
|
|
.dev_led_on = NULL,
|
|
|
|
.dev_led_off = NULL,
|
|
|
|
.flow_ctrl_get = NULL,
|
|
|
|
.flow_ctrl_set = NULL,
|
|
|
|
.priority_flow_ctrl_set = NULL,
|
|
|
|
.mac_addr_add = enicpmd_add_mac_addr,
|
|
|
|
.mac_addr_remove = enicpmd_remove_mac_addr,
|
2015-04-09 14:59:32 +05:30
|
|
|
.filter_ctrl = enicpmd_dev_filter_ctrl,
|
2014-11-25 22:56:43 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
struct enic *enicpmd_list_head = NULL;
|
|
|
|
/* Initialize the driver
|
|
|
|
* It returns 0 on success.
|
|
|
|
*/
|
2015-03-05 19:24:59 +01:00
|
|
|
static int eth_enicpmd_dev_init(struct rte_eth_dev *eth_dev)
|
2014-11-25 22:56:43 +05:30
|
|
|
{
|
|
|
|
struct rte_pci_device *pdev;
|
|
|
|
struct rte_pci_addr *addr;
|
|
|
|
struct enic *enic = pmd_priv(eth_dev);
|
|
|
|
|
|
|
|
ENICPMD_FUNC_TRACE();
|
|
|
|
|
2015-04-14 16:23:55 +02:00
|
|
|
enic->port_id = eth_dev->data->port_id;
|
2014-11-25 22:56:43 +05:30
|
|
|
enic->rte_dev = eth_dev;
|
|
|
|
eth_dev->dev_ops = &enicpmd_eth_dev_ops;
|
2016-03-04 13:09:00 -08:00
|
|
|
eth_dev->rx_pkt_burst = &enic_recv_pkts;
|
2016-06-02 17:22:49 -07:00
|
|
|
eth_dev->tx_pkt_burst = &enic_xmit_pkts;
|
2014-11-25 22:56:43 +05:30
|
|
|
|
|
|
|
pdev = eth_dev->pci_dev;
|
2015-11-03 13:01:56 +00:00
|
|
|
rte_eth_copy_pci_info(eth_dev, pdev);
|
2014-11-25 22:56:43 +05:30
|
|
|
enic->pdev = pdev;
|
|
|
|
addr = &pdev->addr;
|
|
|
|
|
|
|
|
snprintf(enic->bdf_name, ENICPMD_BDF_LENGTH, "%04x:%02x:%02x.%x",
|
|
|
|
addr->domain, addr->bus, addr->devid, addr->function);
|
|
|
|
|
|
|
|
return enic_probe(enic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct eth_driver rte_enic_pmd = {
|
2015-05-29 08:47:51 -07:00
|
|
|
.pci_drv = {
|
2014-11-25 22:56:43 +05:30
|
|
|
.id_table = pci_id_enic_map,
|
|
|
|
.drv_flags = RTE_PCI_DRV_NEED_MAPPING,
|
2016-09-20 18:11:20 +05:30
|
|
|
.probe = rte_eth_dev_pci_probe,
|
|
|
|
.remove = rte_eth_dev_pci_remove,
|
2014-11-25 22:56:43 +05:30
|
|
|
},
|
|
|
|
.eth_dev_init = eth_enicpmd_dev_init,
|
|
|
|
.dev_private_size = sizeof(struct enic),
|
|
|
|
};
|
|
|
|
|
2016-09-20 18:11:20 +05:30
|
|
|
DRIVER_REGISTER_PCI(net_enic, rte_enic_pmd.pci_drv);
|
2016-08-24 23:24:54 +01:00
|
|
|
DRIVER_REGISTER_PCI_TABLE(net_enic, pci_id_enic_map);
|