/* SPDX-License-Identifier: BSD-3-Clause * * Copyright(c) 2019-2021 Xilinx, Inc. * Copyright(c) 2019 Solarflare Communications Inc. * * This software was jointly developed between OKTET Labs (under contract * for Solarflare) and Solarflare Communications, Inc. */ #include #include #include #include #include #include #include #include "efx.h" #include "sfc_log.h" #include "sfc_debug.h" #include "sfc_repr.h" #include "sfc_ethdev_state.h" #include "sfc_repr_proxy_api.h" #include "sfc_switch.h" #include "sfc_dp_tx.h" /** Multi-process shared representor private data */ struct sfc_repr_shared { uint16_t pf_port_id; uint16_t repr_id; uint16_t switch_domain_id; uint16_t switch_port_id; }; struct sfc_repr_queue_stats { union sfc_pkts_bytes packets_bytes; }; struct sfc_repr_rxq { /* Datapath members */ struct rte_ring *ring; struct sfc_repr_queue_stats stats; }; struct sfc_repr_txq { /* Datapath members */ struct rte_ring *ring; efx_mport_id_t egress_mport; struct sfc_repr_queue_stats stats; }; /** Primary process representor private data */ struct sfc_repr { /** * PMD setup and configuration is not thread safe. Since it is not * performance sensitive, it is better to guarantee thread-safety * and add device level lock. Adapter control operations which * change its state should acquire the lock. */ rte_spinlock_t lock; enum sfc_ethdev_state state; }; #define sfcr_err(sr, ...) \ do { \ const struct sfc_repr *_sr = (sr); \ \ (void)_sr; \ SFC_GENERIC_LOG(ERR, __VA_ARGS__); \ } while (0) #define sfcr_warn(sr, ...) \ do { \ const struct sfc_repr *_sr = (sr); \ \ (void)_sr; \ SFC_GENERIC_LOG(WARNING, __VA_ARGS__); \ } while (0) #define sfcr_info(sr, ...) \ do { \ const struct sfc_repr *_sr = (sr); \ \ (void)_sr; \ SFC_GENERIC_LOG(INFO, \ RTE_FMT("%s() " \ RTE_FMT_HEAD(__VA_ARGS__ ,), \ __func__, \ RTE_FMT_TAIL(__VA_ARGS__ ,))); \ } while (0) static inline struct sfc_repr_shared * sfc_repr_shared_by_eth_dev(struct rte_eth_dev *eth_dev) { struct sfc_repr_shared *srs = eth_dev->data->dev_private; return srs; } static inline struct sfc_repr * sfc_repr_by_eth_dev(struct rte_eth_dev *eth_dev) { struct sfc_repr *sr = eth_dev->process_private; return sr; } /* * Add wrapper functions to acquire/release lock to be able to remove or * change the lock in one place. */ static inline void sfc_repr_lock_init(struct sfc_repr *sr) { rte_spinlock_init(&sr->lock); } #if defined(RTE_LIBRTE_SFC_EFX_DEBUG) || defined(RTE_ENABLE_ASSERT) static inline int sfc_repr_lock_is_locked(struct sfc_repr *sr) { return rte_spinlock_is_locked(&sr->lock); } #endif static inline void sfc_repr_lock(struct sfc_repr *sr) { rte_spinlock_lock(&sr->lock); } static inline void sfc_repr_unlock(struct sfc_repr *sr) { rte_spinlock_unlock(&sr->lock); } static inline void sfc_repr_lock_fini(__rte_unused struct sfc_repr *sr) { /* Just for symmetry of the API */ } static void sfc_repr_rx_queue_stop(void *queue) { struct sfc_repr_rxq *rxq = queue; if (rxq == NULL) return; rte_ring_reset(rxq->ring); } static void sfc_repr_tx_queue_stop(void *queue) { struct sfc_repr_txq *txq = queue; if (txq == NULL) return; rte_ring_reset(txq->ring); } static uint16_t sfc_repr_rx_burst(void *rx_queue, struct rte_mbuf **rx_pkts, uint16_t nb_pkts) { struct sfc_repr_rxq *rxq = rx_queue; void **objs = (void *)&rx_pkts[0]; unsigned int n_rx; /* mbufs port is already filled correctly by representors proxy */ n_rx = rte_ring_sc_dequeue_burst(rxq->ring, objs, nb_pkts, NULL); if (n_rx > 0) { unsigned int n_bytes = 0; unsigned int i = 0; do { n_bytes += rx_pkts[i]->pkt_len; } while (++i < n_rx); sfc_pkts_bytes_add(&rxq->stats.packets_bytes, n_rx, n_bytes); } return n_rx; } static uint16_t sfc_repr_tx_burst(void *tx_queue, struct rte_mbuf **tx_pkts, uint16_t nb_pkts) { struct sfc_repr_txq *txq = tx_queue; unsigned int n_bytes = 0; unsigned int n_tx; void **objs; uint16_t i; /* * mbuf is likely cache-hot. Set flag and egress m-port here instead of * doing that in representors proxy. Also, it should help to avoid * cache bounce. Moreover, potentially, it allows to use one * multi-producer single-consumer ring for all representors. * * The only potential problem is doing so many times if enqueue * fails and sender retries. */ for (i = 0; i < nb_pkts; ++i) { struct rte_mbuf *m = tx_pkts[i]; m->ol_flags |= sfc_dp_mport_override; *RTE_MBUF_DYNFIELD(m, sfc_dp_mport_offset, efx_mport_id_t *) = txq->egress_mport; n_bytes += tx_pkts[i]->pkt_len; } objs = (void *)&tx_pkts[0]; n_tx = rte_ring_sp_enqueue_burst(txq->ring, objs, nb_pkts, NULL); /* * Remove m-port override flag from packets that were not enqueued * Setting the flag only for enqueued packets after the burst is * not possible since the ownership of enqueued packets is * transferred to representor proxy. The same logic applies to * counting the enqueued packets' bytes. */ for (i = n_tx; i < nb_pkts; ++i) { struct rte_mbuf *m = tx_pkts[i]; m->ol_flags &= ~sfc_dp_mport_override; n_bytes -= m->pkt_len; } sfc_pkts_bytes_add(&txq->stats.packets_bytes, n_tx, n_bytes); return n_tx; } static int sfc_repr_start(struct rte_eth_dev *dev) { struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); struct sfc_repr_shared *srs; int ret; sfcr_info(sr, "entry"); SFC_ASSERT(sfc_repr_lock_is_locked(sr)); switch (sr->state) { case SFC_ETHDEV_CONFIGURED: break; case SFC_ETHDEV_STARTED: sfcr_info(sr, "already started"); return 0; default: ret = -EINVAL; goto fail_bad_state; } sr->state = SFC_ETHDEV_STARTING; srs = sfc_repr_shared_by_eth_dev(dev); ret = sfc_repr_proxy_start_repr(srs->pf_port_id, srs->repr_id); if (ret != 0) { SFC_ASSERT(ret > 0); ret = -ret; goto fail_start; } sr->state = SFC_ETHDEV_STARTED; sfcr_info(sr, "done"); return 0; fail_start: sr->state = SFC_ETHDEV_CONFIGURED; fail_bad_state: sfcr_err(sr, "%s() failed: %s", __func__, rte_strerror(-ret)); return ret; } static int sfc_repr_dev_start(struct rte_eth_dev *dev) { struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); int ret; sfcr_info(sr, "entry"); sfc_repr_lock(sr); ret = sfc_repr_start(dev); sfc_repr_unlock(sr); if (ret != 0) goto fail_start; sfcr_info(sr, "done"); return 0; fail_start: sfcr_err(sr, "%s() failed: %s", __func__, rte_strerror(-ret)); return ret; } static int sfc_repr_stop(struct rte_eth_dev *dev) { struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); struct sfc_repr_shared *srs; unsigned int i; int ret; sfcr_info(sr, "entry"); SFC_ASSERT(sfc_repr_lock_is_locked(sr)); switch (sr->state) { case SFC_ETHDEV_STARTED: break; case SFC_ETHDEV_CONFIGURED: sfcr_info(sr, "already stopped"); return 0; default: sfcr_err(sr, "stop in unexpected state %u", sr->state); SFC_ASSERT(B_FALSE); ret = -EINVAL; goto fail_bad_state; } srs = sfc_repr_shared_by_eth_dev(dev); ret = sfc_repr_proxy_stop_repr(srs->pf_port_id, srs->repr_id); if (ret != 0) { SFC_ASSERT(ret > 0); ret = -ret; goto fail_stop; } for (i = 0; i < dev->data->nb_rx_queues; i++) sfc_repr_rx_queue_stop(dev->data->rx_queues[i]); for (i = 0; i < dev->data->nb_tx_queues; i++) sfc_repr_tx_queue_stop(dev->data->tx_queues[i]); sr->state = SFC_ETHDEV_CONFIGURED; sfcr_info(sr, "done"); return 0; fail_bad_state: fail_stop: sfcr_err(sr, "%s() failed: %s", __func__, rte_strerror(-ret)); return ret; } static int sfc_repr_dev_stop(struct rte_eth_dev *dev) { struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); int ret; sfcr_info(sr, "entry"); sfc_repr_lock(sr); ret = sfc_repr_stop(dev); if (ret != 0) { sfcr_err(sr, "%s() failed to stop representor", __func__); goto fail_stop; } sfc_repr_unlock(sr); sfcr_info(sr, "done"); return 0; fail_stop: sfc_repr_unlock(sr); sfcr_err(sr, "%s() failed %s", __func__, rte_strerror(-ret)); return ret; } static int sfc_repr_check_conf(struct sfc_repr *sr, uint16_t nb_rx_queues, const struct rte_eth_conf *conf) { const struct rte_eth_rss_conf *rss_conf; int ret = 0; sfcr_info(sr, "entry"); if (conf->link_speeds != 0) { sfcr_err(sr, "specific link speeds not supported"); ret = -EINVAL; } switch (conf->rxmode.mq_mode) { case RTE_ETH_MQ_RX_RSS: if (nb_rx_queues != 1) { sfcr_err(sr, "Rx RSS is not supported with %u queues", nb_rx_queues); ret = -EINVAL; break; } rss_conf = &conf->rx_adv_conf.rss_conf; if (rss_conf->rss_key != NULL || rss_conf->rss_key_len != 0 || rss_conf->rss_hf != 0) { sfcr_err(sr, "Rx RSS configuration is not supported"); ret = -EINVAL; } break; case RTE_ETH_MQ_RX_NONE: break; default: sfcr_err(sr, "Rx mode MQ modes other than RSS not supported"); ret = -EINVAL; break; } if (conf->txmode.mq_mode != RTE_ETH_MQ_TX_NONE) { sfcr_err(sr, "Tx mode MQ modes not supported"); ret = -EINVAL; } if (conf->lpbk_mode != 0) { sfcr_err(sr, "loopback not supported"); ret = -EINVAL; } if (conf->dcb_capability_en != 0) { sfcr_err(sr, "priority-based flow control not supported"); ret = -EINVAL; } if (conf->fdir_conf.mode != RTE_FDIR_MODE_NONE) { sfcr_err(sr, "Flow Director not supported"); ret = -EINVAL; } if (conf->intr_conf.lsc != 0) { sfcr_err(sr, "link status change interrupt not supported"); ret = -EINVAL; } if (conf->intr_conf.rxq != 0) { sfcr_err(sr, "receive queue interrupt not supported"); ret = -EINVAL; } if (conf->intr_conf.rmv != 0) { sfcr_err(sr, "remove interrupt not supported"); ret = -EINVAL; } sfcr_info(sr, "done %d", ret); return ret; } static int sfc_repr_configure(struct sfc_repr *sr, uint16_t nb_rx_queues, const struct rte_eth_conf *conf) { int ret; sfcr_info(sr, "entry"); SFC_ASSERT(sfc_repr_lock_is_locked(sr)); ret = sfc_repr_check_conf(sr, nb_rx_queues, conf); if (ret != 0) goto fail_check_conf; sr->state = SFC_ETHDEV_CONFIGURED; sfcr_info(sr, "done"); return 0; fail_check_conf: sfcr_info(sr, "failed %s", rte_strerror(-ret)); return ret; } static int sfc_repr_dev_configure(struct rte_eth_dev *dev) { struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); struct rte_eth_dev_data *dev_data = dev->data; int ret; sfcr_info(sr, "entry n_rxq=%u n_txq=%u", dev_data->nb_rx_queues, dev_data->nb_tx_queues); sfc_repr_lock(sr); switch (sr->state) { case SFC_ETHDEV_CONFIGURED: /* FALLTHROUGH */ case SFC_ETHDEV_INITIALIZED: ret = sfc_repr_configure(sr, dev_data->nb_rx_queues, &dev_data->dev_conf); break; default: sfcr_err(sr, "unexpected adapter state %u to configure", sr->state); ret = -EINVAL; break; } sfc_repr_unlock(sr); sfcr_info(sr, "done %s", rte_strerror(-ret)); return ret; } static int sfc_repr_dev_infos_get(struct rte_eth_dev *dev, struct rte_eth_dev_info *dev_info) { struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); dev_info->device = dev->device; dev_info->max_rx_queues = SFC_REPR_RXQ_MAX; dev_info->max_tx_queues = SFC_REPR_TXQ_MAX; dev_info->default_rxconf.rx_drop_en = 1; dev_info->switch_info.domain_id = srs->switch_domain_id; dev_info->switch_info.port_id = srs->switch_port_id; return 0; } static int sfc_repr_dev_link_update(struct rte_eth_dev *dev, __rte_unused int wait_to_complete) { struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); struct rte_eth_link link; if (sr->state != SFC_ETHDEV_STARTED) { sfc_port_link_mode_to_info(EFX_LINK_UNKNOWN, &link); } else { memset(&link, 0, sizeof(link)); link.link_status = RTE_ETH_LINK_UP; link.link_speed = RTE_ETH_SPEED_NUM_UNKNOWN; } return rte_eth_linkstatus_set(dev, &link); } static int sfc_repr_ring_create(uint16_t pf_port_id, uint16_t repr_id, const char *type_name, uint16_t qid, uint16_t nb_desc, unsigned int socket_id, struct rte_ring **ring) { char ring_name[RTE_RING_NAMESIZE]; int ret; ret = snprintf(ring_name, sizeof(ring_name), "sfc_%u_repr_%u_%sq%u", pf_port_id, repr_id, type_name, qid); if (ret >= (int)sizeof(ring_name)) return -ENAMETOOLONG; /* * Single producer/consumer rings are used since the API for Tx/Rx * packet burst for representors are guaranteed to be called from * a single thread, and the user of the other end (representor proxy) * is also single-threaded. */ *ring = rte_ring_create(ring_name, nb_desc, socket_id, RING_F_SP_ENQ | RING_F_SC_DEQ); if (*ring == NULL) return -rte_errno; return 0; } static int sfc_repr_rx_qcheck_conf(struct sfc_repr *sr, const struct rte_eth_rxconf *rx_conf) { int ret = 0; sfcr_info(sr, "entry"); if (rx_conf->rx_thresh.pthresh != 0 || rx_conf->rx_thresh.hthresh != 0 || rx_conf->rx_thresh.wthresh != 0) { sfcr_warn(sr, "RxQ prefetch/host/writeback thresholds are not supported"); } if (rx_conf->rx_free_thresh != 0) sfcr_warn(sr, "RxQ free threshold is not supported"); if (rx_conf->rx_drop_en == 0) sfcr_warn(sr, "RxQ drop disable is not supported"); if (rx_conf->rx_deferred_start) { sfcr_err(sr, "Deferred start is not supported"); ret = -EINVAL; } sfcr_info(sr, "done: %s", rte_strerror(-ret)); return ret; } static int sfc_repr_rx_queue_setup(struct rte_eth_dev *dev, uint16_t rx_queue_id, uint16_t nb_rx_desc, unsigned int socket_id, __rte_unused const struct rte_eth_rxconf *rx_conf, struct rte_mempool *mb_pool) { struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); struct sfc_repr_rxq *rxq; int ret; sfcr_info(sr, "entry"); ret = sfc_repr_rx_qcheck_conf(sr, rx_conf); if (ret != 0) goto fail_check_conf; ret = -ENOMEM; rxq = rte_zmalloc_socket("sfc-repr-rxq", sizeof(*rxq), RTE_CACHE_LINE_SIZE, socket_id); if (rxq == NULL) { sfcr_err(sr, "%s() failed to alloc RxQ", __func__); goto fail_rxq_alloc; } ret = sfc_repr_ring_create(srs->pf_port_id, srs->repr_id, "rx", rx_queue_id, nb_rx_desc, socket_id, &rxq->ring); if (ret != 0) { sfcr_err(sr, "%s() failed to create ring", __func__); goto fail_ring_create; } ret = sfc_repr_proxy_add_rxq(srs->pf_port_id, srs->repr_id, rx_queue_id, rxq->ring, mb_pool); if (ret != 0) { SFC_ASSERT(ret > 0); ret = -ret; sfcr_err(sr, "%s() failed to add proxy RxQ", __func__); goto fail_proxy_add_rxq; } dev->data->rx_queues[rx_queue_id] = rxq; sfcr_info(sr, "done"); return 0; fail_proxy_add_rxq: rte_ring_free(rxq->ring); fail_ring_create: rte_free(rxq); fail_rxq_alloc: fail_check_conf: sfcr_err(sr, "%s() failed: %s", __func__, rte_strerror(-ret)); return ret; } static void sfc_repr_rx_queue_release(struct rte_eth_dev *dev, uint16_t rx_queue_id) { struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); struct sfc_repr_rxq *rxq = dev->data->rx_queues[rx_queue_id]; sfc_repr_proxy_del_rxq(srs->pf_port_id, srs->repr_id, rx_queue_id); rte_ring_free(rxq->ring); rte_free(rxq); } static int sfc_repr_tx_qcheck_conf(struct sfc_repr *sr, const struct rte_eth_txconf *tx_conf) { int ret = 0; sfcr_info(sr, "entry"); if (tx_conf->tx_rs_thresh != 0) sfcr_warn(sr, "RS bit in transmit descriptor is not supported"); if (tx_conf->tx_free_thresh != 0) sfcr_warn(sr, "TxQ free threshold is not supported"); if (tx_conf->tx_thresh.pthresh != 0 || tx_conf->tx_thresh.hthresh != 0 || tx_conf->tx_thresh.wthresh != 0) { sfcr_warn(sr, "prefetch/host/writeback thresholds are not supported"); } if (tx_conf->tx_deferred_start) { sfcr_err(sr, "Deferred start is not supported"); ret = -EINVAL; } sfcr_info(sr, "done: %s", rte_strerror(-ret)); return ret; } static int sfc_repr_tx_queue_setup(struct rte_eth_dev *dev, uint16_t tx_queue_id, uint16_t nb_tx_desc, unsigned int socket_id, const struct rte_eth_txconf *tx_conf) { struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); struct sfc_repr_txq *txq; int ret; sfcr_info(sr, "entry"); ret = sfc_repr_tx_qcheck_conf(sr, tx_conf); if (ret != 0) goto fail_check_conf; ret = -ENOMEM; txq = rte_zmalloc_socket("sfc-repr-txq", sizeof(*txq), RTE_CACHE_LINE_SIZE, socket_id); if (txq == NULL) goto fail_txq_alloc; ret = sfc_repr_ring_create(srs->pf_port_id, srs->repr_id, "tx", tx_queue_id, nb_tx_desc, socket_id, &txq->ring); if (ret != 0) goto fail_ring_create; ret = sfc_repr_proxy_add_txq(srs->pf_port_id, srs->repr_id, tx_queue_id, txq->ring, &txq->egress_mport); if (ret != 0) goto fail_proxy_add_txq; dev->data->tx_queues[tx_queue_id] = txq; sfcr_info(sr, "done"); return 0; fail_proxy_add_txq: rte_ring_free(txq->ring); fail_ring_create: rte_free(txq); fail_txq_alloc: fail_check_conf: sfcr_err(sr, "%s() failed: %s", __func__, rte_strerror(-ret)); return ret; } static void sfc_repr_tx_queue_release(struct rte_eth_dev *dev, uint16_t tx_queue_id) { struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); struct sfc_repr_txq *txq = dev->data->tx_queues[tx_queue_id]; sfc_repr_proxy_del_txq(srs->pf_port_id, srs->repr_id, tx_queue_id); rte_ring_free(txq->ring); rte_free(txq); } static void sfc_repr_close(struct sfc_repr *sr) { SFC_ASSERT(sfc_repr_lock_is_locked(sr)); SFC_ASSERT(sr->state == SFC_ETHDEV_CONFIGURED); sr->state = SFC_ETHDEV_CLOSING; /* Put representor close actions here */ sr->state = SFC_ETHDEV_INITIALIZED; } static int sfc_repr_dev_close(struct rte_eth_dev *dev) { struct sfc_repr *sr = sfc_repr_by_eth_dev(dev); struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); unsigned int i; sfcr_info(sr, "entry"); sfc_repr_lock(sr); switch (sr->state) { case SFC_ETHDEV_STARTED: sfc_repr_stop(dev); SFC_ASSERT(sr->state == SFC_ETHDEV_CONFIGURED); /* FALLTHROUGH */ case SFC_ETHDEV_CONFIGURED: sfc_repr_close(sr); SFC_ASSERT(sr->state == SFC_ETHDEV_INITIALIZED); /* FALLTHROUGH */ case SFC_ETHDEV_INITIALIZED: break; default: sfcr_err(sr, "unexpected adapter state %u on close", sr->state); break; } for (i = 0; i < dev->data->nb_rx_queues; i++) { sfc_repr_rx_queue_release(dev, i); dev->data->rx_queues[i] = NULL; } for (i = 0; i < dev->data->nb_tx_queues; i++) { sfc_repr_tx_queue_release(dev, i); dev->data->tx_queues[i] = NULL; } /* * Cleanup all resources. * Rollback primary process sfc_repr_eth_dev_init() below. */ (void)sfc_repr_proxy_del_port(srs->pf_port_id, srs->repr_id); sfc_mae_clear_switch_port(srs->switch_domain_id, srs->switch_port_id); dev->rx_pkt_burst = NULL; dev->tx_pkt_burst = NULL; dev->dev_ops = NULL; sfc_repr_unlock(sr); sfc_repr_lock_fini(sr); sfcr_info(sr, "done"); free(sr); return 0; } static int sfc_repr_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *stats) { union sfc_pkts_bytes queue_stats; uint16_t i; for (i = 0; i < dev->data->nb_rx_queues; i++) { struct sfc_repr_rxq *rxq = dev->data->rx_queues[i]; sfc_pkts_bytes_get(&rxq->stats.packets_bytes, &queue_stats); stats->ipackets += queue_stats.pkts; stats->ibytes += queue_stats.bytes; } for (i = 0; i < dev->data->nb_tx_queues; i++) { struct sfc_repr_txq *txq = dev->data->tx_queues[i]; sfc_pkts_bytes_get(&txq->stats.packets_bytes, &queue_stats); stats->opackets += queue_stats.pkts; stats->obytes += queue_stats.bytes; } return 0; } static int sfc_repr_flow_pick_transfer_proxy(struct rte_eth_dev *dev, uint16_t *transfer_proxy_port, struct rte_flow_error *error) { struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); return rte_flow_pick_transfer_proxy(srs->pf_port_id, transfer_proxy_port, error); } const struct rte_flow_ops sfc_repr_flow_ops = { .pick_transfer_proxy = sfc_repr_flow_pick_transfer_proxy, }; static int sfc_repr_dev_flow_ops_get(struct rte_eth_dev *dev __rte_unused, const struct rte_flow_ops **ops) { *ops = &sfc_repr_flow_ops; return 0; } static const struct eth_dev_ops sfc_repr_dev_ops = { .dev_configure = sfc_repr_dev_configure, .dev_start = sfc_repr_dev_start, .dev_stop = sfc_repr_dev_stop, .dev_close = sfc_repr_dev_close, .dev_infos_get = sfc_repr_dev_infos_get, .link_update = sfc_repr_dev_link_update, .stats_get = sfc_repr_stats_get, .rx_queue_setup = sfc_repr_rx_queue_setup, .rx_queue_release = sfc_repr_rx_queue_release, .tx_queue_setup = sfc_repr_tx_queue_setup, .tx_queue_release = sfc_repr_tx_queue_release, .flow_ops_get = sfc_repr_dev_flow_ops_get, }; struct sfc_repr_init_data { uint16_t pf_port_id; uint16_t switch_domain_id; efx_mport_sel_t mport_sel; efx_pcie_interface_t intf; uint16_t pf; uint16_t vf; }; static int sfc_repr_assign_mae_switch_port(uint16_t switch_domain_id, const struct sfc_mae_switch_port_request *req, uint16_t *switch_port_id) { int rc; rc = sfc_mae_assign_switch_port(switch_domain_id, req, switch_port_id); SFC_ASSERT(rc >= 0); return -rc; } static int sfc_repr_eth_dev_init(struct rte_eth_dev *dev, void *init_params) { const struct sfc_repr_init_data *repr_data = init_params; struct sfc_repr_shared *srs = sfc_repr_shared_by_eth_dev(dev); struct sfc_mae_switch_port_request switch_port_request; efx_mport_sel_t ethdev_mport_sel; struct sfc_repr *sr; int ret; /* * Currently there is no mport we can use for representor's * ethdev. Use an invalid one for now. This way representors * can be instantiated. */ efx_mae_mport_invalid(ðdev_mport_sel); memset(&switch_port_request, 0, sizeof(switch_port_request)); switch_port_request.type = SFC_MAE_SWITCH_PORT_REPRESENTOR; switch_port_request.ethdev_mportp = ðdev_mport_sel; switch_port_request.entity_mportp = &repr_data->mport_sel; switch_port_request.ethdev_port_id = dev->data->port_id; switch_port_request.port_data.repr.intf = repr_data->intf; switch_port_request.port_data.repr.pf = repr_data->pf; switch_port_request.port_data.repr.vf = repr_data->vf; ret = sfc_repr_assign_mae_switch_port(repr_data->switch_domain_id, &switch_port_request, &srs->switch_port_id); if (ret != 0) { SFC_GENERIC_LOG(ERR, "%s() failed to assign MAE switch port (domain id %u)", __func__, repr_data->switch_domain_id); goto fail_mae_assign_switch_port; } ret = sfc_repr_proxy_add_port(repr_data->pf_port_id, srs->switch_port_id, dev->data->port_id, &repr_data->mport_sel); if (ret != 0) { SFC_GENERIC_LOG(ERR, "%s() failed to add repr proxy port", __func__); SFC_ASSERT(ret > 0); ret = -ret; goto fail_create_port; } /* * Allocate process private data from heap, since it should not * be located in shared memory allocated using rte_malloc() API. */ sr = calloc(1, sizeof(*sr)); if (sr == NULL) { ret = -ENOMEM; goto fail_alloc_sr; } sfc_repr_lock_init(sr); sfc_repr_lock(sr); dev->process_private = sr; srs->pf_port_id = repr_data->pf_port_id; srs->repr_id = srs->switch_port_id; srs->switch_domain_id = repr_data->switch_domain_id; dev->data->dev_flags |= RTE_ETH_DEV_REPRESENTOR; dev->data->representor_id = srs->repr_id; dev->data->backer_port_id = srs->pf_port_id; dev->data->mac_addrs = rte_zmalloc("sfcr", RTE_ETHER_ADDR_LEN, 0); if (dev->data->mac_addrs == NULL) { ret = -ENOMEM; goto fail_mac_addrs; } dev->rx_pkt_burst = sfc_repr_rx_burst; dev->tx_pkt_burst = sfc_repr_tx_burst; dev->dev_ops = &sfc_repr_dev_ops; sr->state = SFC_ETHDEV_INITIALIZED; sfc_repr_unlock(sr); return 0; fail_mac_addrs: sfc_repr_unlock(sr); free(sr); fail_alloc_sr: (void)sfc_repr_proxy_del_port(repr_data->pf_port_id, srs->switch_port_id); fail_create_port: fail_mae_assign_switch_port: SFC_GENERIC_LOG(ERR, "%s() failed: %s", __func__, rte_strerror(-ret)); return ret; } int sfc_repr_create(struct rte_eth_dev *parent, struct sfc_repr_entity_info *entity, uint16_t switch_domain_id, const efx_mport_sel_t *mport_sel) { struct sfc_repr_init_data repr_data; char name[RTE_ETH_NAME_MAX_LEN]; int controller; int ret; int rc; struct rte_eth_dev *dev; controller = -1; rc = sfc_mae_switch_domain_get_controller(switch_domain_id, entity->intf, &controller); if (rc != 0) { SFC_GENERIC_LOG(ERR, "%s() failed to get DPDK controller for %d", __func__, entity->intf); return -rc; } switch (entity->type) { case RTE_ETH_REPRESENTOR_VF: ret = snprintf(name, sizeof(name), "net_%s_representor_c%upf%uvf%u", parent->device->name, controller, entity->pf, entity->vf); break; case RTE_ETH_REPRESENTOR_PF: ret = snprintf(name, sizeof(name), "net_%s_representor_c%upf%u", parent->device->name, controller, entity->pf); break; default: return -ENOTSUP; } if (ret >= (int)sizeof(name)) { SFC_GENERIC_LOG(ERR, "%s() failed name too long", __func__); return -ENAMETOOLONG; } dev = rte_eth_dev_allocated(name); if (dev == NULL) { memset(&repr_data, 0, sizeof(repr_data)); repr_data.pf_port_id = parent->data->port_id; repr_data.switch_domain_id = switch_domain_id; repr_data.mport_sel = *mport_sel; repr_data.intf = entity->intf; repr_data.pf = entity->pf; repr_data.vf = entity->vf; ret = rte_eth_dev_create(parent->device, name, sizeof(struct sfc_repr_shared), NULL, NULL, sfc_repr_eth_dev_init, &repr_data); if (ret != 0) { SFC_GENERIC_LOG(ERR, "%s() failed to create device", __func__); return ret; } } return 0; }