diff options
-rw-r--r-- | host/lib/include/uhdlib/transport/dpdk_common.hpp | 81 | ||||
-rw-r--r-- | host/lib/include/uhdlib/transport/dpdk_zero_copy.hpp | 45 | ||||
-rw-r--r-- | host/lib/include/uhdlib/transport/uhd-dpdk.h | 280 | ||||
-rw-r--r-- | host/lib/transport/dpdk_zero_copy.cpp | 505 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk.c | 467 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_ctx.h | 268 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_driver.c | 624 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_driver.h | 33 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_fops.c | 353 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_fops.h | 23 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_udp.c | 525 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_udp.h | 40 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_wait.c | 114 | ||||
-rw-r--r-- | host/lib/transport/uhd-dpdk/uhd_dpdk_wait.h | 62 |
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 *) ð_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, ð_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 *) ð_hdr[1]; - - ether_addr_copy(&port->mac_addr, ð_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 *) ð_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_ */ |