aboutsummaryrefslogtreecommitdiffstats
path: root/host
diff options
context:
space:
mode:
authorMartin Braun <martin.braun@ettus.com>2019-12-02 23:06:39 -0800
committerBrent Stapleton <brent.stapleton@ettus.com>2019-12-20 16:32:22 -0800
commit4547f2de85f0d8b258a89cea71af774fce8616b7 (patch)
treed1239e074c5a35b449638a34aa3a6f74a53adfbb /host
parent4e38eef817813c1bbd8a9cf972e4cf0134d24308 (diff)
downloaduhd-4547f2de85f0d8b258a89cea71af774fce8616b7.tar.gz
uhd-4547f2de85f0d8b258a89cea71af774fce8616b7.tar.bz2
uhd-4547f2de85f0d8b258a89cea71af774fce8616b7.zip
x300,mpmd: Enable DPDK
x300: - Remove obsolete variables from x300_eth_mgr and X300 motherboard components - Added some documentation / comments - Use constrained device args in more places - Enables the use of use_dpdk=1 - Switches between regular (kernel-based) and DPDK UDP mpmd: - Merge link_if_ctrl for udp and dpdk_udp - Update cmake options
Diffstat (limited to 'host')
-rw-r--r--host/lib/usrp/mpmd/CMakeLists.txt30
-rw-r--r--host/lib/usrp/mpmd/mpmd_find.cpp10
-rw-r--r--host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.cpp279
-rw-r--r--host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.hpp48
-rw-r--r--host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.cpp126
-rw-r--r--host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.hpp2
-rw-r--r--host/lib/usrp/mpmd/mpmd_link_if_mgr.cpp3
-rw-r--r--host/lib/usrp/x300/CMakeLists.txt4
-rw-r--r--host/lib/usrp/x300/x300_eth_mgr.cpp152
-rw-r--r--host/lib/usrp/x300/x300_eth_mgr.hpp23
-rw-r--r--host/lib/usrp/x300/x300_impl.cpp3
-rw-r--r--host/lib/usrp/x300/x300_impl.hpp1
12 files changed, 216 insertions, 465 deletions
diff --git a/host/lib/usrp/mpmd/CMakeLists.txt b/host/lib/usrp/mpmd/CMakeLists.txt
index ce06913a3..1eab1f39b 100644
--- a/host/lib/usrp/mpmd/CMakeLists.txt
+++ b/host/lib/usrp/mpmd/CMakeLists.txt
@@ -1,21 +1,11 @@
#
# Copyright 2017 Ettus Research, a National Instruments Company
+# Copyright 2019 Ettus Research, a National Instruments Brand
#
# SPDX-License-Identifier: GPL-3.0-or-later
#
if(ENABLE_MPMD)
- if(ENABLE_LIBERIO)
- message(STATUS "Compiling MPMD with liberio support...")
- add_definitions(-DHAVE_LIBERIO)
- endif(ENABLE_LIBERIO)
-
- if(ENABLE_DPDK)
- message(STATUS "Compiling MPMD without DPDK support for now...")
- # message(STATUS "Compiling MPMD with DPDK support...")
- # add_definitions(-DHAVE_DPDK)
- endif(ENABLE_DPDK)
-
LIBUHD_APPEND_SOURCES(
${CMAKE_CURRENT_SOURCE_DIR}/mpmd_find.cpp
${CMAKE_CURRENT_SOURCE_DIR}/mpmd_image_loader.cpp
@@ -29,15 +19,23 @@ if(ENABLE_MPMD)
)
if(ENABLE_LIBERIO)
+ message(STATUS "Compiling MPMD with liberio support...")
LIBUHD_APPEND_SOURCES(
${CMAKE_CURRENT_SOURCE_DIR}/mpmd_link_if_ctrl_liberio.cpp
)
+ set_property(
+ SOURCE ${CMAKE_CURRENT_SOURCE_DIR}/mpmd_link_if_mgr.cpp
+ PROPERTY COMPILE_DEFINITIONS HAVE_LIBERIO
+ )
endif(ENABLE_LIBERIO)
- # if(ENABLE_DPDK)
- # LIBUHD_APPEND_SOURCES(
- # ${CMAKE_CURRENT_SOURCE_DIR}/mpmd_link_if_ctrl_dpdk_udp.cpp
- # )
- # endif(ENABLE_DPDK)
+ if(ENABLE_DPDK)
+ set_property(
+ SOURCE
+ ${CMAKE_CURRENT_SOURCE_DIR}/mpmd_link_if_ctrl_udp.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/mpmd_find.cpp
+ PROPERTY COMPILE_DEFINITIONS HAVE_DPDK
+ )
+ endif(ENABLE_DPDK)
endif(ENABLE_MPMD)
diff --git a/host/lib/usrp/mpmd/mpmd_find.cpp b/host/lib/usrp/mpmd/mpmd_find.cpp
index 3e8bcc72f..b5f20559c 100644
--- a/host/lib/usrp/mpmd/mpmd_find.cpp
+++ b/host/lib/usrp/mpmd/mpmd_find.cpp
@@ -9,13 +9,15 @@
#include "mpmd_devices.hpp"
#include "mpmd_impl.hpp"
#include "mpmd_link_if_mgr.hpp"
-#include <uhdlib/transport/dpdk_common.hpp>
#include <uhd/transport/if_addrs.hpp>
#include <uhd/transport/udp_simple.hpp>
#include <uhd/types/device_addr.hpp>
#include <boost/algorithm/string.hpp>
#include <boost/asio.hpp>
#include <future>
+#ifdef HAVE_DPDK
+# include <uhdlib/transport/dpdk/common.hpp>
+#endif
using namespace uhd;
using namespace uhd::mpmd;
@@ -198,9 +200,9 @@ device_addrs_t mpmd_find(const device_addr_t& hint_)
#ifdef HAVE_DPDK
// Start DPDK so links come up
if (hint_.has_key("use_dpdk")) {
- auto& dpdk_ctx = uhd::transport::uhd_dpdk_ctx::get();
- if (not dpdk_ctx.is_init_done()) {
- dpdk_ctx.init(hint_);
+ auto dpdk_ctx = uhd::transport::dpdk::dpdk_ctx::get();
+ if (not dpdk_ctx->is_init_done()) {
+ dpdk_ctx->init(hint_);
}
}
#endif
diff --git a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.cpp b/host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.cpp
deleted file mode 100644
index 2a3a48b62..000000000
--- a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.cpp
+++ /dev/null
@@ -1,279 +0,0 @@
-//
-// Copyright 2017 Ettus Research, National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-
-#include "mpmd_impl.hpp"
-#include "mpmd_link_if_mgr.hpp"
-#include "mpmd_link_if_ctrl_dpdk_udp.hpp"
-#include <uhd/transport/udp_simple.hpp>
-#include <uhd/transport/udp_constants.hpp>
-#include <uhdlib/transport/dpdk_zero_copy.hpp>
-#include <arpa/inet.h>
-
-
-using namespace uhd;
-using namespace uhd::mpmd::xport;
-
-namespace {
-constexpr unsigned int MPMD_UDP_RESERVED_FRAME_SIZE = 64;
-
-//! Maximum CHDR packet size in bytes
-const size_t MPMD_10GE_DATA_FRAME_MAX_SIZE = 4000;
-const size_t MPMD_10GE_DATA_FRAME_DEFAULT_SIZE = 4000;
-const size_t MPMD_10GE_MSG_FRAME_DEFAULT_SIZE = 256;
-
-//! Number of send/recv frames
-const size_t MPMD_ETH_NUM_SEND_FRAMES = 32;
-const size_t MPMD_ETH_NUM_RECV_FRAMES = 128;
-const size_t MPMD_ETH_NUM_CTRL_FRAMES = 32;
-
-//! For MTU discovery, the time we wait for a packet before calling it
-// oversized (seconds).
-const double MPMD_MTU_DISCOVERY_TIMEOUT = 0.02;
-
-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
- * \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 auto &ctx = uhd::transport::uhd_dpdk_ctx::get();
- const size_t echo_prefix_offset =
- uhd::mpmd::mpmd_impl::MPM_ECHO_CMD.size();
- const size_t mtu_hdr_len = echo_prefix_offset + 10;
- const int port_id = ctx.get_route(address);
- UHD_ASSERT_THROW(port_id >= 0);
- 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;
- uhd::transport::zero_copy_xport_params buff_args;
- buff_args.recv_frame_size = max_frame_size;
- buff_args.send_frame_size = max_frame_size;
- buff_args.num_send_frames = 1;
- buff_args.num_recv_frames = 1;
- auto dev_addr = uhd::device_addr_t();
- dpdk_zero_copy::sptr sock = dpdk_zero_copy::make(ctx,
- (unsigned int) port_id, address, port, "0", buff_args, dev_addr);
- 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);
-
- // Little helper to check returned packets match the sent ones
- auto require_bufs_match = [&send_buf, mtu_hdr_len](
- const uint8_t *recv_buf,
- 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) {
- managed_send_buffer::sptr msbuf = sock->get_send_buff(0);
- UHD_ASSERT_THROW(msbuf.get() != nullptr);
- max_frame_size = std::min(msbuf->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
- );
- // Copy to real buffer
- UHD_LOG_TRACE("MPMD", "Testing frame size " << test_frame_size);
- auto *tx_buf = msbuf->cast<uint8_t *>();
- std::memcpy(tx_buf, &send_buf[0], test_frame_size);
- msbuf->commit(test_frame_size);
- msbuf.reset();
-
- managed_recv_buffer::sptr mrbuf = sock->get_recv_buff(echo_timeout);
- if (mrbuf.get() == nullptr || mrbuf->size() == 0) {
- // Nothing received, so this is probably too big
- max_frame_size = test_frame_size - 4;
- } else if (mrbuf->size() >= test_frame_size) {
- // Size went through, so bump the minimum
- require_bufs_match(mrbuf->cast<uint8_t *>(), mrbuf->size());
- min_frame_size = test_frame_size;
- } else if (mrbuf->size() < 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(mrbuf->cast<uint8_t *>(), mrbuf->size());
- max_frame_size = mrbuf->size();
- }
- mrbuf.reset();
- }
- UHD_LOG_DEBUG("MPMD",
- "Path MTU for address " << address << ": " << min_frame_size);
- return min_frame_size;
-}
-
-}
-
-
-mpmd_link_if_ctrl_dpdk_udp::mpmd_link_if_ctrl_dpdk_udp(
- const uhd::device_addr_t& mb_args
-) : _mb_args(mb_args)
- , _ctx(uhd::transport::uhd_dpdk_ctx::get())
- , _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)
-{
- if (not _ctx.is_init_done()) {
- _ctx.init(mb_args);
- }
- 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_link_if_ctrl_dpdk_udp::make_transport(
-// mpmd_link_if_mgr::xport_info_t &xport_info,
-// const uhd::transport::link_type_t xport_type,
-// const uhd::device_addr_t& xport_args
-//) {
-
-//// 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 UHD-DPDK UDP transport
-// transport::zero_copy_xport_params default_buff_args;
-// default_buff_args.num_recv_frames = MPMD_ETH_NUM_CTRL_FRAMES;
-// default_buff_args.num_send_frames = MPMD_ETH_NUM_CTRL_FRAMES;
-// default_buff_args.recv_frame_size = MPMD_10GE_MSG_FRAME_DEFAULT_SIZE;
-// default_buff_args.send_frame_size = MPMD_10GE_MSG_FRAME_DEFAULT_SIZE;
-
-// if (xport_type == uhd::transport::link_type_t::RX_DATA) {
-// default_buff_args.num_recv_frames =
-// xport_args.cast<size_t>("num_recv_frames", MPMD_ETH_NUM_RECV_FRAMES);
-// default_buff_args.recv_frame_size = std::min(
-// xport_args.cast<size_t>("recv_frame_size",
-// MPMD_10GE_DATA_FRAME_DEFAULT_SIZE),
-// recv_mtu);
-//} else if (xport_type == uhd::transport::link_type_t::TX_DATA) {
-// default_buff_args.num_send_frames =
-// xport_args.cast<size_t>("num_send_frames", MPMD_ETH_NUM_SEND_FRAMES);
-// default_buff_args.send_frame_size = std::min(
-// xport_args.cast<size_t>("send_frame_size",
-// MPMD_10GE_DATA_FRAME_DEFAULT_SIZE),
-// send_mtu);
-//}
-
-// UHD_LOG_TRACE("BUFF", "num_recv_frames=" << default_buff_args.num_recv_frames
-//<< ", num_send_frames=" << default_buff_args.num_send_frames
-//<< ", recv_frame_size=" << default_buff_args.recv_frame_size
-//<< ", send_frame_size=" << default_buff_args.send_frame_size);
-
-// int dpdk_port_id = _ctx.get_route(xport_info["ipv4"]);
-// if (dpdk_port_id < 0) {
-// throw uhd::runtime_error("Could not find a DPDK port with route to " +
-// xport_info["ipv4"]);
-//}
-// auto recv = transport::dpdk_zero_copy::make(
-//_ctx,
-//(const unsigned int) dpdk_port_id,
-// xport_info["ipv4"],
-// xport_info["port"],
-//"0",
-// default_buff_args,
-// uhd::device_addr_t()
-//);
-// 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 =
-// (default_buff_args.recv_frame_size-MPMD_UDP_RESERVED_FRAME_SIZE)*default_buff_args.num_recv_frames;
-// xports.send_buff_size =
-// (default_buff_args.send_frame_size-MPMD_UDP_RESERVED_FRAME_SIZE)*default_buff_args.num_send_frames;
-// xports.recv = recv; // Note: This is a type cast!
-// xports.send = recv; // This too
-// return xports;
-//}
-
-bool mpmd_link_if_ctrl_dpdk_udp::is_valid(
- const mpmd_link_if_mgr::xport_info_t& xport_info
-) const {
- int dpdk_port_id = _ctx.get_route(xport_info.at("ipv4"));
- return (dpdk_port_id >= 0);
-}
-
-size_t mpmd_link_if_ctrl_dpdk_udp::get_mtu(const uhd::direction_t /*dir*/) const
-{
- return _mtu;
-}
diff --git a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.hpp b/host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.hpp
deleted file mode 100644
index 8f4f1c7d1..000000000
--- a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_dpdk_udp.hpp
+++ /dev/null
@@ -1,48 +0,0 @@
-//
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-
-#ifndef INCLUDED_MPMD_XPORT_CTRL_DPDK_UDP_HPP
-#define INCLUDED_MPMD_XPORT_CTRL_DPDK_UDP_HPP
-
-#include "mpmd_link_if_ctrl_base.hpp"
-#include <uhd/types/device_addr.hpp>
-#include <uhdlib/transport/dpdk_zero_copy.hpp>
-
-namespace uhd { namespace mpmd { namespace xport {
-
-/*! UDP transport manager
- *
- * Opens UDP sockets
- */
-class mpmd_link_if_ctrl_dpdk_udp : public mpmd_link_if_ctrl_base
-{
-public:
- mpmd_link_if_ctrl_dpdk_udp(
- const uhd::device_addr_t& mb_args
- );
-
- bool is_valid(
- const mpmd_link_if_mgr::xport_info_t& xport_info
- ) const;
-
- size_t get_mtu(
- const uhd::direction_t dir
- ) const;
-
-private:
- const uhd::device_addr_t _mb_args;
- uhd::transport::uhd_dpdk_ctx &_ctx;
- const uhd::dict<std::string, std::string> _recv_args;
- const uhd::dict<std::string, std::string> _send_args;
- //! A list of IP addresses we can connect our CHDR connections to
- const std::vector<std::string> _available_addrs;
- //! MTU
- size_t _mtu;
-};
-
-}}} /* namespace uhd::mpmd::xport */
-
-#endif /* INCLUDED_MPMD_XPORT_CTRL_DPDK_UDP_HPP */
diff --git a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.cpp b/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.cpp
index a87a9cada..aeb1c6436 100644
--- a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.cpp
+++ b/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.cpp
@@ -1,5 +1,6 @@
//
// Copyright 2017 Ettus Research, National Instruments Company
+// Copyright 2019 Ettus Research, National Instruments Brand
//
// SPDX-License-Identifier: GPL-3.0-or-later
//
@@ -16,6 +17,10 @@
#include <uhdlib/transport/udp_common.hpp>
#include <uhdlib/utils/narrow.hpp>
#include <string>
+#ifdef HAVE_DPDK
+//# include <uhdlib/transport/dpdk_simple.hpp>
+# include <uhdlib/transport/udp_dpdk_link.hpp>
+#endif
using namespace uhd;
using namespace uhd::transport;
@@ -124,8 +129,19 @@ 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 double echo_timeout,
+ const bool use_dpdk)
{
+ if (use_dpdk) {
+#ifdef HAVE_DPDK
+ // FIXME
+ UHD_LOG_WARNING("MPMD", "Using hard-coded MTU of 8000 for DPDK");
+ return 8000;
+#else
+ UHD_LOG_WARNING("MPMD",
+ "DPDK was requested but is not available, falling back to regular UDP");
+#endif
+ }
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);
@@ -186,6 +202,87 @@ size_t discover_mtu(const std::string& address,
return min_frame_size;
}
+// DPDK version
+//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 auto& ctx = uhd::transport::uhd_dpdk_ctx::get();
+ //const size_t echo_prefix_offset = uhd::mpmd::mpmd_impl::MPM_ECHO_CMD.size();
+ //const size_t mtu_hdr_len = echo_prefix_offset + 10;
+ //const int port_id = ctx.get_route(address);
+ //UHD_ASSERT_THROW(port_id >= 0);
+ //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;
+ //uhd::transport::zero_copy_xport_params buff_args;
+ //buff_args.recv_frame_size = max_frame_size;
+ //buff_args.send_frame_size = max_frame_size;
+ //buff_args.num_send_frames = 1;
+ //buff_args.num_recv_frames = 1;
+ //auto dev_addr = uhd::device_addr_t();
+ //dpdk_zero_copy::sptr sock = dpdk_zero_copy::make(
+ //ctx, (unsigned int)port_id, address, port, "0", buff_args, dev_addr);
+ //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);
+
+ //// Little helper to check returned packets match the sent ones
+ //auto require_bufs_match = [&send_buf, mtu_hdr_len](
+ //const uint8_t* recv_buf, 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) {
+ //managed_send_buffer::sptr msbuf = sock->get_send_buff(0);
+ //UHD_ASSERT_THROW(msbuf.get() != nullptr);
+ //max_frame_size = std::min(msbuf->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);
+ //// Copy to real buffer
+ //UHD_LOG_TRACE("MPMD", "Testing frame size " << test_frame_size);
+ //auto* tx_buf = msbuf->cast<uint8_t*>();
+ //std::memcpy(tx_buf, &send_buf[0], test_frame_size);
+ //msbuf->commit(test_frame_size);
+ //msbuf.reset();
+
+ //managed_recv_buffer::sptr mrbuf = sock->get_recv_buff(echo_timeout);
+ //if (mrbuf.get() == nullptr || mrbuf->size() == 0) {
+ //// Nothing received, so this is probably too big
+ //max_frame_size = test_frame_size - 4;
+ //} else if (mrbuf->size() >= test_frame_size) {
+ //// Size went through, so bump the minimum
+ //require_bufs_match(mrbuf->cast<uint8_t*>(), mrbuf->size());
+ //min_frame_size = test_frame_size;
+ //} else if (mrbuf->size() < 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(mrbuf->cast<uint8_t*>(), mrbuf->size());
+ //max_frame_size = mrbuf->size();
+ //}
+ //mrbuf.reset();
+ //}
+ //UHD_LOG_DEBUG("MPMD", "Path MTU for address " << address << ": " << min_frame_size);
+ //return min_frame_size;
+//}
+
} // namespace
@@ -198,14 +295,16 @@ mpmd_link_if_ctrl_udp::mpmd_link_if_ctrl_udp(const uhd::device_addr_t& mb_args,
, _udp_info(get_udp_info_from_xport_info(xport_info))
, _mtu(MPMD_10GE_DATA_FRAME_MAX_SIZE)
{
+ const bool use_dpdk = mb_args.has_key("use_dpdk"); // FIXME use constrained_device_args
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) {
+ auto discover_mtu_for_ip = [mpm_discovery_port, use_dpdk](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);
+ MPMD_MTU_DISCOVERY_TIMEOUT,
+ use_dpdk);
};
const std::vector<std::string> requested_addrs(
@@ -259,19 +358,26 @@ uhd::transport::both_links_t mpmd_link_if_ctrl_udp::get_link(const size_t link_i
link_args);
// Enforce a minimum bound of the number of receive and send frames.
- link_params.num_send_frames = std::max(uhd::rfnoc::MIN_NUM_FRAMES, link_params.num_send_frames);
- link_params.num_recv_frames = std::max(uhd::rfnoc::MIN_NUM_FRAMES, link_params.num_recv_frames);
+ link_params.num_send_frames =
+ std::max(uhd::rfnoc::MIN_NUM_FRAMES, link_params.num_send_frames);
+ link_params.num_recv_frames =
+ std::max(uhd::rfnoc::MIN_NUM_FRAMES, link_params.num_recv_frames);
+ if (_mb_args.has_key("use_dpdk")) { // FIXME use constrained device args
+#ifdef HAVE_DPDK
+ auto link = uhd::transport::udp_dpdk_link::make(ip_addr, udp_port, link_params);
+ return std::make_tuple(
+ link, link_params.send_buff_size, link, link_params.recv_buff_size, true);
+#else
+ UHD_LOG_WARNING("X300", "Cannot create DPDK transport, falling back to UDP");
+#endif
+ }
auto link = uhd::transport::udp_boost_asio_link::make(ip_addr,
udp_port,
link_params,
link_params.recv_buff_size,
link_params.send_buff_size);
- return std::tuple<send_link_if::sptr,
- size_t,
- recv_link_if::sptr,
- size_t,
- bool>(
+ return std::make_tuple(
link, link_params.send_buff_size, link, link_params.recv_buff_size, true);
}
diff --git a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.hpp b/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.hpp
index 33db83b47..c80f1e613 100644
--- a/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.hpp
+++ b/host/lib/usrp/mpmd/mpmd_link_if_ctrl_udp.hpp
@@ -14,7 +14,7 @@
namespace uhd { namespace mpmd { namespace xport {
-/*! UDP link interface controller
+/*! UDP link interface controller (for both kernel-based UDP and DPDK)
*
* Opens UDP sockets
*/
diff --git a/host/lib/usrp/mpmd/mpmd_link_if_mgr.cpp b/host/lib/usrp/mpmd/mpmd_link_if_mgr.cpp
index 6bb6cae3a..4b36e7ba4 100644
--- a/host/lib/usrp/mpmd/mpmd_link_if_mgr.cpp
+++ b/host/lib/usrp/mpmd/mpmd_link_if_mgr.cpp
@@ -12,9 +12,6 @@
#ifdef HAVE_LIBERIO
# include "mpmd_link_if_ctrl_liberio.hpp"
#endif
-#ifdef HAVE_DPDK
-# include "mpmd_link_if_ctrl_dpdk_udp.hpp"
-#endif
uhd::dict<std::string, std::string> uhd::mpmd::xport::filter_args(
const uhd::device_addr_t& args, const std::string& prefix)
diff --git a/host/lib/usrp/x300/CMakeLists.txt b/host/lib/usrp/x300/CMakeLists.txt
index 1bd71dab4..b00ee357c 100644
--- a/host/lib/usrp/x300/CMakeLists.txt
+++ b/host/lib/usrp/x300/CMakeLists.txt
@@ -35,4 +35,8 @@ if(ENABLE_X300)
${CMAKE_CURRENT_SOURCE_DIR}/x300_prop_tree.cpp
${CMAKE_CURRENT_SOURCE_DIR}/cdecode.c
)
+
+ if(ENABLE_DPDK)
+ add_definitions(-DHAVE_DPDK)
+ endif(ENABLE_DPDK)
endif(ENABLE_X300)
diff --git a/host/lib/usrp/x300/x300_eth_mgr.cpp b/host/lib/usrp/x300/x300_eth_mgr.cpp
index 7177032c6..fd826e484 100644
--- a/host/lib/usrp/x300/x300_eth_mgr.cpp
+++ b/host/lib/usrp/x300/x300_eth_mgr.cpp
@@ -7,6 +7,7 @@
#include "x300_eth_mgr.hpp"
#include "x300_claim.hpp"
#include "x300_defaults.hpp"
+#include "x300_device_args.hpp"
#include "x300_fw_common.h"
#include "x300_mb_eeprom.hpp"
#include "x300_mb_eeprom_iface.hpp"
@@ -17,16 +18,17 @@
#include <uhd/transport/udp_constants.hpp>
#include <uhd/transport/udp_simple.hpp>
#include <uhd/transport/udp_zero_copy.hpp>
+#include <uhd/utils/algorithm.hpp>
#include <uhd/utils/byteswap.hpp>
#include <uhdlib/rfnoc/device_id.hpp>
#include <uhdlib/rfnoc/rfnoc_common.hpp>
#include <uhdlib/transport/udp_boost_asio_link.hpp>
#include <uhdlib/transport/udp_common.hpp>
#include <uhdlib/usrp/cores/i2c_core_100_wb32.hpp>
-//#ifdef HAVE_DPDK
-//# include <uhdlib/transport/dpdk_simple.hpp>
-//# include <uhdlib/transport/dpdk_zero_copy.hpp>
-//#endif
+#ifdef HAVE_DPDK
+# include <uhdlib/transport/dpdk_simple.hpp>
+# include <uhdlib/transport/udp_dpdk_link.hpp>
+#endif
#include <boost/asio.hpp>
#include <string>
@@ -69,42 +71,41 @@ constexpr size_t MAX_RATE_1GIGE = (size_t)( // bytes/s
/******************************************************************************
* Static Methods
*****************************************************************************/
-eth_manager::udp_simple_factory_t eth_manager::x300_get_udp_factory(
- const device_addr_t& args)
+eth_manager::udp_simple_factory_t eth_manager::x300_get_udp_factory(const bool use_dpdk)
{
udp_simple_factory_t udp_make_connected = udp_simple::make_connected;
- if (args.has_key("use_dpdk")) {
-//#ifdef HAVE_DPDK
-// udp_make_connected = [](const std::string& addr, const std::string& port) {
-// auto& ctx = uhd::transport::uhd_dpdk_ctx::get();
-// return dpdk_simple::make_connected(ctx, addr, port);
-// };
-//#else
+ if (use_dpdk) {
+#ifdef HAVE_DPDK
+ udp_make_connected = [](const std::string& addr, const std::string& port) {
+ return dpdk_simple::make_connected(addr, port);
+ };
+#else
UHD_LOG_WARNING(
"DPDK", "Detected use_dpdk argument, but DPDK support not built in.");
-//#endif
+#endif
}
return udp_make_connected;
}
device_addrs_t eth_manager::find(const device_addr_t& hint)
{
+ x300_device_args_t hint_args;
+ hint_args.parse(hint);
+
udp_simple_factory_t udp_make_broadcast = udp_simple::make_broadcast;
- udp_simple_factory_t udp_make_connected = x300_get_udp_factory(hint);
+ udp_simple_factory_t udp_make_connected =
+ x300_get_udp_factory(hint_args.get_use_dpdk());
#ifdef HAVE_DPDK
- if (hint.has_key("use_dpdk")) {
-// auto& dpdk_ctx = uhd::transport::uhd_dpdk_ctx::get();
-// if (not dpdk_ctx.is_init_done()) {
-// dpdk_ctx.init(hint);
-// }
-// udp_make_broadcast = [](const std::string& addr, const std::string& port) {
-// auto& ctx = uhd::transport::uhd_dpdk_ctx::get();
-// return dpdk_simple::make_broadcast(ctx, addr, port);
-// };
+ if (hint_args.get_use_dpdk()) {
+ auto dpdk_ctx = uhd::transport::dpdk::dpdk_ctx::get();
+ if (not dpdk_ctx->is_init_done()) {
+ dpdk_ctx->init(hint);
+ }
+ udp_make_broadcast = dpdk_simple::make_broadcast;
}
#endif
- udp_simple::sptr comm =
- udp_make_broadcast(hint["addr"], BOOST_STRINGIZE(X300_FW_COMMS_UDP_PORT));
+ udp_simple::sptr comm = udp_make_broadcast(
+ hint_args.get_first_addr(), BOOST_STRINGIZE(X300_FW_COMMS_UDP_PORT));
// load request struct
x300_fw_comms_t request = x300_fw_comms_t();
@@ -177,9 +178,8 @@ device_addrs_t eth_manager::find(const device_addr_t& hint)
/******************************************************************************
* Structors
*****************************************************************************/
-eth_manager::eth_manager(const x300_device_args_t& args,
- uhd::property_tree::sptr tree,
- const uhd::fs_path& root_path)
+eth_manager::eth_manager(
+ const x300_device_args_t& args, uhd::property_tree::sptr, const uhd::fs_path&)
: _args(args)
{
UHD_ASSERT_THROW(!args.get_first_addr().empty());
@@ -201,10 +201,7 @@ eth_manager::eth_manager(const x300_device_args_t& args,
_local_device_ids.push_back(device_id);
eth_conns[device_id] = init;
- _x300_make_udp_connected = x300_get_udp_factory(dev_addr);
-
- tree->create<double>(root_path / "link_max_rate").set(10e9);
- _tree = tree->subtree(root_path);
+ _x300_make_udp_connected = x300_get_udp_factory(args.get_use_dpdk());
}
both_links_t eth_manager::get_links(link_type_t link_type,
@@ -213,12 +210,13 @@ both_links_t eth_manager::get_links(link_type_t link_type,
const sep_id_t& /*remote_epid*/,
const device_addr_t& link_args)
{
- if (std::find(_local_device_ids.cbegin(), _local_device_ids.cend(), local_device_id)
- == _local_device_ids.cend()) {
- throw uhd::runtime_error(
- std::string("[X300] Cannot create Ethernet link through local device ID ")
+ if (!uhd::has(_local_device_ids, local_device_id)) {
+ const std::string err_msg =
+ std::string("Cannot create Ethernet link through local device ID ")
+ std::to_string(local_device_id)
- + ", no such device associated with this motherboard!");
+ + ", no such device associated with this motherboard!";
+ UHD_LOG_ERROR("X300", err_msg);
+ throw uhd::runtime_error(err_msg);
}
// FIXME: We now need to properly associate local_device_id with the right
// entry in eth_conn. We should probably do the load balancing elsewhere,
@@ -236,56 +234,6 @@ both_links_t eth_manager::get_links(link_type_t link_type,
default_buff_args.send_frame_size = std::min(send_mtu, ETH_MSG_FRAME_SIZE);
default_buff_args.recv_frame_size = std::min(recv_mtu, ETH_MSG_FRAME_SIZE);
- if (_args.get_use_dpdk()) {
-//#ifdef HAVE_DPDK
- // auto& dpdk_ctx = uhd::transport::uhd_dpdk_ctx::get();
-
-// default_buff_args.num_recv_frames = ETH_MSG_NUM_FRAMES;
-// default_buff_args.num_send_frames = ETH_MSG_NUM_FRAMES;
-// if (link_type == link_type_t::CTRL) {
-//// Increasing number of recv frames here because ctrl_iface uses it
-//// to determine how many control packets can be in flight before it
-//// must wait for an ACK
-// default_buff_args.num_recv_frames =
-// uhd::rfnoc::CMD_FIFO_SIZE / uhd::rfnoc::MAX_CMD_PKT_SIZE;
-//} else if (xport_type == uhd::transport::link_type_t::TX_DATA) {
-// size_t default_frame_size = conn.link_rate == MAX_RATE_1GIGE
-//? GE_DATA_FRAME_SEND_SIZE
-//: XGE_DATA_FRAME_SEND_SIZE;
-// default_buff_args.send_frame_size = args.cast<size_t>(
-//"send_frame_size", std::min(default_frame_size, send_mtu));
-// default_buff_args.num_send_frames =
-// args.cast<size_t>("num_send_frames", default_buff_args.num_send_frames);
-// default_buff_args.send_buff_size = args.cast<size_t>("send_buff_size", 0);
-//} else if (xport_type == uhd::transport::link_type_t::RX_DATA) {
-// size_t default_frame_size = conn.link_rate == MAX_RATE_1GIGE
-//? GE_DATA_FRAME_RECV_SIZE
-//: XGE_DATA_FRAME_RECV_SIZE;
-// default_buff_args.recv_frame_size = args.cast<size_t>(
-//"recv_frame_size", std::min(default_frame_size, recv_mtu));
-// default_buff_args.num_recv_frames =
-// args.cast<size_t>("num_recv_frames", default_buff_args.num_recv_frames);
-// default_buff_args.recv_buff_size = args.cast<size_t>("recv_buff_size", 0);
-//}
-
-// int dpdk_port_id = dpdk_ctx.get_route(conn.addr);
-// if (dpdk_port_id < 0) {
-// throw uhd::runtime_error(
-//"Could not find a DPDK port with route to " + conn.addr);
-//}
-// auto recv = transport::dpdk_zero_copy::make(dpdk_ctx,
-//(const unsigned int)dpdk_port_id,
-// conn.addr,
-// BOOST_STRINGIZE(X300_VITA_UDP_PORT),
-//"0",
-// default_buff_args,
-// uhd::device_addr_t());
-
-//#else
- UHD_LOG_WARNING("X300", "Cannot create DPDK transport, falling back to UDP");
-//#endif
- }
-
// Buffering is done in the socket buffers, so size them relative to
// the link rate
link_params_t default_link_params;
@@ -315,7 +263,19 @@ both_links_t eth_manager::get_links(link_type_t link_type,
link_params.num_send_frames = std::max(uhd::rfnoc::MIN_NUM_FRAMES, link_params.num_send_frames);
link_params.num_recv_frames = std::max(uhd::rfnoc::MIN_NUM_FRAMES, link_params.num_recv_frames);
+ // Dummy variables for populating the return tuple
size_t recv_buff_size, send_buff_size;
+ if (_args.get_use_dpdk()) {
+#ifdef HAVE_DPDK
+ auto link = uhd::transport::udp_dpdk_link::make(
+ conn.addr, BOOST_STRINGIZE(X300_VITA_UDP_PORT), link_params);
+ return std::make_tuple(link, send_buff_size, link, recv_buff_size, true);
+#else
+ UHD_LOG_WARNING("X300", "Cannot create DPDK transport, falling back to UDP");
+#endif
+ }
+
+ // No DPDK, then return regular/kernel UDP
auto link = uhd::transport::udp_boost_asio_link::make(conn.addr,
BOOST_STRINGIZE(X300_VITA_UDP_PORT),
link_params,
@@ -333,11 +293,10 @@ wb_iface::sptr eth_manager::get_ctrl_iface()
get_pri_eth().addr, BOOST_STRINGIZE(X300_FW_COMMS_UDP_PORT)));
}
+// - Populates _max_frame_sizes
void eth_manager::init_link(
const mboard_eeprom_t& mb_eeprom, const std::string& loaded_fpga_image)
{
- double link_max_rate = 0.0;
-
// Discover ethernet interfaces on the device
discover_eth(mb_eeprom, loaded_fpga_image);
@@ -392,6 +351,8 @@ void eth_manager::init_link(
UHD_LOGGER_ERROR("X300") << e.what();
}
+ // Check actual frame sizes against user-requested frame sizes, and print
+ // warnings if they don't match
if ((recv_args.has_key("recv_frame_size"))
&& (req_max_frame_size.recv_frame_size > _max_frame_sizes.recv_frame_size)) {
UHD_LOGGER_WARNING("X300")
@@ -416,11 +377,10 @@ void eth_manager::init_link(
<< "UHD will use the auto-detected max frame size for this connection.";
}
- // Check frame sizes
+ // Check actual frame sizes against detected frame sizes, and print
+ // warnings if they don't match
for (auto conn_pair : eth_conns) {
auto conn = conn_pair.second;
- link_max_rate += conn.link_rate;
-
size_t rec_send_frame_size = conn.link_rate == MAX_RATE_1GIGE
? GE_DATA_FRAME_SEND_SIZE
: XGE_DATA_FRAME_SEND_SIZE;
@@ -452,10 +412,6 @@ void eth_manager::init_link(
"argument.";
}
}
-
- _tree->create<size_t>("mtu/recv").set(_max_frame_sizes.recv_frame_size);
- _tree->create<size_t>("mtu/send").set(_max_frame_sizes.send_frame_size);
- _tree->access<double>("link_max_rate").set(link_max_rate);
}
size_t eth_manager::get_mtu(uhd::direction_t dir)
@@ -468,7 +424,7 @@ size_t eth_manager::get_mtu(uhd::direction_t dir)
void eth_manager::discover_eth(
const mboard_eeprom_t mb_eeprom, const std::string& loaded_fpga_image)
{
- udp_simple_factory_t udp_make_connected = x300_get_udp_factory(_args.get_orig_args());
+ udp_simple_factory_t udp_make_connected = x300_get_udp_factory(_args.get_use_dpdk());
// Load all valid, non-duplicate IP addrs
std::vector<std::string> ip_addrs{_args.get_first_addr()};
if (not _args.get_second_addr().empty()
diff --git a/host/lib/usrp/x300/x300_eth_mgr.hpp b/host/lib/usrp/x300/x300_eth_mgr.hpp
index 1f4013e17..19b6064dd 100644
--- a/host/lib/usrp/x300/x300_eth_mgr.hpp
+++ b/host/lib/usrp/x300/x300_eth_mgr.hpp
@@ -99,7 +99,13 @@ private:
return eth_conns.at(_local_device_ids.at(0));
}
- static udp_simple_factory_t x300_get_udp_factory(const device_addr_t& args);
+ //! Create a factory function for UDP traffic
+ //
+ // \note This is static rather than local to x300_eth_mgr.cpp to get access
+ // to udp_simple_factory_t
+ // \param use_dpdk If true, use a DPDK transport instead of a regular UDP
+ // transport
+ static udp_simple_factory_t x300_get_udp_factory(const bool use_dpdk);
/*!
* Automatically determine the maximum frame size available by sending a UDP packet
@@ -109,15 +115,22 @@ private:
frame_size_t determine_max_frame_size(
const std::string& addr, const frame_size_t& user_mtu);
- // Discover the ethernet connections per motherboard
+ //! Discover the ethernet connections per motherboard
+ //
+ // - Gets called during init_link()
+ // - Populates eth_conn
+ // - Populates _local_device_ids
+ //
+ // \throws uhd::runtime_error if no Ethernet connections can be found
void discover_eth(
const uhd::usrp::mboard_eeprom_t mb_eeprom, const std::string& loaded_fpga_image);
-
+ /**************************************************************************
+ * Attributes
+ *************************************************************************/
+ // Cache the initial device args that brought up this motherboard
const x300_device_args_t _args;
- uhd::property_tree::sptr _tree;
-
udp_simple_factory_t _x300_make_udp_connected;
std::map<uhd::rfnoc::device_id_t, x300_eth_conn_t> eth_conns;
diff --git a/host/lib/usrp/x300/x300_impl.cpp b/host/lib/usrp/x300/x300_impl.cpp
index 636202d33..273485577 100644
--- a/host/lib/usrp/x300/x300_impl.cpp
+++ b/host/lib/usrp/x300/x300_impl.cpp
@@ -23,6 +23,9 @@
#include <chrono>
#include <fstream>
#include <thread>
+#ifdef HAVE_DPDK
+# include <uhdlib/transport/dpdk/common.hpp>
+#endif
uhd::uart_iface::sptr x300_make_uart_iface(uhd::wb_iface::sptr iface);
diff --git a/host/lib/usrp/x300/x300_impl.hpp b/host/lib/usrp/x300/x300_impl.hpp
index a3276152a..a493d4e8c 100644
--- a/host/lib/usrp/x300/x300_impl.hpp
+++ b/host/lib/usrp/x300/x300_impl.hpp
@@ -90,7 +90,6 @@ private:
std::string loaded_fpga_image;
size_t hw_rev;
- std::string current_refclk_src;
uhd::usrp::x300::conn_manager::sptr conn_mgr;
};