In the case where the device is created by the primary process. Currently, secondary process only contains the eth dev without being able to do any Rx/Tx.
When attaching the device from secondary process this patch adds queues info got from IPC massaging. Signed-off-by: Raslan Darawsheh <rasl...@mellanox.com> --- drivers/net/tap/Makefile | 2 + drivers/net/tap/rte_eth_tap.c | 105 +++++++++++++++++++++++++++++++++++++++++- 2 files changed, 106 insertions(+), 1 deletion(-) diff --git a/drivers/net/tap/Makefile b/drivers/net/tap/Makefile index ccc5c5f..913423c 100644 --- a/drivers/net/tap/Makefile +++ b/drivers/net/tap/Makefile @@ -27,6 +27,8 @@ LDLIBS += -lrte_ethdev -lrte_net -lrte_kvargs -lrte_hash LDLIBS += -lrte_bus_vdev CFLAGS += -DTAP_MAX_QUEUES=$(TAP_MAX_QUEUES) +CFLAGS += -DALLOW_EXPERIMENTAL_API + # # all source are stored in SRCS-y diff --git a/drivers/net/tap/rte_eth_tap.c b/drivers/net/tap/rte_eth_tap.c index 5531fe9..0f4c8d9 100644 --- a/drivers/net/tap/rte_eth_tap.c +++ b/drivers/net/tap/rte_eth_tap.c @@ -16,6 +16,8 @@ #include <rte_debug.h> #include <rte_ip.h> #include <rte_string_fns.h> +#include <rte_ethdev.h> +#include <rte_errno.h> #include <sys/types.h> #include <sys/stat.h> @@ -55,6 +57,9 @@ #define ETH_TAP_CMP_MAC_FMT "0123456789ABCDEFabcdef" #define ETH_TAP_MAC_ARG_FMT ETH_TAP_MAC_FIXED "|" ETH_TAP_USR_MAC_FMT +/* IPC key for communication of queue fds between processes. */ +#define TAP_MP_KEY "tap_mp_exchange_fds" + static struct rte_vdev_driver pmd_tap_drv; static struct rte_vdev_driver pmd_tun_drv; @@ -93,6 +98,15 @@ enum ioctl_mode { REMOTE_ONLY, }; +/* To communicate queue infos between processes */ +struct queues_fds { + char name[RTE_DEV_NAME_MAX_LEN]; + int num_rxq_fds; + int num_txq_fds; + int rxq_fds[RTE_PMD_TAP_MAX_QUEUES]; + int txq_fds[RTE_PMD_TAP_MAX_QUEUES]; +}; + static int tap_intr_handle_set(struct rte_eth_dev *dev, int set); /** @@ -1731,6 +1745,47 @@ rte_pmd_tun_probe(struct rte_vdev_device *dev) return ret; } +/* + * Send the queues fds from the primary process to secondary. + */ +static int +tap_exchange_queues_fds(const struct rte_mp_msg *mp_msg, const void *peer) +{ + struct rte_eth_dev *eth_dev; + struct rte_mp_msg mp_resp; + struct queues_fds *out = (struct queues_fds *)&mp_resp.param; + const struct queues_fds *in = (const struct queues_fds *)mp_msg->param; + uint16_t port_id; + int i, ret; + + TAP_LOG(DEBUG, "received request"); + strlcpy(out->name, in->name, sizeof(out->name)); + ret = rte_eth_dev_get_port_by_name(in->name, &port_id); + if (ret) { + TAP_LOG(ERR, "Failed to get dev %s", in->name); + return -1; + } + eth_dev = &rte_eth_devices[port_id]; + struct pmd_internals *pmd = eth_dev->data->dev_private; + + /* fill the queues fds data in the reply msg */ + strlcpy(mp_resp.name, TAP_MP_KEY, sizeof(mp_resp.name)); + out->num_rxq_fds = eth_dev->data->nb_rx_queues; + for (i = 0; i < eth_dev->data->nb_rx_queues; i++) + out->rxq_fds[i] = pmd->rxq[i].fd; + out->num_txq_fds = eth_dev->data->nb_tx_queues; + for (i = 0; i < eth_dev->data->nb_tx_queues; i++) + out->txq_fds[i] = pmd->txq[i].fd; + mp_resp.len_param = sizeof(*out); + mp_resp.num_fds = 0; + if (rte_mp_reply(&mp_resp, peer) < 0) { + TAP_LOG(ERR, "Failed to reply a fds request"); + return -1; + } + + return 0; +} + /* Open a TAP interface device. */ static int @@ -1744,6 +1799,7 @@ rte_pmd_tap_probe(struct rte_vdev_device *dev) char remote_iface[RTE_ETH_NAME_MAX_LEN]; struct ether_addr user_mac = { .addr_bytes = {0} }; struct rte_eth_dev *eth_dev; + int queue_id; strcpy(tuntap_name, "TAP"); @@ -1757,8 +1813,46 @@ rte_pmd_tap_probe(struct rte_vdev_device *dev) TAP_LOG(ERR, "Failed to probe %s", name); return -1; } - /* TODO: request info from primary to set up Rx and Tx */ eth_dev->dev_ops = &ops; + /* request a sync from the primary process to get queues fds */ + eth_dev->rx_pkt_burst = pmd_rx_burst; + eth_dev->tx_pkt_burst = pmd_tx_burst; + if (!rte_eal_primary_proc_alive(NULL)) { + TAP_LOG(ERR, "cannot initialize secondary process " + "without a primary one"); + return -1; + } + struct rte_mp_msg mp_req, *mp_rep; + struct rte_mp_reply mp_reply; + struct timespec ts = {.tv_sec = 5, .tv_nsec = 0}; + struct queues_fds *req = (struct queues_fds *)mp_req.param; + struct queues_fds *resp; + + strlcpy(req->name, name, sizeof(mp_req.name)); + strlcpy(mp_req.name, TAP_MP_KEY, sizeof(mp_req.name)); + mp_req.len_param = sizeof(*req); + /* request for sync from primary process to get queues fds. */ + if (rte_mp_request_sync(&mp_req, &mp_reply, &ts) == 0 && + mp_reply.nb_received == 1) { + mp_rep = &mp_reply.msgs[0]; + resp = (struct queues_fds *)mp_rep->param; + TAP_LOG(INFO, "Received fds for %d rx_queues and " + "%d tx_queues", resp->num_rxq_fds, + resp->num_txq_fds); + } else { + TAP_LOG(ERR, "Failed to request queues from primary"); + return -1; + } + + struct pmd_internals *pmd = eth_dev->data->dev_private; + for (queue_id = 0; queue_id < resp->num_rxq_fds; queue_id++) + pmd->rxq[queue_id].fd = resp->rxq_fds[queue_id]; + + for (queue_id = 0; queue_id < resp->num_txq_fds; queue_id++) + pmd->txq[queue_id].fd = resp->txq_fds[queue_id]; + + TAP_LOG(NOTICE, "Initializing secondary process pmd_tap for %s", + name); rte_eth_dev_probing_finish(eth_dev); return 0; } @@ -1806,6 +1900,14 @@ rte_pmd_tap_probe(struct rte_vdev_device *dev) TAP_LOG(NOTICE, "Initializing pmd_tap for %s as %s", name, tap_name); + /* register for mp communication between secondary and primary */ + if (rte_mp_action_register(TAP_MP_KEY, tap_exchange_queues_fds) && + rte_errno != EEXIST) { + TAP_LOG(ERR, "%s : %s fail to register MP action : %s", + tuntap_name, name, strerror(errno)); + return -1; + } + ret = eth_dev_tap_create(dev, tap_name, remote_iface, &user_mac, ETH_TUNTAP_TYPE_TAP); @@ -1813,6 +1915,7 @@ rte_pmd_tap_probe(struct rte_vdev_device *dev) if (ret == -1) { TAP_LOG(ERR, "Failed to create pmd for %s as %s", name, tap_name); + rte_mp_action_unregister(TAP_MP_KEY); tap_unit--; /* Restore the unit number */ } rte_kvargs_free(kvlist); -- 2.7.4