aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/usrp/x300
diff options
context:
space:
mode:
authorMichael West <michael.west@ettus.com>2019-06-19 13:25:32 -0700
committermichael-west <michael.west@ettus.com>2019-06-28 08:21:09 -0700
commit3f7b937d3fbabe0a885bdcaf8137502528db6e68 (patch)
treeb10d741b615cf9707420a7e24276d0ca67858c4b /host/lib/usrp/x300
parent55b9f7309f6f49eccd1157e898b77dc9a6afd238 (diff)
downloaduhd-3f7b937d3fbabe0a885bdcaf8137502528db6e68.tar.gz
uhd-3f7b937d3fbabe0a885bdcaf8137502528db6e68.tar.bz2
uhd-3f7b937d3fbabe0a885bdcaf8137502528db6e68.zip
Device3: Fix MTU and default frame sizes
The latest changes to the get_*x_stream() functions to calculate the MTU for the channel caused default frame size values to be ignored. This change fixes that by changing the key from "send/recv_frame_size" to "mtu" and then changing the implementations of make_transport() constrain the frame size values based on the "mtu" value as well as any device and/or transport-specific limits. Signed-off-by: Michael West <michael.west@ettus.com>
Diffstat (limited to 'host/lib/usrp/x300')
-rw-r--r--host/lib/usrp/x300/x300_impl.cpp158
1 files changed, 90 insertions, 68 deletions
diff --git a/host/lib/usrp/x300/x300_impl.cpp b/host/lib/usrp/x300/x300_impl.cpp
index f6eda34ff..3106c0e19 100644
--- a/host/lib/usrp/x300/x300_impl.cpp
+++ b/host/lib/usrp/x300/x300_impl.cpp
@@ -872,9 +872,12 @@ void x300_impl::setup_mb(const size_t mb_i, const uhd::device_addr_t& dev_addr)
_tree->create<std::string>(mb_path / "codename").set("Yetti");
////////////////////////////////////////////////////////////////////
- // discover ethernet interfaces, frame sizes, and link rates
+ // discover interfaces, frame sizes, and link rates
////////////////////////////////////////////////////////////////////
- if (mb.xport_path == "eth") {
+ if (mb.xport_path == "nirio") {
+ _max_frame_sizes.recv_frame_size = PCIE_RX_DATA_FRAME_SIZE;
+ _max_frame_sizes.send_frame_size = PCIE_TX_DATA_FRAME_SIZE;
+ } else if (mb.xport_path == "eth") {
double link_max_rate = 0.0;
// Discover ethernet interfaces
@@ -1320,12 +1323,17 @@ uhd::both_xports_t x300_impl::make_transport(const uhd::sid_t& address,
{
const size_t mb_index = address.get_dst_addr() - x300::DST_ADDR;
mboard_members_t& mb = _mb[mb_index];
- const uhd::device_addr_t& xport_args = (xport_type == CTRL) ? uhd::device_addr_t()
- : args;
zero_copy_xport_params default_buff_args;
both_xports_t xports;
xports.endianness = mb.if_pkt_is_big_endian ? ENDIANNESS_BIG : ENDIANNESS_LITTLE;
+
+ // Calculate MTU based on MTU in args and device limitations
+ const size_t send_mtu = args.cast<size_t>("mtu",
+ get_mtu(mb_index, uhd::TX_DIRECTION));
+ const size_t recv_mtu = args.cast<size_t>("mtu",
+ get_mtu(mb_index, uhd::RX_DIRECTION));
+
if (mb.xport_path == "nirio") {
xports.lossless = true;
xports.send_sid =
@@ -1355,26 +1363,30 @@ uhd::both_xports_t x300_impl::make_transport(const uhd::sid_t& address,
}
// Create a virtual async message transport
xports.recv = mb.async_msg_dma_xport->make_stream(xports.recv_sid.get_dst());
- } else {
- // Transport for data stream
- default_buff_args.send_frame_size = (xport_type == TX_DATA)
- ? x300::PCIE_TX_DATA_FRAME_SIZE
- : x300::PCIE_MSG_FRAME_SIZE;
-
- default_buff_args.recv_frame_size = (xport_type == RX_DATA)
- ? x300::PCIE_RX_DATA_FRAME_SIZE
- : x300::PCIE_MSG_FRAME_SIZE;
-
- default_buff_args.num_send_frames = (xport_type == TX_DATA)
- ? x300::PCIE_TX_DATA_NUM_FRAMES
- : x300::PCIE_MSG_NUM_FRAMES;
-
- default_buff_args.num_recv_frames = (xport_type == RX_DATA)
- ? x300::PCIE_RX_DATA_NUM_FRAMES
- : x300::PCIE_MSG_NUM_FRAMES;
-
+ } else if (xport_type == TX_DATA) {
+ default_buff_args.send_frame_size = args.cast<size_t>(
+ "send_frame_size", std::min(send_mtu,
+ x300::PCIE_TX_DATA_FRAME_SIZE));
+ default_buff_args.num_send_frames = args.cast<size_t>(
+ "num_send_frames", x300::PCIE_TX_DATA_NUM_FRAMES);
+ default_buff_args.send_buff_size = args.cast<size_t>(
+ "send_buff_size", 0);
+ default_buff_args.recv_frame_size = x300::PCIE_MSG_FRAME_SIZE;
+ default_buff_args.num_recv_frames = x300::PCIE_MSG_NUM_FRAMES;
xports.recv = nirio_zero_copy::make(
- mb.rio_fpga_interface, dma_channel_num, default_buff_args, xport_args);
+ mb.rio_fpga_interface, dma_channel_num, default_buff_args);
+ } else if (xport_type == RX_DATA) {
+ default_buff_args.send_frame_size = x300::PCIE_MSG_FRAME_SIZE;
+ default_buff_args.num_send_frames = x300::PCIE_MSG_NUM_FRAMES;
+ default_buff_args.recv_frame_size = args.cast<size_t>(
+ "recv_frame_size", std::min(recv_mtu,
+ x300::PCIE_RX_DATA_FRAME_SIZE));
+ default_buff_args.num_recv_frames = args.cast<size_t>(
+ "num_recv_frames", x300::PCIE_RX_DATA_NUM_FRAMES);
+ default_buff_args.recv_buff_size = args.cast<size_t>(
+ "recv_buff_size", 0);
+ xports.recv = nirio_zero_copy::make(
+ mb.rio_fpga_interface, dma_channel_num, default_buff_args);
}
xports.send = xports.recv;
@@ -1418,22 +1430,42 @@ uhd::both_xports_t x300_impl::make_transport(const uhd::sid_t& address,
xports.recv_sid = xports.send_sid.reversed();
// Set size and number of frames
- size_t system_max_send_frame_size = (size_t)_max_frame_sizes.send_frame_size;
- size_t system_max_recv_frame_size = (size_t)_max_frame_sizes.recv_frame_size;
- default_buff_args.send_frame_size = xport_args.cast<size_t>("send_frame_size",
- std::min(system_max_send_frame_size, x300::ETH_MSG_FRAME_SIZE));
- default_buff_args.recv_frame_size = xport_args.cast<size_t>("recv_frame_size",
- std::min(system_max_recv_frame_size, x300::ETH_MSG_FRAME_SIZE));
- default_buff_args.num_recv_frames =
- xport_args.cast<size_t>("num_recv_frames", x300::ETH_MSG_NUM_FRAMES);
- default_buff_args.num_send_frames =
- xport_args.cast<size_t>("num_send_frames", x300::ETH_MSG_NUM_FRAMES);
+ default_buff_args.send_frame_size = std::min(send_mtu,
+ x300::ETH_MSG_FRAME_SIZE);
+ default_buff_args.recv_frame_size = std::min(recv_mtu,
+ x300::ETH_MSG_FRAME_SIZE);
+ default_buff_args.num_recv_frames = x300::ETH_MSG_NUM_FRAMES;
+ default_buff_args.num_send_frames = x300::ETH_MSG_NUM_FRAMES;
if (xport_type == 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 == TX_DATA) {
+ size_t default_frame_size = conn.link_rate == x300::MAX_RATE_1GIGE
+ ? x300::GE_DATA_FRAME_SEND_SIZE
+ : x300::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 == RX_DATA) {
+ size_t default_frame_size = conn.link_rate == x300::MAX_RATE_1GIGE
+ ? x300::GE_DATA_FRAME_RECV_SIZE
+ : x300::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);
@@ -1448,7 +1480,7 @@ uhd::both_xports_t x300_impl::make_transport(const uhd::sid_t& address,
BOOST_STRINGIZE(X300_VITA_UDP_PORT),
"0",
default_buff_args,
- xport_args
+ uhd::device_addr_t()
);
xports.recv = recv; // Note: This is a type cast!
@@ -1505,12 +1537,10 @@ uhd::both_xports_t x300_impl::make_transport(const uhd::sid_t& address,
xports.recv_sid = xports.send_sid.reversed();
// Set size and number of frames
- size_t system_max_send_frame_size = (size_t)_max_frame_sizes.send_frame_size;
- size_t system_max_recv_frame_size = (size_t)_max_frame_sizes.recv_frame_size;
- default_buff_args.send_frame_size =
- std::min(system_max_send_frame_size, x300::ETH_MSG_FRAME_SIZE);
- default_buff_args.recv_frame_size =
- std::min(system_max_recv_frame_size, x300::ETH_MSG_FRAME_SIZE);
+ default_buff_args.send_frame_size = std::min(send_mtu,
+ x300::ETH_MSG_FRAME_SIZE);
+ default_buff_args.recv_frame_size = std::min(recv_mtu,
+ x300::ETH_MSG_FRAME_SIZE);
// Buffering is done in the socket buffers, so size them relative to
// the link rate
default_buff_args.send_buff_size = conn.link_rate / 50; // 20ms
@@ -1532,33 +1562,29 @@ uhd::both_xports_t x300_impl::make_transport(const uhd::sid_t& address,
size_t default_frame_size = conn.link_rate == x300::MAX_RATE_1GIGE
? x300::GE_DATA_FRAME_SEND_SIZE
: x300::XGE_DATA_FRAME_SEND_SIZE;
- default_buff_args.send_frame_size = args.cast<size_t>("send_frame_size",
- std::min(default_frame_size, system_max_send_frame_size));
- if (default_buff_args.send_frame_size > system_max_send_frame_size) {
- UHD_LOGGER_WARNING("X300")
- << boost::format("Requested send_frame_size of %d exceeds the "
- "maximum allowed on the %s connection. Using %d.")
- % default_buff_args.send_frame_size % conn.addr
- % system_max_send_frame_size;
- default_buff_args.send_frame_size = system_max_send_frame_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",
+ default_buff_args.send_buff_size);
} else if (xport_type == RX_DATA) {
size_t default_frame_size = conn.link_rate == x300::MAX_RATE_1GIGE
? x300::GE_DATA_FRAME_RECV_SIZE
: x300::XGE_DATA_FRAME_RECV_SIZE;
- default_buff_args.recv_frame_size = args.cast<size_t>("recv_frame_size",
- std::min(default_frame_size, system_max_recv_frame_size));
- if (default_buff_args.recv_frame_size > system_max_recv_frame_size) {
- UHD_LOGGER_WARNING("X300")
- << boost::format("Requested recv_frame_size of %d exceeds the "
- "maximum allowed on the %s connection. Using %d.")
- % default_buff_args.recv_frame_size % conn.addr
- % system_max_recv_frame_size;
- default_buff_args.recv_frame_size = system_max_recv_frame_size;
- }
+ default_buff_args.recv_frame_size =
+ args.cast<size_t>("recv_frame_size",
+ std::min(default_frame_size, recv_mtu));
+ // set some buffers so the offload thread actually offloads the
+ // socket I/O
default_buff_args.num_recv_frames =
- 2; // set some buffers so the offload thread actually offloads the socket
- // I/O
+ args.cast<size_t>("num_recv_frames", 2);
+ default_buff_args.recv_buff_size =
+ args.cast<size_t>("recv_buff_size",
+ default_buff_args.recv_buff_size);
}
// make a new transport - fpga has no idea how to talk to us on this yet
@@ -1566,8 +1592,7 @@ uhd::both_xports_t x300_impl::make_transport(const uhd::sid_t& address,
xports.recv = udp_zero_copy::make(conn.addr,
BOOST_STRINGIZE(X300_VITA_UDP_PORT),
default_buff_args,
- buff_params,
- xport_args);
+ buff_params);
// Create a threaded transport for the receive chain only
// Note that this shouldn't affect PCIe
@@ -1993,10 +2018,7 @@ x300_impl::frame_size_t x300_impl::determine_max_frame_size(
return frame_size;
}
-size_t x300_impl::get_mtu(const size_t mb_index, const uhd::direction_t dir) {
- if (_mb[mb_index].xport_path == "nirio") {
- return (dir == RX_DIRECTION ? x300::PCIE_RX_DATA_FRAME_SIZE : x300::PCIE_TX_DATA_FRAME_SIZE);
- }
+size_t x300_impl::get_mtu(const size_t /*mb_index*/, const uhd::direction_t dir) {
return (dir == RX_DIRECTION) ? _max_frame_sizes.recv_frame_size :
_max_frame_sizes.send_frame_size;
}