aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--host/lib/usrp/multi_usrp_rfnoc.cpp255
1 files changed, 142 insertions, 113 deletions
diff --git a/host/lib/usrp/multi_usrp_rfnoc.cpp b/host/lib/usrp/multi_usrp_rfnoc.cpp
index d7bc49639..b3851ba9e 100644
--- a/host/lib/usrp/multi_usrp_rfnoc.cpp
+++ b/host/lib/usrp/multi_usrp_rfnoc.cpp
@@ -8,10 +8,12 @@
#include <uhd/rfnoc/ddc_block_control.hpp>
#include <uhd/rfnoc/duc_block_control.hpp>
#include <uhd/rfnoc/filter_node.hpp>
+#include <uhd/rfnoc/graph_edge.hpp>
#include <uhd/rfnoc/radio_control.hpp>
#include <uhd/rfnoc_graph.hpp>
#include <uhd/types/device_addr.hpp>
#include <uhd/usrp/multi_usrp.hpp>
+#include <uhd/utils/graph_utils.hpp>
#include <uhdlib/rfnoc/rfnoc_device.hpp>
#include <uhdlib/usrp/gpio_defs.hpp>
#include <unordered_set>
@@ -76,6 +78,7 @@ public:
radio_control::sptr radio;
ddc_block_control::sptr ddc; // can be nullptr
size_t block_chan;
+ std::vector<graph_edge_t> edge_list;
};
struct tx_chan_t
@@ -83,6 +86,7 @@ public:
radio_control::sptr radio;
duc_block_control::sptr duc; // can be nullptr
size_t block_chan;
+ std::vector<graph_edge_t> edge_list;
};
/**************************************************************************
@@ -118,76 +122,39 @@ public:
size_t musrp_tx_channel = 0;
for (auto radio_id : radio_blk_ids) {
auto radio_blk = _graph->get_block<uhd::rfnoc::radio_control>(radio_id);
- // We assume that the DDC connected to this radio block has the same mboard,
- // instance, and port number
- auto ddc_id =
- block_id_t(radio_id.get_device_no(), "DDC", radio_id.get_block_count());
- uhd::rfnoc::ddc_block_control::sptr ddc_blk;
- try {
- ddc_blk = _graph->get_block<uhd::rfnoc::ddc_block_control>(ddc_id);
- } catch (const uhd::exception&) {
- UHD_LOGGER_TRACE("MULTI_USRP")
- << boost::format("No DDC found: %s") % ddc_id.to_string();
- }
for (size_t block_chan = 0; block_chan < radio_blk->get_num_output_ports();
++block_chan) {
- // Figure out if this channel has a DDC available
- auto this_chan_ddc =
- ddc_blk
- && _graph->is_connectable(
- radio_id, block_chan, ddc_id, block_chan)
- ? ddc_blk
- : nullptr;
- _rx_chans.emplace(
- musrp_rx_channel, rx_chan_t({radio_blk, this_chan_ddc, block_chan}));
- if (!this_chan_ddc) {
- UHD_LOGGER_DEBUG("MULTI_USRP")
- << boost::format(
- "Radio %s unable to connect to DDC %s on channel %d")
- % radio_id.to_string() % ddc_id.to_string() % block_chan;
- } else {
- UHD_LOG_DEBUG("MULTI_USRP",
- "RX Channel " << musrp_rx_channel << " has "
- << radio_id.to_string() << " and DDC "
- << ddc_id.to_string());
+ // Create the RX chan
+ uhd::usrp::subdev_spec_t rx_radio_subdev;
+ rx_radio_subdev.push_back(uhd::usrp::subdev_spec_pair_t(
+ radio_blk->get_slot_name(),
+ radio_blk->get_dboard_fe_from_chan(block_chan, uhd::RX_DIRECTION)));
+ auto rx_chans =
+ _generate_mboard_rx_chans(rx_radio_subdev, radio_id.get_device_no());
+ // TODO: we're passing the same info around here; there has to be a
+ // cleaner way
+ for (auto rx_chan : rx_chans) {
+ _rx_chans.emplace(musrp_rx_channel, rx_chan);
+ ++musrp_rx_channel; // Increment after logging so we print the correct
+ // value
}
- ++musrp_rx_channel; // Increment after logging so we print the correct
- // value
- }
- // We assume that the DUC connected to this radio block has the same mboard,
- // instance, and port number
- auto duc_id =
- block_id_t(radio_id.get_device_no(), "DUC", radio_id.get_block_count());
- uhd::rfnoc::duc_block_control::sptr duc_blk;
- try {
- duc_blk = _graph->get_block<uhd::rfnoc::duc_block_control>(duc_id);
- } catch (const uhd::exception&) {
- UHD_LOGGER_TRACE("MULTI_USRP")
- << boost::format("No DUC found: %s") % duc_id.to_string();
}
for (size_t block_chan = 0; block_chan < radio_blk->get_num_input_ports();
++block_chan) {
- auto this_chan_duc =
- duc_blk
- && _graph->is_connectable(
- duc_id, block_chan, radio_id, block_chan)
- ? duc_blk
- : nullptr;
- _tx_chans.emplace(
- musrp_tx_channel, tx_chan_t({radio_blk, this_chan_duc, block_chan}));
- if (!this_chan_duc) {
- UHD_LOGGER_DEBUG("MULTI_USRP")
- << boost::format(
- "Radio %s unable to connect to DUC %s on channel %d")
- % radio_id.to_string() % duc_id.to_string() % block_chan;
- } else {
- UHD_LOG_DEBUG("MULTI_USRP",
- "TX Channel " << musrp_tx_channel << " has "
- << radio_id.to_string() << " and DUC "
- << duc_id.to_string());
+ // Create the TX chan
+ uhd::usrp::subdev_spec_t tx_radio_subdev;
+ tx_radio_subdev.push_back(uhd::usrp::subdev_spec_pair_t(
+ radio_blk->get_slot_name(),
+ radio_blk->get_dboard_fe_from_chan(block_chan, uhd::TX_DIRECTION)));
+ auto tx_chans =
+ _generate_mboard_tx_chans(tx_radio_subdev, radio_id.get_device_no());
+ // TODO: we're passing the same info around here; there has to be a
+ // cleaner way
+ for (auto tx_chan : tx_chans) {
+ _tx_chans.emplace(musrp_tx_channel, tx_chan);
+ ++musrp_tx_channel; // Increment after logging so we print the correct
+ // value
}
- ++musrp_tx_channel; // Increment after logging so we print the correct
- // value
}
}
_graph->commit();
@@ -218,15 +185,24 @@ public:
for (size_t strm_port = 0; strm_port < args.channels.size(); ++strm_port) {
auto rx_channel = args.channels.at(strm_port);
auto rx_chain = _get_rx_chan(rx_channel);
- if (rx_chain.ddc) {
- _graph->connect(rx_chain.radio->get_block_id(),
- rx_chain.block_chan,
- rx_chain.ddc->get_block_id(),
- rx_chain.block_chan);
+ // Make all of the connections in our chain
+ for (auto edge : rx_chain.edge_list) {
+ if (block_id_t(edge.dst_blockid).match(NODE_ID_SEP)) {
+ break;
+ }
+ UHD_LOG_TRACE("MULTI_USRP",
+ boost::format("Connecting RX edge: %s:%d -> %s:%d") % edge.src_blockid
+ % edge.src_port % edge.dst_blockid % edge.dst_port);
+ _graph->connect(
+ edge.src_blockid, edge.src_port, edge.dst_blockid, edge.dst_port);
}
- _graph->connect((rx_chain.ddc) ? rx_chain.ddc->get_block_id()
- : rx_chain.radio->get_block_id(),
- rx_chain.block_chan,
+ // Including the connection to the streamer
+ UHD_LOG_TRACE("MULTI_USRP",
+ boost::format("Connecting %s:%d -> RxStreamer:%d")
+ % rx_chain.edge_list.back().src_blockid
+ % rx_chain.edge_list.back().src_port % strm_port);
+ _graph->connect(rx_chain.edge_list.back().src_blockid,
+ rx_chain.edge_list.back().src_port,
rx_streamer,
strm_port);
const double chan_rate =
@@ -277,17 +253,26 @@ public:
for (size_t strm_port = 0; strm_port < args.channels.size(); ++strm_port) {
auto tx_channel = args.channels.at(strm_port);
auto tx_chain = _get_tx_chan(tx_channel);
- if (tx_chain.duc) {
- _graph->connect(tx_chain.duc->get_block_id(),
- tx_chain.block_chan,
- tx_chain.radio->get_block_id(),
- tx_chain.block_chan);
+ // Make all of the connections in our chain
+ for (auto edge : tx_chain.edge_list) {
+ if (block_id_t(edge.src_blockid).match(NODE_ID_SEP)) {
+ break;
+ }
+ UHD_LOG_TRACE("MULTI_USRP",
+ boost::format("Connecting TX edge %s:%d -> %s:%d") % edge.src_blockid
+ % edge.src_port % edge.dst_blockid % edge.dst_port);
+ _graph->connect(
+ edge.src_blockid, edge.src_port, edge.dst_blockid, edge.dst_port);
}
+ // Including the connection to the streamer
+ UHD_LOG_TRACE("MULTI_USRP",
+ boost::format("Connecting TxStreamer:%d -> %s:%d") % strm_port
+ % tx_chain.edge_list.back().dst_blockid
+ % tx_chain.edge_list.back().dst_port);
_graph->connect(tx_streamer,
strm_port,
- (tx_chain.duc) ? tx_chain.duc->get_block_id()
- : tx_chain.radio->get_block_id(),
- tx_chain.block_chan);
+ tx_chain.edge_list.back().dst_blockid,
+ tx_chain.edge_list.back().dst_port);
const double chan_rate =
_tx_rates.count(tx_channel) ? _tx_rates.at(tx_channel) : 1.0;
if (chan_rate > 1.0 && rate != chan_rate) {
@@ -862,23 +847,45 @@ public:
******************************************************************/
rx_chan_t _generate_rx_radio_chan(block_id_t radio_id, size_t block_chan)
{
- auto radio_blk = _graph->get_block<uhd::rfnoc::radio_control>(radio_id);
- // We assume that the DDC connected to this radio block has the same mboard,
- // instance, and port number
- auto ddc_id =
- block_id_t(radio_id.get_device_no(), "DDC", radio_id.get_block_count());
- uhd::rfnoc::ddc_block_control::sptr ddc_blk;
- try {
- ddc_blk = _graph->get_block<uhd::rfnoc::ddc_block_control>(ddc_id);
- } catch (const uhd::lookup_error&) {
- UHD_LOGGER_TRACE("MULTI_USRP") << "No DDC found: " << ddc_id.to_string();
- }
- // Figure out if this channel has a DDC available
- auto this_chan_ddc =
- ddc_blk && _graph->is_connectable(radio_id, block_chan, ddc_id, block_chan)
- ? ddc_blk
- : nullptr;
- return {radio_blk, this_chan_ddc, block_chan};
+ auto radio_blk = _graph->get_block<uhd::rfnoc::radio_control>(radio_id);
+ auto radio_source_chain = get_block_chain(_graph, radio_id, block_chan, true);
+
+ // Find out if we have a DDC in the radio block chain
+ auto ddc_port_def = [this, radio_source_chain, radio_id, block_chan]() {
+ try {
+ for (auto edge : radio_source_chain) {
+ if (block_id_t(edge.dst_blockid).match("DDC")) {
+ if (edge.dst_port != block_chan) {
+ /* We don't expect this to happen very often. But in
+ * the case that port numbers don't match, we need to
+ * disable DDC control to ensure we're not controlling
+ * another channel's DDC
+ */
+ UHD_LOGGER_WARNING("MULTI_USRP")
+ << "DDC in radio chain " << radio_id << ":"
+ << std::to_string(block_chan)
+ << " not connected to the same port number! "
+ "Disabling DDC control.";
+ break;
+ }
+ auto ddc_blk = _graph->get_block<uhd::rfnoc::ddc_block_control>(
+ edge.dst_blockid);
+ return std::tuple<uhd::rfnoc::ddc_block_control::sptr, size_t>(
+ ddc_blk, block_chan);
+ }
+ }
+ } catch (const uhd::exception&) {
+ UHD_LOGGER_DEBUG("MULTI_USRP")
+ << "No DDC found for radio block " << radio_id << ":"
+ << std::to_string(block_chan);
+ // Then just return a nullptr
+ }
+ return std::tuple<uhd::rfnoc::ddc_block_control::sptr, size_t>(nullptr, 0);
+ }();
+
+ // Create the RX chan
+ return rx_chan_t(
+ {radio_blk, std::get<0>(ddc_port_def), block_chan, radio_source_chain});
}
std::vector<rx_chan_t> _generate_mboard_rx_chans(
@@ -1446,23 +1453,45 @@ public:
tx_chan_t _generate_tx_radio_chan(block_id_t radio_id, size_t block_chan)
{
auto radio_blk = _graph->get_block<uhd::rfnoc::radio_control>(radio_id);
- // We assume that the duc connected to this radio block has the same mboard,
- // instance, and port number
- auto duc_id =
- block_id_t(radio_id.get_device_no(), "DUC", radio_id.get_block_count());
- uhd::rfnoc::duc_block_control::sptr duc_blk;
- try {
- duc_blk = _graph->get_block<uhd::rfnoc::duc_block_control>(duc_id);
- } catch (const uhd::exception&) {
- UHD_LOGGER_TRACE("MULTI_USRP_RFNOC")
- << boost::format("No DUC found: %s") % duc_id.to_string();
- }
- // Figure out if this channel has a DUC available
- auto this_chan_duc =
- duc_blk && _graph->is_connectable(duc_id, block_chan, radio_id, block_chan)
- ? duc_blk
- : nullptr;
- return {radio_blk, this_chan_duc, block_chan};
+ // Now on to the DUC chain
+ auto radio_sink_chain = get_block_chain(_graph, radio_id, block_chan, false);
+
+ // Find out if we have a DUC in the radio block chain
+ auto duc_port_def = [this, radio_sink_chain, radio_id, block_chan]() {
+ try {
+ for (auto edge : radio_sink_chain) {
+ if (block_id_t(edge.src_blockid).match("DUC")) {
+ if (edge.src_port != block_chan) {
+ /* We don't expect this to happen very often. But in
+ * the case that port numbers don't match, we need to
+ * disable DUC control to ensure we're not controlling
+ * another channel's DDC
+ */
+ UHD_LOGGER_WARNING("MULTI_USRP")
+ << "DUC in radio chain " << radio_id << ":"
+ << std::to_string(block_chan)
+ << " not connected to the same port number! "
+ "Disabling DUC control.";
+ break;
+ }
+ auto ddc_blk = _graph->get_block<uhd::rfnoc::duc_block_control>(
+ edge.src_blockid);
+ return std::tuple<uhd::rfnoc::duc_block_control::sptr, size_t>(
+ ddc_blk, block_chan);
+ }
+ }
+ } catch (const uhd::exception&) {
+ UHD_LOGGER_DEBUG("MULTI_USRP")
+ << "No DDC found for radio block " << radio_id << ":"
+ << std::to_string(block_chan);
+ // Then just return a nullptr
+ }
+ return std::tuple<uhd::rfnoc::duc_block_control::sptr, size_t>(nullptr, 0);
+ }();
+
+ // Create the TX chan
+ return tx_chan_t(
+ {radio_blk, std::get<0>(duc_port_def), block_chan, radio_sink_chain});
}
std::vector<tx_chan_t> _generate_mboard_tx_chans(