mirror of
https://github.com/saymrwulf/uhd.git
synced 2026-05-16 21:10:10 +00:00
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>
198 lines
7.2 KiB
C++
198 lines
7.2 KiB
C++
//
|
|
// 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);
|
|
}
|
|
}
|