diff options
author | Martin Braun <martin.braun@ettus.com> | 2019-07-03 20:15:35 -0700 |
---|---|---|
committer | Martin Braun <martin.braun@ettus.com> | 2019-11-26 12:16:25 -0800 |
commit | c256b9df6502536c2e451e690f1ad5962c664d1a (patch) | |
tree | a83ad13e6f5978bbe14bb3ecf8294ba1e3d28db4 /host/lib/usrp/mpmd/mpmd_xport_ctrl_udp.cpp | |
parent | 9a8435ed998fc5c65257f4c55768750b227ab19e (diff) | |
download | uhd-c256b9df6502536c2e451e690f1ad5962c664d1a.tar.gz uhd-c256b9df6502536c2e451e690f1ad5962c664d1a.tar.bz2 uhd-c256b9df6502536c2e451e690f1ad5962c664d1a.zip |
x300/mpmd: Port all RFNoC devices to the new RFNoC framework
Co-Authored-By: Alex Williams <alex.williams@ni.com>
Co-Authored-By: Sugandha Gupta <sugandha.gupta@ettus.com>
Co-Authored-By: Brent Stapleton <brent.stapleton@ettus.com>
Co-Authored-By: Ciro Nishiguchi <ciro.nishiguchi@ni.com>
Diffstat (limited to 'host/lib/usrp/mpmd/mpmd_xport_ctrl_udp.cpp')
-rw-r--r-- | host/lib/usrp/mpmd/mpmd_xport_ctrl_udp.cpp | 265 |
1 files changed, 0 insertions, 265 deletions
diff --git a/host/lib/usrp/mpmd/mpmd_xport_ctrl_udp.cpp b/host/lib/usrp/mpmd/mpmd_xport_ctrl_udp.cpp deleted file mode 100644 index 20b94899c..000000000 --- a/host/lib/usrp/mpmd/mpmd_xport_ctrl_udp.cpp +++ /dev/null @@ -1,265 +0,0 @@ -// -// Copyright 2017 Ettus Research, National Instruments Company -// Copyright 2019 Ettus Research, National Instruments Brand -// -// SPDX-License-Identifier: GPL-3.0-or-later -// - -#include "mpmd_xport_ctrl_udp.hpp" -#include "mpmd_impl.hpp" -#include "mpmd_xport_mgr.hpp" -#include <uhd/transport/udp_constants.hpp> -#include <uhd/transport/udp_simple.hpp> -#include <uhd/transport/udp_zero_copy.hpp> - - -using namespace uhd; -using namespace uhd::mpmd::xport; - -namespace { - -//! Maximum CHDR packet size in bytes -const size_t MPMD_10GE_DATA_FRAME_MAX_SIZE = 4000; - -//! Default number of send frames -const size_t MPMD_UDP_DEFAULT_NUM_SEND_FRAMES = 1; -//! Default number of recv frames -const size_t MPMD_UDP_DEFAULT_NUM_RECV_FRAMES = 1; -//! Default message frame size -const size_t MPMD_UDP_MSG_FRAME_SIZE = 256; -//! Default 1GbE send frame size -const size_t MPMD_UDP_1GE_DEFAULT_SEND_FRAME_SIZE = 1472; -//! Default 1GbE receive frame size -const size_t MPMD_UDP_1GE_DEFAULT_RECV_FRAME_SIZE = 1472; -//! Default 10GbE send frame size -const size_t MPMD_UDP_10GE_DEFAULT_SEND_FRAME_SIZE = 4000; -//! Default 10GbE receive frame size -const size_t MPMD_UDP_10GE_DEFAULT_RECV_FRAME_SIZE = 4000; - -//! -const double MPMD_BUFFER_DEPTH = 50.0e-3; // s -//! For MTU discovery, the time we wait for a packet before calling it -// oversized (seconds). -const double MPMD_MTU_DISCOVERY_TIMEOUT = 0.02; - -// TODO: move these to appropriate header file for all other devices -const double MAX_RATE_1GIGE = 1e9 / 8; // byte/s -const double MAX_RATE_10GIGE = 10e9 / 8; // byte/s - -std::vector<std::string> get_addrs_from_mb_args(const uhd::device_addr_t& mb_args) -{ - // mb_args must always include addr - if (not mb_args.has_key(FIRST_ADDR_KEY)) { - throw uhd::runtime_error( - "The " + FIRST_ADDR_KEY - + " key must be specified in " - "device args to create an Ethernet transport to an RFNoC block"); - } - std::vector<std::string> addrs{mb_args[FIRST_ADDR_KEY]}; - if (mb_args.has_key(SECOND_ADDR_KEY)) { - addrs.push_back(mb_args[SECOND_ADDR_KEY]); - } - return addrs; -} - -/*! Do a binary search to discover MTU - * - * Uses the MPM echo service to figure out MTU. We simply send a bunch of - * packets and see if they come back until we converged on the path MTU. - * The end result must lie between \p min_frame_size and \p max_frame_size. - * - * \param address IP address - * \param port UDP port (yeah it's a string!) - * \param min_frame_size Minimum frame size, initialize algorithm to start - * with this value - * \param max_frame_size Maximum frame size, initialize algorithm to start - * with this value - * \param echo_timeout Timeout value in seconds. For frame sizes that - * exceed the MTU, we don't expect a response, and this - * is the amount of time we'll wait before we assume - * the frame size exceeds the MTU. - */ -size_t discover_mtu(const std::string& address, - const std::string& port, - size_t min_frame_size, - size_t max_frame_size, - const double echo_timeout = 0.020) -{ - const size_t echo_prefix_offset = uhd::mpmd::mpmd_impl::MPM_ECHO_CMD.size(); - const size_t mtu_hdr_len = echo_prefix_offset + 10; - UHD_ASSERT_THROW(min_frame_size < max_frame_size); - UHD_ASSERT_THROW(min_frame_size % 4 == 0); - UHD_ASSERT_THROW(max_frame_size % 4 == 0); - UHD_ASSERT_THROW(min_frame_size >= echo_prefix_offset + mtu_hdr_len); - using namespace uhd::transport; - // The return port will probably differ from the discovery port, so we - // need a "broadcast" UDP connection; using make_connected() would - // drop packets - udp_simple::sptr udp = udp_simple::make_broadcast(address, port); - std::string send_buf(uhd::mpmd::mpmd_impl::MPM_ECHO_CMD); - send_buf.resize(max_frame_size, '#'); - UHD_ASSERT_THROW(send_buf.size() == max_frame_size); - std::vector<uint8_t> recv_buf; - recv_buf.resize(max_frame_size, ' '); - - // Little helper to check returned packets match the sent ones - auto require_bufs_match = [&recv_buf, &send_buf, mtu_hdr_len](const size_t len) { - if (len < mtu_hdr_len - or std::memcmp((void*)&recv_buf[0], (void*)&send_buf[0], mtu_hdr_len) != 0) { - throw uhd::runtime_error("Unexpected content of MTU " - "discovery return packet!"); - } - }; - UHD_LOG_TRACE("MPMD", "Determining UDP MTU... "); - size_t seq_no = 0; - while (min_frame_size < max_frame_size) { - // Only test multiples of 4 bytes! - const size_t test_frame_size = (max_frame_size / 2 + min_frame_size / 2 + 3) - & ~size_t(3); - // Encode sequence number and current size in the string, makes it - // easy to debug in code or Wireshark. Is also used for identifying - // response packets. - std::sprintf( - &send_buf[echo_prefix_offset], ";%04lu,%04lu", seq_no++, test_frame_size); - UHD_LOG_TRACE("MPMD", "Testing frame size " << test_frame_size); - udp->send(boost::asio::buffer(&send_buf[0], test_frame_size)); - - const size_t len = udp->recv(boost::asio::buffer(recv_buf), echo_timeout); - if (len == 0) { - // Nothing received, so this is probably too big - max_frame_size = test_frame_size - 4; - } else if (len >= test_frame_size) { - // Size went through, so bump the minimum - require_bufs_match(len); - min_frame_size = test_frame_size; - } else if (len < test_frame_size) { - // This is an odd case. Something must have snipped the packet - // on the way back. Still, we'll just back off and try - // something smaller. - UHD_LOG_DEBUG("MPMD", "Unexpected packet truncation during MTU discovery."); - require_bufs_match(len); - max_frame_size = len; - } - } - UHD_LOG_DEBUG("MPMD", "Path MTU for address " << address << ": " << min_frame_size); - return min_frame_size; -} - -} // namespace - - -mpmd_xport_ctrl_udp::mpmd_xport_ctrl_udp(const uhd::device_addr_t& mb_args) - : _mb_args(mb_args) - , _recv_args(filter_args(mb_args, "recv")) - , _send_args(filter_args(mb_args, "send")) - , _available_addrs(get_addrs_from_mb_args(mb_args)) - , _mtu(MPMD_10GE_DATA_FRAME_MAX_SIZE) -{ - const std::string mpm_discovery_port = _mb_args.get( - mpmd_impl::MPM_DISCOVERY_PORT_KEY, std::to_string(mpmd_impl::MPM_DISCOVERY_PORT)); - auto discover_mtu_for_ip = [mpm_discovery_port](const std::string& ip_addr) { - return discover_mtu(ip_addr, - mpm_discovery_port, - IP_PROTOCOL_MIN_MTU_SIZE - IP_PROTOCOL_UDP_PLUS_IP_HEADER, - MPMD_10GE_DATA_FRAME_MAX_SIZE, - MPMD_MTU_DISCOVERY_TIMEOUT); - }; - - for (const auto& ip_addr : _available_addrs) { - _mtu = std::min(_mtu, discover_mtu_for_ip(ip_addr)); - } -} - -uhd::both_xports_t mpmd_xport_ctrl_udp::make_transport( - mpmd_xport_mgr::xport_info_t& xport_info, - const usrp::device3_impl::xport_type_t xport_type, - const uhd::device_addr_t& xport_args) -{ - - double link_speed = MAX_RATE_1GIGE; - if (xport_info.count("link_speed") == 0) { - UHD_LOG_WARNING("MPMD", - "Could not determine link speed; using 1GibE max speed of " - << MAX_RATE_1GIGE); - } else { - link_speed = xport_info.at("link_speed") == "10000" ? MAX_RATE_10GIGE - : MAX_RATE_1GIGE; - } - - // Constrain by this transport's MTU and the MTU in the xport_args - const size_t send_mtu = std::min(get_mtu(uhd::TX_DIRECTION), - xport_args.cast<size_t>("mtu", get_mtu(uhd::TX_DIRECTION))); - const size_t recv_mtu = std::min(get_mtu(uhd::RX_DIRECTION), - xport_args.cast<size_t>("mtu", get_mtu(uhd::RX_DIRECTION))); - - // Create actual UDP transport - transport::zero_copy_xport_params default_buff_args; - default_buff_args.num_send_frames = MPMD_UDP_DEFAULT_NUM_SEND_FRAMES; - default_buff_args.num_recv_frames = MPMD_UDP_DEFAULT_NUM_RECV_FRAMES; - default_buff_args.recv_frame_size = MPMD_UDP_MSG_FRAME_SIZE; - default_buff_args.send_frame_size = MPMD_UDP_MSG_FRAME_SIZE; - default_buff_args.recv_buff_size = link_speed * MPMD_BUFFER_DEPTH; - default_buff_args.send_buff_size = link_speed * MPMD_BUFFER_DEPTH; - if (xport_type == usrp::device3_impl::CTRL) { - default_buff_args.num_recv_frames = - uhd::rfnoc::CMD_FIFO_SIZE / uhd::rfnoc::MAX_CMD_PKT_SIZE; - } else if (xport_type == usrp::device3_impl::TX_DATA) { - const size_t default_frame_size = (link_speed == MAX_RATE_10GIGE ? - MPMD_UDP_10GE_DEFAULT_SEND_FRAME_SIZE : - MPMD_UDP_1GE_DEFAULT_SEND_FRAME_SIZE); - default_buff_args.send_frame_size = - xport_args.cast<size_t>("send_frame_size", - std::min(default_frame_size, send_mtu)); - default_buff_args.num_send_frames = - xport_args.cast<size_t>("num_send_frames", - default_buff_args.num_send_frames); - default_buff_args.send_buff_size = - xport_args.cast<size_t>("send_buff_size", - default_buff_args.send_buff_size); - } else if (xport_type == usrp::device3_impl::RX_DATA) { - const size_t default_frame_size = (link_speed == MAX_RATE_10GIGE ? - MPMD_UDP_10GE_DEFAULT_RECV_FRAME_SIZE : - MPMD_UDP_1GE_DEFAULT_RECV_FRAME_SIZE); - default_buff_args.recv_frame_size = - xport_args.cast<size_t>("recv_frame_size", - std::min(default_frame_size, recv_mtu)); - default_buff_args.num_recv_frames = - xport_args.cast<size_t>("num_recv_frames", - default_buff_args.num_recv_frames); - default_buff_args.recv_buff_size = - xport_args.cast<size_t>("recv_buff_size", - default_buff_args.recv_buff_size); - } - transport::udp_zero_copy::buff_params buff_params; - auto recv = transport::udp_zero_copy::make(xport_info["ipv4"], - xport_info["port"], - default_buff_args, - buff_params); - const uint16_t port = recv->get_local_port(); - const std::string src_ip_addr = recv->get_local_addr(); - xport_info["src_port"] = std::to_string(port); - xport_info["src_ipv4"] = src_ip_addr; - - // Create both_xports_t object and finish: - both_xports_t xports; - xports.endianness = uhd::ENDIANNESS_BIG; - xports.send_sid = sid_t(xport_info["send_sid"]); - xports.recv_sid = xports.send_sid.reversed(); - xports.recv_buff_size = buff_params.recv_buff_size; - xports.send_buff_size = buff_params.send_buff_size; - xports.recv = recv; // Note: This is a type cast! - xports.send = recv; // This too - return xports; -} - -bool mpmd_xport_ctrl_udp::is_valid(const mpmd_xport_mgr::xport_info_t& xport_info) const -{ - return std::find( - _available_addrs.cbegin(), _available_addrs.cend(), xport_info.at("ipv4")) - != _available_addrs.cend(); -} - -size_t mpmd_xport_ctrl_udp::get_mtu(const uhd::direction_t /*dir*/) const -{ - return _mtu; -} |