diff options
Diffstat (limited to 'host/lib/transport/udp_dpdk_link.cpp')
-rw-r--r-- | host/lib/transport/udp_dpdk_link.cpp | 198 |
1 files changed, 198 insertions, 0 deletions
diff --git a/host/lib/transport/udp_dpdk_link.cpp b/host/lib/transport/udp_dpdk_link.cpp new file mode 100644 index 000000000..dc56de43c --- /dev/null +++ b/host/lib/transport/udp_dpdk_link.cpp @@ -0,0 +1,198 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#include <uhd/config.hpp> +#include <uhd/utils/log.hpp> +#include <uhd/utils/static.hpp> +#include <uhdlib/transport/adapter.hpp> +#include <uhdlib/transport/dpdk/udp.hpp> +#include <uhdlib/transport/udp_dpdk_link.hpp> +#include <arpa/inet.h> +#include <memory> + +using namespace uhd::transport; +using namespace uhd::transport::dpdk; + +udp_dpdk_link::udp_dpdk_link(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) + : _num_recv_frames(params.num_recv_frames) + , _recv_frame_size(params.recv_frame_size) + , _num_send_frames(params.num_send_frames) + , _send_frame_size(params.send_frame_size) +{ + // Get a reference to the context, since this class manages DPDK memory + _ctx = dpdk_ctx::get(); + UHD_ASSERT_THROW(_ctx); + + // Fill in remote IPv4 address and UDP port + // NOTE: Remote MAC address is filled in later by I/O service + int status = inet_pton(AF_INET, remote_addr.c_str(), &_remote_ipv4); + if (status != 1) { + UHD_LOG_ERROR("DPDK", std::string("Invalid destination address ") + remote_addr); + throw uhd::runtime_error( + std::string("DPDK: Invalid destination address ") + remote_addr); + } + _remote_port = rte_cpu_to_be_16(std::stoul(remote_port)); + + // Grab the port with a route to the remote host + _port = _ctx->get_port(port_id); + + uint16_t local_port_num = rte_cpu_to_be_16(std::stoul(local_port)); + // Get an unused UDP port for listening + _local_port = _port->alloc_udp_port(local_port_num); + + // Validate params + const size_t max_frame_size = _port->get_mtu() - dpdk::HDR_SIZE_UDP_IPV4; + UHD_ASSERT_THROW(params.send_frame_size <= max_frame_size); + UHD_ASSERT_THROW(params.recv_frame_size <= max_frame_size); + + // Register the adapter + auto info = _port->get_adapter_info(); + auto& adap_ctx = adapter_ctx::get(); + _adapter_id = adap_ctx.register_adapter(info); + UHD_LOGGER_TRACE("DPDK") << boost::format("Created udp_dpdk_link to (%s:%s)") + % remote_addr % remote_port; + UHD_LOGGER_TRACE("DPDK") + << boost::format("num_recv_frames=%d, recv_frame_size=%d, num_send_frames=%d, " + "send_frame_size=%d") + % params.num_recv_frames % params.recv_frame_size % params.num_send_frames + % params.send_frame_size; +} + +udp_dpdk_link::~udp_dpdk_link() {} + +udp_dpdk_link::sptr udp_dpdk_link::make(const std::string& remote_addr, + const std::string& remote_port, + const link_params_t& params) +{ + auto ctx = dpdk::dpdk_ctx::get(); + auto port = ctx->get_route(remote_addr); + if (!port) { + UHD_LOG_ERROR("DPDK", + std::string("Could not find route to destination address ") + remote_addr); + throw uhd::runtime_error( + std::string("DPDK: Could not find route to destination address ") + + remote_addr); + } + return make(port->get_port_id(), remote_addr, remote_port, "0", params); +} + +udp_dpdk_link::sptr udp_dpdk_link::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) +{ + UHD_ASSERT_THROW(params.recv_frame_size > 0); + UHD_ASSERT_THROW(params.send_frame_size > 0); + UHD_ASSERT_THROW(params.num_send_frames > 0); + UHD_ASSERT_THROW(params.num_recv_frames > 0); + + return std::make_shared<udp_dpdk_link>( + port_id, remote_addr, remote_port, local_port, params); +} + +void udp_dpdk_link::enqueue_recv_mbuf(struct rte_mbuf* mbuf) +{ + // Get packet size + struct udp_hdr* hdr = rte_pktmbuf_mtod_offset( + mbuf, struct udp_hdr*, sizeof(struct ether_hdr) + sizeof(struct ipv4_hdr)); + size_t packet_size = rte_be_to_cpu_16(hdr->dgram_len) - sizeof(struct udp_hdr); + // Prepare the dpdk_frame_buff + auto buff = new (rte_mbuf_to_priv(mbuf)) dpdk_frame_buff(mbuf); + buff->header_jump( + sizeof(struct ether_hdr) + sizeof(struct ipv4_hdr) + sizeof(struct udp_hdr)); + buff->set_packet_size(packet_size); + // Add the dpdk_frame_buff to the list + if (_recv_buff_head) { + buff->prev = _recv_buff_head->prev; + buff->next = _recv_buff_head; + _recv_buff_head->prev->next = buff; + _recv_buff_head->prev = buff; + } else { + _recv_buff_head = buff; + buff->next = buff; + buff->prev = buff; + } +} + +frame_buff::uptr udp_dpdk_link::get_recv_buff(int32_t /*timeout_ms*/) +{ + auto buff = _recv_buff_head; + if (buff) { + if (_recv_buff_head->next == buff) { + /* Only had the one buff, so the list is empty */ + _recv_buff_head = nullptr; + } else { + /* Make the next buff the new list head */ + _recv_buff_head->next->prev = _recv_buff_head->prev; + _recv_buff_head->prev->next = _recv_buff_head->next; + _recv_buff_head = _recv_buff_head->next; + } + buff->next = nullptr; + buff->prev = nullptr; + return frame_buff::uptr(buff); + } + return frame_buff::uptr(); +} + +void udp_dpdk_link::release_recv_buff(frame_buff::uptr buff) +{ + dpdk_frame_buff* buff_ptr = (dpdk_frame_buff*)buff.release(); + assert(buff_ptr); + rte_pktmbuf_free(buff_ptr->get_pktmbuf()); +} + +frame_buff::uptr udp_dpdk_link::get_send_buff(int32_t /*timeout_ms*/) +{ + auto mbuf = rte_pktmbuf_alloc(_port->get_tx_pktbuf_pool()); + if (mbuf) { + auto buff = new (rte_mbuf_to_priv(mbuf)) dpdk_frame_buff(mbuf); + buff->header_jump( + sizeof(struct ether_hdr) + sizeof(struct ipv4_hdr) + sizeof(struct udp_hdr)); + return frame_buff::uptr(buff); + } + return frame_buff::uptr(); +} + +void udp_dpdk_link::release_send_buff(frame_buff::uptr buff) +{ + dpdk_frame_buff* buff_ptr = (dpdk_frame_buff*)buff.release(); + assert(buff_ptr); + auto mbuf = buff_ptr->get_pktmbuf(); + if (buff_ptr->packet_size()) { + // Fill in L2 header + auto local_mac = _port->get_mac_addr(); + struct ether_hdr* l2_hdr = rte_pktmbuf_mtod(mbuf, struct ether_hdr*); + ether_addr_copy(&_remote_mac, &l2_hdr->d_addr); + ether_addr_copy(&local_mac, &l2_hdr->s_addr); + l2_hdr->ether_type = rte_cpu_to_be_16(ETHER_TYPE_IPv4); + // Fill in L3 and L4 headers + dpdk::fill_udp_hdr(mbuf, + _port, + _remote_ipv4, + _local_port, + _remote_port, + buff_ptr->packet_size()); + // Prepare the packet buffer and send it out + int status = rte_eth_tx_prepare(_port->get_port_id(), _queue, &mbuf, 1); + if (status != 1) { + throw uhd::runtime_error("DPDK: Failed to prepare TX buffer for send"); + } + status = rte_eth_tx_burst(_port->get_port_id(), _queue, &mbuf, 1); + while (status != 1) { + status = rte_eth_tx_burst(_port->get_port_id(), _queue, &mbuf, 1); + // FIXME: Should we make available retrying? + // throw uhd::runtime_error("DPDK: Failed to send TX buffer"); + } + } else { + // Release the buffer if there is nothing in it + rte_pktmbuf_free(mbuf); + } +} |