mirror of https://github.com/F-Stack/f-stack.git
491 lines
12 KiB
C
491 lines
12 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright(c) 2020 Intel Corporation
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <sys/types.h>
|
|
|
|
#include <rte_ethdev.h>
|
|
|
|
#include "ice_dcf_ethdev.h"
|
|
#include "ice_rxtx.h"
|
|
|
|
static uint16_t
|
|
ice_dcf_vf_repr_rx_burst(__rte_unused void *rxq,
|
|
__rte_unused struct rte_mbuf **rx_pkts,
|
|
__rte_unused uint16_t nb_pkts)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static uint16_t
|
|
ice_dcf_vf_repr_tx_burst(__rte_unused void *txq,
|
|
__rte_unused struct rte_mbuf **tx_pkts,
|
|
__rte_unused uint16_t nb_pkts)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_dev_configure(struct rte_eth_dev *dev)
|
|
{
|
|
ice_dcf_vf_repr_init_vlan(dev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_dev_start(struct rte_eth_dev *dev)
|
|
{
|
|
dev->data->dev_link.link_status = RTE_ETH_LINK_UP;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_dev_stop(struct rte_eth_dev *dev)
|
|
{
|
|
dev->data->dev_link.link_status = RTE_ETH_LINK_DOWN;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_dev_close(struct rte_eth_dev *dev)
|
|
{
|
|
return ice_dcf_vf_repr_uninit(dev);
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_rx_queue_setup(__rte_unused struct rte_eth_dev *dev,
|
|
__rte_unused uint16_t queue_id,
|
|
__rte_unused uint16_t nb_desc,
|
|
__rte_unused unsigned int socket_id,
|
|
__rte_unused const struct rte_eth_rxconf *conf,
|
|
__rte_unused struct rte_mempool *pool)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_tx_queue_setup(__rte_unused struct rte_eth_dev *dev,
|
|
__rte_unused uint16_t queue_id,
|
|
__rte_unused uint16_t nb_desc,
|
|
__rte_unused unsigned int socket_id,
|
|
__rte_unused const struct rte_eth_txconf *conf)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_promiscuous_enable(__rte_unused struct rte_eth_dev *ethdev)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_promiscuous_disable(__rte_unused struct rte_eth_dev *ethdev)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_allmulticast_enable(__rte_unused struct rte_eth_dev *dev)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_allmulticast_disable(__rte_unused struct rte_eth_dev *dev)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_link_update(__rte_unused struct rte_eth_dev *ethdev,
|
|
__rte_unused int wait_to_complete)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static __rte_always_inline struct ice_dcf_hw *
|
|
ice_dcf_vf_repr_hw(struct ice_dcf_vf_repr *repr)
|
|
{
|
|
struct ice_dcf_adapter *dcf_adapter =
|
|
repr->dcf_eth_dev->data->dev_private;
|
|
|
|
if (!dcf_adapter) {
|
|
PMD_DRV_LOG(ERR, "DCF for VF representor has been released\n");
|
|
return NULL;
|
|
}
|
|
|
|
return &dcf_adapter->real_hw;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_dev_info_get(struct rte_eth_dev *dev,
|
|
struct rte_eth_dev_info *dev_info)
|
|
{
|
|
struct ice_dcf_vf_repr *repr = dev->data->dev_private;
|
|
struct ice_dcf_hw *dcf_hw = ice_dcf_vf_repr_hw(repr);
|
|
|
|
if (!dcf_hw)
|
|
return -EIO;
|
|
|
|
dev_info->device = dev->device;
|
|
dev_info->max_mac_addrs = 1;
|
|
dev_info->max_rx_queues = dcf_hw->vsi_res->num_queue_pairs;
|
|
dev_info->max_tx_queues = dcf_hw->vsi_res->num_queue_pairs;
|
|
dev_info->min_rx_bufsize = ICE_BUF_SIZE_MIN;
|
|
dev_info->max_rx_pktlen = ICE_FRAME_SIZE_MAX;
|
|
dev_info->hash_key_size = dcf_hw->vf_res->rss_key_size;
|
|
dev_info->reta_size = dcf_hw->vf_res->rss_lut_size;
|
|
dev_info->flow_type_rss_offloads = ICE_RSS_OFFLOAD_ALL;
|
|
|
|
dev_info->rx_offload_capa =
|
|
RTE_ETH_RX_OFFLOAD_VLAN_STRIP |
|
|
RTE_ETH_RX_OFFLOAD_IPV4_CKSUM |
|
|
RTE_ETH_RX_OFFLOAD_UDP_CKSUM |
|
|
RTE_ETH_RX_OFFLOAD_TCP_CKSUM |
|
|
RTE_ETH_RX_OFFLOAD_OUTER_IPV4_CKSUM |
|
|
RTE_ETH_RX_OFFLOAD_SCATTER |
|
|
RTE_ETH_RX_OFFLOAD_VLAN_FILTER |
|
|
RTE_ETH_RX_OFFLOAD_VLAN_EXTEND |
|
|
RTE_ETH_RX_OFFLOAD_RSS_HASH;
|
|
dev_info->tx_offload_capa =
|
|
RTE_ETH_TX_OFFLOAD_VLAN_INSERT |
|
|
RTE_ETH_TX_OFFLOAD_IPV4_CKSUM |
|
|
RTE_ETH_TX_OFFLOAD_UDP_CKSUM |
|
|
RTE_ETH_TX_OFFLOAD_TCP_CKSUM |
|
|
RTE_ETH_TX_OFFLOAD_SCTP_CKSUM |
|
|
RTE_ETH_TX_OFFLOAD_OUTER_IPV4_CKSUM |
|
|
RTE_ETH_TX_OFFLOAD_TCP_TSO |
|
|
RTE_ETH_TX_OFFLOAD_VXLAN_TNL_TSO |
|
|
RTE_ETH_TX_OFFLOAD_GRE_TNL_TSO |
|
|
RTE_ETH_TX_OFFLOAD_IPIP_TNL_TSO |
|
|
RTE_ETH_TX_OFFLOAD_GENEVE_TNL_TSO |
|
|
RTE_ETH_TX_OFFLOAD_MULTI_SEGS;
|
|
|
|
dev_info->default_rxconf = (struct rte_eth_rxconf) {
|
|
.rx_thresh = {
|
|
.pthresh = ICE_DEFAULT_RX_PTHRESH,
|
|
.hthresh = ICE_DEFAULT_RX_HTHRESH,
|
|
.wthresh = ICE_DEFAULT_RX_WTHRESH,
|
|
},
|
|
.rx_free_thresh = ICE_DEFAULT_RX_FREE_THRESH,
|
|
.rx_drop_en = 0,
|
|
.offloads = 0,
|
|
};
|
|
|
|
dev_info->default_txconf = (struct rte_eth_txconf) {
|
|
.tx_thresh = {
|
|
.pthresh = ICE_DEFAULT_TX_PTHRESH,
|
|
.hthresh = ICE_DEFAULT_TX_HTHRESH,
|
|
.wthresh = ICE_DEFAULT_TX_WTHRESH,
|
|
},
|
|
.tx_free_thresh = ICE_DEFAULT_TX_FREE_THRESH,
|
|
.tx_rs_thresh = ICE_DEFAULT_TX_RSBIT_THRESH,
|
|
.offloads = 0,
|
|
};
|
|
|
|
dev_info->rx_desc_lim = (struct rte_eth_desc_lim) {
|
|
.nb_max = ICE_MAX_RING_DESC,
|
|
.nb_min = ICE_MIN_RING_DESC,
|
|
.nb_align = ICE_ALIGN_RING_DESC,
|
|
};
|
|
|
|
dev_info->tx_desc_lim = (struct rte_eth_desc_lim) {
|
|
.nb_max = ICE_MAX_RING_DESC,
|
|
.nb_min = ICE_MIN_RING_DESC,
|
|
.nb_align = ICE_ALIGN_RING_DESC,
|
|
};
|
|
|
|
dev_info->switch_info.name = dcf_hw->eth_dev->device->name;
|
|
dev_info->switch_info.domain_id = repr->switch_domain_id;
|
|
dev_info->switch_info.port_id = repr->vf_id;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static __rte_always_inline bool
|
|
ice_dcf_vlan_offload_ena(struct ice_dcf_vf_repr *repr)
|
|
{
|
|
return !!(ice_dcf_vf_repr_hw(repr)->vf_res->vf_cap_flags &
|
|
VIRTCHNL_VF_OFFLOAD_VLAN_V2);
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vlan_offload_config(struct ice_dcf_vf_repr *repr,
|
|
struct virtchnl_dcf_vlan_offload *vlan_offload)
|
|
{
|
|
struct dcf_virtchnl_cmd args;
|
|
int err;
|
|
|
|
memset(&args, 0, sizeof(args));
|
|
args.v_op = VIRTCHNL_OP_DCF_VLAN_OFFLOAD;
|
|
args.req_msg = (uint8_t *)vlan_offload;
|
|
args.req_msglen = sizeof(*vlan_offload);
|
|
|
|
err = ice_dcf_execute_virtchnl_cmd(ice_dcf_vf_repr_hw(repr), &args);
|
|
if (err)
|
|
PMD_DRV_LOG(ERR,
|
|
"Failed to execute command of VIRTCHNL_OP_DCF_VLAN_OFFLOAD");
|
|
|
|
return err;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_vlan_offload_set(struct rte_eth_dev *dev, int mask)
|
|
{
|
|
struct ice_dcf_vf_repr *repr = dev->data->dev_private;
|
|
struct rte_eth_conf *dev_conf = &dev->data->dev_conf;
|
|
struct virtchnl_dcf_vlan_offload vlan_offload;
|
|
int err;
|
|
|
|
if (!ice_dcf_vlan_offload_ena(repr))
|
|
return -ENOTSUP;
|
|
|
|
/* Vlan stripping setting */
|
|
if (mask & RTE_ETH_VLAN_STRIP_MASK) {
|
|
bool enable = !!(dev_conf->rxmode.offloads &
|
|
RTE_ETH_RX_OFFLOAD_VLAN_STRIP);
|
|
|
|
if (enable && repr->outer_vlan_info.port_vlan_ena) {
|
|
PMD_DRV_LOG(ERR,
|
|
"Disable the port VLAN firstly\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
memset(&vlan_offload, 0, sizeof(vlan_offload));
|
|
|
|
if (enable)
|
|
vlan_offload.vlan_flags =
|
|
VIRTCHNL_DCF_VLAN_STRIP_INTO_RX_DESC <<
|
|
VIRTCHNL_DCF_VLAN_STRIP_MODE_S;
|
|
else if (repr->outer_vlan_info.stripping_ena && !enable)
|
|
vlan_offload.vlan_flags =
|
|
VIRTCHNL_DCF_VLAN_STRIP_DISABLE <<
|
|
VIRTCHNL_DCF_VLAN_STRIP_MODE_S;
|
|
|
|
if (vlan_offload.vlan_flags) {
|
|
vlan_offload.vf_id = repr->vf_id;
|
|
vlan_offload.tpid = repr->outer_vlan_info.tpid;
|
|
vlan_offload.vlan_flags |=
|
|
VIRTCHNL_DCF_VLAN_TYPE_OUTER <<
|
|
VIRTCHNL_DCF_VLAN_TYPE_S;
|
|
|
|
err = ice_dcf_vlan_offload_config(repr, &vlan_offload);
|
|
if (err)
|
|
return -EIO;
|
|
|
|
repr->outer_vlan_info.stripping_ena = enable;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_vlan_pvid_set(struct rte_eth_dev *dev,
|
|
uint16_t pvid, int on)
|
|
{
|
|
struct ice_dcf_vf_repr *repr = dev->data->dev_private;
|
|
struct virtchnl_dcf_vlan_offload vlan_offload;
|
|
int err;
|
|
|
|
if (!ice_dcf_vlan_offload_ena(repr))
|
|
return -ENOTSUP;
|
|
|
|
if (repr->outer_vlan_info.stripping_ena) {
|
|
PMD_DRV_LOG(ERR,
|
|
"Disable the VLAN stripping firstly\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (pvid > RTE_ETHER_MAX_VLAN_ID)
|
|
return -EINVAL;
|
|
|
|
memset(&vlan_offload, 0, sizeof(vlan_offload));
|
|
|
|
if (on)
|
|
vlan_offload.vlan_flags =
|
|
(VIRTCHNL_DCF_VLAN_INSERT_PORT_BASED <<
|
|
VIRTCHNL_DCF_VLAN_INSERT_MODE_S);
|
|
else
|
|
vlan_offload.vlan_flags =
|
|
(VIRTCHNL_DCF_VLAN_INSERT_DISABLE <<
|
|
VIRTCHNL_DCF_VLAN_INSERT_MODE_S);
|
|
|
|
vlan_offload.vf_id = repr->vf_id;
|
|
vlan_offload.tpid = repr->outer_vlan_info.tpid;
|
|
vlan_offload.vlan_flags |= (VIRTCHNL_DCF_VLAN_TYPE_OUTER <<
|
|
VIRTCHNL_DCF_VLAN_TYPE_S);
|
|
vlan_offload.vlan_id = pvid;
|
|
|
|
err = ice_dcf_vlan_offload_config(repr, &vlan_offload);
|
|
if (!err) {
|
|
if (on) {
|
|
repr->outer_vlan_info.port_vlan_ena = true;
|
|
repr->outer_vlan_info.vid = pvid;
|
|
} else {
|
|
repr->outer_vlan_info.port_vlan_ena = false;
|
|
}
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
static int
|
|
ice_dcf_vf_repr_vlan_tpid_set(struct rte_eth_dev *dev,
|
|
enum rte_vlan_type vlan_type, uint16_t tpid)
|
|
{
|
|
struct ice_dcf_vf_repr *repr = dev->data->dev_private;
|
|
int err = 0;
|
|
|
|
if (!ice_dcf_vlan_offload_ena(repr))
|
|
return -ENOTSUP;
|
|
|
|
if (vlan_type != RTE_ETH_VLAN_TYPE_OUTER) {
|
|
PMD_DRV_LOG(ERR,
|
|
"Can accelerate only outer VLAN in QinQ\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (tpid != RTE_ETHER_TYPE_QINQ &&
|
|
tpid != RTE_ETHER_TYPE_VLAN &&
|
|
tpid != RTE_ETHER_TYPE_QINQ1) {
|
|
PMD_DRV_LOG(ERR,
|
|
"Invalid TPID: 0x%04x\n", tpid);
|
|
return -EINVAL;
|
|
}
|
|
|
|
repr->outer_vlan_info.tpid = tpid;
|
|
|
|
if (repr->outer_vlan_info.port_vlan_ena) {
|
|
err = ice_dcf_vf_repr_vlan_pvid_set(dev,
|
|
repr->outer_vlan_info.vid,
|
|
true);
|
|
if (err) {
|
|
PMD_DRV_LOG(ERR,
|
|
"Failed to reset port VLAN : %d\n",
|
|
err);
|
|
return err;
|
|
}
|
|
}
|
|
|
|
if (repr->outer_vlan_info.stripping_ena) {
|
|
err = ice_dcf_vf_repr_vlan_offload_set(dev,
|
|
RTE_ETH_VLAN_STRIP_MASK);
|
|
if (err) {
|
|
PMD_DRV_LOG(ERR,
|
|
"Failed to reset VLAN stripping : %d\n",
|
|
err);
|
|
return err;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct eth_dev_ops ice_dcf_vf_repr_dev_ops = {
|
|
.dev_configure = ice_dcf_vf_repr_dev_configure,
|
|
.dev_start = ice_dcf_vf_repr_dev_start,
|
|
.dev_stop = ice_dcf_vf_repr_dev_stop,
|
|
.dev_close = ice_dcf_vf_repr_dev_close,
|
|
.dev_infos_get = ice_dcf_vf_repr_dev_info_get,
|
|
.rx_queue_setup = ice_dcf_vf_repr_rx_queue_setup,
|
|
.tx_queue_setup = ice_dcf_vf_repr_tx_queue_setup,
|
|
.promiscuous_enable = ice_dcf_vf_repr_promiscuous_enable,
|
|
.promiscuous_disable = ice_dcf_vf_repr_promiscuous_disable,
|
|
.allmulticast_enable = ice_dcf_vf_repr_allmulticast_enable,
|
|
.allmulticast_disable = ice_dcf_vf_repr_allmulticast_disable,
|
|
.link_update = ice_dcf_vf_repr_link_update,
|
|
.vlan_offload_set = ice_dcf_vf_repr_vlan_offload_set,
|
|
.vlan_pvid_set = ice_dcf_vf_repr_vlan_pvid_set,
|
|
.vlan_tpid_set = ice_dcf_vf_repr_vlan_tpid_set,
|
|
};
|
|
|
|
int
|
|
ice_dcf_vf_repr_init(struct rte_eth_dev *vf_rep_eth_dev, void *init_param)
|
|
{
|
|
struct ice_dcf_vf_repr *repr = vf_rep_eth_dev->data->dev_private;
|
|
struct ice_dcf_vf_repr_param *param = init_param;
|
|
|
|
repr->dcf_eth_dev = param->dcf_eth_dev;
|
|
repr->switch_domain_id = param->switch_domain_id;
|
|
repr->vf_id = param->vf_id;
|
|
repr->outer_vlan_info.port_vlan_ena = false;
|
|
repr->outer_vlan_info.stripping_ena = false;
|
|
repr->outer_vlan_info.tpid = RTE_ETHER_TYPE_VLAN;
|
|
|
|
vf_rep_eth_dev->dev_ops = &ice_dcf_vf_repr_dev_ops;
|
|
|
|
vf_rep_eth_dev->rx_pkt_burst = ice_dcf_vf_repr_rx_burst;
|
|
vf_rep_eth_dev->tx_pkt_burst = ice_dcf_vf_repr_tx_burst;
|
|
|
|
vf_rep_eth_dev->data->dev_flags |= RTE_ETH_DEV_REPRESENTOR;
|
|
vf_rep_eth_dev->data->representor_id = repr->vf_id;
|
|
vf_rep_eth_dev->data->backer_port_id = repr->dcf_eth_dev->data->port_id;
|
|
|
|
vf_rep_eth_dev->data->mac_addrs = &repr->mac_addr;
|
|
|
|
rte_eth_random_addr(repr->mac_addr.addr_bytes);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
ice_dcf_vf_repr_uninit(struct rte_eth_dev *vf_rep_eth_dev)
|
|
{
|
|
vf_rep_eth_dev->data->mac_addrs = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
ice_dcf_vf_repr_init_vlan(struct rte_eth_dev *vf_rep_eth_dev)
|
|
{
|
|
struct ice_dcf_vf_repr *repr = vf_rep_eth_dev->data->dev_private;
|
|
int err;
|
|
|
|
err = ice_dcf_vf_repr_vlan_offload_set(vf_rep_eth_dev,
|
|
RTE_ETH_VLAN_STRIP_MASK);
|
|
if (err) {
|
|
PMD_DRV_LOG(ERR, "Failed to set VLAN offload");
|
|
return err;
|
|
}
|
|
|
|
if (repr->outer_vlan_info.port_vlan_ena) {
|
|
err = ice_dcf_vf_repr_vlan_pvid_set(vf_rep_eth_dev,
|
|
repr->outer_vlan_info.vid,
|
|
true);
|
|
if (err) {
|
|
PMD_DRV_LOG(ERR, "Failed to enable port VLAN");
|
|
return err;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
ice_dcf_vf_repr_stop_all(struct ice_dcf_adapter *dcf_adapter)
|
|
{
|
|
uint16_t vf_id;
|
|
int ret;
|
|
|
|
if (!dcf_adapter->repr_infos)
|
|
return;
|
|
|
|
for (vf_id = 0; vf_id < dcf_adapter->real_hw.num_vfs; vf_id++) {
|
|
struct rte_eth_dev *vf_rep_eth_dev =
|
|
dcf_adapter->repr_infos[vf_id].vf_rep_eth_dev;
|
|
if (!vf_rep_eth_dev || vf_rep_eth_dev->data->dev_started == 0)
|
|
continue;
|
|
|
|
ret = ice_dcf_vf_repr_dev_stop(vf_rep_eth_dev);
|
|
if (!ret)
|
|
vf_rep_eth_dev->data->dev_started = 0;
|
|
}
|
|
}
|