aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/usrp/device3
diff options
context:
space:
mode:
Diffstat (limited to 'host/lib/usrp/device3')
-rw-r--r--host/lib/usrp/device3/CMakeLists.txt25
-rw-r--r--host/lib/usrp/device3/device3_impl.cpp188
-rw-r--r--host/lib/usrp/device3/device3_impl.hpp210
-rw-r--r--host/lib/usrp/device3/device3_io_impl.cpp851
4 files changed, 1274 insertions, 0 deletions
diff --git a/host/lib/usrp/device3/CMakeLists.txt b/host/lib/usrp/device3/CMakeLists.txt
new file mode 100644
index 000000000..83f01a2e7
--- /dev/null
+++ b/host/lib/usrp/device3/CMakeLists.txt
@@ -0,0 +1,25 @@
+#
+# Copyright 2014 Ettus Research LLC
+#
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program. If not, see <http://www.gnu.org/licenses/>.
+#
+
+########################################################################
+# This file included, use CMake directory variables
+########################################################################
+
+LIBUHD_APPEND_SOURCES(
+ ${CMAKE_CURRENT_SOURCE_DIR}/device3_impl.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/device3_io_impl.cpp
+)
diff --git a/host/lib/usrp/device3/device3_impl.cpp b/host/lib/usrp/device3/device3_impl.cpp
new file mode 100644
index 000000000..7fcbc01b2
--- /dev/null
+++ b/host/lib/usrp/device3/device3_impl.cpp
@@ -0,0 +1,188 @@
+//
+// Copyright 2014 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#include "device3_impl.hpp"
+#include "graph_impl.hpp"
+#include <uhd/utils/msg.hpp>
+#include <uhd/rfnoc/block_ctrl_base.hpp>
+#include <boost/make_shared.hpp>
+#include <algorithm>
+
+#define UHD_DEVICE3_LOG() UHD_LOGV(never)
+
+using namespace uhd::usrp;
+
+device3_impl::device3_impl()
+ : _sid_framer(0)
+{
+ _type = uhd::device::USRP;
+ _async_md.reset(new async_md_type(1000/*messages deep*/));
+ _tree = uhd::property_tree::make();
+};
+
+//! Returns true if the integer value stored in lhs is smaller than that in rhs
+bool _compare_string_indexes(const std::string &lhs, const std::string &rhs)
+{
+ return boost::lexical_cast<size_t>(lhs) < boost::lexical_cast<size_t>(rhs);
+}
+
+void device3_impl::merge_channel_defs(
+ const std::vector<uhd::rfnoc::block_id_t> &chan_ids,
+ const std::vector<uhd::device_addr_t> &chan_args,
+ const uhd::direction_t dir
+) {
+ UHD_ASSERT_THROW(chan_ids.size() == chan_args.size());
+ if (dir == uhd::DX_DIRECTION) {
+ merge_channel_defs(chan_ids, chan_args, RX_DIRECTION);
+ merge_channel_defs(chan_ids, chan_args, TX_DIRECTION);
+ return;
+ }
+
+ uhd::fs_path chans_root = uhd::fs_path("/channels/") / (dir == RX_DIRECTION ? "rx" : "tx");
+ // Store the new positions of the channels:
+ std::vector<size_t> chan_idxs;
+
+ // 1. Get sorted list of currently defined channels
+ std::vector<std::string> curr_channels;
+ if (_tree->exists(chans_root)) {
+ curr_channels = _tree->list(chans_root);
+ std::sort(curr_channels.begin(), curr_channels.end(), _compare_string_indexes);
+ }
+
+ // 2. Cycle through existing channels to find out where to merge
+ // the new channels. Rules are:
+ // - The order of chan_ids must be preserved
+ // - All block indices that are in chan_ids may be overwritten in the channel definition
+ // - If the channels in chan_ids are not yet in the property tree channel list,
+ // they are appended.
+ BOOST_FOREACH(const std::string &chan_idx, curr_channels) {
+ if (_tree->exists(chans_root / chan_idx)) {
+ rfnoc::block_id_t chan_block_id = _tree->access<rfnoc::block_id_t>(chans_root / chan_idx).get();
+ if (std::find(chan_ids.begin(), chan_ids.end(), chan_block_id) != chan_ids.end()) {
+ chan_idxs.push_back(boost::lexical_cast<size_t>(chan_idx));
+ }
+ }
+ }
+ size_t last_chan_idx = curr_channels.empty() ? 0 : (boost::lexical_cast<size_t>(curr_channels.back()) + 1);
+ while (chan_idxs.size() < chan_ids.size()) {
+ chan_idxs.push_back(last_chan_idx);
+ last_chan_idx++;
+ }
+
+ // 3. Write the new channels
+ for (size_t i = 0; i < chan_ids.size(); i++) {
+ if (not _tree->exists(chans_root / chan_idxs[i])) {
+ _tree->create<rfnoc::block_id_t>(chans_root / chan_idxs[i]);
+ }
+ _tree->access<rfnoc::block_id_t>(chans_root / chan_idxs[i]).set(chan_ids[i]);
+ if (not _tree->exists(chans_root / chan_idxs[i] / "args")) {
+ _tree->create<uhd::device_addr_t>(chans_root / chan_idxs[i] / "args");
+ }
+ _tree->access<uhd::device_addr_t>(chans_root / chan_idxs[i] / "args").set(chan_args[i]);
+ }
+}
+
+/***********************************************************************
+ * RFNoC-Specific
+ **********************************************************************/
+void device3_impl::enumerate_rfnoc_blocks(
+ size_t device_index,
+ size_t n_blocks,
+ size_t base_port,
+ const uhd::sid_t &base_sid,
+ uhd::device_addr_t transport_args,
+ uhd::endianness_t endianness
+) {
+ // entries that are already connected to this block
+ uhd::sid_t ctrl_sid = base_sid;
+ uhd::property_tree::sptr subtree = _tree->subtree(uhd::fs_path("/mboards") / device_index);
+ // 1) Clean property tree entries
+ // TODO put this back once radios are actual rfnoc blocks!!!!!!
+ //if (subtree->exists("xbar")) {
+ //subtree->remove("xbar");
+ //}
+ // 2) Destroy existing block controllers
+ // TODO: Clear out all the old block control classes
+ // 3) Create new block controllers
+ for (size_t i = 0; i < n_blocks; i++) {
+ UHD_DEVICE3_LOG() << "[RFNOC] ------- Block Setup -----------" << std::endl;
+ // First, make a transport for port number zero, because we always need that:
+ ctrl_sid.set_dst_xbarport(base_port + i);
+ ctrl_sid.set_dst_blockport(0);
+ both_xports_t xport = this->make_transport(
+ ctrl_sid,
+ CTRL,
+ transport_args
+ );
+ UHD_DEVICE3_LOG() << str(boost::format("Setting up NoC-Shell Control for port #0 (SID: %s)...") % xport.send_sid.to_pp_string_hex());
+ uhd::rfnoc::ctrl_iface::sptr ctrl = uhd::rfnoc::ctrl_iface::make(
+ endianness == ENDIANNESS_BIG,
+ xport.send,
+ xport.recv,
+ xport.send_sid,
+ str(boost::format("CE_%02d_Port_%02X") % i % ctrl_sid.get_dst_endpoint())
+ );
+ UHD_DEVICE3_LOG() << "OK" << std::endl;
+ uint64_t noc_id = ctrl->peek64(uhd::rfnoc::SR_READBACK_REG_ID);
+ UHD_DEVICE3_LOG() << str(boost::format("Port %d: Found NoC-Block with ID %016X.") % int(ctrl_sid.get_dst_endpoint()) % noc_id) << std::endl;
+ uhd::rfnoc::make_args_t make_args;
+ uhd::rfnoc::blockdef::sptr block_def = uhd::rfnoc::blockdef::make_from_noc_id(noc_id);
+ if (not block_def) {
+ UHD_DEVICE3_LOG() << "Using default block configuration." << std::endl;
+ block_def = uhd::rfnoc::blockdef::make_from_noc_id(uhd::rfnoc::DEFAULT_NOC_ID);
+ }
+ UHD_ASSERT_THROW(block_def);
+ make_args.ctrl_ifaces[0] = ctrl;
+ BOOST_FOREACH(const size_t port_number, block_def->get_all_port_numbers()) {
+ if (port_number == 0) { // We've already set this up
+ continue;
+ }
+ ctrl_sid.set_dst_blockport(port_number);
+ both_xports_t xport1 = this->make_transport(
+ ctrl_sid,
+ CTRL,
+ transport_args
+ );
+ UHD_DEVICE3_LOG() << str(boost::format("Setting up NoC-Shell Control for port #%d (SID: %s)...") % port_number % xport1.send_sid.to_pp_string_hex());
+ uhd::rfnoc::ctrl_iface::sptr ctrl1 = uhd::rfnoc::ctrl_iface::make(
+ endianness == ENDIANNESS_BIG,
+ xport1.send,
+ xport1.recv,
+ xport1.send_sid,
+ str(boost::format("CE_%02d_Port_%02d") % i % ctrl_sid.get_dst_endpoint())
+ );
+ UHD_DEVICE3_LOG() << "OK" << std::endl;
+ make_args.ctrl_ifaces[port_number] = ctrl1;
+ }
+
+ make_args.base_address = xport.send_sid.get_dst();
+ make_args.device_index = device_index;
+ make_args.tree = subtree;
+ make_args.is_big_endian = (endianness == ENDIANNESS_BIG);
+ _rfnoc_block_ctrl.push_back(uhd::rfnoc::block_ctrl_base::make(make_args, noc_id));
+ }
+}
+
+
+uhd::rfnoc::graph::sptr device3_impl::create_graph(const std::string &name)
+{
+ return boost::make_shared<uhd::rfnoc::graph_impl>(
+ name,
+ shared_from_this()
+ );
+}
+
diff --git a/host/lib/usrp/device3/device3_impl.hpp b/host/lib/usrp/device3/device3_impl.hpp
new file mode 100644
index 000000000..0d94ae21c
--- /dev/null
+++ b/host/lib/usrp/device3/device3_impl.hpp
@@ -0,0 +1,210 @@
+//
+// Copyright 2014-2015 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+// Declares the device3_impl class which is a layer between device3 and
+// the different 3-rd gen device impls (e.g. x300_impl)
+
+#ifndef INCLUDED_DEVICE3_IMPL_HPP
+#define INCLUDED_DEVICE3_IMPL_HPP
+
+#include <uhd/transport/bounded_buffer.hpp>
+#include <uhd/transport/vrt_if_packet.hpp>
+#include <uhd/transport/chdr.hpp>
+#include <uhd/transport/zero_copy.hpp>
+#include <uhd/types/sid.hpp>
+#include <uhd/types/metadata.hpp>
+#include <uhd/types/endianness.hpp>
+#include <uhd/types/direction.hpp>
+#include <uhd/utils/tasks.hpp>
+#include <uhd/device3.hpp>
+#include "xports.hpp"
+// Common FPGA cores:
+#include "ctrl_iface.hpp"
+#include "rx_dsp_core_3000.hpp"
+#include "tx_dsp_core_3000.hpp"
+#include "rx_vita_core_3000.hpp"
+#include "tx_vita_core_3000.hpp"
+#include "rx_frontend_core_200.hpp"
+#include "tx_frontend_core_200.hpp"
+#include "time_core_3000.hpp"
+#include "gpio_atr_3000.hpp"
+// RFNoC-specific includes:
+#include "radio_ctrl_impl.hpp"
+
+namespace uhd { namespace usrp {
+
+/***********************************************************************
+ * Default settings (any device3 may override these)
+ **********************************************************************/
+static const size_t DEVICE3_RX_FC_REQUEST_FREQ = 32; //per flow-control window
+static const size_t DEVICE3_TX_FC_RESPONSE_FREQ = 8;
+static const size_t DEVICE3_TX_FC_RESPONSE_CYCLES = 0; // Cycles: Off.
+
+static const size_t DEVICE3_TX_MAX_HDR_LEN = uhd::transport::vrt::chdr::max_if_hdr_words64 * sizeof(boost::uint64_t); // Bytes
+static const size_t DEVICE3_RX_MAX_HDR_LEN = uhd::transport::vrt::chdr::max_if_hdr_words64 * sizeof(boost::uint64_t); // Bytes
+
+class device3_impl : public uhd::device3, public boost::enable_shared_from_this<device3_impl>
+{
+public:
+ /***********************************************************************
+ * device3-specific Types
+ **********************************************************************/
+ typedef uhd::transport::bounded_buffer<uhd::async_metadata_t> async_md_type;
+
+ //! The purpose of a transport
+ enum xport_type_t {
+ CTRL = 0,
+ TX_DATA,
+ RX_DATA
+ };
+
+ enum xport_t {AXI, ETH, PCIE};
+
+ //! Stores all streaming-related options
+ struct stream_options_t
+ {
+ //! Max size of the header in bytes for TX
+ size_t tx_max_len_hdr;
+ //! Max size of the header in bytes for RX
+ size_t rx_max_len_hdr;
+ //! How often we send ACKs to the upstream block per one full FC window
+ size_t rx_fc_request_freq;
+ //! How often the downstream block should send ACKs per one full FC window
+ size_t tx_fc_response_freq;
+ //! How often the downstream block should send ACKs in cycles
+ size_t tx_fc_response_cycles;
+ stream_options_t(void)
+ : tx_max_len_hdr(DEVICE3_TX_MAX_HDR_LEN)
+ , rx_max_len_hdr(DEVICE3_RX_MAX_HDR_LEN)
+ , rx_fc_request_freq(DEVICE3_RX_FC_REQUEST_FREQ)
+ , tx_fc_response_freq(DEVICE3_TX_FC_RESPONSE_FREQ)
+ , tx_fc_response_cycles(DEVICE3_TX_FC_RESPONSE_CYCLES)
+ {};
+ };
+
+ /***********************************************************************
+ * I/O Interface
+ **********************************************************************/
+ uhd::tx_streamer::sptr get_tx_stream(const uhd::stream_args_t &);
+ uhd::rx_streamer::sptr get_rx_stream(const uhd::stream_args_t &);
+ bool recv_async_msg(uhd::async_metadata_t &async_metadata, double timeout);
+
+ /***********************************************************************
+ * Other public APIs
+ **********************************************************************/
+ rfnoc::graph::sptr create_graph(const std::string &name="");
+
+protected:
+ /***********************************************************************
+ * Structors
+ **********************************************************************/
+ device3_impl();
+ virtual ~device3_impl() {};
+
+ /***********************************************************************
+ * Streaming-related
+ **********************************************************************/
+ // The 'rate' argument is so we can use these as subscribers to rate changes
+public: // TODO make these protected again
+ void update_rx_streamers(double rate=-1.0);
+ void update_tx_streamers(double rate=-1.0);
+protected:
+
+ /***********************************************************************
+ * Transport-related
+ **********************************************************************/
+ stream_options_t stream_options;
+
+ /*! \brief Create a transport to a given endpoint.
+ *
+ * \param address The endpoint address of the block we're creating a transport to.
+ * The source address in this value is not considered, only the
+ * destination address.
+ * \param xport_type Specify which kind of transport this is.
+ * \param args Additional arguments for the transport generation. See \ref page_transport
+ * for valid arguments.
+ */
+ virtual uhd::both_xports_t make_transport(
+ const uhd::sid_t &address,
+ const xport_type_t xport_type,
+ const uhd::device_addr_t& args
+ ) = 0;
+
+ virtual uhd::device_addr_t get_tx_hints(size_t) { return uhd::device_addr_t(); };
+ virtual uhd::device_addr_t get_rx_hints(size_t) { return uhd::device_addr_t(); };
+ virtual uhd::endianness_t get_transport_endianness(size_t mb_index) = 0;
+
+ //! Is called after a streamer is generated
+ virtual void post_streamer_hooks(uhd::direction_t) {};
+
+ /***********************************************************************
+ * Channel-related
+ **********************************************************************/
+ /*! Merge a list of channels into the existing channel definition.
+ *
+ * Intelligently merge the channels described in \p chan_ids
+ * into the current channel definition. If none of the channels in
+ * \p chan_ids is in the current definition, they simply get appended.
+ * Otherwise, they get overwritten in the order of \p chan_ids.
+ *
+ * \param chan_ids List of block IDs for the channels.
+ * \param chan_args New channel args. Must have same length as chan_ids.
+ *
+ */
+ void merge_channel_defs(
+ const std::vector<rfnoc::block_id_t> &chan_ids,
+ const std::vector<uhd::device_addr_t> &chan_args,
+ const uhd::direction_t dir
+ );
+
+ /***********************************************************************
+ * RFNoC-Specific
+ **********************************************************************/
+ void enumerate_rfnoc_blocks(
+ size_t device_index,
+ size_t n_blocks,
+ size_t base_port,
+ const uhd::sid_t &base_sid,
+ uhd::device_addr_t transport_args,
+ uhd::endianness_t endianness
+ );
+
+ /***********************************************************************
+ * Members
+ **********************************************************************/
+ //! A counter, designed to create unique SIDs
+ size_t _sid_framer;
+
+ // TODO: Maybe move these to private
+ uhd::dict<std::string, boost::weak_ptr<uhd::rx_streamer> > _rx_streamers;
+ uhd::dict<std::string, boost::weak_ptr<uhd::tx_streamer> > _tx_streamers;
+
+private:
+ /***********************************************************************
+ * Private Members
+ **********************************************************************/
+ //! Buffer for async metadata
+ boost::shared_ptr<async_md_type> _async_md;
+
+ //! This mutex locks the get_xx_stream() functions.
+ boost::mutex _transport_setup_mutex;
+};
+
+}} /* namespace uhd::usrp */
+
+#endif /* INCLUDED_DEVICE3_IMPL_HPP */
+// vim: sw=4 expandtab:
diff --git a/host/lib/usrp/device3/device3_io_impl.cpp b/host/lib/usrp/device3/device3_io_impl.cpp
new file mode 100644
index 000000000..8c61f8f15
--- /dev/null
+++ b/host/lib/usrp/device3/device3_io_impl.cpp
@@ -0,0 +1,851 @@
+//
+// Copyright 2014-2016 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+// Provides streaming-related functions which are used by device3 objects.
+
+#define DEVICE3_STREAMER // For the super_*_packet_handlers
+
+#include "device3_impl.hpp"
+#include <uhd/rfnoc/constants.hpp>
+#include <uhd/rfnoc/source_block_ctrl_base.hpp>
+#include <uhd/rfnoc/sink_block_ctrl_base.hpp>
+#include <uhd/utils/byteswap.hpp>
+#include <uhd/utils/log.hpp>
+#include <uhd/utils/msg.hpp>
+#include "../common/async_packet_handler.hpp"
+#include "../../transport/super_recv_packet_handler.hpp"
+#include "../../transport/super_send_packet_handler.hpp"
+#include "../../rfnoc/rx_stream_terminator.hpp"
+#include "../../rfnoc/tx_stream_terminator.hpp"
+#include <uhd/rfnoc/rate_node_ctrl.hpp>
+#include <uhd/rfnoc/radio_ctrl.hpp>
+
+#define UHD_STREAMER_LOG() UHD_LOGV(never)
+
+using namespace uhd;
+using namespace uhd::usrp;
+using namespace uhd::transport;
+
+//! CVITA uses 12-Bit sequence numbers
+static const boost::uint32_t HW_SEQ_NUM_MASK = 0xfff;
+
+
+/***********************************************************************
+ * Helper functions for get_?x_stream()
+ **********************************************************************/
+static uhd::stream_args_t sanitize_stream_args(const uhd::stream_args_t &args_)
+{
+ uhd::stream_args_t args = args_;
+ if (args.channels.empty()) {
+ args.channels = std::vector<size_t>(1, 0);
+ }
+
+ return args;
+}
+
+static void check_stream_sig_compatible(const rfnoc::stream_sig_t &stream_sig, stream_args_t &args, const std::string &tx_rx)
+{
+ if (args.otw_format.empty()) {
+ if (stream_sig.item_type.empty()) {
+ throw uhd::runtime_error(str(
+ boost::format("[%s Streamer] No otw_format defined!") % tx_rx
+ ));
+ } else {
+ args.otw_format = stream_sig.item_type;
+ }
+ } else if (not stream_sig.item_type.empty() and stream_sig.item_type != args.otw_format) {
+ throw uhd::runtime_error(str(
+ boost::format("[%s Streamer] Conflicting OTW types defined: args.otw_format = '%s' <=> stream_sig.item_type = '%s'")
+ % tx_rx % args.otw_format % stream_sig.item_type
+ ));
+ }
+ const size_t bpi = convert::get_bytes_per_item(args.otw_format); // bytes per item
+ if (stream_sig.packet_size) {
+ if (args.args.has_key("spp")) {
+ size_t args_spp = args.args.cast<size_t>("spp", 0);
+ if (args_spp * bpi != stream_sig.packet_size) {
+ throw uhd::runtime_error(str(
+ boost::format("[%s Streamer] Conflicting packet sizes defined: args yields %d bytes but stream_sig.packet_size is %d bytes")
+ % tx_rx % (args_spp * bpi) % stream_sig.packet_size
+ ));
+ }
+ } else {
+ args.args["spp"] = str(boost::format("%d") % (stream_sig.packet_size / bpi));
+ }
+ }
+}
+
+/*! \brief Returns a list of rx or tx channels for a streamer.
+ *
+ * If the given stream args contain instructions to set up channels,
+ * those are used. Otherwise, the current device's channel definition
+ * is consulted.
+ *
+ * \param args_ Stream args.
+ * \param[out] chan_list The list of channels in the correct order.
+ * \param[out] chan_args Channel args for every channel. `chan_args.size() == chan_list.size()`
+ */
+void generate_channel_list(
+ const uhd::stream_args_t &args_,
+ std::vector<uhd::rfnoc::block_id_t> &chan_list,
+ std::vector<device_addr_t> &chan_args
+) {
+ uhd::stream_args_t args = args_;
+ BOOST_FOREACH(const size_t chan_idx, args.channels) {
+ //// Find block ID for this channel:
+ if (args.args.has_key(str(boost::format("block_id%d") % chan_idx))) {
+ chan_list.push_back(
+ uhd::rfnoc::block_id_t(
+ args.args.pop(str(boost::format("block_id%d") % chan_idx))
+ )
+ );
+ chan_args.push_back(args.args);
+ } else if (args.args.has_key("block_id")) {
+ chan_list.push_back(args.args.get("block_id"));
+ chan_args.push_back(args.args);
+ chan_args.back().pop("block_id");
+ } else {
+ throw uhd::runtime_error(str(
+ boost::format("Cannot create streamers: No block_id specified for channel %d.")
+ % chan_idx
+ ));
+ }
+ //// Find block port for this channel
+ if (args.args.has_key(str(boost::format("block_port%d") % chan_idx))) {
+ chan_args.back()["block_port"] = args.args.pop(str(boost::format("block_port%d") % chan_idx));
+ } else if (args.args.has_key("block_port")) {
+ // We have to write it again, because the chan args from the
+ // property tree might have overwritten this
+ chan_args.back()["block_port"] = args.args.get("block_port");
+ }
+ }
+}
+
+
+/***********************************************************************
+ * RX Flow Control Functions
+ **********************************************************************/
+//! Stores the state of RX flow control
+struct rx_fc_cache_t
+{
+ rx_fc_cache_t():
+ last_seq_in(0){}
+ size_t last_seq_in;
+};
+
+/*! Determine the size of the flow control window in number of packets.
+ *
+ * This value depends on three things:
+ * - The packet size (in bytes), P
+ * - The size of the software buffer (in bytes), B
+ * - The desired buffer fullness, F
+ *
+ * The FC window size is thus X = floor(B*F/P).
+ *
+ * \param pkt_size The maximum packet size in bytes
+ * \param sw_buff_size Software buffer size in bytes
+ * \param rx_args If this has a key 'recv_buff_fullness', this value will
+ * be used for said fullness. Must be between 0.01 and 1.
+ *
+ * \returns The size of the flow control window in number of packets
+ */
+static size_t get_rx_flow_control_window(
+ size_t pkt_size,
+ size_t sw_buff_size,
+ const device_addr_t& rx_args
+) {
+ double fullness_factor = rx_args.cast<double>(
+ "recv_buff_fullness",
+ uhd::rfnoc::DEFAULT_FC_RX_SW_BUFF_FULL_FACTOR
+ );
+
+ if (fullness_factor < 0.01 || fullness_factor > 1) {
+ throw uhd::value_error("recv_buff_fullness must be in [0.01, 1] inclusive (1% to 100%)");
+ }
+
+ size_t window_in_pkts = (static_cast<size_t>(sw_buff_size * fullness_factor) / pkt_size);
+ if (rx_args.has_key("max_recv_window")) {
+ window_in_pkts = std::min(
+ window_in_pkts,
+ rx_args.cast<size_t>("max_recv_window", window_in_pkts)
+ );
+ }
+ if (window_in_pkts == 0) {
+ throw uhd::value_error("recv_buff_size must be larger than the recv_frame_size.");
+ }
+ UHD_ASSERT_THROW(size_t(sw_buff_size * fullness_factor) >= pkt_size * window_in_pkts);
+ return window_in_pkts;
+}
+
+
+/*! Send out RX flow control packets.
+ *
+ * For an rx stream, this function takes care of sending back
+ * a flow control packet to the source telling it which
+ * packets have been consumed.
+ *
+ * This function should only be called by the function handling
+ * the rx stream, usually recv() in super_recv_packet_handler.
+ *
+ * \param sid The SID that goes into this packet. This is the reversed()
+ * version of the data stream's SID.
+ * \param xport A transport object over which to send the data
+ * \param big_endian Endianness of the transport
+ * \param seq32_state Pointer to a variable that saves the 32-Bit state
+ * of the sequence numbers, since we only have 12 Bit
+ * sequence numbers in CHDR.
+ * \param last_seq The value to send: The last consumed packet's sequence number.
+ */
+static void handle_rx_flowctrl(
+ const sid_t &sid,
+ zero_copy_if::sptr xport,
+ endianness_t endianness,
+ boost::shared_ptr<rx_fc_cache_t> fc_cache,
+ const size_t last_seq
+) {
+ static const size_t RXFC_PACKET_LEN_IN_WORDS = 2;
+ static const size_t RXFC_CMD_CODE_OFFSET = 0;
+ static const size_t RXFC_SEQ_NUM_OFFSET = 1;
+
+ managed_send_buffer::sptr buff = xport->get_send_buff(0.0);
+ if (not buff) {
+ throw uhd::runtime_error("handle_rx_flowctrl timed out getting a send buffer");
+ }
+ boost::uint32_t *pkt = buff->cast<boost::uint32_t *>();
+
+ // Recover sequence number. The sequence numbers handled by the streamers
+ // are 12 Bits, but we want to know the 32-Bit sequence number.
+ size_t &seq32 = fc_cache->last_seq_in;
+ const size_t seq12 = seq32 & HW_SEQ_NUM_MASK;
+ if (last_seq < seq12)
+ seq32 += (HW_SEQ_NUM_MASK + 1);
+ seq32 &= ~HW_SEQ_NUM_MASK;
+ seq32 |= last_seq;
+
+ // Super-verbose mode:
+ //static size_t fc_pkt_count = 0;
+ //UHD_MSG(status) << "sending flow ctrl packet " << fc_pkt_count++ << ", acking " << str(boost::format("%04d\tseq_sw==0x%08x") % last_seq % seq32) << std::endl;
+
+ //load packet info
+ vrt::if_packet_info_t packet_info;
+ packet_info.packet_type = vrt::if_packet_info_t::PACKET_TYPE_FC;
+ packet_info.num_payload_words32 = RXFC_PACKET_LEN_IN_WORDS;
+ packet_info.num_payload_bytes = packet_info.num_payload_words32*sizeof(boost::uint32_t);
+ packet_info.packet_count = seq32;
+ packet_info.sob = false;
+ packet_info.eob = false;
+ packet_info.sid = sid.get();
+ packet_info.has_sid = true;
+ packet_info.has_cid = false;
+ packet_info.has_tsi = false;
+ packet_info.has_tsf = false;
+ packet_info.has_tlr = false;
+
+ if (endianness == ENDIANNESS_BIG) {
+ // Load Header:
+ vrt::chdr::if_hdr_pack_be(pkt, packet_info);
+ // Load Payload: (the sequence number)
+ pkt[packet_info.num_header_words32+RXFC_CMD_CODE_OFFSET] = uhd::htonx<boost::uint32_t>(0);
+ pkt[packet_info.num_header_words32+RXFC_SEQ_NUM_OFFSET] = uhd::htonx<boost::uint32_t>(seq32);
+ } else {
+ // Load Header:
+ vrt::chdr::if_hdr_pack_le(pkt, packet_info);
+ // Load Payload: (the sequence number)
+ pkt[packet_info.num_header_words32+RXFC_CMD_CODE_OFFSET] = uhd::htowx<boost::uint32_t>(0);
+ pkt[packet_info.num_header_words32+RXFC_SEQ_NUM_OFFSET] = uhd::htowx<boost::uint32_t>(seq32);
+ }
+
+ //std::cout << " SID=" << std::hex << sid << " hdr bits=" << packet_info.packet_type << " seq32=" << seq32 << std::endl;
+ //std::cout << "num_packet_words32: " << packet_info.num_packet_words32 << std::endl;
+ //for (size_t i = 0; i < packet_info.num_packet_words32; i++) {
+ //std::cout << str(boost::format("0x%08x") % pkt[i]) << " ";
+ //if (i % 2) {
+ //std::cout << std::endl;
+ //}
+ //}
+
+ //send the buffer over the interface
+ buff->commit(sizeof(boost::uint32_t)*(packet_info.num_packet_words32));
+}
+
+/***********************************************************************
+ * TX Flow Control Functions
+ **********************************************************************/
+//! Stores the state of TX flow control
+struct tx_fc_cache_t
+{
+ tx_fc_cache_t(void):
+ stream_channel(0),
+ device_channel(0),
+ last_seq_out(0),
+ last_seq_ack(0),
+ seq_queue(1){}
+ size_t stream_channel;
+ size_t device_channel;
+ size_t last_seq_out;
+ size_t last_seq_ack;
+ uhd::transport::bounded_buffer<size_t> seq_queue;
+ boost::shared_ptr<device3_impl::async_md_type> async_queue;
+ boost::shared_ptr<device3_impl::async_md_type> old_async_queue;
+};
+
+/*! Return the size of the flow control window in packets.
+ *
+ * If the return value of this function is F, the last tx'd packet
+ * has index N and the last ack'd packet has index M, the amount of
+ * FC credit we have is C = F + M - N (i.e. we can send C more packets
+ * before getting another ack).
+ *
+ * Note: If `send_buff_size` is set in \p tx_hints, this will
+ * override hw_buff_size_.
+ */
+static size_t get_tx_flow_control_window(
+ size_t pkt_size,
+ const double hw_buff_size_,
+ const device_addr_t& tx_hints
+) {
+ double hw_buff_size = tx_hints.cast<double>("send_buff_size", hw_buff_size_);
+ size_t window_in_pkts = (static_cast<size_t>(hw_buff_size) / pkt_size);
+ if (window_in_pkts == 0) {
+ throw uhd::value_error("send_buff_size must be larger than the send_frame_size.");
+ }
+ return window_in_pkts;
+}
+
+static managed_send_buffer::sptr get_tx_buff_with_flowctrl(
+ task::sptr /*holds ref*/,
+ boost::shared_ptr<tx_fc_cache_t> fc_cache,
+ zero_copy_if::sptr xport,
+ size_t fc_window,
+ const double timeout
+){
+ while (true)
+ {
+ // delta is the amount of FC credit we've used up
+ const size_t delta = (fc_cache->last_seq_out & HW_SEQ_NUM_MASK) - (fc_cache->last_seq_ack & HW_SEQ_NUM_MASK);
+ // If we want to send another packet, we must have FC credit left
+ if ((delta & HW_SEQ_NUM_MASK) < fc_window)
+ break;
+
+ // If credit is all used up, we check seq_queue for more.
+ const bool ok = fc_cache->seq_queue.pop_with_timed_wait(fc_cache->last_seq_ack, timeout);
+ if (not ok) {
+ return managed_send_buffer::sptr(); //timeout waiting for flow control
+ }
+ }
+
+ managed_send_buffer::sptr buff = xport->get_send_buff(timeout);
+ if (buff) {
+ fc_cache->last_seq_out++; //update seq, this will actually be a send
+ }
+ return buff;
+}
+
+#define DEVICE3_ASYNC_EVENT_CODE_FLOW_CTRL 0
+/*! Handle incoming messages. If they're flow control, update the TX FC cache.
+ * Otherwise, send them to the async message queue for the user to poll.
+ *
+ * This is run inside a uhd::task as long as this streamer lives.
+ */
+static void handle_tx_async_msgs(
+ boost::shared_ptr<tx_fc_cache_t> fc_cache,
+ zero_copy_if::sptr xport,
+ endianness_t endianness,
+ boost::function<double(void)> get_tick_rate
+) {
+ managed_recv_buffer::sptr buff = xport->get_recv_buff();
+ if (not buff)
+ return;
+
+ //extract packet info
+ vrt::if_packet_info_t if_packet_info;
+ if_packet_info.num_packet_words32 = buff->size()/sizeof(boost::uint32_t);
+ const boost::uint32_t *packet_buff = buff->cast<const boost::uint32_t *>();
+
+ //unpacking can fail
+ boost::uint32_t (*endian_conv)(boost::uint32_t) = uhd::ntohx;
+ try
+ {
+ if (endianness == ENDIANNESS_BIG)
+ {
+ vrt::chdr::if_hdr_unpack_be(packet_buff, if_packet_info);
+ endian_conv = uhd::ntohx;
+ }
+ else
+ {
+ vrt::chdr::if_hdr_unpack_le(packet_buff, if_packet_info);
+ endian_conv = uhd::wtohx;
+ }
+ }
+ catch(const std::exception &ex)
+ {
+ UHD_MSG(error) << "Error parsing async message packet: " << ex.what() << std::endl;
+ return;
+ }
+
+ double tick_rate = get_tick_rate();
+ if (tick_rate == rfnoc::tick_node_ctrl::RATE_UNDEFINED) {
+ tick_rate = 1;
+ }
+
+ //fill in the async metadata
+ async_metadata_t metadata;
+ load_metadata_from_buff(
+ endian_conv,
+ metadata,
+ if_packet_info,
+ packet_buff,
+ tick_rate,
+ fc_cache->stream_channel
+ );
+
+ // TODO: Shouldn't we be polling if_packet_info.packet_type == PACKET_TYPE_FC?
+ // Thing is, on X300, packet_type == 0, so that wouldn't work. But it seems it should.
+ //The FC response and the burst ack are two indicators that the radio
+ //consumed packets. Use them to update the FC metadata
+ if (metadata.event_code == DEVICE3_ASYNC_EVENT_CODE_FLOW_CTRL) {
+ const size_t seq = metadata.user_payload[0];
+ fc_cache->seq_queue.push_with_pop_on_full(seq);
+ }
+
+ //FC responses don't propagate up to the user so filter them here
+ if (metadata.event_code != DEVICE3_ASYNC_EVENT_CODE_FLOW_CTRL) {
+ fc_cache->async_queue->push_with_pop_on_full(metadata);
+ metadata.channel = fc_cache->device_channel;
+ fc_cache->old_async_queue->push_with_pop_on_full(metadata);
+ standard_async_msg_prints(metadata);
+ }
+}
+
+
+
+/***********************************************************************
+ * Async Data
+ **********************************************************************/
+bool device3_impl::recv_async_msg(
+ async_metadata_t &async_metadata, double timeout
+)
+{
+ return _async_md->pop_with_timed_wait(async_metadata, timeout);
+}
+
+/***********************************************************************
+ * Receive streamer
+ **********************************************************************/
+void device3_impl::update_rx_streamers(double /* rate */)
+{
+ BOOST_FOREACH(const std::string &block_id, _rx_streamers.keys()) {
+ UHD_STREAMER_LOG() << "[Device3] updating RX streamer to " << block_id << std::endl;
+ boost::shared_ptr<sph::recv_packet_streamer> my_streamer =
+ boost::dynamic_pointer_cast<sph::recv_packet_streamer>(_rx_streamers[block_id].lock());
+ if (my_streamer) {
+ double tick_rate = my_streamer->get_terminator()->get_tick_rate();
+ if (tick_rate == rfnoc::tick_node_ctrl::RATE_UNDEFINED) {
+ tick_rate = 1.0;
+ }
+ my_streamer->set_tick_rate(tick_rate);
+ double samp_rate = my_streamer->get_terminator()->get_output_samp_rate();
+ if (samp_rate == rfnoc::rate_node_ctrl::RATE_UNDEFINED) {
+ samp_rate = 1.0;
+ }
+ // This formula is not derived by any scientific means -- we just need to
+ // increase the failure threshold as we increase rates. For 1 Msps, we use
+ // the default.
+ const size_t alignment_failure_factor = std::max(size_t(1), size_t(samp_rate * 1000 / tick_rate));
+ double scaling = my_streamer->get_terminator()->get_output_scale_factor();
+ if (scaling == rfnoc::scalar_node_ctrl::SCALE_UNDEFINED) {
+ scaling = 1/32767.;
+ }
+ UHD_STREAMER_LOG() << " New tick_rate == " << tick_rate << " New samp_rate == " << samp_rate << " New scaling == " << scaling << std::endl;
+
+ my_streamer->set_tick_rate(tick_rate);
+ my_streamer->set_samp_rate(samp_rate);
+ // 1000 packets is the default alignment failure threshold
+ my_streamer->set_alignment_failure_threshold(1000 * alignment_failure_factor);
+ my_streamer->set_scale_factor(scaling);
+ }
+ }
+}
+
+rx_streamer::sptr device3_impl::get_rx_stream(const stream_args_t &args_)
+{
+ boost::mutex::scoped_lock lock(_transport_setup_mutex);
+ stream_args_t args = sanitize_stream_args(args_);
+
+ // I. Generate the channel list
+ std::vector<uhd::rfnoc::block_id_t> chan_list;
+ std::vector<device_addr_t> chan_args;
+ generate_channel_list(args, chan_list, chan_args);
+ // Note: All 'args.args' are merged into chan_args now.
+
+ // II. Iterate over all channels
+ boost::shared_ptr<sph::recv_packet_streamer> my_streamer;
+ // The terminator's lifetime is coupled to the streamer.
+ // There is only one terminator. If the streamer has multiple channels,
+ // it will be connected to each upstream block.
+ rfnoc::rx_stream_terminator::sptr recv_terminator = rfnoc::rx_stream_terminator::make();
+ for (size_t stream_i = 0; stream_i < chan_list.size(); stream_i++) {
+ // Get block ID and mb index
+ uhd::rfnoc::block_id_t block_id = chan_list[stream_i];
+ UHD_STREAMER_LOG() << "[RX Streamer] chan " << stream_i << " connecting to " << block_id << std::endl;
+ // Update args so args.args is always valid for this particular channel:
+ args.args = chan_args[stream_i];
+ size_t mb_index = block_id.get_device_no();
+ size_t suggested_block_port = args.args.cast<size_t>("block_port", rfnoc::ANY_PORT);
+
+ // Access to this channel's block control
+ uhd::rfnoc::source_block_ctrl_base::sptr blk_ctrl =
+ boost::dynamic_pointer_cast<uhd::rfnoc::source_block_ctrl_base>(get_block_ctrl(block_id));
+
+ // Connect the terminator with this channel's block.
+ size_t block_port = blk_ctrl->connect_downstream(
+ recv_terminator,
+ suggested_block_port,
+ args.args
+ );
+ const size_t terminator_port = recv_terminator->connect_upstream(blk_ctrl);
+ blk_ctrl->set_downstream_port(block_port, terminator_port);
+ recv_terminator->set_upstream_port(terminator_port, block_port);
+
+ // Check if the block connection is compatible (spp and item type)
+ check_stream_sig_compatible(blk_ctrl->get_output_signature(block_port), args, "RX");
+
+ // Setup the DSP transport hints
+ device_addr_t rx_hints = get_rx_hints(mb_index);
+
+ //allocate sid and create transport
+ uhd::sid_t stream_address = blk_ctrl->get_address(block_port);
+ UHD_STREAMER_LOG() << "[RX Streamer] creating rx stream " << rx_hints.to_string() << std::endl;
+ both_xports_t xport = make_transport(stream_address, RX_DATA, rx_hints);
+ UHD_STREAMER_LOG() << std::hex << "[RX Streamer] data_sid = " << xport.send_sid << std::dec << " actual recv_buff_size = " << xport.recv_buff_size << std::endl;
+
+ // Configure the block
+ blk_ctrl->set_destination(xport.send_sid.get_src(), block_port);
+
+ blk_ctrl->sr_write(uhd::rfnoc::SR_RESP_OUT_DST_SID, xport.send_sid.get_src(), block_port);
+ UHD_STREAMER_LOG() << "[RX Streamer] resp_out_dst_sid == " << xport.send_sid.get_src() << std::endl;
+
+ // Find all upstream radio nodes and set their response in SID to the host
+ std::vector<boost::shared_ptr<uhd::rfnoc::radio_ctrl> > upstream_radio_nodes = blk_ctrl->find_upstream_node<uhd::rfnoc::radio_ctrl>();
+ UHD_STREAMER_LOG() << "[RX Streamer] Number of upstream radio nodes: " << upstream_radio_nodes.size() << std::endl;
+ BOOST_FOREACH(const boost::shared_ptr<uhd::rfnoc::radio_ctrl> &node, upstream_radio_nodes) {
+ node->sr_write(uhd::rfnoc::SR_RESP_OUT_DST_SID, xport.send_sid.get_src(), block_port);
+ }
+
+ // To calculate the max number of samples per packet, we assume the maximum header length
+ // to avoid fragmentation should the entire header be used.
+ const size_t bpp = xport.recv->get_recv_frame_size() - stream_options.rx_max_len_hdr; // bytes per packet
+ const size_t bpi = convert::get_bytes_per_item(args.otw_format); // bytes per item
+ const size_t spp = std::min(args.args.cast<size_t>("spp", bpp/bpi), bpp/bpi); // samples per packet
+ UHD_STREAMER_LOG() << "[RX Streamer] spp == " << spp << std::endl;
+
+ //make the new streamer given the samples per packet
+ if (not my_streamer)
+ my_streamer = boost::make_shared<sph::recv_packet_streamer>(spp);
+ my_streamer->resize(chan_list.size());
+
+ //init some streamer stuff
+ std::string conv_endianness;
+ if (get_transport_endianness(mb_index) == ENDIANNESS_BIG) {
+ my_streamer->set_vrt_unpacker(&vrt::chdr::if_hdr_unpack_be);
+ conv_endianness = "be";
+ } else {
+ my_streamer->set_vrt_unpacker(&vrt::chdr::if_hdr_unpack_le);
+ conv_endianness = "le";
+ }
+
+ //set the converter
+ uhd::convert::id_type id;
+ id.input_format = args.otw_format + "_item32_" + conv_endianness;
+ id.num_inputs = 1;
+ id.output_format = args.cpu_format;
+ id.num_outputs = 1;
+ my_streamer->set_converter(id);
+
+ //flow control setup
+ const size_t pkt_size = spp * bpi + stream_options.rx_max_len_hdr;
+ const size_t fc_window = get_rx_flow_control_window(pkt_size, xport.recv_buff_size, rx_hints);
+ const size_t fc_handle_window = std::max<size_t>(1, fc_window / stream_options.rx_fc_request_freq);
+ UHD_STREAMER_LOG()<< "[RX Streamer] Flow Control Window (minus one) = " << fc_window-1 << ", Flow Control Handler Window = " << fc_handle_window << std::endl;
+ blk_ctrl->configure_flow_control_out(
+ fc_window-1, // Leave one space for overrun packets TODO make this obsolete
+ block_port
+ );
+
+ //Give the streamer a functor to get the recv_buffer
+ //bind requires a zero_copy_if::sptr to add a streamer->xport lifetime dependency
+ my_streamer->set_xport_chan_get_buff(
+ stream_i,
+ boost::bind(&zero_copy_if::get_recv_buff, xport.recv, _1),
+ true /*flush*/
+ );
+
+ //Give the streamer a functor to handle overruns
+ //bind requires a weak_ptr to break the a streamer->streamer circular dependency
+ //Using "this" is OK because we know that this device3_impl will outlive the streamer
+ my_streamer->set_overflow_handler(
+ stream_i,
+ boost::bind(
+ &uhd::rfnoc::rx_stream_terminator::handle_overrun, recv_terminator,
+ boost::weak_ptr<uhd::rx_streamer>(my_streamer), stream_i
+ )
+ );
+
+ //Give the streamer a functor to send flow control messages
+ //handle_rx_flowctrl is static and has no lifetime issues
+ boost::shared_ptr<rx_fc_cache_t> fc_cache(new rx_fc_cache_t());
+ my_streamer->set_xport_handle_flowctrl(
+ stream_i, boost::bind(
+ &handle_rx_flowctrl,
+ xport.send_sid,
+ xport.send,
+ get_transport_endianness(mb_index),
+ fc_cache,
+ _1
+ ),
+ fc_handle_window,
+ true/*init*/
+ );
+
+ //Give the streamer a functor issue stream cmd
+ //bind requires a shared pointer to add a streamer->framer lifetime dependency
+ my_streamer->set_issue_stream_cmd(
+ stream_i,
+ boost::bind(&uhd::rfnoc::source_block_ctrl_base::issue_stream_cmd, blk_ctrl, _1, block_port)
+ );
+
+ // Tell the streamer which SID is valid for this channel
+ my_streamer->set_xport_chan_sid(stream_i, true, xport.send_sid);
+ }
+
+ // Connect the terminator to the streamer
+ my_streamer->set_terminator(recv_terminator);
+
+ // Notify all blocks in this chain that they are connected to an active streamer
+ recv_terminator->set_rx_streamer(true, 0);
+
+ // Store a weak pointer to prevent a streamer->device3_impl->streamer circular dependency.
+ // Note that we store the streamer only once, and use its terminator's
+ // ID to do so.
+ _rx_streamers[recv_terminator->unique_id()] = boost::weak_ptr<sph::recv_packet_streamer>(my_streamer);
+
+ // Sets tick rate, samp rate and scaling on this streamer.
+ // A registered terminator is required to do this.
+ update_rx_streamers();
+
+ post_streamer_hooks(RX_DIRECTION);
+ return my_streamer;
+}
+
+/***********************************************************************
+ * Transmit streamer
+ **********************************************************************/
+void device3_impl::update_tx_streamers(double /* rate */)
+{
+ BOOST_FOREACH(const std::string &block_id, _tx_streamers.keys()) {
+ UHD_STREAMER_LOG() << "[Device3] updating TX streamer: " << block_id << std::endl;
+ boost::shared_ptr<sph::send_packet_streamer> my_streamer =
+ boost::dynamic_pointer_cast<sph::send_packet_streamer>(_tx_streamers[block_id].lock());
+ if (my_streamer) {
+ double tick_rate = my_streamer->get_terminator()->get_tick_rate();
+ if (tick_rate == rfnoc::tick_node_ctrl::RATE_UNDEFINED) {
+ tick_rate = 1.0;
+ }
+ double samp_rate = my_streamer->get_terminator()->get_input_samp_rate();
+ if (samp_rate == rfnoc::rate_node_ctrl::RATE_UNDEFINED) {
+ samp_rate = 1.0;
+ }
+ double scaling = my_streamer->get_terminator()->get_input_scale_factor();
+ if (scaling == rfnoc::scalar_node_ctrl::SCALE_UNDEFINED) {
+ scaling = 32767.;
+ }
+ UHD_STREAMER_LOG() << " New tick_rate == " << tick_rate << " New samp_rate == " << samp_rate << " New scaling == " << scaling << std::endl;
+ my_streamer->set_tick_rate(tick_rate);
+ my_streamer->set_samp_rate(samp_rate);
+ my_streamer->set_scale_factor(scaling);
+ }
+ }
+}
+
+tx_streamer::sptr device3_impl::get_tx_stream(const uhd::stream_args_t &args_)
+{
+ boost::mutex::scoped_lock lock(_transport_setup_mutex);
+ stream_args_t args = sanitize_stream_args(args_);
+
+ // I. Generate the channel list
+ std::vector<uhd::rfnoc::block_id_t> chan_list;
+ std::vector<device_addr_t> chan_args;
+ generate_channel_list(args, chan_list, chan_args);
+ // Note: All 'args.args' are merged into chan_args now.
+
+ //shared async queue for all channels in streamer
+ boost::shared_ptr<async_md_type> async_md(new async_md_type(1000/*messages deep*/));
+
+ // II. Iterate over all channels
+ boost::shared_ptr<sph::send_packet_streamer> my_streamer;
+ // The terminator's lifetime is coupled to the streamer.
+ // There is only one terminator. If the streamer has multiple channels,
+ // it will be connected to each downstream block.
+ rfnoc::tx_stream_terminator::sptr send_terminator = rfnoc::tx_stream_terminator::make();
+ for (size_t stream_i = 0; stream_i < chan_list.size(); stream_i++) {
+ // Get block ID and mb index
+ uhd::rfnoc::block_id_t block_id = chan_list[stream_i];
+ // Update args so args.args is always valid for this particular channel:
+ args.args = chan_args[stream_i];
+ size_t mb_index = block_id.get_device_no();
+ size_t suggested_block_port = args.args.cast<size_t>("block_port", rfnoc::ANY_PORT);
+
+ // Access to this channel's block control
+ uhd::rfnoc::sink_block_ctrl_base::sptr blk_ctrl =
+ boost::dynamic_pointer_cast<uhd::rfnoc::sink_block_ctrl_base>(get_block_ctrl(block_id));
+
+ // Connect the terminator with this channel's block.
+ // This will throw if the connection is not possible.
+ size_t block_port = blk_ctrl->connect_upstream(
+ send_terminator,
+ suggested_block_port,
+ args.args
+ );
+ const size_t terminator_port = send_terminator->connect_downstream(blk_ctrl);
+ blk_ctrl->set_upstream_port(block_port, terminator_port);
+ send_terminator->set_downstream_port(terminator_port, block_port);
+
+ // Check if the block connection is compatible (spp and item type)
+ check_stream_sig_compatible(blk_ctrl->get_input_signature(block_port), args, "TX");
+
+ // Setup the dsp transport hints
+ device_addr_t tx_hints = get_tx_hints(mb_index);
+
+ //allocate sid and create transport
+ uhd::sid_t stream_address = blk_ctrl->get_address(block_port);
+ UHD_STREAMER_LOG() << "[TX Streamer] creating tx stream " << tx_hints.to_string() << std::endl;
+ both_xports_t xport = make_transport(stream_address, TX_DATA, tx_hints);
+ UHD_STREAMER_LOG() << std::hex << "[TX Streamer] data_sid = " << xport.send_sid << std::dec << std::endl;
+
+ // To calculate the max number of samples per packet, we assume the maximum header length
+ // to avoid fragmentation should the entire header be used.
+ const size_t bpp = tx_hints.cast<size_t>("bpp", xport.send->get_send_frame_size()) - stream_options.tx_max_len_hdr;
+ const size_t bpi = convert::get_bytes_per_item(args.otw_format); // bytes per item
+ const size_t spp = std::min(args.args.cast<size_t>("spp", bpp/bpi), bpp/bpi); // samples per packet
+ UHD_STREAMER_LOG() << "[TX Streamer] spp == " << spp << std::endl;
+
+ //make the new streamer given the samples per packet
+ if (not my_streamer)
+ my_streamer = boost::make_shared<sph::send_packet_streamer>(spp);
+ my_streamer->resize(chan_list.size());
+
+ //init some streamer stuff
+ std::string conv_endianness;
+ if (get_transport_endianness(mb_index) == ENDIANNESS_BIG) {
+ my_streamer->set_vrt_packer(&vrt::chdr::if_hdr_pack_be);
+ conv_endianness = "be";
+ } else {
+ my_streamer->set_vrt_packer(&vrt::chdr::if_hdr_pack_le);
+ conv_endianness = "le";
+ }
+
+ //set the converter
+ uhd::convert::id_type id;
+ id.input_format = args.cpu_format;
+ id.num_inputs = 1;
+ id.output_format = args.otw_format + "_item32_" + conv_endianness;
+ id.num_outputs = 1;
+ my_streamer->set_converter(id);
+
+ //flow control setup
+ const size_t pkt_size = spp * bpi + stream_options.tx_max_len_hdr;
+ // For flow control, this value is used to determine the window size in *packets*
+ size_t fc_window = get_tx_flow_control_window(
+ pkt_size, // This is the maximum packet size
+ blk_ctrl->get_fifo_size(block_port),
+ tx_hints // This can override the value reported by the block!
+ );
+ const size_t fc_handle_window = std::max<size_t>(1, fc_window / stream_options.tx_fc_response_freq);
+ UHD_STREAMER_LOG() << "[TX Streamer] Flow Control Window = " << fc_window << ", Flow Control Handler Window = " << fc_handle_window << std::endl;
+ blk_ctrl->configure_flow_control_in(
+ stream_options.tx_fc_response_cycles,
+ fc_handle_window, /*pkts*/
+ block_port
+ );
+
+ boost::shared_ptr<tx_fc_cache_t> fc_cache(new tx_fc_cache_t());
+ fc_cache->stream_channel = stream_i;
+ fc_cache->device_channel = mb_index;
+ fc_cache->async_queue = async_md;
+ fc_cache->old_async_queue = _async_md;
+
+ boost::function<double(void)> tick_rate_retriever = boost::bind(
+ &rfnoc::tick_node_ctrl::get_tick_rate,
+ send_terminator,
+ std::set< rfnoc::node_ctrl_base::sptr >() // Need to specify default args with bind
+ );
+ task::sptr task = task::make(
+ boost::bind(
+ &handle_tx_async_msgs,
+ fc_cache,
+ xport.recv,
+ get_transport_endianness(mb_index),
+ tick_rate_retriever
+ )
+ );
+
+ blk_ctrl->sr_write(uhd::rfnoc::SR_RESP_IN_DST_SID, xport.recv_sid.get_dst(), block_port);
+ UHD_STREAMER_LOG() << "[TX Streamer] resp_in_dst_sid == " << boost::format("0x%04X") % xport.recv_sid.get_dst() << std::endl;
+ // Find all downstream radio nodes and set their response in SID to the host
+ std::vector<boost::shared_ptr<uhd::rfnoc::radio_ctrl> > downstream_radio_nodes = blk_ctrl->find_downstream_node<uhd::rfnoc::radio_ctrl>();
+ UHD_STREAMER_LOG() << "[TX Streamer] Number of downstream radio nodes: " << downstream_radio_nodes.size() << std::endl;
+ BOOST_FOREACH(const boost::shared_ptr<uhd::rfnoc::radio_ctrl> &node, downstream_radio_nodes) {
+ node->sr_write(uhd::rfnoc::SR_RESP_IN_DST_SID, xport.send_sid.get_src(), block_port);
+ }
+
+ //Give the streamer a functor to get the send buffer
+ //get_tx_buff_with_flowctrl is static so bind has no lifetime issues
+ //xport.send (sptr) is required to add streamer->data-transport lifetime dependency
+ //task (sptr) is required to add a streamer->async-handler lifetime dependency
+ my_streamer->set_xport_chan_get_buff(
+ stream_i,
+ boost::bind(&get_tx_buff_with_flowctrl, task, fc_cache, xport.send, fc_window, _1)
+ );
+ //Give the streamer a functor handled received async messages
+ my_streamer->set_async_receiver(
+ boost::bind(&async_md_type::pop_with_timed_wait, async_md, _1, _2)
+ );
+ my_streamer->set_xport_chan_sid(stream_i, true, xport.send_sid);
+ // CHDR does not support trailers
+ my_streamer->set_enable_trailer(false);
+ }
+
+ // Connect the terminator to the streamer
+ my_streamer->set_terminator(send_terminator);
+
+ // Notify all blocks in this chain that they are connected to an active streamer
+ send_terminator->set_tx_streamer(true, 0);
+
+ // Store a weak pointer to prevent a streamer->device3_impl->streamer circular dependency.
+ // Note that we store the streamer only once, and use its terminator's
+ // ID to do so.
+ _tx_streamers[send_terminator->unique_id()] = boost::weak_ptr<sph::send_packet_streamer>(my_streamer);
+
+ // Sets tick rate, samp rate and scaling on this streamer
+ // A registered terminator is required to do this.
+ update_tx_streamers();
+
+ post_streamer_hooks(TX_DIRECTION);
+ return my_streamer;
+}
+
+