aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--host/lib/include/uhdlib/transport/dpdk_common.hpp81
-rw-r--r--host/lib/include/uhdlib/transport/dpdk_zero_copy.hpp45
-rw-r--r--host/lib/include/uhdlib/transport/uhd-dpdk.h280
-rw-r--r--host/lib/transport/dpdk_zero_copy.cpp505
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk.c467
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_ctx.h268
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_driver.c624
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_driver.h33
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_fops.c353
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_fops.h23
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_udp.c525
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_udp.h40
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_wait.c114
-rw-r--r--host/lib/transport/uhd-dpdk/uhd_dpdk_wait.h62
14 files changed, 0 insertions, 3420 deletions
diff --git a/host/lib/include/uhdlib/transport/dpdk_common.hpp b/host/lib/include/uhdlib/transport/dpdk_common.hpp
deleted file mode 100644
index 0db51e80f..000000000
--- a/host/lib/include/uhdlib/transport/dpdk_common.hpp
+++ /dev/null
@@ -1,81 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-
-#ifndef INCLUDED_DPDK_COMMON_HPP
-#define INCLUDED_DPDK_COMMON_HPP
-
-#include <uhd/config.hpp>
-#include <uhd/types/device_addr.hpp>
-#include <uhd/utils/static.hpp>
-#include <uhd/utils/noncopyable.hpp>
-#include <array>
-#include <atomic>
-#include <mutex>
-#include <string>
-
-namespace uhd { namespace transport {
-
-class uhd_dpdk_ctx : uhd::noncopyable {
-public:
- UHD_SINGLETON_FCN(uhd_dpdk_ctx, get);
-
- ~uhd_dpdk_ctx(void);
-
- /*!
- * Initialize uhd-dpdk (and do only once)
- * \param user_args User args passed in to override config files
- */
- void init(const device_addr_t &user_args);
-
- /*!
- * Get MTU of NICs used by DPDK
- *
- * \return Number of Bytes in MTU
- */
- size_t get_mtu(void) const;
-
- /*!
- * Get port ID from provided MAC address
- * \param mac_addr MAC address
- * \param port_id Int to write ID of port corresponding to MAC address
- * \return 0 if match found, else no match
- */
- int get_port_id(std::array<uint8_t, 6> mac_addr, unsigned int &port_id) const;
-
- /*!
- * Get port ID for routing packet destined for given address
- * \param addr Destination address
- * \return port ID from routing table
- */
- int get_route(const std::string &addr) const;
-
- /*!
- * Set IPv4 address and subnet mask of given NIC port
- * Not thread-safe. Should only be written before ports are in use.
- * \param port_id NIC port ID
- * \param ipv4_addr IPv4 address to write
- * \param netmask Subnet mask identifying network number in ipv4_addr
- * \return 0 if successful, else error
- */
- int set_ipv4_addr(unsigned int port_id, uint32_t ipv4_addr, uint32_t netmask);
-
- /*!
- * \return whether init() has been called
- */
- bool is_init_done(void) const;
-
-private:
- uhd_dpdk_ctx(void);
-
- size_t _mtu;
- std::mutex _init_mutex;
- std::atomic<bool> _init_done;
- uhd::dict<uint32_t, unsigned int> _routes;
-};
-
-}} // namespace uhd::transport
-
-#endif /* INCLUDED_DPDK_COMMON_HPP */
diff --git a/host/lib/include/uhdlib/transport/dpdk_zero_copy.hpp b/host/lib/include/uhdlib/transport/dpdk_zero_copy.hpp
deleted file mode 100644
index 3497f0598..000000000
--- a/host/lib/include/uhdlib/transport/dpdk_zero_copy.hpp
+++ /dev/null
@@ -1,45 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-
-#ifndef DPDK_ZERO_COPY_HPP
-#define DPDK_ZERO_COPY_HPP
-
-#include <uhdlib/transport/dpdk_common.hpp>
-#include <uhd/types/device_addr.hpp>
-#include <uhd/transport/zero_copy.hpp>
-#include <memory>
-#include <string>
-
-
-namespace uhd { namespace transport {
-
-/*!
- * A zero copy transport interface to the dpdk DMA library.
- */
-class dpdk_zero_copy : public virtual zero_copy_if {
-public:
- typedef std::shared_ptr<dpdk_zero_copy> sptr;
-
- static sptr make(
- const struct uhd_dpdk_ctx &ctx,
- const unsigned int dpdk_port_id,
- const std::string &addr,
- const std::string &remote_port,
- const std::string &local_port, /* 0 = auto-assign */
- const zero_copy_xport_params &default_buff_args,
- const device_addr_t &hints
- );
-
- virtual uint16_t get_local_port(void) const = 0;
-
- virtual std::string get_local_addr(void) const = 0;
-
- virtual uint32_t get_drop_count(void) const = 0;
-};
-
-}} // namespace uhd::transport
-
-#endif /* DPDK_ZERO_COPY_HPP */
diff --git a/host/lib/include/uhdlib/transport/uhd-dpdk.h b/host/lib/include/uhdlib/transport/uhd-dpdk.h
deleted file mode 100644
index ae7d31383..000000000
--- a/host/lib/include/uhdlib/transport/uhd-dpdk.h
+++ /dev/null
@@ -1,280 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#ifndef _UHD_DPDK_H_
-#define _UHD_DPDK_H_
-
-#ifdef __cplusplus
-extern "C" {
-#endif
-
-#include <stdbool.h>
-#include <stdint.h>
-#include <rte_mbuf.h>
-
-/* For MAC address */
-struct eth_addr {
- uint8_t addr[6];
-};
-
-/* Opaque type representing a socket
- * May NOT be shared between threads
- */
-struct uhd_dpdk_socket;
-
-/* Only support UDP sockets currently */
-enum uhd_dpdk_sock_type {
- UHD_DPDK_SOCK_UDP = 0,
- UHD_DPDK_SOCK_TYPE_COUNT
-};
-
-/**
- * Init UHD-DPDK environment, including DPDK's EAL.
- * This will make available information about the DPDK-assigned NIC devices.
- *
- * @param argc passed directly to rte_eal_init()
- * @param argv passed directly to rte_eal_init()
- *
- * @return Returns negative error code if there were issues, else 0
- */
-int uhd_dpdk_init(int argc, const char **argv);
-
-/**
- * Start UHD-DPDK networking stack. Bring ports up (link UP).
- * uhd_dpdk_init() must be called first.
- *
- * Offload capabilities will be used if available
- *
- * @param num_ports number of network interfaces to map
- * @param port_thread_mapping array of num_ports entries specifying which thread
- * will drive the I/O for a given port (determined by array index)
- * @param num_mbufs number of packets in each packet buffer pool (multiplied by num_ports)
- * There is one RX and one TX buffer pool per CPU socket
- * @param mbuf_cache_size Number of packet buffers to put in core-local cache
- * @param mtu Maximum frame size
- *
- * @return Returns negative error code if there were issues, else 0
- */
-int uhd_dpdk_start(unsigned int num_ports, int *port_thread_mapping,
- int num_mbufs, int mbuf_cache_size, int mtu);
-
-/**
- * @return Returns number of ports registered to DPDK.
- * Returns negative error value if uhd-dpdk hasn't been init'd
- */
-int uhd_dpdk_port_count(void);
-
-/**
- * @return Returns 0 if link is down, 1 if link is up, and negative error code
- * if error occurred.
- */
-int uhd_dpdk_port_link_status(unsigned int portid);
-
-/**
- * @return Returns Ethernet MAC address of requested port
- *
- * @param portid ID number of network interface
- */
-struct eth_addr uhd_dpdk_get_eth_addr(unsigned int portid);
-
-/**
- * Get IPv4 address of requested port
- *
- * @param portid ID number of network interface
- * @param ipv4_addr pointer to uint32_t where ipv4 address is stored
- * Must be non-NULL
- * @param netmask pointer to uint32_t where netmask is stored
- * May be left NULL
- *
- * @return Returns
- * 0 = success
- * nonzero = failure
- */
-int uhd_dpdk_get_ipv4_addr(unsigned int portid, uint32_t *ipv4_addr, uint32_t *netmask);
-
-/**
- * Sets IPv4 address of requested port
- *
- * @param portid ID number of network interface
- * @param ipv4_addr must be in network format
- * @param netmask must be in network format
- *
- * @return Return values:
- * 0 = success
- * nonzero = failure
- */
-int uhd_dpdk_set_ipv4_addr(unsigned int portid, uint32_t ipv4_addr, uint32_t netmask);
-
-/**
- * Create new socket of type sock_type on port portid
- * Copies needed info from sockarg
- * Do NOT share struct uhd_dpdk_socket between threads!
- *
- * @param portid ID number of network interface
- * @param t Type of socket to create (only UDP supported currently)
- * @param sockarg Pointer to arguments for corresponding socket type
- *
- * @return Returns pointer to socket structure on success, else NULL
- */
-struct uhd_dpdk_socket* uhd_dpdk_sock_open(unsigned int portid,
- enum uhd_dpdk_sock_type t, void *sockarg);
-
-/**
- * Close socket created by uhd_dpdk_sock_open
- *
- * Note: Outstanding packet buffers must still be freed by user
- *
- * @param sock Socket to close
- *
- * @return Returns
- * 0 = success
- * nonzero = failure
- */
-int uhd_dpdk_sock_close(struct uhd_dpdk_socket *sock);
-
-/**
- * Arguments for a UDP socket
- * All address/port data should be provided in network format
- */
-struct uhd_dpdk_sockarg_udp {
- /*! True for TX socket, false for RX socket */
- bool is_tx;
- /*! True to filter broadcast packets, else recv */
- bool filter_bcast;
- /*! Local udp port. This is dst_port for RX, src_port for TX */
- uint16_t local_port;
- /*! Remote udp port. This is dst_port for TX */
- uint16_t remote_port;
- /*! IPv4 address for destination (TX) */
- uint32_t dst_addr;
- /*! Number of buffers in ring */
- size_t num_bufs;
-};
-
-/**
- * Brings all ports and threads down in preparation for a clean program exit
- *
- * All sockets will need to be closed by the user for a thread to terminate in
- * this function.
- */
-int uhd_dpdk_destroy(void);
-
-/**
- * Requests num_bufs buffers from sock. Places pointers to buffers in bufs table.
- *
- * @param sock pointer to socket
- * @param bufs pointer to array of buffers (to store buffer locations)
- * @param num_bufs number of buffers requested
- * @param timeout Time (in us) to wait for a buffer
- *
- * @return Returns number of buffers retrieved or negative error code
- */
-int uhd_dpdk_request_tx_bufs(struct uhd_dpdk_socket *sock, struct rte_mbuf **bufs, unsigned int num_bufs, int timeout);
-
-/**
- * Enqueues num_bufs buffers in sock TX buffer. Uses pointers to buffers in bufs table.
- *
- * @param sock pointer to socket
- * @param bufs pointer to array of buffers (to retrieve buffer locations)
- * @param num_bufs number of buffers requested
- *
- * @return Returns number of buffers enqueued or negative error code
- */
-int uhd_dpdk_send(struct uhd_dpdk_socket *sock, struct rte_mbuf **bufs, unsigned int num_bufs);
-
-/**
- * Dequeues num_bufs buffers from sock RX buffer. Uses pointers to buffers in bufs table.
- *
- * @param sock pointer to socket
- * @param bufs pointer to array of buffers (to store buffer locations)
- * @param num_bufs number of buffers requested
- * @param timeout Time (in us) to wait for a packet
- *
- * @return Returns number of buffers dequeued or negative error code
- *
- * NOTE: MUST free buffers with uhd_dpdk_free_buf once finished
- */
-int uhd_dpdk_recv(struct uhd_dpdk_socket *sock, struct rte_mbuf **bufs,
- unsigned int num_bufs, int timeout);
-
-/**
- * Frees buffer previously received from uhd_dpdk_recv
- * (or unused ones from uhd_dpdk_request_tx_bufs)
- *
- * @param buf pointer to packet buffer
- */
-void uhd_dpdk_free_buf(struct rte_mbuf *buf);
-
-/**
- * Returns pointer to start of data segment of packet buffer
- *
- * @param sock Socket associated with packet buffer
- * @param buf pointer to packet buffer
- */
-void * uhd_dpdk_buf_to_data(struct uhd_dpdk_socket *sock, struct rte_mbuf *buf);
-
-/**
- * Returns size of data segment of packet buffer (in bytes)
- *
- * This is protocol-dependent. A UDP socket will return the UDP payload size.
- *
- * @param sock Socket associated with packet buffer
- * @param buf pointer to packet buffer
- *
- * @return Return 0 for success, else failed
- */
-int uhd_dpdk_get_len(struct uhd_dpdk_socket *sock, struct rte_mbuf *buf);
-
-/**
- * Get IPv4 address of sender (for UDP RX socket)
- *
- * @param sock Socket associated with packet buffer
- * @param buf pointer to packet buffer
- * @param ipv4_addr pointer to buffer where ipv4 address will be written
- *
- * @return Return 0 for success, else failed
- */
-int uhd_dpdk_get_src_ipv4(struct uhd_dpdk_socket *sock, struct rte_mbuf *buf,
- uint32_t *ipv4_addr);
-
-/**
- * Get info (local port, remote port, dst addr, etc.) for UDP socket
- *
- * @param sock Socket to get information from
- * @param sockarg Pointer to location where information will be stored
- *
- * @return Return 0 for success, else failed
- */
-int uhd_dpdk_udp_get_info(struct uhd_dpdk_socket *sock, struct uhd_dpdk_sockarg_udp *sockarg);
-
-
-/***********************************************
- * Statistics
- ***********************************************/
-/**
- * Get dropped packet count of provided socket
- *
- * @param sock Socket to get information from
- * @param count Pointer to location where information will be stored
- *
- * @return Return 0 for success, else failed
- */
-int uhd_dpdk_get_drop_count(struct uhd_dpdk_socket *sock, size_t *count);
-
-/**
- * Get transferred packet count of provided socket
- * Currently only tracks received packets (i.e. for RX)
- *
- * @param sock Socket to get information from
- * @param count Pointer to location where information will be stored
- *
- * @return Return 0 for success, else failed
- */
-int uhd_dpdk_get_xfer_count(struct uhd_dpdk_socket *sock, size_t *count);
-
-#ifdef __cplusplus
-}
-#endif
-#endif /* _UHD_DPDK_H_ */
diff --git a/host/lib/transport/dpdk_zero_copy.cpp b/host/lib/transport/dpdk_zero_copy.cpp
deleted file mode 100644
index 99df3b059..000000000
--- a/host/lib/transport/dpdk_zero_copy.cpp
+++ /dev/null
@@ -1,505 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-
-#include <uhd/config.hpp>
-#include <uhd/utils/log.hpp>
-#include <uhdlib/transport/uhd-dpdk.h>
-#include <uhdlib/transport/dpdk_zero_copy.hpp>
-#include <uhdlib/utils/prefs.hpp>
-#include <boost/algorithm/string.hpp>
-#include <boost/format.hpp>
-#include <memory>
-#include <stack>
-#include <sys/syslog.h>
-#include <arpa/inet.h>
-
-namespace uhd { namespace transport {
-
-namespace {
-constexpr uint64_t USEC = 1000000;
-constexpr size_t DEFAULT_FRAME_SIZE = 8000;
-constexpr int DEFAULT_NUM_MBUFS = 4095;
-constexpr int DEFAULT_MBUF_CACHE_SIZE = 315;
-constexpr size_t UHD_DPDK_HEADERS_SIZE = 14 + 20 + 8; // Ethernet + IPv4 + UDP
-
-inline char * eal_add_opt(std::vector<const char*> &argv, size_t n,
- char *dst, const char *opt, const char *arg)
-{
- char *ptr = dst;
- strncpy(ptr, opt, n);
- argv.push_back(ptr);
- ptr += strlen(opt) + 1;
- n -= ptr - dst;
- strncpy(ptr, arg, n);
- argv.push_back(ptr);
- ptr += strlen(arg) + 1;
- return ptr;
-}
-
-inline void uhd_dpdk_eal_init(const device_addr_t &eal_args)
-{
- /* Build up argc and argv */
- std::vector<const char *> argv;
- argv.push_back("uhd-dpdk");
- auto args = new std::array<char, 4096>();
- char *opt = args->data();
- char *end = args->data() + args->size();
- for (std::string &key : eal_args.keys()) {
- std::string val = eal_args[key];
- if (key == "dpdk-coremask") {
- opt = eal_add_opt(argv, end - opt, opt, "-c",
- val.c_str());
- } else if (key == "dpdk-corelist") {
- /* NOTE: This arg may have commas, so limited to config file */
- opt = eal_add_opt(argv, end - opt, opt, "-l",
- val.c_str());
- } else if (key == "dpdk-coremap") {
- opt = eal_add_opt(argv, end - opt, opt, "--lcores",
- val.c_str());
- } else if (key == "dpdk-master-lcore") {
- opt = eal_add_opt(argv, end - opt, opt, "--master-lcore",
- val.c_str());
- } else if (key == "dpdk-pci-blacklist") {
- opt = eal_add_opt(argv, end - opt, opt, "-b",
- val.c_str());
- } else if (key == "dpdk-pci-whitelist") {
- opt = eal_add_opt(argv, end - opt, opt, "-w",
- val.c_str());
- } else if (key == "dpdk-log-level") {
- opt = eal_add_opt(argv, end - opt, opt, "--log-level",
- val.c_str());
- } else if (key == "dpdk-huge-dir") {
- opt = eal_add_opt(argv, end - opt, opt, "--huge-dir",
- val.c_str());
- } else if (key == "dpdk-file-prefix") {
- opt = eal_add_opt(argv, end - opt, opt, "--file-prefix",
- val.c_str());
- } else if (key == "dpdk-driver") {
- opt = eal_add_opt(argv, end - opt, opt, "-d",
- val.c_str());
- }
- /* TODO: Change where log goes?
- int rte_openlog_stream( FILE * f)
- */
- }
- /* Init DPDK's EAL */
- uhd_dpdk_init(argv.size(), argv.data());
- delete args;
-}
-
-inline std::string eth_addr_to_string(struct eth_addr mac_addr)
-{
- auto mac_stream = boost::format("%02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx");
- mac_stream % (uint32_t) mac_addr.addr[0] % (uint32_t) mac_addr.addr[1]
- % (uint32_t) mac_addr.addr[2] % (uint32_t) mac_addr.addr[3]
- % (uint32_t) mac_addr.addr[4] % (uint32_t) mac_addr.addr[5];
- return mac_stream.str();
-}
-
-inline void separate_ipv4_addr(const std::string ipv4,
- uint32_t &ipv4_addr, uint32_t &netmask)
-{
- std::vector<std::string> result;
- boost::algorithm::split(result, ipv4,
- [](const char &in) {return in == '/';}, boost::token_compress_on);
- UHD_ASSERT_THROW(result.size() == 2);
- ipv4_addr = (uint32_t) inet_addr(result[0].c_str());
- int netbits = std::atoi(result[1].c_str());
- netmask = htonl(0xffffffff << (32-netbits));
-}
-} // namespace
-
-uhd_dpdk_ctx::uhd_dpdk_ctx(void) : _init_done(false) {}
-
-uhd_dpdk_ctx::~uhd_dpdk_ctx(void) {}
-
-/* Initialize uhd-dpdk (and do only once) */
-void uhd_dpdk_ctx::init(const device_addr_t &user_args)
-{
- std::lock_guard<std::mutex> lock(_init_mutex);
- if (!_init_done) {
- /* Gather global config, build args for EAL, and init UHD-DPDK */
- const device_addr_t dpdk_args = uhd::prefs::get_dpdk_args(user_args);
- UHD_LOG_TRACE("DPDK", "Configuration:" << std::endl
- << dpdk_args.to_pp_string());
- uhd_dpdk_eal_init(dpdk_args);
-
- _mtu = dpdk_args.has_key("dpdk-mtu")
- ? dpdk_args.cast<size_t>("dpdk-mtu", 0)
- : DEFAULT_FRAME_SIZE;
- const int num_mbufs = dpdk_args.has_key("dpdk-num-mbufs")
- ? dpdk_args.cast<int>("dpdk-num-mbufs", 0)
- : DEFAULT_NUM_MBUFS;
- const int mbuf_cache_size = dpdk_args.has_key("dpdk-mbuf-cache-size")
- ? dpdk_args.cast<int>("dpdk-mbuf-cache-size", 0)
- : DEFAULT_MBUF_CACHE_SIZE;
-
- /* Get configuration for all the NIC ports */
- device_addrs_t args = separate_device_addr(user_args);
- int num_ports = uhd_dpdk_port_count();
- std::vector<int> io_cpu_map(num_ports);
- device_addrs_t nics(num_ports);
- for (ssize_t i = 0; i < num_ports; i++) {
- struct eth_addr mac_addr = uhd_dpdk_get_eth_addr(i);
- nics[i]["dpdk-mac"] = eth_addr_to_string(mac_addr);
- for (const auto &arg: args) {
- if (arg.has_key("dpdk-mac")
- && arg["dpdk-mac"] == nics[i]["dpdk-mac"]) {
- for (const auto& key: arg.keys()) {
- nics[i][key] = arg[key];
- }
- break;
- }
- }
- nics[i] = uhd::prefs::get_dpdk_nic_args(nics[i]);
- if (nics[i].has_key("dpdk-ipv4")
- && nics[i].has_key("dpdk-io-cpu")) {
- uint32_t ipv4_addr, netmask;
- io_cpu_map[i] = std::atoi(nics[i]["dpdk-io-cpu"].c_str());
- separate_ipv4_addr(nics[i]["dpdk-ipv4"], ipv4_addr, netmask);
- uhd_dpdk_set_ipv4_addr((unsigned int) i, ipv4_addr, netmask);
- } else {
- /* Not enough configuration to use NIC */
- io_cpu_map[i] = -1;
- }
- UHD_LOG_TRACE("DPDK", "Found NIC(" << i << "):" << std::endl
- << nics[i].to_pp_string());
- }
- uhd_dpdk_start(num_ports, io_cpu_map.data(), num_mbufs,
- mbuf_cache_size, _mtu);
- _init_done = true;
- }
-}
-
-size_t uhd_dpdk_ctx::get_mtu(void) const
-{
- UHD_ASSERT_THROW(is_init_done());
- return _mtu;
-}
-
-int uhd_dpdk_ctx::get_port_id(std::array<uint8_t, 6> mac_addr,
- unsigned int &port_id) const
-{
- UHD_ASSERT_THROW(is_init_done());
- int num_ports = uhd_dpdk_port_count();
- for (int i = 0; i < num_ports; i++) {
- struct eth_addr port_mac_addr = uhd_dpdk_get_eth_addr((unsigned int) i);
- for (int j = 0; j < 6; j++) {
- if (mac_addr[j] != port_mac_addr.addr[j]) {
- break;
- }
- if (j == 5) {
- port_id = (unsigned int) i;
- return 0;
- }
- }
- }
- return -1;
-}
-
-int uhd_dpdk_ctx::get_route(const std::string &addr) const
-{
- const uint32_t dst_ipv4 = (uint32_t) inet_addr(addr.c_str());
- const unsigned int num_ports = uhd_dpdk_port_count();
- for (unsigned int port = 0; port < num_ports; port++) {
- uint32_t src_ipv4;
- uint32_t netmask;
- if (uhd_dpdk_port_link_status(port) < 1)
- continue;
- uhd_dpdk_get_ipv4_addr(port, &src_ipv4, &netmask);
- if ((src_ipv4 & netmask) == (dst_ipv4 & netmask)) {
- return (int) port;
- }
- }
- return -ENODEV;
-}
-
-int uhd_dpdk_ctx::set_ipv4_addr(unsigned int port_id, uint32_t ipv4_addr,
- uint32_t netmask)
-{
- return uhd_dpdk_set_ipv4_addr(port_id, ipv4_addr, netmask);
-}
-
-bool uhd_dpdk_ctx::is_init_done(void) const
-{
- return _init_done.load();
-}
-
-
-class dpdk_zero_copy_msb : public managed_send_buffer {
-public:
- dpdk_zero_copy_msb(struct uhd_dpdk_socket *sock,
- std::stack<dpdk_zero_copy_msb *, std::vector<dpdk_zero_copy_msb *>> &free_bufs,
- size_t frame_size)
- : _sock(sock), _buf(nullptr), _free_bufs(free_bufs),
- _frame_size(frame_size) {};
-
- ~dpdk_zero_copy_msb(void) {}
-
- void release(void)
- {
- if (_buf) {
- _buf->pkt_len = _length;
- _buf->data_len = _length;
- int num_tx = uhd_dpdk_send(_sock, &_buf, 1);
- if (num_tx == 0) {
- /* Drop packet and free buffer (do not share sockets!) */
- UHD_LOG_ERROR("DPDK", "Invalid shared socket usage detected. Dropping packet...");
- uhd_dpdk_free_buf(_buf);
- }
- // Push back into pool
- _free_bufs.push(this);
- }
- }
-
- sptr get_new(double timeout)
- {
- int bufs = uhd_dpdk_request_tx_bufs(_sock, &_buf, 1, timeout);
- if (bufs != 1 || !_buf)
- return sptr();
-
- return make(this, uhd_dpdk_buf_to_data(_sock, _buf),
- _frame_size);
- }
-
-private:
- struct uhd_dpdk_socket *_sock;
- struct rte_mbuf *_buf;
- std::stack<dpdk_zero_copy_msb *, std::vector<dpdk_zero_copy_msb *>> &_free_bufs;
- size_t _frame_size;
-};
-
-class dpdk_zero_copy_mrb : public managed_recv_buffer {
-public:
- dpdk_zero_copy_mrb(struct uhd_dpdk_socket *sock,
- std::stack<dpdk_zero_copy_mrb*, std::vector<dpdk_zero_copy_mrb*>> &free_bufs)
- : _sock(sock), _buf(nullptr), _free_bufs(free_bufs) {};
- ~dpdk_zero_copy_mrb(void) {}
-
- void release(void)
- {
- if (_buf) {
- uhd_dpdk_free_buf(_buf);
- _free_bufs.push(this);
- }
- }
-
- sptr get_new(double timeout)
- {
- int bufs = uhd_dpdk_recv(_sock, &_buf, 1, (int) (timeout*USEC));
- if (bufs != 1 || _buf == nullptr) {
- // Push back into pool if we didn't get a real buffer
- _free_bufs.push(this);
- return sptr();
- }
-
- return make(this, uhd_dpdk_buf_to_data(_sock, _buf),
- uhd_dpdk_get_len(_sock, _buf));
- }
-
-private:
- struct uhd_dpdk_socket *_sock;
- struct rte_mbuf *_buf;
- std::stack<dpdk_zero_copy_mrb*, std::vector<dpdk_zero_copy_mrb*>> &_free_bufs;
-};
-
-class dpdk_zero_copy_impl : public dpdk_zero_copy {
-public:
-
- dpdk_zero_copy_impl(const struct uhd_dpdk_ctx &ctx,
- const unsigned int dpdk_port_id,
- const std::string &addr,
- const std::string &remote_port,
- const std::string &local_port,
- const zero_copy_xport_params& xport_params)
- : _num_send_frames(xport_params.num_send_frames),
- _send_frame_size(xport_params.send_frame_size),
- _num_recv_frames(xport_params.num_recv_frames),
- _recv_frame_size(xport_params.recv_frame_size),
- _port_id(dpdk_port_id),
- _rx_empty_count(0),
- _tx_empty_count(0)
- {
- UHD_ASSERT_THROW(xport_params.recv_frame_size > 0);
- UHD_ASSERT_THROW(xport_params.send_frame_size > 0);
- UHD_ASSERT_THROW(xport_params.num_send_frames > 0);
- UHD_ASSERT_THROW(xport_params.num_recv_frames > 0);
-
- UHD_ASSERT_THROW(ctx.is_init_done());
- UHD_ASSERT_THROW(xport_params.recv_frame_size < ctx.get_mtu() - UHD_DPDK_HEADERS_SIZE);
- UHD_ASSERT_THROW(xport_params.send_frame_size < ctx.get_mtu() - UHD_DPDK_HEADERS_SIZE);
-
- const int num_ports = uhd_dpdk_port_count();
- UHD_ASSERT_THROW(num_ports > 0);
- UHD_ASSERT_THROW(dpdk_port_id < (unsigned int) num_ports);
-
- // Convert ipv4 addr from string to uint32_t, network format
- uint32_t dst_ipv4 = (uint32_t) inet_addr(addr.c_str());
- // Convert port from string to uint16_t, network format
- uint16_t dst_port = htons(std::stoi(remote_port, NULL, 0));
- uint16_t src_port = htons(std::stoi(local_port, NULL, 0));
-
- // Create RX socket first
- struct uhd_dpdk_sockarg_udp sockarg = {
- .is_tx = false,
- .filter_bcast = true,
- .local_port = src_port,
- .remote_port = dst_port,
- .dst_addr = dst_ipv4,
- .num_bufs = _num_recv_frames
- };
- _rx_sock = uhd_dpdk_sock_open(dpdk_port_id, UHD_DPDK_SOCK_UDP, &sockarg);
- UHD_ASSERT_THROW(_rx_sock != nullptr);
-
- // Backfill the local port, in case it was auto-assigned
- uhd_dpdk_udp_get_info(_rx_sock, &sockarg);
-
- sockarg.is_tx = true;
- sockarg.num_bufs = _num_send_frames;
- sockarg.remote_port = dst_port;
- sockarg.dst_addr = dst_ipv4;
- _tx_sock = uhd_dpdk_sock_open(dpdk_port_id, UHD_DPDK_SOCK_UDP, &sockarg);
- UHD_ASSERT_THROW(_tx_sock != nullptr);
-
- // Create managed_buffer containers
- for (size_t i = 0; i < _num_recv_frames; i++) {
- _mrb_pool.push(new dpdk_zero_copy_mrb(_rx_sock, _mrb_pool));
- }
- for (size_t i = 0; i < _num_send_frames; i++) {
- _msb_pool.push(new dpdk_zero_copy_msb(_tx_sock, _msb_pool, _send_frame_size));
- }
-
- UHD_LOG_TRACE("DPDK", "Created transports between " << addr << ":"
- << remote_port << " and NIC(" << dpdk_port_id
- << "):" << ntohs(sockarg.local_port));
- }
-
- ~dpdk_zero_copy_impl(void)
- {
- struct uhd_dpdk_sockarg_udp sockarg;
- size_t count;
- uhd_dpdk_udp_get_info(_rx_sock, &sockarg);
- uhd_dpdk_get_drop_count(_rx_sock, &count);
- UHD_LOG_TRACE("DPDK", "Closing transports between " << sockarg.dst_addr << ":"
- << ntohs(sockarg.remote_port) << " and local:"
- << ntohs(sockarg.local_port));
- UHD_LOG_TRACE("DPDK", "(" << ntohs(sockarg.remote_port) << "," << ntohs(sockarg.local_port) << ") "
- << " Dropped "<< count << " packets");
- uhd_dpdk_get_xfer_count(_rx_sock, &count);
- UHD_LOG_TRACE("DPDK", "(" << ntohs(sockarg.remote_port) << "," << ntohs(sockarg.local_port) << ") "
- << " Received "<< count << " packets");
- UHD_LOG_TRACE("DPDK", "(" << ntohs(sockarg.remote_port) << "," << ntohs(sockarg.local_port) << ") "
- << "RX empty count is " << _rx_empty_count);
- UHD_LOG_TRACE("DPDK", "(" << ntohs(sockarg.remote_port) << "," << ntohs(sockarg.local_port) << ") "
- << "TX empty count is " << _tx_empty_count);
- uhd_dpdk_sock_close(_rx_sock);
- uhd_dpdk_sock_close(_tx_sock);
- }
-
- managed_recv_buffer::sptr get_recv_buff(double timeout = 0.1)
- {
- if (_mrb_pool.empty()) {
- _rx_empty_count++;
- return managed_recv_buffer::sptr();
- }
-
- dpdk_zero_copy_mrb *mrb = _mrb_pool.top();
- _mrb_pool.pop();
- managed_recv_buffer::sptr buff = mrb->get_new(timeout);
- if (!buff)
- _rx_empty_count++;
- return buff;
- }
-
- size_t get_num_recv_frames(void) const
- {
- return _num_recv_frames;
- }
-
- size_t get_recv_frame_size(void) const
- {
- return _recv_frame_size;
- }
-
- managed_send_buffer::sptr get_send_buff(double timeout = 0.1)
- {
- if (_msb_pool.empty()) {
- _tx_empty_count++;
- return managed_send_buffer::sptr();
- }
-
- dpdk_zero_copy_msb *msb = _msb_pool.top();
- _msb_pool.pop();
- managed_send_buffer::sptr buff = msb->get_new(timeout);
- if (!buff)
- _tx_empty_count++;
- return buff;
- }
-
- size_t get_num_send_frames(void) const
- {
- return _num_send_frames;
- }
-
- size_t get_send_frame_size(void) const
- {
- return _send_frame_size;
- }
-
- uint16_t get_local_port(void) const
- {
- struct uhd_dpdk_sockarg_udp sockarg;
- int status = uhd_dpdk_udp_get_info(_rx_sock, &sockarg);
- UHD_ASSERT_THROW(status == 0);
- return ntohs(sockarg.local_port);
- }
-
- std::string get_local_addr(void) const
- {
- struct in_addr ipv4_addr;
- int status = uhd_dpdk_get_ipv4_addr(_port_id, &ipv4_addr.s_addr, NULL);
- UHD_ASSERT_THROW(status == 0);
- char addr_str[INET_ADDRSTRLEN];
- inet_ntop(AF_INET, &ipv4_addr, addr_str, sizeof(addr_str));
- return std::string(addr_str);
- }
-
- uint32_t get_drop_count(void) const
- {
- size_t drop_count = 0;
- uhd_dpdk_get_drop_count(_rx_sock, &drop_count);
- return drop_count;
- }
-private:
- struct uhd_dpdk_socket *_rx_sock;
- struct uhd_dpdk_socket *_tx_sock;
- const size_t _num_send_frames;
- const size_t _send_frame_size;
- const size_t _num_recv_frames;
- const size_t _recv_frame_size;
- const unsigned int _port_id;
- unsigned int _rx_empty_count;
- unsigned int _tx_empty_count;
-
- std::stack<dpdk_zero_copy_mrb *, std::vector<dpdk_zero_copy_mrb *>> _mrb_pool;
- std::stack<dpdk_zero_copy_msb *, std::vector<dpdk_zero_copy_msb *>> _msb_pool;
-};
-
-dpdk_zero_copy::sptr dpdk_zero_copy::make(
- const struct uhd_dpdk_ctx &ctx,
- const unsigned int dpdk_port_id,
- const std::string &addr,
- const std::string &remote_port,
- const std::string &local_port,
- const zero_copy_xport_params &default_buff_args,
- const device_addr_t &/*hints*/)
-{
- return dpdk_zero_copy::sptr(
- new dpdk_zero_copy_impl(ctx, dpdk_port_id, addr, remote_port, local_port, default_buff_args)
- );
-}
-
-}} // namespace uhd::transport
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk.c b/host/lib/transport/uhd-dpdk/uhd_dpdk.c
deleted file mode 100644
index b83b539fd..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk.c
+++ /dev/null
@@ -1,467 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#include "uhd_dpdk_ctx.h"
-#include "uhd_dpdk_wait.h"
-#include "uhd_dpdk_udp.h"
-#include "uhd_dpdk_driver.h"
-#include <stdlib.h>
-#include <sched.h>
-#include <rte_cycles.h>
-#include <rte_errno.h>
-#include <rte_malloc.h>
-#include <rte_log.h>
-
-/* FIXME: Descriptor ring size: Replace with configurable values */
-#define DEFAULT_RING_SIZE 512
-
-/* FIXME: This needs to be protected */
-struct uhd_dpdk_ctx *ctx = NULL;
-
-/**
- * TODO: Probably should provide way to get access to thread for a given port
- * UHD's first calling thread will be the master thread
- * In UHD, maybe check thread, and if it is different, pass work to that thread and optionally wait() on it (some condition variable)
- */
-
-/* TODO: For nice scheduling options later, make sure to separate RX and TX activity */
-
-int uhd_dpdk_port_count(void)
-{
- if (!ctx)
- return -ENODEV;
- return ctx->num_ports;
-}
-
-int uhd_dpdk_port_link_status(unsigned int portid)
-{
- if (!ctx)
- return -ENODEV;
-
- struct uhd_dpdk_port *p = find_port(portid);
- if (p) {
- struct rte_eth_link link;
- rte_eth_link_get_nowait(p->id, &link);
- return link.link_status;
- }
- return -ENODEV;
-}
-
-struct eth_addr uhd_dpdk_get_eth_addr(unsigned int portid)
-{
- struct eth_addr retval;
- memset(retval.addr, 0xff, ETHER_ADDR_LEN);
-
- struct uhd_dpdk_port *p = find_port(portid);
- if (p) {
- memcpy(retval.addr, p->mac_addr.addr_bytes, ETHER_ADDR_LEN);
- }
-
- return retval;
-}
-
-int uhd_dpdk_get_ipv4_addr(unsigned int portid, uint32_t *ipv4_addr, uint32_t *netmask)
-{
- if (!ipv4_addr)
- return -EINVAL;
- struct uhd_dpdk_port *p = find_port(portid);
- if (p) {
- *ipv4_addr = p->ipv4_addr;
- if (netmask) {
- *netmask = p->netmask;
- }
- return 0;
- }
- return -ENODEV;
-}
-
-int uhd_dpdk_set_ipv4_addr(unsigned int portid, uint32_t ipv4_addr, uint32_t netmask)
-{
- struct uhd_dpdk_port *p = find_port(portid);
- if (p) {
- p->ipv4_addr = ipv4_addr;
- p->netmask = netmask;
- return 0;
- }
- return -ENODEV;
-}
-
-/*
- * Initialize a given port using default settings and with the RX buffers
- * coming from the mbuf_pool passed as a parameter.
- * FIXME: Starting with assumption of one thread/core per port
- */
-static inline int uhd_dpdk_port_init(struct uhd_dpdk_port *port,
- struct rte_mempool *rx_mbuf_pool,
- unsigned int mtu)
-{
- int retval;
-
- /* Check for a valid port */
- if (port->id >= rte_eth_dev_count())
- return -ENODEV;
-
- /* Set up Ethernet device with defaults (1 RX ring, 1 TX ring) */
- retval = rte_eth_dev_set_mtu(port->id, mtu);
- if (retval) {
- uint16_t actual_mtu;
- RTE_LOG(WARNING, EAL, "%d: Could not set mtu to %d\n", retval, mtu);
- rte_eth_dev_get_mtu(port->id, &actual_mtu);
- RTE_LOG(WARNING, EAL, "Current mtu=%d\n", actual_mtu);
- mtu = actual_mtu;
- }
-
- // Require checksum offloads
- struct rte_eth_dev_info dev_info;
- rte_eth_dev_info_get(port->id, &dev_info);
- uint64_t rx_offloads = DEV_RX_OFFLOAD_IPV4_CKSUM;
- uint64_t tx_offloads = DEV_TX_OFFLOAD_IPV4_CKSUM;
- if ((dev_info.rx_offload_capa & rx_offloads) != rx_offloads) {
- RTE_LOG(WARNING, EAL, "%d: Only supports RX offloads 0x%0llx\n", port->id, dev_info.rx_offload_capa);
- rte_exit(EXIT_FAILURE, "Missing required RX offloads\n");
- }
- if ((dev_info.tx_offload_capa & tx_offloads) != tx_offloads) {
- RTE_LOG(WARNING, EAL, "%d: Only supports TX offloads 0x%0llx\n", port->id, dev_info.tx_offload_capa);
- rte_exit(EXIT_FAILURE, "Missing required TX offloads\n");
- }
-
- struct rte_eth_conf port_conf = {
- .rxmode = {
- .offloads = rx_offloads | DEV_RX_OFFLOAD_JUMBO_FRAME,
- .max_rx_pkt_len = mtu,
- .jumbo_frame = 1,
- .hw_ip_checksum = 1,
- .ignore_offload_bitfield = 0,
- },
- .txmode = {
- .offloads = tx_offloads,
- }
- };
- retval = rte_eth_dev_configure(port->id, 1, 1, &port_conf);
- if (retval != 0)
- return retval;
-
- uint16_t rx_desc = DEFAULT_RING_SIZE;
- uint16_t tx_desc = DEFAULT_RING_SIZE;
- retval = rte_eth_dev_adjust_nb_rx_tx_desc(port->id, &rx_desc, &tx_desc);
- if (retval != 0)
- return retval;
-
- if (rx_desc != DEFAULT_RING_SIZE)
- RTE_LOG(WARNING, EAL, "RX descriptors changed to %d\n", rx_desc);
- if (tx_desc != DEFAULT_RING_SIZE)
- RTE_LOG(WARNING, EAL, "TX descriptors changed to %d\n", tx_desc);
-
- retval = rte_eth_rx_queue_setup(port->id, 0, rx_desc,
- rte_eth_dev_socket_id(port->id), NULL, rx_mbuf_pool);
- if (retval < 0)
- return retval;
-
- struct rte_eth_txconf txconf = {
- .offloads = DEV_TX_OFFLOAD_IPV4_CKSUM
- };
- retval = rte_eth_tx_queue_setup(port->id, 0, tx_desc,
- rte_eth_dev_socket_id(port->id), &txconf);
- if (retval < 0)
- goto port_init_fail;
-
- /* Create the hash table for the RX sockets */
- char name[32];
- snprintf(name, sizeof(name), "rx_table_%u", port->id);
- struct rte_hash_parameters hash_params = {
- .name = name,
- .entries = UHD_DPDK_MAX_SOCKET_CNT,
- .key_len = sizeof(struct uhd_dpdk_ipv4_5tuple),
- .hash_func = NULL,
- .hash_func_init_val = 0,
- };
- port->rx_table = rte_hash_create(&hash_params);
- if (port->rx_table == NULL) {
- retval = rte_errno;
- goto port_init_fail;
- }
-
- /* Create ARP table */
- snprintf(name, sizeof(name), "arp_table_%u", port->id);
- hash_params.name = name;
- hash_params.entries = UHD_DPDK_MAX_SOCKET_CNT;
- hash_params.key_len = sizeof(uint32_t);
- hash_params.hash_func = NULL;
- hash_params.hash_func_init_val = 0;
- port->arp_table = rte_hash_create(&hash_params);
- if (port->arp_table == NULL) {
- retval = rte_errno;
- goto free_rx_table;
- }
-
- /* Set up list for TX queues */
- LIST_INIT(&port->txq_list);
-
- /* Start the Ethernet port. */
- retval = rte_eth_dev_start(port->id);
- if (retval < 0) {
- goto free_arp_table;
- }
-
- /* Display the port MAC address. */
- rte_eth_macaddr_get(port->id, &port->mac_addr);
- RTE_LOG(INFO, EAL, "Port %u MAC: %02x %02x %02x %02x %02x %02x\n",
- (unsigned)port->id,
- port->mac_addr.addr_bytes[0], port->mac_addr.addr_bytes[1],
- port->mac_addr.addr_bytes[2], port->mac_addr.addr_bytes[3],
- port->mac_addr.addr_bytes[4], port->mac_addr.addr_bytes[5]);
-
- return 0;
-
-free_arp_table:
- rte_hash_free(port->arp_table);
-free_rx_table:
- rte_hash_free(port->rx_table);
-port_init_fail:
- return rte_errno;
-}
-
-static int uhd_dpdk_thread_init(struct uhd_dpdk_thread *thread, unsigned int lcore)
-{
- if (!ctx || !thread)
- return -EINVAL;
-
- unsigned int socket_id = rte_lcore_to_socket_id(lcore);
- thread->lcore = lcore;
- thread->rx_pktbuf_pool = ctx->rx_pktbuf_pools[socket_id];
- thread->tx_pktbuf_pool = ctx->tx_pktbuf_pools[socket_id];
- LIST_INIT(&thread->port_list);
-
- char name[32];
- snprintf(name, sizeof(name), "sockreq_ring_%u", lcore);
- thread->sock_req_ring = rte_ring_create(
- name,
- UHD_DPDK_MAX_PENDING_SOCK_REQS,
- socket_id,
- RING_F_SC_DEQ
- );
- if (!thread->sock_req_ring)
- return -ENOMEM;
- snprintf(name, sizeof(name), "waiter_ring_%u", lcore);
- thread->waiter_ring = rte_ring_create(
- name,
- UHD_DPDK_MAX_WAITERS,
- socket_id,
- RING_F_SC_DEQ
- );
- if (!thread->waiter_ring)
- return -ENOMEM;
- return 0;
-}
-
-int uhd_dpdk_init(int argc, const char **argv)
-{
- /* Init context only once */
- if (ctx)
- return 1;
-
- /* Grabs arguments intended for DPDK's EAL */
- int ret = rte_eal_init(argc, (char **) argv);
- if (ret < 0)
- rte_exit(EXIT_FAILURE, "Error with EAL initialization\n");
-
- ctx = (struct uhd_dpdk_ctx *) rte_zmalloc("uhd_dpdk_ctx", sizeof(*ctx), rte_socket_id());
- if (!ctx)
- return -ENOMEM;
-
- ctx->num_threads = rte_lcore_count();
- if (ctx->num_threads <= 1)
- rte_exit(EXIT_FAILURE, "Error: No worker threads enabled\n");
-
- /* Check that we have ports to send/receive on */
- ctx->num_ports = rte_eth_dev_count();
- if (ctx->num_ports < 1)
- rte_exit(EXIT_FAILURE, "Error: Found no ports\n");
-
- /* Get memory for thread and port data structures */
- ctx->threads = rte_zmalloc("uhd_dpdk_thread", RTE_MAX_LCORE*sizeof(struct uhd_dpdk_thread), 0);
- if (!ctx->threads)
- rte_exit(EXIT_FAILURE, "Error: Could not allocate memory for thread data\n");
- ctx->ports = rte_zmalloc("uhd_dpdk_port", ctx->num_ports*sizeof(struct uhd_dpdk_port), 0);
- if (!ctx->ports)
- rte_exit(EXIT_FAILURE, "Error: Could not allocate memory for port data\n");
-
- for (size_t i = 0; i < ctx->num_ports; i++) {
- struct uhd_dpdk_port *port = &ctx->ports[i];
- port->id = i;
- rte_eth_macaddr_get(port->id, &port->mac_addr);
- }
-
- return 0;
-}
-
-int uhd_dpdk_start(unsigned int num_ports, int *port_thread_mapping,
- int num_mbufs, int mbuf_cache_size, int mtu)
-{
- if (!ctx)
- return -EIO;
-
- if ((num_ports == 0) || (port_thread_mapping == NULL)) {
- return -EINVAL;
- }
-
- if (ctx->num_ports < num_ports)
- rte_exit(EXIT_FAILURE, "Error: User requested more ports than available\n");
-
- /* Initialize the thread data structures */
- for (int i = rte_get_next_lcore(-1, 1, 0);
- (i < RTE_MAX_LCORE);
- i = rte_get_next_lcore(i, 1, 0))
- {
- /* Do one mempool of RX/TX per socket */
- unsigned int socket_id = rte_lcore_to_socket_id(i);
- /* FIXME Probably want to take into account actual number of ports per socket */
- if (ctx->tx_pktbuf_pools[socket_id] == NULL) {
- /* Creates a new mempool in memory to hold the mbufs.
- * This is done for each CPU socket
- */
- const int mbuf_size = mtu + 2048 + RTE_PKTMBUF_HEADROOM;
- char name[32];
- snprintf(name, sizeof(name), "rx_mbuf_pool_%u", socket_id);
- ctx->rx_pktbuf_pools[socket_id] = rte_pktmbuf_pool_create(
- name,
- ctx->num_ports*num_mbufs,
- mbuf_cache_size,
- 0,
- mbuf_size,
- socket_id
- );
- snprintf(name, sizeof(name), "tx_mbuf_pool_%u", socket_id);
- ctx->tx_pktbuf_pools[socket_id] = rte_pktmbuf_pool_create(
- name,
- ctx->num_ports*num_mbufs,
- mbuf_cache_size,
- 0,
- mbuf_size,
- socket_id
- );
- if ((ctx->rx_pktbuf_pools[socket_id]== NULL) ||
- (ctx->tx_pktbuf_pools[socket_id]== NULL))
- rte_exit(EXIT_FAILURE, "Cannot create mbuf pool\n");
- }
-
- if (uhd_dpdk_thread_init(&ctx->threads[i], i) < 0)
- rte_exit(EXIT_FAILURE, "Error initializing thread %i\n", i);
- }
-
- unsigned master_lcore = rte_get_master_lcore();
-
- /* Assign ports to threads and initialize the port data structures */
- for (unsigned int i = 0; i < num_ports; i++) {
- int thread_id = port_thread_mapping[i];
- if (thread_id < 0)
- continue;
- if (((unsigned int) thread_id) == master_lcore)
- RTE_LOG(WARNING, EAL, "User requested master lcore for port %u\n", i);
- if (ctx->threads[thread_id].lcore != (unsigned int) thread_id)
- rte_exit(EXIT_FAILURE, "Requested inactive lcore %u for port %u\n", (unsigned int) thread_id, i);
-
- struct uhd_dpdk_port *port = &ctx->ports[i];
- port->parent = &ctx->threads[thread_id];
- ctx->threads[thread_id].num_ports++;
- LIST_INSERT_HEAD(&ctx->threads[thread_id].port_list, port, port_entry);
-
- /* Initialize port. */
- if (uhd_dpdk_port_init(port, port->parent->rx_pktbuf_pool, mtu) != 0)
- rte_exit(EXIT_FAILURE, "Cannot init port %"PRIu8 "\n",
- i);
- }
-
- RTE_LOG(INFO, EAL, "Waiting for links to come up...\n");
- rte_delay_ms(1000);
- for (unsigned int i = 0; i < num_ports; i++) {
- struct uhd_dpdk_port *port = &ctx->ports[i];
- if (!port->parent)
- continue;
- struct rte_eth_link link;
- rte_eth_link_get(i, &link);
- RTE_LOG(INFO, EAL, "Port %u UP: %d, %u Mbps\n", i,
- link.link_status, link.link_speed);
- }
- RTE_LOG(INFO, EAL, "Init DONE!\n");
-
- /* FIXME: Create functions to do this */
- RTE_LOG(INFO, EAL, "Starting I/O threads!\n");
-
- cpu_set_t io_cpuset;
- CPU_ZERO(&io_cpuset);
- for (int i = rte_get_next_lcore(-1, 1, 0);
- (i < RTE_MAX_LCORE);
- i = rte_get_next_lcore(i, 1, 0))
- {
- struct uhd_dpdk_thread *t = &ctx->threads[i];
- if (!LIST_EMPTY(&t->port_list)) {
- rte_eal_remote_launch(_uhd_dpdk_driver_main, NULL, ctx->threads[i].lcore);
- struct uhd_dpdk_wait_req *waiter = uhd_dpdk_waiter_alloc(UHD_DPDK_WAIT_SIMPLE);
- if (!waiter) {
- rte_exit(EXIT_FAILURE, "%s: Failed to get wait request\n", __func__);
- }
- uhd_dpdk_waiter_get(waiter);
- uhd_dpdk_waiter_wait(waiter, -1, &ctx->threads[i]);
- uhd_dpdk_waiter_put(waiter);
- CPU_OR(&io_cpuset, &io_cpuset, &t->cpu_affinity);
- }
- }
- cpu_set_t user_cpuset;
- CPU_ZERO(&user_cpuset);
- for (int i = 0; i < CPU_SETSIZE; i++) {
- CPU_SET(i, &user_cpuset);
- }
- CPU_XOR(&user_cpuset, &user_cpuset, &io_cpuset);
- if (pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &user_cpuset)) {
- rte_exit(EXIT_FAILURE, "%s: Failed to set CPU affinity\n", __func__);
- }
- return 0;
-}
-
-/* FIXME: This will be changed once we have functions to handle the threads */
-int uhd_dpdk_destroy(void)
-{
- if (!ctx)
- return -ENODEV;
-
- struct uhd_dpdk_config_req *req = (struct uhd_dpdk_config_req *) rte_zmalloc(NULL, sizeof(*req), 0);
- if (!req)
- return -ENOMEM;
-
- req->waiter = uhd_dpdk_waiter_alloc(UHD_DPDK_WAIT_SIMPLE);
- if (!req->waiter) {
- rte_free(req);
- return -ENOMEM;
- }
-
- req->req_type = UHD_DPDK_LCORE_TERM;
-
- for (int i = rte_get_next_lcore(-1, 1, 0);
- (i < RTE_MAX_LCORE);
- i = rte_get_next_lcore(i, 1, 0))
- {
- struct uhd_dpdk_thread *t = &ctx->threads[i];
-
- if (LIST_EMPTY(&t->port_list))
- continue;
-
- if (rte_eal_get_lcore_state(t->lcore) == FINISHED)
- continue;
-
- if (rte_ring_enqueue(t->sock_req_ring, req)) {
- RTE_LOG(ERR, USER2, "Failed to terminate thread %d\n", i);
- rte_free(req->waiter);
- rte_free(req);
- return -ENOSPC;
- }
- uhd_dpdk_config_req_submit(req, 1, t);
- }
-
- uhd_dpdk_waiter_put(req->waiter);
- rte_free(req);
- return 0;
-}
-
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_ctx.h b/host/lib/transport/uhd-dpdk/uhd_dpdk_ctx.h
deleted file mode 100644
index 6f43ae1cf..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_ctx.h
+++ /dev/null
@@ -1,268 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#ifndef _UHD_DPDK_CTX_H_
-#define _UHD_DPDK_CTX_H_
-
-#include <stdint.h>
-#include <sys/queue.h>
-#include <sys/types.h>
-#include <rte_ethdev.h>
-#include <rte_mbuf.h>
-#include <rte_hash.h>
-#include <rte_eal.h>
-#include <rte_atomic.h>
-#include <uhdlib/transport/uhd-dpdk.h>
-//#include <pthread.h>
-
-/* For nice scheduling options later, make sure to separate RX and TX activity */
-
-#define UHD_DPDK_MAX_SOCKET_CNT 1024
-#define UHD_DPDK_MAX_PENDING_SOCK_REQS 16
-#define UHD_DPDK_MAX_WAITERS UHD_DPDK_MAX_SOCKET_CNT
-#define UHD_DPDK_TXQ_SIZE 64
-#define UHD_DPDK_TX_BURST_SIZE (UHD_DPDK_TXQ_SIZE - 1)
-#define UHD_DPDK_RXQ_SIZE 128
-#define UHD_DPDK_RX_BURST_SIZE (UHD_DPDK_RXQ_SIZE - 1)
-
-struct uhd_dpdk_port;
-struct uhd_dpdk_tx_queue;
-
-/**
- *
- * All memory allocation for port, rx_ring, and tx_queue owned by I/O thread
- * Rest owned by user thread
- *
- * port: port servicing this socket
- * tid: thread ID that owns this socket (to be associated with TX queue)
- * sock_type: Type of socket
- * priv: Private data, based on sock_type
- * rx_ring: pointer to individual rx_ring (created during init--Also used as free buffer ring for TX)
- * tx_queue: pointer to tx queue structure
- * tx_buf_count: Number of buffers currently outside the rings
- * tx_entry: List node for TX Queue tracking
- *
- * If a user closes a socket without outstanding TX buffers, user must free the
- * buffers. Otherwise, that memory will be leaked, and usage will grow.
- */
-struct uhd_dpdk_socket {
- struct uhd_dpdk_port *port;
- pthread_t tid;
- enum uhd_dpdk_sock_type sock_type;
- void *priv;
- struct rte_ring *rx_ring;
- struct uhd_dpdk_tx_queue *tx_queue;
- int tx_buf_count;
- LIST_ENTRY(uhd_dpdk_socket) tx_entry;
-};
-LIST_HEAD(uhd_dpdk_tx_head, uhd_dpdk_socket);
-
-/************************************************
- * Configuration and Blocking
- ************************************************/
-struct uhd_dpdk_wait_req;
-
-enum uhd_dpdk_sock_req {
- UHD_DPDK_SOCK_OPEN = 0,
- UHD_DPDK_SOCK_CLOSE,
- UHD_DPDK_LCORE_TERM,
- UHD_DPDK_SOCK_REQ_COUNT
-};
-
-/**
- * port: port associated with this request
- * sock: socket associated with this request
- * req_type: Open, Close, or terminate lcore
- * cond: Used to sleep until socket creation is finished
- * mutex: associated with cond
- * entry: List node for requests pending ARP responses
- * priv: private data
- * retval: Result of call (needed post-wakeup)
- *
- * config_reqs are assumed not to time out
- * The interaction with wait_reqs currently makes this impossible to do safely
- */
-struct uhd_dpdk_config_req {
- struct uhd_dpdk_port *port;
- struct uhd_dpdk_socket *sock;
- enum uhd_dpdk_sock_req req_type;
- struct uhd_dpdk_wait_req *waiter;
- LIST_ENTRY(uhd_dpdk_config_req) entry;
- void *priv;
- int retval;
-};
-LIST_HEAD(uhd_dpdk_config_head, uhd_dpdk_config_req);
-
-/************************************************
- * RX Table
- ************************************************/
-struct uhd_dpdk_arp_entry {
- struct ether_addr mac_addr;
- struct uhd_dpdk_config_head pending_list; /* Config reqs pending ARP--Thread-unsafe */
-};
-
-struct uhd_dpdk_ipv4_5tuple {
- enum uhd_dpdk_sock_type sock_type;
- uint32_t src_ip;
- uint32_t dst_ip;
- uint16_t src_port;
- uint16_t dst_port;
-};
-
-/**
- * Entry for RX table
- * req used for blocking calls to RX
- */
-struct uhd_dpdk_rx_entry {
- struct uhd_dpdk_socket *sock;
- struct uhd_dpdk_wait_req *waiter;
-};
-
-/************************************************
- * TX Queues
- *
- * 1 TX Queue per socket sending through a hardware port
- * All memory allocation owned by I/O thread
- *
- * tid: thread id
- * queue: TX queue holding threads prepared packets (via send())
- * retry_queue: queue holding packets that couldn't be sent
- * freebufs: queue holding empty buffers
- * waiter: Request to wait for a free buffer
- * tx_list: list of sockets using this queue
- * entry: list node for port to track TX queues
- *
- * queue, retry_queue, and freebufs are single-producer, single-consumer queues
- * retry_queue wholly-owned by I/O thread
- * For queue, user thread is producer, I/O thread is consumer
- * For freebufs, user thread is consumer, I/O thread is consumer
- *
- * All queues are same size
- * 1. Buffers start in freebufs (user gets buffers from freebufs)
- * 2. User submits packet to queue
- * 3. If packet couldn't be sent, it is (re)enqueued on retry_queue
- ************************************************/
-struct uhd_dpdk_tx_queue {
- pthread_t tid;
- struct rte_ring *queue;
- struct rte_ring *retry_queue;
- struct rte_ring *freebufs;
- struct uhd_dpdk_wait_req *waiter;
- struct uhd_dpdk_tx_head tx_list;
- LIST_ENTRY(uhd_dpdk_tx_queue) entry;
-};
-LIST_HEAD(uhd_dpdk_txq_head, uhd_dpdk_tx_queue);
-
-/************************************************
- * Port structure
- *
- * All memory allocation owned by I/O thread
- *
- * id: hardware port id (for DPDK)
- * parent: I/O thread servicing this port
- * mac_addr: MAC address of this port
- * ipv4_addr: IPv4 address of this port
- * netmask: Subnet mask of this port
- * arp_table: ARP cache for this port
- * rx_table: Mapping of 5-tuple key to sockets for RX
- * txq_list: List of TX queues associated with this port
- * port_entry: List node entry for I/O thread to track
- ************************************************/
-struct uhd_dpdk_port {
- unsigned int id;
- struct uhd_dpdk_thread *parent;
- struct ether_addr mac_addr;
- uint32_t ipv4_addr; /* FIXME: Check this before allowing a socket!!! */
- uint32_t netmask;
- /* Key = IP addr
- * Value = MAC addr (ptr to uhd_dpdk_arp_entry)
- */
- struct rte_hash *arp_table;
- /* hash map of RX sockets
- * Key = uhd_dpdk_ipv4_5tuple
- * Value = uhd_dpdk_socket
- */
- struct rte_hash *rx_table;
- /* doubly-linked list of TX sockets */
- struct uhd_dpdk_txq_head txq_list;
- LIST_ENTRY(uhd_dpdk_port) port_entry;
-};
-
-LIST_HEAD(uhd_dpdk_port_head, uhd_dpdk_port);
-
-/************************************************
- * Thread/lcore-private data structure
- *
- * All data owned by global context
- *
- * id: lcore id (from DPDK)
- * rx_pktbuf_pool: memory pool for generating buffers for RX packets
- * tx_pktbuf_pool: memory pool for generating buffers for TX packets
- * num_ports: Number of ports this lcore is servicing
- * port_list: List of ports this lcore is servicing
- * sock_req_ring: Queue for user threads to submit service requests to the lcore
- *
- * sock_req_ring is a multi-producer, single-consumer queue
- * It must NOT BE ACCESSED SIMULTANEOUSLY by two threads not using SCHED_OTHER(cfs)
- *
- * For threads that have ports:
- * Launch individually
- * For threads without ports:
- * Do not launch unless user specifically does it themselves.
- * Should also have master lcore returned to user
- * REMEMBER: Without args, DPDK creates an lcore for each CPU core!
- */
-struct uhd_dpdk_thread {
- unsigned int lcore;
- cpu_set_t cpu_affinity;
- struct rte_mempool *rx_pktbuf_pool;
- struct rte_mempool *tx_pktbuf_pool;
- int num_ports;
- struct uhd_dpdk_port_head port_list;
- struct rte_ring *sock_req_ring;
- struct rte_ring *waiter_ring;
-};
-
-
-/************************************************
- * One global context
- *
- * num_threads: Number of DPDK lcores tracked
- * num_ports: Number of DPDK/NIC ports tracked
- * threads: Array of all lcores/threads
- * ports: Array of all DPDK/NIC ports
- * rx_pktbuf_pools: Array of all packet buffer pools for RX
- * tx_pktbuf_pools: Array of all packet buffer pools for TX
- *
- * The packet buffer pools are memory pools that are associated with a CPU
- * socket. They will provide storage close to the socket to accommodate NUMA
- * nodes.
- ************************************************/
-struct uhd_dpdk_ctx {
- unsigned int num_threads;
- unsigned int num_ports;
- struct uhd_dpdk_thread *threads;
- struct uhd_dpdk_port *ports;
- struct rte_mempool *rx_pktbuf_pools[RTE_MAX_NUMA_NODES];
- struct rte_mempool *tx_pktbuf_pools[RTE_MAX_NUMA_NODES];
-};
-
-extern struct uhd_dpdk_ctx *ctx;
-
-static inline struct uhd_dpdk_port * find_port(unsigned int portid)
-{
- if (!ctx)
- return NULL;
-
- for (unsigned int i = 0; i < ctx->num_ports; i++) {
- struct uhd_dpdk_port *p = &ctx->ports[i];
- if (p->id == portid) {
- return p;
- }
- }
- return NULL;
-}
-
-#endif /* _UHD_DPDK_CTX_H_ */
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_driver.c b/host/lib/transport/uhd-dpdk/uhd_dpdk_driver.c
deleted file mode 100644
index 7b4cf8df5..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_driver.c
+++ /dev/null
@@ -1,624 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#include "uhd_dpdk_driver.h"
-#include "uhd_dpdk_fops.h"
-#include "uhd_dpdk_udp.h"
-#include "uhd_dpdk_wait.h"
-#include <rte_malloc.h>
-#include <rte_mempool.h>
-#include <arpa/inet.h>
-#include <unistd.h>
-
-int _uhd_dpdk_arp_reply(struct uhd_dpdk_port *port, struct arp_hdr *arp_req)
-{
- struct rte_mbuf *mbuf;
- struct ether_hdr *hdr;
- struct arp_hdr *arp_frame;
-
- mbuf = rte_pktmbuf_alloc(port->parent->tx_pktbuf_pool);
- if (unlikely(mbuf == NULL)) {
- RTE_LOG(WARNING, MEMPOOL, "Could not allocate packet buffer for ARP response\n");
- return -ENOMEM;
- }
-
- hdr = rte_pktmbuf_mtod(mbuf, struct ether_hdr *);
- arp_frame = (struct arp_hdr *) &hdr[1];
-
- ether_addr_copy(&arp_req->arp_data.arp_sha, &hdr->d_addr);
- ether_addr_copy(&port->mac_addr, &hdr->s_addr);
- hdr->ether_type = rte_cpu_to_be_16(ETHER_TYPE_ARP);
-
- arp_frame->arp_hrd = rte_cpu_to_be_16(ARP_HRD_ETHER);
- arp_frame->arp_pro = rte_cpu_to_be_16(ETHER_TYPE_IPv4);
- arp_frame->arp_hln = 6;
- arp_frame->arp_pln = 4;
- arp_frame->arp_op = rte_cpu_to_be_16(ARP_OP_REPLY);
- ether_addr_copy(&port->mac_addr, &arp_frame->arp_data.arp_sha);
- arp_frame->arp_data.arp_sip = port->ipv4_addr;
- ether_addr_copy(&hdr->d_addr, &arp_frame->arp_data.arp_tha);
- arp_frame->arp_data.arp_tip = arp_req->arp_data.arp_sip;
-
- mbuf->pkt_len = 42;
- mbuf->data_len = 42;
-
- if (rte_eth_tx_burst(port->id, 0, &mbuf, 1) != 1) {
- RTE_LOG(WARNING, RING, "%s: TX descriptor ring is full\n", __func__);
- rte_pktmbuf_free(mbuf);
- return -EAGAIN;
- }
- return 0;
-}
-
-int _uhd_dpdk_process_arp(struct uhd_dpdk_port *port, struct arp_hdr *arp_frame)
-{
- uint32_t dest_ip = arp_frame->arp_data.arp_sip;
- struct ether_addr dest_addr = arp_frame->arp_data.arp_sha;
-
- /* Add entry to ARP table */
- struct uhd_dpdk_arp_entry *entry = NULL;
- rte_hash_lookup_data(port->arp_table, &dest_ip, (void **) &entry);
- if (!entry) {
- entry = rte_zmalloc(NULL, sizeof(*entry), 0);
- if (!entry) {
- return -ENOMEM;
- }
- LIST_INIT(&entry->pending_list);
- ether_addr_copy(&dest_addr, &entry->mac_addr);
- if (rte_hash_add_key_data(port->arp_table, &dest_ip, entry) < 0) {
- rte_free(entry);
- return -ENOSPC;
- }
- } else {
- struct uhd_dpdk_config_req *req = NULL;
- ether_addr_copy(&dest_addr, &entry->mac_addr);
- /* Now wake any config reqs waiting for the ARP */
- LIST_FOREACH(req, &entry->pending_list, entry) {
- _uhd_dpdk_config_req_compl(req, 0);
- }
- while (entry->pending_list.lh_first != NULL) {
- LIST_REMOVE(entry->pending_list.lh_first, entry);
- }
- }
-
- /* Respond if this was an ARP request */
- if (arp_frame->arp_op == rte_cpu_to_be_16(ARP_OP_REQUEST) &&
- arp_frame->arp_data.arp_tip == port->ipv4_addr) {
- _uhd_dpdk_arp_reply(port, arp_frame);
- }
-
- return 0;
-}
-
-/* Send ARP request */
-int _uhd_dpdk_arp_request(struct uhd_dpdk_port *port, uint32_t ip)
-{
- struct rte_mbuf *mbuf;
- struct ether_hdr *hdr;
- struct arp_hdr *arp_frame;
-
- mbuf = rte_pktmbuf_alloc(port->parent->tx_pktbuf_pool);
- if (unlikely(mbuf == NULL)) {
- RTE_LOG(WARNING, MEMPOOL, "Could not allocate packet buffer for ARP request\n");
- return -ENOMEM;
- }
-
- hdr = rte_pktmbuf_mtod(mbuf, struct ether_hdr *);
- arp_frame = (struct arp_hdr *) &hdr[1];
-
- memset(hdr->d_addr.addr_bytes, 0xFF, ETHER_ADDR_LEN);
- ether_addr_copy(&port->mac_addr, &hdr->s_addr);
- hdr->ether_type = rte_cpu_to_be_16(ETHER_TYPE_ARP);
-
- arp_frame->arp_hrd = rte_cpu_to_be_16(ARP_HRD_ETHER);
- arp_frame->arp_pro = rte_cpu_to_be_16(ETHER_TYPE_IPv4);
- arp_frame->arp_hln = 6;
- arp_frame->arp_pln = 4;
- arp_frame->arp_op = rte_cpu_to_be_16(ARP_OP_REQUEST);
- ether_addr_copy(&port->mac_addr, &arp_frame->arp_data.arp_sha);
- arp_frame->arp_data.arp_sip = port->ipv4_addr;
- memset(arp_frame->arp_data.arp_tha.addr_bytes, 0x00, ETHER_ADDR_LEN);
- arp_frame->arp_data.arp_tip = ip;
-
- mbuf->pkt_len = 42;
- mbuf->data_len = 42;
-
- if (rte_eth_tx_burst(port->id, 0, &mbuf, 1) != 1) {
- RTE_LOG(WARNING, RING, "%s: TX descriptor ring is full\n", __func__);
- rte_pktmbuf_free(mbuf);
- return -EAGAIN;
- }
- return 0;
-}
-
-int _uhd_dpdk_process_udp(struct uhd_dpdk_port *port, struct rte_mbuf *mbuf,
- struct udp_hdr *pkt, bool bcast)
-{
- int status = 0;
- struct uhd_dpdk_ipv4_5tuple ht_key = {
- .sock_type = UHD_DPDK_SOCK_UDP,
- .src_ip = 0,
- .src_port = 0,
- .dst_ip = 0,
- .dst_port = pkt->dst_port
- };
-
- struct uhd_dpdk_rx_entry *entry = NULL;
- rte_hash_lookup_data(port->rx_table, &ht_key, (void **) &entry);
- if (!entry) {
- status = -ENODEV;
- //RTE_LOG(WARNING, USER1, "%s: Dropping packet to UDP port %d\n", __func__, ntohs(pkt->dst_port));
- goto udp_rx_drop;
- }
-
- struct uhd_dpdk_udp_priv *pdata = (struct uhd_dpdk_udp_priv *) entry->sock->priv;
- if (bcast && pdata->filter_bcast) {
- // Filter broadcast packets if not listening
- goto udp_rx_drop;
- }
- status = rte_ring_enqueue(entry->sock->rx_ring, mbuf);
- if (entry->waiter) {
- _uhd_dpdk_waiter_wake(entry->waiter, port->parent);
- entry->waiter = NULL;
- }
- if (status) {
- pdata->dropped_pkts++;
- goto udp_rx_drop;
- }
- pdata->xferd_pkts++;
- return 0;
-
-udp_rx_drop:
- rte_pktmbuf_free(mbuf);
- return status;
-}
-
-int _uhd_dpdk_process_ipv4(struct uhd_dpdk_port *port, struct rte_mbuf *mbuf,
- struct ipv4_hdr *pkt)
-{
- bool bcast = is_broadcast(port, pkt->dst_addr);
- if (pkt->dst_addr != port->ipv4_addr && !bcast) {
- rte_pktmbuf_free(mbuf);
- return -ENODEV;
- }
- if (pkt->next_proto_id == 0x11) {
- return _uhd_dpdk_process_udp(port, mbuf, (struct udp_hdr *) &pkt[1], bcast);
- }
- rte_pktmbuf_free(mbuf);
- return -EINVAL;
-}
-
-static int _uhd_dpdk_fill_ipv4_addr(struct uhd_dpdk_port *port,
- struct rte_mbuf *mbuf)
-{
- struct ether_hdr *eth_hdr = rte_pktmbuf_mtod(mbuf, struct ether_hdr *);
- struct ipv4_hdr *ip_hdr = (struct ipv4_hdr *) &eth_hdr[1];
- if (is_broadcast(port, ip_hdr->dst_addr)) {
- memset(eth_hdr->d_addr.addr_bytes, 0xff, ETHER_ADDR_LEN);
- } else {
- /* Lookup dest_addr */
- struct uhd_dpdk_arp_entry *entry = NULL;
- rte_hash_lookup_data(port->arp_table, &ip_hdr->dst_addr, (void **) &entry);
- if (!entry) {
- RTE_LOG(ERR, USER1, "TX packet on port %d to addr 0x%08x has no ARP entry\n", port->id, ip_hdr->dst_addr);
- return -ENODEV;
- }
-
- ether_addr_copy(&entry->mac_addr, &eth_hdr->d_addr);
- }
- return 0;
-}
-
-static int _uhd_dpdk_send(struct uhd_dpdk_port *port,
- struct uhd_dpdk_tx_queue *txq,
- struct rte_ring *q)
-{
- struct rte_mbuf *buf;
-
- unsigned int num_tx = rte_ring_count(q);
- num_tx = (num_tx < UHD_DPDK_TX_BURST_SIZE) ? num_tx : UHD_DPDK_TX_BURST_SIZE;
- for (unsigned int i = 0; i < num_tx; i++) {
- int status = rte_ring_dequeue(q, (void **) &buf);
- if (status) {
- RTE_LOG(ERR, USER1, "%s: Q Count doesn't match actual\n", __func__);
- break;
- }
- struct ether_hdr *eth_hdr = rte_pktmbuf_mtod(buf, struct ether_hdr *);
- if (eth_hdr->ether_type == rte_cpu_to_be_16(ETHER_TYPE_IPv4)) {
- status = _uhd_dpdk_fill_ipv4_addr(port, buf);
- if (status) {
- return status;
- }
- }
-
- status = rte_eth_tx_prepare(port->id, 0, &buf, 1);
- if (status != 1) {
- status = rte_ring_enqueue(txq->retry_queue, buf);
- if (status) {
- RTE_LOG(WARNING, USER1, "%s: Could not re-enqueue pkt %d\n", __func__, i);
- rte_pktmbuf_free(buf);
- }
- num_tx = i;
- break;
- }
-
- status = rte_eth_tx_burst(port->id, 0, &buf, 1); /* Automatically frees mbuf */
- if (status != 1) {
- status = rte_ring_enqueue(txq->retry_queue, buf);
- if (status) {
- RTE_LOG(WARNING, USER1, "%s: Could not re-enqueue pkt %d\n", __func__, i);
- rte_pktmbuf_free(buf);
- }
- num_tx = i;
- break;
- }
- }
-
- return num_tx;
-}
-
-static inline int _uhd_dpdk_restore_bufs(struct uhd_dpdk_port *port,
- struct uhd_dpdk_tx_queue *q,
- unsigned int num_bufs)
-{
- /* Allocate more buffers to replace the sent ones */
- struct rte_mbuf *freebufs[UHD_DPDK_TXQ_SIZE];
- int status = rte_pktmbuf_alloc_bulk(port->parent->tx_pktbuf_pool, freebufs, num_bufs);
- if (status) {
- RTE_LOG(ERR, USER1, "%d %s: Could not restore %u pktmbufs in bulk!\n", status, __func__, num_bufs);
- }
-
- /* Enqueue the buffers for the user thread to retrieve */
- unsigned int enqd = rte_ring_enqueue_bulk(q->freebufs, (void **) freebufs, num_bufs, NULL);
- if (q->waiter && rte_ring_count(q->freebufs) > 0) {
- _uhd_dpdk_waiter_wake(q->waiter, port->parent);
- q->waiter = NULL;
- }
- if (enqd != num_bufs) {
- RTE_LOG(ERR, USER1, "Could not enqueue pktmbufs!\n");
- return status;
- }
-
- return num_bufs;
-}
-
-static inline void _uhd_dpdk_disable_ports(struct uhd_dpdk_thread *t)
-{
- struct uhd_dpdk_port *port = NULL;
- LIST_FOREACH(port, &t->port_list, port_entry) {
- rte_eth_dev_stop(port->id);
- }
-}
-
-static inline int _uhd_dpdk_driver_cleanup(struct uhd_dpdk_thread *t)
-{
- /* Close sockets upon request, but reply to other service requests with
- * errors
- */
- struct uhd_dpdk_config_req *sock_req;
- if (rte_ring_dequeue(t->sock_req_ring, (void **) &sock_req)) {
- switch (sock_req->req_type) {
- case UHD_DPDK_SOCK_CLOSE:
- _uhd_dpdk_sock_release(sock_req);
- break;
- default:
- _uhd_dpdk_config_req_compl(sock_req, -ENODEV);
- break;
- }
- }
-
- /* Do nothing if there are users remaining */
- struct uhd_dpdk_port *port = NULL;
- LIST_FOREACH(port, &t->port_list, port_entry) {
- /* Check for RX sockets */
- const void *hash_key;
- void *hash_sock;
- uint32_t hash_next = 0;
- if (rte_hash_iterate(port->rx_table, &hash_key,
- &hash_sock, &hash_next) != -ENOENT)
- return -EAGAIN;
-
- /* Check for TX sockets */
- struct uhd_dpdk_tx_queue *q = NULL;
- LIST_FOREACH(q, &port->txq_list, entry) {
- if (!LIST_EMPTY(&q->tx_list))
- return -EAGAIN;
- }
- }
-
- /* Now clean up waiters
- * TODO: Determine if better to wake threads
- */
- int num_waiters = rte_ring_count(t->waiter_ring);
- for (int i = 0; i < num_waiters; i++) {
- struct uhd_dpdk_wait_req *req = NULL;
- rte_ring_dequeue(t->waiter_ring, (void **) &req);
- uhd_dpdk_waiter_put(req);
- }
- if (rte_ring_count(t->waiter_ring))
- return -EAGAIN;
-
- /* Now can free memory, except sock_req_ring and waiter_ring */
- LIST_FOREACH(port, &t->port_list, port_entry) {
- rte_hash_free(port->rx_table);
-
- struct uhd_dpdk_tx_queue *q = LIST_FIRST(&port->txq_list);
- while (!LIST_EMPTY(&port->txq_list)) {
- struct uhd_dpdk_tx_queue *nextq = LIST_NEXT(q, entry);
- while (!rte_ring_empty(q->queue)) {
- struct rte_buf *buf = NULL;
- rte_ring_dequeue(q->queue, (void **) &buf);
- rte_free(buf);
- }
- while (!rte_ring_empty(q->freebufs)) {
- struct rte_buf *buf = NULL;
- rte_ring_dequeue(q->freebufs, (void **) &buf);
- rte_free(buf);
- }
- while (!rte_ring_empty(q->retry_queue)) {
- struct rte_buf *buf = NULL;
- rte_ring_dequeue(q->retry_queue, (void **) &buf);
- rte_free(buf);
- }
- rte_ring_free(q->queue);
- rte_ring_free(q->freebufs);
- rte_ring_free(q->retry_queue);
- rte_free(q);
- q = nextq;
- }
-
- const void *arp_key;
- uint32_t arp_key_next = 0;
- struct uhd_dpdk_arp_entry *arp_entry = NULL;
- while (rte_hash_iterate(port->arp_table, &arp_key,
- (void **) &arp_entry, &arp_key_next) >= 0) {
- rte_free(arp_entry);
- }
- rte_hash_free(port->arp_table);
- }
-
- return 0;
-}
-
-static inline int _uhd_dpdk_service_config_req(struct rte_ring *sock_req_ring)
-{
- int status = 0;
- struct uhd_dpdk_config_req *sock_req;
- if (rte_ring_dequeue(sock_req_ring, (void **) &sock_req) == 0) {
- if (sock_req) {
- /* FIXME: Not checking return vals */
- switch (sock_req->req_type) {
- case UHD_DPDK_SOCK_OPEN:
- _uhd_dpdk_sock_setup(sock_req);
- break;
- case UHD_DPDK_SOCK_CLOSE:
- _uhd_dpdk_sock_release(sock_req);
- break;
- case UHD_DPDK_LCORE_TERM:
- RTE_LOG(INFO, EAL, "Terminating lcore %u\n", rte_lcore_id());
- status = 1;
- _uhd_dpdk_config_req_compl(sock_req, 0);
- break;
- default:
- RTE_LOG(ERR, USER2, "Invalid socket request %d\n", sock_req->req_type);
- break;
- }
- } else {
- RTE_LOG(ERR, USER1, "%s: NULL service request received\n", __func__);
- }
- }
- return status;
-}
-
-/* Do a burst of RX on port */
-static inline void _uhd_dpdk_rx_burst(struct uhd_dpdk_port *port)
-{
- struct ether_hdr *hdr;
- char *l2_data;
- struct rte_mbuf *bufs[UHD_DPDK_RX_BURST_SIZE];
- const uint16_t num_rx = rte_eth_rx_burst(port->id, 0,
- bufs, UHD_DPDK_RX_BURST_SIZE);
- if (unlikely(num_rx == 0)) {
- return;
- }
-
- for (int buf = 0; buf < num_rx; buf++) {
- uint64_t ol_flags = bufs[buf]->ol_flags;
- hdr = rte_pktmbuf_mtod(bufs[buf], struct ether_hdr *);
- l2_data = (char *) &hdr[1];
- switch (rte_be_to_cpu_16(hdr->ether_type)) {
- case ETHER_TYPE_ARP:
- _uhd_dpdk_process_arp(port, (struct arp_hdr *) l2_data);
- rte_pktmbuf_free(bufs[buf]);
- break;
- case ETHER_TYPE_IPv4:
- if ((ol_flags & PKT_RX_IP_CKSUM_MASK) == PKT_RX_IP_CKSUM_BAD) {
- RTE_LOG(WARNING, RING, "Buf %d: Bad IP cksum\n", buf);
- } else if ((ol_flags & PKT_RX_IP_CKSUM_MASK) == PKT_RX_IP_CKSUM_NONE) {
- RTE_LOG(WARNING, RING, "Buf %d: Missing IP cksum\n", buf);
- } else {
- _uhd_dpdk_process_ipv4(port, bufs[buf], (struct ipv4_hdr *) l2_data);
- }
- break;
- default:
- rte_pktmbuf_free(bufs[buf]);
- break;
- }
- }
-}
-
-/* Do a burst of TX on port's tx q */
-static inline int _uhd_dpdk_tx_burst(struct uhd_dpdk_port *port,
- struct uhd_dpdk_tx_queue *q)
-{
- if (!rte_ring_empty(q->retry_queue)) {
- int num_retry = _uhd_dpdk_send(port, q, q->retry_queue);
- _uhd_dpdk_restore_bufs(port, q, num_retry);
- if (!rte_ring_empty(q->retry_queue)) {
- return -EAGAIN;
- }
- }
- if (rte_ring_empty(q->queue)) {
- return 0;
- }
- int num_tx = _uhd_dpdk_send(port, q, q->queue);
- if (num_tx > 0) {
- _uhd_dpdk_restore_bufs(port, q, num_tx);
- return 0;
- } else {
- return num_tx;
- }
-}
-
-/* Process threads requesting to block on RX */
-static inline void _uhd_dpdk_process_rx_wait(struct uhd_dpdk_thread *t,
- struct uhd_dpdk_wait_req *req)
-{
- struct uhd_dpdk_socket *sock = req->sock;
- if (!sock)
- goto rx_wait_skip;
- if (!sock->port)
- goto rx_wait_skip;
- if (!sock->port->rx_table)
- goto rx_wait_skip;
-
- if (!rte_ring_empty(sock->rx_ring))
- goto rx_wait_skip;
-
- struct uhd_dpdk_ipv4_5tuple ht_key;
- if (_uhd_dpdk_sock_rx_key(sock, &ht_key))
- goto rx_wait_skip;
-
- struct uhd_dpdk_rx_entry *entry = NULL;
- rte_hash_lookup_data(sock->port->rx_table, &ht_key, (void **) &entry);
- entry->waiter = req;
- return;
-
-rx_wait_skip:
- _uhd_dpdk_waiter_wake(req, t);
-}
-
-/* Process threads requesting to block on TX bufs*/
-static inline void _uhd_dpdk_process_tx_buf_wait(struct uhd_dpdk_thread *t,
- struct uhd_dpdk_wait_req *req)
-{
- struct uhd_dpdk_socket *sock = req->sock;
- if (!sock)
- goto tx_wait_skip;
- if (!sock->port)
- goto tx_wait_skip;
- if (!sock->tx_queue)
- goto tx_wait_skip;
-
- struct uhd_dpdk_tx_queue *q = sock->tx_queue;
- if (!q->freebufs || !q->retry_queue || !q->queue)
- goto tx_wait_skip;
-
- if (!rte_ring_empty(q->freebufs))
- goto tx_wait_skip;
-
- sock->tx_queue->waiter = req;
-
- // Attempt to restore bufs only if failed before
- unsigned int num_bufs = sock->tx_buf_count + rte_ring_count(q->queue) +
- rte_ring_count(q->retry_queue);
- unsigned int max_bufs = rte_ring_get_capacity(q->freebufs);
- if (num_bufs < max_bufs) {
- _uhd_dpdk_restore_bufs(sock->port, q, max_bufs - num_bufs);
- }
- return;
-
-tx_wait_skip:
- _uhd_dpdk_waiter_wake(req, t);
-}
-
-/* Process threads making requests to wait */
-static inline void _uhd_dpdk_process_waiters(struct uhd_dpdk_thread *t)
-{
- int num_waiters = rte_ring_count(t->waiter_ring);
- num_waiters = (num_waiters > UHD_DPDK_MAX_PENDING_SOCK_REQS) ?
- UHD_DPDK_MAX_PENDING_SOCK_REQS :
- num_waiters;
- for (int i = 0; i < num_waiters; i++) {
- /* Dequeue */
- struct uhd_dpdk_wait_req *req = NULL;
- if (rte_ring_dequeue(t->waiter_ring, (void **) &req))
- break;
- switch (req->reason) {
- case UHD_DPDK_WAIT_SIMPLE:
- _uhd_dpdk_waiter_wake(req, t);
- break;
- case UHD_DPDK_WAIT_RX:
- _uhd_dpdk_process_rx_wait(t, req);
- break;
- default:
- RTE_LOG(ERR, USER2, "Invalid reason associated with wait request\n");
- _uhd_dpdk_waiter_wake(req, t);
- break;
- }
- }
-}
-
-int _uhd_dpdk_driver_main(void *arg)
-{
-
- /* Don't currently have use for arguments */
- if (arg)
- return -EINVAL;
-
- /* Check that this is a valid lcore */
- unsigned int lcore_id = rte_lcore_id();
- if (lcore_id == LCORE_ID_ANY)
- return -ENODEV;
-
- /* Check that this lcore has ports */
- struct uhd_dpdk_thread *t = &ctx->threads[lcore_id];
- if (t->lcore != lcore_id)
- return -ENODEV;
-
- pthread_getaffinity_np(pthread_self(), sizeof(cpu_set_t),
- &t->cpu_affinity);
- char name[16];
- snprintf(name, sizeof(name), "dpdk-io_%u", lcore_id);
- pthread_setname_np(pthread_self(), name);
-
- RTE_LOG(INFO, USER2, "Thread %d started\n", lcore_id);
- int status = 0;
- while (!status) {
- /* Check for open()/close()/term() requests and service 1 at a time */
- status = _uhd_dpdk_service_config_req(t->sock_req_ring);
- /* For each port, attempt to receive packets and process */
- struct uhd_dpdk_port *port = NULL;
- LIST_FOREACH(port, &t->port_list, port_entry) {
- _uhd_dpdk_rx_burst(port);
- }
-
- /* TODO: Handle waiter_ring
- * Also use it for config_req wake retries
- * Also take care of RX table with new struct w/ waiter
- * (construction, adding, destruction)
- */
- _uhd_dpdk_process_waiters(t);
-
- /* For each port's TX queues, do TX */
- LIST_FOREACH(port, &t->port_list, port_entry) {
- struct uhd_dpdk_tx_queue *q = NULL;
- LIST_FOREACH(q, &port->txq_list, entry) {
- if (_uhd_dpdk_tx_burst(port, q))
- break;
- }
- }
- }
-
- /* Now turn off ports */
- _uhd_dpdk_disable_ports(t);
-
- /* Now clean up before exiting */
- int cleaning = -EAGAIN;
- while (cleaning == -EAGAIN) {
- cleaning = _uhd_dpdk_driver_cleanup(t);
- }
- return status;
-}
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_driver.h b/host/lib/transport/uhd-dpdk/uhd_dpdk_driver.h
deleted file mode 100644
index f94a678ba..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_driver.h
+++ /dev/null
@@ -1,33 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#ifndef _UHD_DPDK_DRIVER_H_
-#define _UHD_DPDK_DRIVER_H_
-
-#include "uhd_dpdk_ctx.h"
-#include <rte_mbuf.h>
-#include <rte_arp.h>
-#include <rte_udp.h>
-#include <rte_ip.h>
-
-static inline bool is_broadcast(struct uhd_dpdk_port *port, uint32_t dst_ipv4_addr)
-{
- uint32_t network = port->netmask | ((~port->netmask) & dst_ipv4_addr);
- return (network == 0xffffffff);
-}
-
-
-int _uhd_dpdk_process_arp(struct uhd_dpdk_port *port, struct arp_hdr *arp_frame);
-int _uhd_dpdk_process_udp(struct uhd_dpdk_port *port, struct rte_mbuf *mbuf,
- struct udp_hdr *pkt, bool bcast);
-int _uhd_dpdk_process_ipv4(struct uhd_dpdk_port *port, struct rte_mbuf *mbuf, struct ipv4_hdr *pkt);
-int _uhd_dpdk_send_udp(struct uhd_dpdk_port *port,
- struct uhd_dpdk_socket *sock,
- struct rte_mbuf *mbuf);
-int _uhd_dpdk_arp_request(struct uhd_dpdk_port *port,
- uint32_t ip);
-
-int _uhd_dpdk_driver_main(void *arg);
-#endif /* _UHD_DPDK_DRIVER_H_ */
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_fops.c b/host/lib/transport/uhd-dpdk/uhd_dpdk_fops.c
deleted file mode 100644
index 605f01de3..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_fops.c
+++ /dev/null
@@ -1,353 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#include "uhd_dpdk_fops.h"
-#include "uhd_dpdk_udp.h"
-#include "uhd_dpdk_wait.h"
-#include <rte_malloc.h>
-#include <rte_ip.h>
-
-/************************************************
- * I/O thread ONLY
- *
- * TODO: Decide whether to allow blocking on mutex
- * This would cause the I/O thread to sleep, which isn't desireable
- * Could throw in a "request completion cleanup" section in I/O thread's
- * main loop, though. Just keep trying until the requesting thred is woken
- * up. This would be to handle the case where the thread hadn't finished
- * setting itself up to wait on the condition variable, but the I/O thread
- * still got the request.
- */
-int _uhd_dpdk_config_req_compl(struct uhd_dpdk_config_req *req, int retval)
-{
- req->retval = retval;
- int stat = _uhd_dpdk_waiter_wake(req->waiter, req->sock->port->parent);
- return stat;
-}
-
-int _uhd_dpdk_sock_setup(struct uhd_dpdk_config_req *req)
-{
- int stat = 0;
- switch (req->sock->sock_type) {
- case UHD_DPDK_SOCK_UDP:
- stat = _uhd_dpdk_udp_setup(req);
- break;
- default:
- stat = -EINVAL;
- _uhd_dpdk_config_req_compl(req, -EINVAL);
- }
- return stat;
-}
-
-int _uhd_dpdk_sock_release(struct uhd_dpdk_config_req *req)
-{
- int stat = 0;
- switch (req->sock->sock_type) {
- case UHD_DPDK_SOCK_UDP:
- stat = _uhd_dpdk_udp_release(req);
- break;
- default:
- stat = -EINVAL;
- _uhd_dpdk_config_req_compl(req, -EINVAL);
- }
-
- return stat;
-}
-
-int _uhd_dpdk_sock_rx_key(struct uhd_dpdk_socket *sock,
- struct uhd_dpdk_ipv4_5tuple *key)
-{
- int stat = 0;
- if (!key)
- return -EINVAL;
-
- switch (sock->sock_type) {
- case UHD_DPDK_SOCK_UDP:
- stat = _uhd_dpdk_udp_rx_key(sock, key);
- break;
- default:
- stat = -EINVAL;
- }
- return stat;
-}
-/************************************************
- * API calls
- */
-struct uhd_dpdk_socket* uhd_dpdk_sock_open(unsigned int portid,
- enum uhd_dpdk_sock_type t, void *sockarg)
-{
- if (!ctx || (t >= UHD_DPDK_SOCK_TYPE_COUNT)) {
- return NULL;
- }
-
- struct uhd_dpdk_port *port = find_port(portid);
- if (!port) {
- return NULL;
- }
-
- if (!port->ipv4_addr) {
- RTE_LOG(WARNING, EAL, "Please set IPv4 address for port %u before opening socket\n", portid);
- return NULL;
- }
-
- struct uhd_dpdk_config_req *req = (struct uhd_dpdk_config_req *) rte_zmalloc(NULL, sizeof(*req), 0);
- if (!req) {
- return NULL;
- }
-
- req->waiter = uhd_dpdk_waiter_alloc(UHD_DPDK_WAIT_SIMPLE);
- if (!req->waiter) {
- req->retval = -ENOMEM;
- goto sock_open_end;
- }
-
- struct uhd_dpdk_socket *s = (struct uhd_dpdk_socket *) rte_zmalloc(NULL, sizeof(*s), 0);
- if (!s) {
- goto sock_open_end;
- }
-
- s->port = port;
- req->sock = s;
- req->req_type = UHD_DPDK_SOCK_OPEN;
- req->sock->sock_type = t;
- req->retval = -ETIMEDOUT;
-
- switch (t) {
- case UHD_DPDK_SOCK_UDP:
- uhd_dpdk_udp_open(req, sockarg);
- break;
- default:
- break;
- }
-
- if (req->retval) {
- rte_free(s);
- s = NULL;
- }
-
-sock_open_end:
- if (req->waiter)
- uhd_dpdk_waiter_put(req->waiter);
- rte_free(req);
- return s;
-}
-
-int uhd_dpdk_sock_close(struct uhd_dpdk_socket *sock)
-{
- if (!ctx || !sock)
- return -EINVAL;
-
- struct uhd_dpdk_config_req *req = (struct uhd_dpdk_config_req *) rte_zmalloc(NULL, sizeof(*req), 0);
- if (!req)
- return -ENOMEM;
-
- req->waiter = uhd_dpdk_waiter_alloc(UHD_DPDK_WAIT_SIMPLE);
- if (!req->waiter) {
- rte_free(req);
- return -ENOMEM;
- }
- req->sock = sock;
- req->req_type = UHD_DPDK_SOCK_CLOSE;
- req->retval = -ETIMEDOUT;
-
- switch (sock->sock_type) {
- case UHD_DPDK_SOCK_UDP:
- uhd_dpdk_udp_close(req);
- break;
- default:
- break;
- }
-
- uhd_dpdk_waiter_put(req->waiter);
-
- if (req->retval) {
- rte_free(req);
- return req->retval;
- }
-
- rte_free(sock);
- return 0;
-}
-
-int uhd_dpdk_request_tx_bufs(struct uhd_dpdk_socket *sock, struct rte_mbuf **bufs,
- unsigned int num_bufs, int timeout)
-{
- if (!sock || !bufs || !num_bufs) {
- return -EINVAL;
- }
- *bufs = NULL;
-
- if (!sock->tx_queue)
- return -EINVAL;
-
- if (!sock->tx_queue->freebufs)
- return -EINVAL;
-
- struct rte_ring *freebufs = sock->tx_queue->freebufs;
- unsigned int num_tx = rte_ring_count(freebufs);
- if (timeout != 0 && num_tx == 0) {
- struct uhd_dpdk_wait_req *req =
- uhd_dpdk_waiter_alloc(UHD_DPDK_WAIT_TX_BUF);
- req->sock = sock;
- uhd_dpdk_waiter_wait(req, timeout, sock->port->parent);
- uhd_dpdk_waiter_put(req);
- num_tx = rte_ring_count(freebufs);
- if (!num_tx)
- return -ETIMEDOUT;
- }
- num_tx = (num_tx < num_bufs) ? num_tx : num_bufs;
- if (rte_ring_dequeue_bulk(freebufs, (void **) bufs, num_tx, NULL) == 0)
- return -ENOENT;
- sock->tx_buf_count += num_tx;
- return num_tx;
-}
-
-int uhd_dpdk_send(struct uhd_dpdk_socket *sock, struct rte_mbuf **bufs,
- unsigned int num_bufs)
-{
- if (!sock || !bufs || !num_bufs)
- return -EINVAL;
- if (!sock->tx_queue)
- return -EINVAL;
- if (!sock->tx_queue->queue)
- return -EINVAL;
- struct rte_ring *tx_ring = sock->tx_queue->queue;
- unsigned int num_tx = rte_ring_free_count(tx_ring);
- num_tx = (num_tx < num_bufs) ? num_tx : num_bufs;
- switch (sock->sock_type) {
- case UHD_DPDK_SOCK_UDP:
- for (unsigned int i = 0; i < num_tx; i++) {
- uhd_dpdk_udp_prep(sock, bufs[i]);
- }
- break;
- default:
- RTE_LOG(ERR, USER1, "%s: Unsupported sock type\n", __func__);
- return -EINVAL;
- }
- int status = rte_ring_enqueue_bulk(tx_ring, (void **) bufs, num_tx, NULL);
- if (status == 0) {
- RTE_LOG(ERR, USER1, "Invalid shared usage of TX ring detected\n");
- return status;
- }
- sock->tx_buf_count -= num_tx;
- return num_tx;
-}
-
-int uhd_dpdk_recv(struct uhd_dpdk_socket *sock, struct rte_mbuf **bufs,
- unsigned int num_bufs, int timeout)
-{
- if (!sock || !bufs || !num_bufs)
- return -EINVAL;
- if (!sock->rx_ring)
- return -EINVAL;
-
- unsigned int num_rx = rte_ring_count(sock->rx_ring);
- if (timeout != 0 && num_rx == 0) {
- struct uhd_dpdk_wait_req *req =
- uhd_dpdk_waiter_alloc(UHD_DPDK_WAIT_RX);
- req->sock = sock;
- uhd_dpdk_waiter_wait(req, timeout, sock->port->parent);
- uhd_dpdk_waiter_put(req);
- num_rx = rte_ring_count(sock->rx_ring);
- if (!num_rx)
- return -ETIMEDOUT;
- }
-
- num_rx = (num_rx < num_bufs) ? num_rx : num_bufs;
- if (num_rx) {
- unsigned int avail = 0;
- unsigned int status = rte_ring_dequeue_bulk(sock->rx_ring,
- (void **) bufs, num_rx, &avail);
- if (status == 0) {
- RTE_LOG(ERR, USER1, "Invalid shared usage of RX ring detected\n");
- RTE_LOG(ERR, USER1, "Requested %u, but %u available\n",
- num_rx, avail);
- return -ENOENT;
- }
- }
- return num_rx;
-}
-
-void uhd_dpdk_free_buf(struct rte_mbuf *buf)
-{
- rte_pktmbuf_free(buf);
-}
-
-void * uhd_dpdk_buf_to_data(struct uhd_dpdk_socket *sock, struct rte_mbuf *buf)
-{
- if (!sock || !buf)
- return NULL;
-
- /* TODO: Support for more types? */
- switch (sock->sock_type) {
- case UHD_DPDK_SOCK_UDP:
- return rte_pktmbuf_mtod_offset(buf, void *, sizeof(struct ether_hdr) +
- sizeof(struct ipv4_hdr) +
- sizeof(struct udp_hdr));
- default:
- return NULL;
- }
-}
-
-
-int uhd_dpdk_get_len(struct uhd_dpdk_socket *sock, struct rte_mbuf *buf)
-{
- if (!sock || !buf)
- return -EINVAL;
-
- if (sock->sock_type != UHD_DPDK_SOCK_UDP)
- return -EINVAL;
-
- struct udp_hdr *hdr = (struct udp_hdr *) ((uint8_t *) uhd_dpdk_buf_to_data(sock, buf) - sizeof(struct udp_hdr));
- if (!hdr)
- return -EINVAL;
-
- /* Report dgram length - header */
- return ntohs(hdr->dgram_len) - 8;
-}
-
-int uhd_dpdk_get_src_ipv4(struct uhd_dpdk_socket *sock, struct rte_mbuf *buf,
- uint32_t *ipv4_addr)
-{
- if (!sock || !buf || !ipv4_addr)
- return -EINVAL;
-
- if (sock->sock_type != UHD_DPDK_SOCK_UDP)
- return -EINVAL;
-
- struct ipv4_hdr *hdr = rte_pktmbuf_mtod_offset(buf, struct ipv4_hdr *,
- sizeof(struct ether_hdr));
-
- *ipv4_addr = hdr->src_addr;
- return 0;
-}
-
-int uhd_dpdk_get_drop_count(struct uhd_dpdk_socket *sock, size_t *count)
-{
- if (!sock)
- return -EINVAL;
- if (sock->sock_type != UHD_DPDK_SOCK_UDP)
- return -EINVAL;
- if (!sock->priv)
- return -ENODEV;
-
- struct uhd_dpdk_udp_priv *pdata = (struct uhd_dpdk_udp_priv *) sock->priv;
- *count = pdata->dropped_pkts;
- return 0;
-}
-
-int uhd_dpdk_get_xfer_count(struct uhd_dpdk_socket *sock, size_t *count)
-{
- if (!sock)
- return -EINVAL;
- if (sock->sock_type != UHD_DPDK_SOCK_UDP)
- return -EINVAL;
- if (!sock->priv)
- return -ENODEV;
-
- struct uhd_dpdk_udp_priv *pdata = (struct uhd_dpdk_udp_priv *) sock->priv;
- *count = pdata->xferd_pkts;
- return 0;
-}
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_fops.h b/host/lib/transport/uhd-dpdk/uhd_dpdk_fops.h
deleted file mode 100644
index 66adb0f54..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_fops.h
+++ /dev/null
@@ -1,23 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#ifndef _UHD_DPDK_FOPS_H_
-#define _UHD_DPDK_FOPS_H_
-
-#include "uhd_dpdk_ctx.h"
-
-int _uhd_dpdk_config_req_compl(struct uhd_dpdk_config_req *req, int retval);
-
-int _uhd_dpdk_sock_setup(struct uhd_dpdk_config_req *req);
-int _uhd_dpdk_sock_release(struct uhd_dpdk_config_req *req);
-
-/*
- * Get key for RX table corresponding to this socket
- *
- * This is primarily used to get access to the waiter entry
- */
-int _uhd_dpdk_sock_rx_key(struct uhd_dpdk_socket *sock,
- struct uhd_dpdk_ipv4_5tuple *key);
-#endif /* _UHD_DPDK_FOPS_H_ */
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_udp.c b/host/lib/transport/uhd-dpdk/uhd_dpdk_udp.c
deleted file mode 100644
index 9429cd6cb..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_udp.c
+++ /dev/null
@@ -1,525 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#include "uhd_dpdk_fops.h"
-#include "uhd_dpdk_udp.h"
-#include "uhd_dpdk_driver.h"
-#include "uhd_dpdk_wait.h"
-#include <rte_ring.h>
-#include <rte_malloc.h>
-#include <unistd.h>
-#include <sys/syscall.h>
-#include <arpa/inet.h>
-
-#define MAX_UDP_PORT 65535
-
-/************************************************
- * I/O thread ONLY
- */
-
-static int _alloc_txq(struct uhd_dpdk_port *port, pthread_t tid,
- struct uhd_dpdk_tx_queue **queue, size_t num_bufs)
-{
- *queue = NULL;
- struct uhd_dpdk_tx_queue *q = rte_zmalloc(NULL, sizeof(*q), 0);
- if (!q) {
- RTE_LOG(ERR, USER1, "%s: Cannot allocate TX queue\n", __func__);
- return -ENOMEM;
- }
- q->tid = tid;
- LIST_INIT(&q->tx_list);
-
- char name[32];
- snprintf(name, sizeof(name), "tx_q%u.%0lx", port->id, (unsigned long) q);
- q->queue = rte_ring_create(
- name,
- num_bufs,
- rte_socket_id(),
- RING_F_SC_DEQ | RING_F_SP_ENQ
- );
- snprintf(name, sizeof(name), "free_q%u.%0lx", port->id, (unsigned long) q);
- q->freebufs = rte_ring_create(
- name,
- num_bufs,
- rte_socket_id(),
- RING_F_SC_DEQ | RING_F_SP_ENQ
- );
- /* Set up retry queue */
- snprintf(name, sizeof(name), "redo_q%u.%0lx", port->id, (unsigned long) q);
- q->retry_queue = rte_ring_create(
- name,
- num_bufs,
- rte_socket_id(),
- RING_F_SC_DEQ | RING_F_SP_ENQ
- );
-
- if (!q->queue || !q->freebufs || !q->retry_queue) {
- RTE_LOG(ERR, USER1, "%s: Cannot allocate TX rings\n", __func__);
- if (q->queue)
- rte_ring_free(q->queue);
- if (q->freebufs)
- rte_ring_free(q->freebufs);
- if (q->retry_queue)
- rte_ring_free(q->retry_queue);
- rte_free(q);
- return -ENOMEM;
- }
-
- do {
- struct rte_mbuf *bufs[UHD_DPDK_TXQ_SIZE];
- num_bufs = rte_ring_free_count(q->freebufs);
- if (num_bufs > 0) {
- num_bufs = num_bufs > UHD_DPDK_TXQ_SIZE ? UHD_DPDK_TXQ_SIZE : num_bufs;
- int buf_stat = rte_pktmbuf_alloc_bulk(port->parent->tx_pktbuf_pool, bufs, num_bufs);
- if (buf_stat) {
- RTE_LOG(ERR, USER1, "%s: Cannot allocate packet buffers\n", __func__);
- goto unwind_txq;
- }
- unsigned int enqd = rte_ring_enqueue_bulk(q->freebufs, (void **) bufs, num_bufs, NULL);
- if (enqd != num_bufs) {
- RTE_LOG(ERR, USER1, "%s: Cannot enqueue freebufs\n", __func__);
- goto unwind_txq;
- }
- }
- } while (num_bufs > 0);
- LIST_INSERT_HEAD(&port->txq_list, q, entry);
- *queue = q;
- return 0;
-
-unwind_txq:
- while (!rte_ring_empty(q->freebufs)) {
- struct rte_mbuf *buf;
- if (rte_ring_dequeue(q->freebufs, (void **) &buf) == 0)
- rte_free(buf);
- }
- rte_ring_free(q->freebufs);
- rte_ring_free(q->queue);
- rte_ring_free(q->retry_queue);
- rte_free(q);
- return -ENOENT;
-}
-
-/* Finish setting up UDP socket (unless ARP needs to be done)
- * Not multi-thread safe!
- * This call should only be used by the thread servicing the port
- * In addition, the code below assumes simplex sockets and unique receive ports
- * FIXME: May need some way to help clean up abandoned socket requests (refcnt check...)
- */
-int _uhd_dpdk_udp_setup(struct uhd_dpdk_config_req *req)
-{
- int retval = 0;
- struct uhd_dpdk_socket *sock = req->sock;
- struct uhd_dpdk_udp_priv *pdata = sock->priv;
- struct uhd_dpdk_port *port = req->sock->port;
-
- struct uhd_dpdk_ipv4_5tuple ht_key = {
- .sock_type = UHD_DPDK_SOCK_UDP,
- .src_ip = 0,
- .src_port = 0,
- .dst_ip = 0,
- .dst_port = pdata->dst_port
- };
-
- /* Are we doing RX? */
- if (sock->rx_ring) {
- /* Add to rx table */
- if (pdata->dst_port == 0) {
- /* Assign unused one in a very slow fashion */
- for (uint16_t i = MAX_UDP_PORT; i > 0; i--) {
- ht_key.dst_port = htons(i);
- if (rte_hash_lookup(port->rx_table, &ht_key) == -ENOENT) {
- pdata->dst_port = htons(i);
- break;
- }
- }
- }
-
- /* Is the port STILL invalid? */
- if (pdata->dst_port == 0) {
- RTE_LOG(ERR, USER1, "%s: No available UDP ports\n", __func__);
- _uhd_dpdk_config_req_compl(req, -EADDRINUSE);
- return -EADDRINUSE;
- }
-
- ht_key.dst_port = pdata->dst_port;
- if (rte_hash_lookup(port->rx_table, &ht_key) > 0) {
- RTE_LOG(ERR, USER1, "%s: Cannot add to RX table\n", __func__);
- _uhd_dpdk_config_req_compl(req, -EADDRINUSE);
- return -EADDRINUSE;
- }
-
- size_t num_bufs = (pdata->xferd_pkts < (UHD_DPDK_RX_BURST_SIZE + 1)) ?
- UHD_DPDK_RX_BURST_SIZE + 1 : pdata->xferd_pkts;
- pdata->xferd_pkts = 0;
- char name[32];
- snprintf(name, sizeof(name), "rx_ring_udp_%u.%u", port->id, ntohs(pdata->dst_port));
- sock->rx_ring = rte_ring_create(
- name,
- num_bufs,
- rte_socket_id(),
- RING_F_SC_DEQ | RING_F_SP_ENQ
- );
- if (!sock->rx_ring) {
- RTE_LOG(ERR, USER1, "%s: Cannot allocate RX ring\n", __func__);
- _uhd_dpdk_config_req_compl(req, -ENOMEM);
- return -ENOMEM;
- }
-
- struct uhd_dpdk_rx_entry *entry = (struct uhd_dpdk_rx_entry *)
- rte_zmalloc(NULL, sizeof(*entry), 0);
- if (!entry) {
- rte_ring_free(sock->rx_ring);
- RTE_LOG(ERR, USER1, "%s: Cannot create RX entry\n", __func__);
- _uhd_dpdk_config_req_compl(req, -ENOMEM);
- return -ENOMEM;
- }
- entry->sock = sock;
- entry->waiter = NULL;
-
- retval = rte_hash_add_key_data(port->rx_table, &ht_key, entry);
- if (retval != 0) {
- RTE_LOG(WARNING, TABLE, "Could not add new RX socket to port %d: %d\n", port->id, retval);
- rte_free(entry);
- rte_ring_free(sock->rx_ring);
- _uhd_dpdk_config_req_compl(req, retval);
- return retval;
- }
- _uhd_dpdk_config_req_compl(req, 0);
- }
-
- /* Are we doing TX? */
- if (sock->tx_queue) {
- size_t num_bufs = (pdata->xferd_pkts < (UHD_DPDK_TX_BURST_SIZE + 1)) ?
- UHD_DPDK_TX_BURST_SIZE + 1 : pdata->xferd_pkts;
- pdata->xferd_pkts = 0;
- sock->tx_queue = NULL;
- struct uhd_dpdk_tx_queue *q = NULL;
- // FIXME Not sharing txq across all thread's sockets for now
- //LIST_FOREACH(q, &port->txq_list, entry) {
- // if (pthread_equal(q->tid, sock->tid)) {
- // LIST_INSERT_HEAD(&q->tx_list, sock, tx_entry);
- // sock->tx_ring = q->queue;
- // sock->rx_ring = q->freebufs;
- // break;
- // }
- //}
- if (!sock->tx_queue) {
- retval = _alloc_txq(port, sock->tid, &q, num_bufs);
- if (retval) {
- _uhd_dpdk_config_req_compl(req, retval);
- return retval;
- }
- sock->tx_queue = q;
- }
- /* If a broadcast type, just finish setup and return */
- if (is_broadcast(port, pdata->dst_ipv4_addr)) {
- LIST_INSERT_HEAD(&q->tx_list, sock, tx_entry);
- _uhd_dpdk_config_req_compl(req, 0);
- return 0;
- }
- /* Otherwise... Check for entry in ARP table */
- struct uhd_dpdk_arp_entry *entry = NULL;
- int arp_table_stat = rte_hash_lookup_data(port->arp_table, &pdata->dst_ipv4_addr, (void **) &entry);
- if (entry) {
- /* Check for null entry */
- if ((entry->mac_addr.addr_bytes[0] == 0xFF) &&
- (entry->mac_addr.addr_bytes[1] == 0xFF) &&
- (entry->mac_addr.addr_bytes[2] == 0xFF) &&
- (entry->mac_addr.addr_bytes[3] == 0xFF) &&
- (entry->mac_addr.addr_bytes[4] == 0xFF) &&
- (entry->mac_addr.addr_bytes[5] == 0xFF)) {
- arp_table_stat = -ENOENT;
- }
- } else {
- /* No entry -> Add null entry */
- entry = rte_zmalloc(NULL, sizeof(*entry), 0);
- if (!entry) {
- RTE_LOG(ERR, USER1, "%s: Cannot allocate ARP entry\n", __func__);
- _uhd_dpdk_config_req_compl(req, -ENOMEM);
- return -ENOMEM;
- }
- memset(entry->mac_addr.addr_bytes, 0xFF, ETHER_ADDR_LEN);
- LIST_INIT(&entry->pending_list);
-
- if (rte_hash_add_key_data(port->arp_table, &pdata->dst_ipv4_addr, entry) < 0) {
- rte_free(entry);
- RTE_LOG(ERR, USER1, "%s: Cannot add entry to ARP table\n", __func__);
- _uhd_dpdk_config_req_compl(req, -ENOMEM);
- return -ENOMEM;
- }
- }
-
- /* Was there a valid address? */
- if (arp_table_stat == -ENOENT) {
- /* Get valid address and have requestor continue waiting */
- int arp_stat = 0;
- do { /* Keep trying to send request if no descriptor */
- arp_stat = _uhd_dpdk_arp_request(port, pdata->dst_ipv4_addr);
- } while (arp_stat == -EAGAIN);
-
- if (arp_stat) {
- /* Config request errors out */
- RTE_LOG(ERR, USER1, "%s: Cannot make ARP request\n", __func__);
- _uhd_dpdk_config_req_compl(req, arp_stat);
- return arp_stat;
- }
- /* Append req to pending list. Will signal later. */
- LIST_INSERT_HEAD(&entry->pending_list, req, entry);
- LIST_INSERT_HEAD(&q->tx_list, sock, tx_entry);
- } else {
- /* We have a valid address. All good. */
- LIST_INSERT_HEAD(&q->tx_list, sock, tx_entry);
- _uhd_dpdk_config_req_compl(req, 0);
- }
- }
- return 0;
-}
-
-int _uhd_dpdk_udp_release(struct uhd_dpdk_config_req *req)
-{
- struct uhd_dpdk_socket *sock = req->sock;
- if (req->sock == NULL) {
- RTE_LOG(ERR, USER1, "%s: no sock in req\n", __func__);
- return -EINVAL;
- }
- struct uhd_dpdk_port *port = req->sock->port;
- struct uhd_dpdk_config_req *conf_req = NULL;
- struct uhd_dpdk_udp_priv *pdata = (struct uhd_dpdk_udp_priv *) sock->priv;
- if (pdata == NULL) {
- RTE_LOG(ERR, USER1, "%s: no pdata in sock\n", __func__);
- return -EINVAL;
- }
- if (sock->tx_queue) {
- // FIXME not sharing buffers anymore
- LIST_REMOVE(sock->tx_queue, entry);
- rte_ring_free(sock->tx_queue->queue);
- rte_ring_free(sock->tx_queue->retry_queue);
-
- /* Remove from tx_list */
- LIST_REMOVE(sock, tx_entry);
- /* Check for entry in ARP table */
- struct uhd_dpdk_arp_entry *entry = NULL;
- rte_hash_lookup_data(port->arp_table, &pdata->dst_ipv4_addr, (void **) &entry);
- if (entry) {
- LIST_FOREACH(conf_req, &entry->pending_list, entry) {
- if (conf_req->sock == sock) {
- LIST_REMOVE(conf_req, entry);
- break;
- }
- }
- }
-
- // FIXME not sharing buffers anymore
- // Remove outstanding buffers from TX queue's freebufs */
- unsigned int bufs = rte_ring_count(sock->tx_queue->freebufs);
- for (unsigned int i = 0; i < bufs; i++) {
- struct rte_mbuf *buf = NULL;
- if (rte_ring_dequeue(sock->tx_queue->freebufs, (void **) &buf)) {
- RTE_LOG(ERR, USER1, "%s: Could not dequeue freebufs\n", __func__);
- } else if (buf) {
- rte_pktmbuf_free(buf);
- }
- }
- rte_ring_free(sock->tx_queue->freebufs);
- rte_free(sock->tx_queue);
-
- /* Add outstanding buffers back to TX queue's freebufs */
- //struct rte_mbuf *freebufs[UHD_DPDK_TXQ_SIZE];
- //int status = rte_pktmbuf_alloc_bulk(port->parent->tx_pktbuf_pool, freebufs, sock->tx_buf_count);
- //if (status) {
- // RTE_LOG(ERR, USER1, "%d %s: Could not restore %u TX buffers in bulk!\n", status, __func__, sock->tx_buf_count);
- //}
-
- //unsigned int enqd = rte_ring_enqueue_bulk(sock->rx_ring, (void **) freebufs, sock->tx_buf_count, NULL);
- //if (enqd != (unsigned int) sock->tx_buf_count) {
- // RTE_LOG(ERR, USER1, "%s: Could not enqueue TX buffers!\n", __func__);
- // return status;
- //}
- } else if (sock->rx_ring) {
- struct uhd_dpdk_ipv4_5tuple ht_key = {
- .sock_type = UHD_DPDK_SOCK_UDP,
- .src_ip = 0,
- .src_port = 0,
- .dst_ip = 0,
- .dst_port = pdata->dst_port
- };
- struct uhd_dpdk_rx_entry *entry = NULL;
- rte_hash_lookup_data(port->rx_table, &ht_key, (void **) &entry);
- if (entry) {
- if (entry->waiter)
- uhd_dpdk_waiter_put(entry->waiter);
- rte_free(entry);
- }
- rte_hash_del_key(port->rx_table, &ht_key);
- struct rte_mbuf *mbuf = NULL;
- while (rte_ring_dequeue(sock->rx_ring, (void **) &mbuf) == 0) {
- rte_pktmbuf_free(mbuf);
- }
- rte_ring_free(sock->rx_ring);
- }
-
- _uhd_dpdk_config_req_compl(req, 0);
- return 0;
-}
-
-int _uhd_dpdk_udp_rx_key(struct uhd_dpdk_socket *sock,
- struct uhd_dpdk_ipv4_5tuple *key)
-{
- struct uhd_dpdk_udp_priv *pdata = (struct uhd_dpdk_udp_priv *) sock->priv;
- if (!pdata)
- return -EINVAL;
- key->sock_type = UHD_DPDK_SOCK_UDP;
- key->src_ip = 0;
- key->src_port = 0;
- key->dst_ip = 0;
- key->dst_port = pdata->dst_port;
- return 0;
-}
-
-/* Configure a socket for UDP
- */
-void uhd_dpdk_udp_open(struct uhd_dpdk_config_req *req,
- struct uhd_dpdk_sockarg_udp *arg)
-{
- if (!req)
- return;
-
- if (!arg) {
- req->retval = -EINVAL;
- return;
- }
-
- struct uhd_dpdk_socket *sock = req->sock;
- sock->tid = pthread_self();
-
- /* Create private data */
- struct uhd_dpdk_udp_priv *data = (struct uhd_dpdk_udp_priv *) rte_zmalloc(NULL, sizeof(*data), 0);
- if (!data) {
- req->retval = -ENOMEM;
- return;
- }
- sock->priv = data;
-
- data->dst_ipv4_addr = arg->dst_addr;
- if (arg->is_tx) {
- data->src_port = arg->local_port;
- data->dst_port = arg->remote_port;
- sock->tx_queue = (struct uhd_dpdk_tx_queue *) sock;
- data->xferd_pkts = arg->num_bufs;
- } else {
- data->src_port = arg->remote_port;
- data->dst_port = arg->local_port;
- sock->rx_ring = (struct rte_ring *) sock;
- data->xferd_pkts = arg->num_bufs;
- data->filter_bcast = arg->filter_bcast;
- }
-
- /* TODO: Add support for I/O thread calling (skip locking and sleep) */
- /* Add to port's config queue */
- int status = uhd_dpdk_config_req_submit(req, -1, sock->port->parent);
- if (status)
- req->retval = status;
-
- if (req->retval)
- rte_free(data);
-}
-
-void uhd_dpdk_udp_close(struct uhd_dpdk_config_req *req)
-{
- if (!req)
- return;
-
- uhd_dpdk_config_req_submit(req, -1, req->sock->port->parent);
- rte_free(req->sock->priv);
-}
-
-/*
- * Note: I/O thread will fill in destination MAC address (doesn't happen here)
- */
-static void uhd_dpdk_ipv4_prep(struct uhd_dpdk_port *port,
- struct rte_mbuf *mbuf,
- uint32_t dst_ipv4_addr,
- uint8_t proto_id,
- uint32_t payload_len)
-{
- struct ether_hdr *eth_hdr = rte_pktmbuf_mtod(mbuf, struct ether_hdr *);
- struct ipv4_hdr *ip_hdr = (struct ipv4_hdr *) &eth_hdr[1];
-
- ether_addr_copy(&port->mac_addr, &eth_hdr->s_addr);
- eth_hdr->ether_type = rte_cpu_to_be_16(ETHER_TYPE_IPv4);
-
- ip_hdr->version_ihl = 0x40 | 5;
- ip_hdr->type_of_service = 0;
- ip_hdr->total_length = rte_cpu_to_be_16(20 + payload_len);
- ip_hdr->packet_id = 0;
- ip_hdr->fragment_offset = rte_cpu_to_be_16(IPV4_HDR_DF_FLAG);
- ip_hdr->time_to_live = 64;
- ip_hdr->next_proto_id = proto_id;
- ip_hdr->hdr_checksum = 0; // Require HW offload
- ip_hdr->src_addr = port->ipv4_addr;
- ip_hdr->dst_addr = dst_ipv4_addr;
-
- mbuf->ol_flags = PKT_TX_IP_CKSUM | PKT_TX_IPV4;
- mbuf->l2_len = sizeof(struct ether_hdr);
- mbuf->l3_len = sizeof(struct ipv4_hdr);
- mbuf->pkt_len = sizeof(struct ether_hdr) + sizeof(struct ipv4_hdr) + payload_len;
- mbuf->data_len = sizeof(struct ether_hdr) + sizeof(struct ipv4_hdr) + payload_len;
-}
-
-int uhd_dpdk_udp_prep(struct uhd_dpdk_socket *sock,
- struct rte_mbuf *mbuf)
-{
- struct ether_hdr *eth_hdr;
- struct ipv4_hdr *ip_hdr;
- struct udp_hdr *tx_hdr;
- struct uhd_dpdk_port *port = sock->port;
- struct uhd_dpdk_udp_priv *pdata = (struct uhd_dpdk_udp_priv *) sock->priv;
-
- if (unlikely(mbuf == NULL || pdata == NULL || port == NULL))
- return -EINVAL;
-
- uint32_t udp_data_len = mbuf->data_len;
- uhd_dpdk_ipv4_prep(port,
- mbuf,
- pdata->dst_ipv4_addr,
- 0x11,
- 8 + udp_data_len);
-
- eth_hdr = rte_pktmbuf_mtod(mbuf, struct ether_hdr *);
- ip_hdr = (struct ipv4_hdr *) &eth_hdr[1];
- tx_hdr = (struct udp_hdr *) &ip_hdr[1];
-
- tx_hdr->src_port = pdata->src_port;
- tx_hdr->dst_port = pdata->dst_port;
- tx_hdr->dgram_len = rte_cpu_to_be_16(8 + udp_data_len);
- tx_hdr->dgram_cksum = 0;
- mbuf->l4_len = sizeof(struct udp_hdr);
-
- return 0;
-}
-
-int uhd_dpdk_udp_get_info(struct uhd_dpdk_socket *sock,
- struct uhd_dpdk_sockarg_udp *sockarg)
-{
- if (unlikely(sock == NULL || sockarg == NULL))
- return -EINVAL;
- if (sock->sock_type != UHD_DPDK_SOCK_UDP)
- return -EINVAL;
-
- struct uhd_dpdk_udp_priv *pdata = (struct uhd_dpdk_udp_priv *) sock->priv;
- if (sock->tx_queue) {
- sockarg->is_tx = true;
- sockarg->local_port = pdata->src_port;
- sockarg->remote_port = pdata->dst_port;
- sockarg->dst_addr = pdata->dst_ipv4_addr;
- } else {
- sockarg->is_tx = false;
- sockarg->local_port = pdata->dst_port;
- sockarg->remote_port = pdata->src_port;
- sockarg->dst_addr = 0;
- }
- return 0;
-}
-
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_udp.h b/host/lib/transport/uhd-dpdk/uhd_dpdk_udp.h
deleted file mode 100644
index d7ca5609b..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_udp.h
+++ /dev/null
@@ -1,40 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#ifndef _UHD_DPDK_UDP_H_
-#define _UHD_DPDK_UDP_H_
-
-#include "uhd_dpdk_ctx.h"
-#include <rte_udp.h>
-
-struct uhd_dpdk_udp_priv {
- uint16_t src_port;
- uint16_t dst_port;
- uint32_t dst_ipv4_addr;
- size_t dropped_pkts;
- size_t xferd_pkts;
- bool filter_bcast;
- /* TODO: Cache destination address ptr to avoid ARP table lookup cost? */
- //struct uhd_dpdk_arp_entry *arp_entry;
-};
-
-int _uhd_dpdk_udp_setup(struct uhd_dpdk_config_req *req);
-int _uhd_dpdk_udp_release(struct uhd_dpdk_config_req *req);
-
-void uhd_dpdk_udp_open(struct uhd_dpdk_config_req *req,
- struct uhd_dpdk_sockarg_udp *arg);
-void uhd_dpdk_udp_close(struct uhd_dpdk_config_req *req);
-
-int uhd_dpdk_udp_prep(struct uhd_dpdk_socket *sock,
- struct rte_mbuf *mbuf);
-
-/*
- * Get key for RX table corresponding to this socket
- *
- * This is primarily used to get access to the waiter entry
- */
-int _uhd_dpdk_udp_rx_key(struct uhd_dpdk_socket *sock,
- struct uhd_dpdk_ipv4_5tuple *key);
-#endif /* _UHD_DPDK_UDP_H_ */
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_wait.c b/host/lib/transport/uhd-dpdk/uhd_dpdk_wait.c
deleted file mode 100644
index c00eaa3c4..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_wait.c
+++ /dev/null
@@ -1,114 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#include "uhd_dpdk_wait.h"
-
-int _uhd_dpdk_waiter_wake(struct uhd_dpdk_wait_req *req,
- struct uhd_dpdk_thread *t)
-{
- int stat = pthread_mutex_trylock(&req->mutex);
- if (stat) {
- if (rte_ring_full(t->waiter_ring)) {
- RTE_LOG(ERR, USER2, "%s: Could not lock req mutex\n", __func__);
- return -ENOBUFS;
- } else {
- req->reason = UHD_DPDK_WAIT_SIMPLE;
- rte_ring_enqueue(t->waiter_ring, req);
- return -EAGAIN;
- }
- }
- stat = pthread_cond_signal(&req->cond);
- if (stat)
- RTE_LOG(ERR, USER2, "%s: Could not signal req cond\n", __func__);
- pthread_mutex_unlock(&req->mutex);
- uhd_dpdk_waiter_put(req);
- return stat;
-}
-
-struct uhd_dpdk_wait_req *uhd_dpdk_waiter_alloc(enum uhd_dpdk_wait_type reason)
-{
- struct uhd_dpdk_wait_req *req;
- req = (struct uhd_dpdk_wait_req *) rte_zmalloc(NULL, sizeof(*req), 0);
- if (!req)
- return NULL;
-
- pthread_mutex_init(&req->mutex, NULL);
- pthread_condattr_t condattr;
- pthread_condattr_init(&condattr);
- pthread_condattr_setclock(&condattr, CLOCK_MONOTONIC);
- pthread_cond_init(&req->cond, &condattr);
- rte_atomic32_set(&req->refcnt, 1);
- req->reason = reason;
- return req;
-}
-
-static inline void uhd_dpdk_waiter_prepare(struct uhd_dpdk_wait_req *req)
-{
- pthread_mutex_lock(&req->mutex);
- /* Get a reference here, to be consumed by other thread (handshake) */
- uhd_dpdk_waiter_get(req);
-}
-
-static inline int uhd_dpdk_waiter_submit(struct uhd_dpdk_wait_req *req,
- int timeout)
-{
- int retval = 0;
- if (timeout < 0) {
- retval = pthread_cond_wait(&req->cond, &req->mutex);
- } else {
- struct timespec timeout_spec;
- clock_gettime(CLOCK_MONOTONIC, &timeout_spec);
- timeout_spec.tv_sec += timeout/1000000;
- timeout_spec.tv_nsec += (timeout % 1000000)*1000;
- if (timeout_spec.tv_nsec > 1000000000) {
- timeout_spec.tv_sec++;
- timeout_spec.tv_nsec -= 1000000000;
- }
- retval = pthread_cond_timedwait(&req->cond, &req->mutex, &timeout_spec);
- }
- return retval;
-}
-
-int uhd_dpdk_waiter_wait(struct uhd_dpdk_wait_req *req, int timeout,
- struct uhd_dpdk_thread *t)
-{
- int ret;
- if (!req || !t)
- return -EINVAL;
-
- uhd_dpdk_waiter_prepare(req);
-
- ret = rte_ring_enqueue(t->waiter_ring, req);
- if (ret) {
- uhd_dpdk_waiter_put(req);
- pthread_mutex_unlock(&req->mutex);
- return ret;
- }
-
- uhd_dpdk_waiter_submit(req, timeout);
- pthread_mutex_unlock(&req->mutex);
- return 0;
-}
-
-int uhd_dpdk_config_req_submit(struct uhd_dpdk_config_req *req,
- int timeout, struct uhd_dpdk_thread *t)
-{
- int ret;
- if (!req || !t)
- return -EINVAL;
-
- uhd_dpdk_waiter_prepare(req->waiter);
-
- ret = rte_ring_enqueue(t->sock_req_ring, req);
- if (ret) {
- uhd_dpdk_waiter_put(req->waiter);
- pthread_mutex_unlock(&req->waiter->mutex);
- return ret;
- }
-
- uhd_dpdk_waiter_submit(req->waiter, timeout);
- pthread_mutex_unlock(&req->waiter->mutex);
- return 0;
-}
diff --git a/host/lib/transport/uhd-dpdk/uhd_dpdk_wait.h b/host/lib/transport/uhd-dpdk/uhd_dpdk_wait.h
deleted file mode 100644
index 465608810..000000000
--- a/host/lib/transport/uhd-dpdk/uhd_dpdk_wait.h
+++ /dev/null
@@ -1,62 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-#ifndef _UHD_DPDK_WAIT_H_
-#define _UHD_DPDK_WAIT_H_
-
-#include "uhd_dpdk_ctx.h"
-#include <rte_malloc.h>
-
-enum uhd_dpdk_wait_type {
- UHD_DPDK_WAIT_SIMPLE,
- UHD_DPDK_WAIT_RX,
- UHD_DPDK_WAIT_TX_BUF,
- UHD_DPDK_WAIT_TYPE_COUNT
-};
-
-struct uhd_dpdk_wait_req {
- enum uhd_dpdk_wait_type reason;
- struct uhd_dpdk_socket *sock;
- pthread_cond_t cond;
- pthread_mutex_t mutex;
- rte_atomic32_t refcnt; /* free resources only when refcnt = 0 */
-};
-
-static inline void uhd_dpdk_waiter_put(struct uhd_dpdk_wait_req *req)
-{
- if (rte_atomic32_dec_and_test(&req->refcnt)) {
- rte_free(req);
- }
-}
-
-static inline void uhd_dpdk_waiter_get(struct uhd_dpdk_wait_req *req)
-{
- rte_atomic32_inc(&req->refcnt);
-}
-
-/*
- * Attempt to wake thread
- * Re-enqueue waiter to thread's waiter_queue if fail
- */
-int _uhd_dpdk_waiter_wake(struct uhd_dpdk_wait_req *req,
- struct uhd_dpdk_thread *t);
-
-/*
- * Allocates wait request and sets refcnt to 1
- */
-struct uhd_dpdk_wait_req *uhd_dpdk_waiter_alloc(enum uhd_dpdk_wait_type reason);
-
-/*
- * Block and send wait request to thread t
- */
-int uhd_dpdk_waiter_wait(struct uhd_dpdk_wait_req *req, int timeout,
- struct uhd_dpdk_thread *t);
-
-/*
- * Block and submit config request to thread t
- */
-int uhd_dpdk_config_req_submit(struct uhd_dpdk_config_req *req,
- int timeout, struct uhd_dpdk_thread *t);
-#endif /* _UHD_DPDK_WAIT_H_ */