aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/include/uhdlib/transport/udp_dpdk_link.hpp
diff options
context:
space:
mode:
authorAlex Williams <alex.williams@ni.com>2019-12-01 21:58:13 -0800
committerBrent Stapleton <brent.stapleton@ettus.com>2019-12-20 16:32:22 -0800
commit4e38eef817813c1bbd8a9cf972e4cf0134d24308 (patch)
treef6200a048a7da5b7b588a4a9aae881ce7551825e /host/lib/include/uhdlib/transport/udp_dpdk_link.hpp
parent797d54bc2573688eebcb2c639cb07e4ab6d5ab9d (diff)
downloaduhd-4e38eef817813c1bbd8a9cf972e4cf0134d24308.tar.gz
uhd-4e38eef817813c1bbd8a9cf972e4cf0134d24308.tar.bz2
uhd-4e38eef817813c1bbd8a9cf972e4cf0134d24308.zip
dpdk: Add new DPDK stack to integrate with I/O services
docs: Update DPDK docs with new parameters: Parameter names have had their hyphens changed to underscores, and the I/O CPU argument is now named after the lcores and reflects the naming used by DPDK. transport: Add new udp_dpdk_link, based atop the new APIs: This link is tightly coupled with the DPDK I/O service. The link class carries all the address information to communicate with the other host, and it can send packets directly through the DPDK NIC ports. However, for receiving packets, the I/O service must pull the packets from the DMA queue and attach them to the appropriate link object. The link object merely formats the frame_buff object underneath, which is embedded in the rte_mbuf container. For get_recv_buff, the link will pull buffers only from its internal queue (the one filled by the I/O service). transport: Add DPDK-specific I/O service: The I/O service is split into two parts, the user threads and the I/O worker threads. The user threads submit requests through various appropriate queues, and the I/O threads perform all the I/O on their behalf. This includes routing UDP packets to the correct receiver and getting the MAC address of a destination (by performing the ARP request and handling the ARP replies). The DPDK context stores I/O services. The context spawns all I/O services on init(), and I/O services can be fetched from the dpdk_ctx object by using a port ID. I/O service clients: The clients have two lockless ring buffers. One is to get a buffer from the I/O service; the other is to release a buffer back to the I/O service. Threads sleeping on buffer I/O are kept in a separate list from the service queue and are processed in the course of doing RX or TX. The list nodes are embedded in the dpdk_io_if, and the head of the list is on the dpdk_io_service. The I/O service will transfer the embedded wait_req to the list if it cannot acquire the mutex to complete the condition for waking. Co-authored-by: Martin Braun <martin.braun@ettus.com> Co-authored-by: Ciro Nishiguchi <ciro.nishiguchi@ni.com> Co-authored-by: Brent Stapleton <brent.stapleton@ettus.com>
Diffstat (limited to 'host/lib/include/uhdlib/transport/udp_dpdk_link.hpp')
-rw-r--r--host/lib/include/uhdlib/transport/udp_dpdk_link.hpp267
1 files changed, 267 insertions, 0 deletions
diff --git a/host/lib/include/uhdlib/transport/udp_dpdk_link.hpp b/host/lib/include/uhdlib/transport/udp_dpdk_link.hpp
new file mode 100644
index 000000000..eaf3cf7c4
--- /dev/null
+++ b/host/lib/include/uhdlib/transport/udp_dpdk_link.hpp
@@ -0,0 +1,267 @@
+//
+// Copyright 2019 Ettus Research, a National Instruments brand
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+#ifndef INCLUDED_UHDLIB_TRANSPORT_UDP_DPDK_LINK_HPP
+#define INCLUDED_UHDLIB_TRANSPORT_UDP_DPDK_LINK_HPP
+
+#include <uhd/config.hpp>
+#include <uhd/transport/buffer_pool.hpp>
+#include <uhd/types/device_addr.hpp>
+#include <uhdlib/transport/dpdk/common.hpp>
+#include <uhdlib/transport/link_if.hpp>
+#include <uhdlib/transport/links.hpp>
+#include <rte_udp.h>
+#include <cassert>
+#include <string>
+#include <vector>
+
+namespace uhd { namespace transport {
+
+/*!
+ * A zero copy transport interface to the dpdk DMA library.
+ */
+class udp_dpdk_link : public virtual recv_link_if, public virtual send_link_if
+{
+public:
+ using sptr = std::shared_ptr<udp_dpdk_link>;
+
+ udp_dpdk_link(const dpdk::port_id_t port_id,
+ const std::string& remote_addr,
+ const std::string& remote_port,
+ const std::string& local_port,
+ const link_params_t& params);
+
+ ~udp_dpdk_link();
+
+ /*!
+ * Make a new dpdk link. Get port ID from routing table.
+ *
+ * \param remote_addr Remote IP address
+ * \param remote_port Remote UDP port
+ * \param params Values for frame sizes, num frames, and buffer sizes
+ * \return a shared_ptr to a new udp dpdk link
+ */
+ static sptr make(const std::string& remote_addr,
+ const std::string& remote_port,
+ const link_params_t& params);
+
+ /*!
+ * Make a new dpdk link. User specifies DPDK port ID directly.
+ *
+ * \param port_id DPDK port ID to use for communication
+ * \param remote_addr Remote IP address
+ * \param remote_port Remote UDP port
+ * \param local_port Local UDP port
+ * \param params Values for frame sizes, num frames, and buffer sizes
+ * \return a shared_ptr to a new udp dpdk link
+ */
+ static sptr make(const dpdk::port_id_t port_id,
+ const std::string& remote_addr,
+ const std::string& remote_port,
+ const std::string& local_port,
+ const link_params_t& params);
+
+ /*!
+ * Get the associated dpdk_port
+ *
+ * \return a pointer to the dpdk_port used by this link
+ */
+ inline dpdk::dpdk_port* get_port()
+ {
+ return _port;
+ }
+
+ /*!
+ * Get the DMA queue associated with this link
+ *
+ * \return the queue ID for this link's DMA queue
+ */
+ inline dpdk::queue_id_t get_queue_id()
+ {
+ return _queue;
+ }
+
+ /*!
+ * Get the local UDP port used by this link
+ *
+ * \return the local UDP port, in network order
+ */
+ inline uint16_t get_local_port()
+ {
+ return _local_port;
+ }
+
+ /*!
+ * Get the remote UDP port used by this link
+ *
+ * \return the remote UDP port, in network order
+ */
+ inline uint16_t get_remote_port()
+ {
+ return _remote_port;
+ }
+
+ /*!
+ * Get the remote IPv4 address used by this link
+ *
+ * \return the remote IPv4 address, in network order
+ */
+ inline uint32_t get_remote_ipv4()
+ {
+ return _remote_ipv4;
+ }
+
+ /*!
+ * Set the remote host's MAC address
+ * This MAC address must be filled in for the remote IPv4 address before
+ * the link can reach its destination.
+ *
+ * \param mac the remote host's MAC address
+ */
+ inline void set_remote_mac(struct ether_addr& mac)
+ {
+ ether_addr_copy(&mac, &_remote_mac);
+ }
+
+ /*!
+ * Get the remote host's MAC address
+ *
+ * \param mac Where to write the MAC address
+ */
+ inline void get_remote_mac(struct ether_addr& dst)
+ {
+ ether_addr_copy(&_remote_mac, &dst);
+ }
+
+ /*!
+ * Get the number of frame buffers that can be queued by this link.
+ */
+ size_t get_num_send_frames() const
+ {
+ return _num_send_frames;
+ }
+
+ /*!
+ * Get the maximum capacity of a frame buffer.
+ */
+ size_t get_send_frame_size() const
+ {
+ return _send_frame_size;
+ }
+
+ /*!
+ * Get the physical adapter ID used for this link
+ */
+ inline adapter_id_t get_send_adapter_id() const
+ {
+ return _adapter_id;
+ }
+
+ /*!
+ * Get the number of frame buffers that can be queued by this link.
+ */
+ size_t get_num_recv_frames() const
+ {
+ return _num_recv_frames;
+ }
+
+ /*!
+ * Get the maximum capacity of a frame buffer.
+ */
+ size_t get_recv_frame_size() const
+ {
+ return _recv_frame_size;
+ }
+
+ /*!
+ * Get the physical adapter ID used for this link
+ */
+ inline adapter_id_t get_recv_adapter_id() const
+ {
+ return _adapter_id;
+ }
+
+ /*!
+ * Enqueue a received mbuf, which can be pulled via get_recv_buff()
+ */
+ void enqueue_recv_mbuf(struct rte_mbuf* mbuf);
+
+ /*!
+ * Receive a packet and return a frame buffer containing the packet data.
+ * The timeout argument is ignored.
+ *
+ * Received buffers are pulled from the frame buffer list. No buffers can
+ * be retrieved unless the corresponding rte_mbufs were placed in the list
+ * via the enqueue_recv_mbuf() method.
+ *
+ * \return a frame buffer, or null uptr if timeout occurs
+ */
+ frame_buff::uptr get_recv_buff(int32_t /*timeout_ms*/);
+
+ /*!
+ * Release a frame buffer, allowing the link driver to reuse it.
+ *
+ * \param buffer frame buffer to release for reuse by the link
+ */
+ void release_recv_buff(frame_buff::uptr buff);
+
+ /*!
+ * Get an empty frame buffer in which to write packet contents.
+ *
+ * \param timeout_ms a positive timeout value specifies the maximum number
+ of ms to wait, a negative value specifies to block
+ until successful, and a value of 0 specifies no wait.
+ * \return a frame buffer, or null uptr if timeout occurs
+ */
+ frame_buff::uptr get_send_buff(int32_t /*timeout_ms*/);
+
+ /*!
+ * Send a packet with the contents of the frame buffer and release the
+ * buffer, allowing the link driver to reuse it. If the size of the frame
+ * buffer is 0, the buffer is released with no packet being sent.
+ *
+ * Note that this function will only fill in the L2 header and send the
+ * mbuf. The L3 and L4 headers, in addition to the lengths in the rte_mbuf
+ * fields, must be set in the I/O service.
+ *
+ * \param buffer frame buffer containing packet data
+ *
+ * Throws an exception if an I/O error occurs while sending
+ */
+ void release_send_buff(frame_buff::uptr buff);
+
+private:
+ //! A reference to the DPDK context
+ dpdk::dpdk_ctx::sptr _ctx;
+ //! The DPDK NIC port used by this link
+ dpdk::dpdk_port* _port;
+ //! Local UDP port, in network order
+ uint16_t _local_port;
+ //! Remote UDP port, in network order
+ uint16_t _remote_port;
+ //! Remote IPv4 address, in network order
+ uint32_t _remote_ipv4;
+ //! Remote host's MAC address
+ struct ether_addr _remote_mac;
+ //! Number of recv frames is not validated
+ size_t _num_recv_frames;
+ //! Maximum bytes of UDP payload data in recv frame
+ size_t _recv_frame_size;
+ //! Number of send frames is not validated
+ size_t _num_send_frames;
+ //! Maximum bytes of UDP payload data in send frame
+ size_t _send_frame_size;
+ //! Registered adapter ID for this link's DPDK NIC port
+ adapter_id_t _adapter_id;
+ //! The RX frame buff list head
+ dpdk::dpdk_frame_buff* _recv_buff_head = nullptr;
+ // TODO: Implement ability to use multiple queues
+ dpdk::queue_id_t _queue = 0;
+};
+
+}} // namespace uhd::transport
+
+#endif /* INCLUDED_UHDLIB_TRANSPORT_UDP_DPDK_LINK_HPP */