aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMark Meserve <mark.meserve@ni.com>2019-07-25 16:12:13 -0500
committerBrent Stapleton <brent.stapleton@ettus.com>2019-08-06 16:33:50 -0700
commitc6eb38369729d7ff32046a11533d15201c3e58fe (patch)
tree0a4e1c5bee33bc1af98b785ee148a1f1c6fac637
parentd25614e6ae3504f34afe7f330ce1e0079fa7ab52 (diff)
downloaduhd-c6eb38369729d7ff32046a11533d15201c3e58fe.tar.gz
uhd-c6eb38369729d7ff32046a11533d15201c3e58fe.tar.bz2
uhd-c6eb38369729d7ff32046a11533d15201c3e58fe.zip
rfnoc: fix multidevice graph connections
- Fixes an issue where a multi-device device3 object would cause connections to be generated across devices in legacy_compat - The connect_blocks function previously assumed that find_blocks would always return blocks in the same device number order
-rw-r--r--host/lib/rfnoc/legacy_compat.cpp203
1 files changed, 107 insertions, 96 deletions
diff --git a/host/lib/rfnoc/legacy_compat.cpp b/host/lib/rfnoc/legacy_compat.cpp
index 30c3f628f..91de361df 100644
--- a/host/lib/rfnoc/legacy_compat.cpp
+++ b/host/lib/rfnoc/legacy_compat.cpp
@@ -20,6 +20,7 @@
#include <uhd/utils/log.hpp>
#include <uhdlib/rfnoc/legacy_compat.hpp>
#include <boost/make_shared.hpp>
+#include <set>
#define UHD_LEGACY_LOG() UHD_LOGGER_TRACE("RFNOC")
@@ -880,9 +881,10 @@ private: // methods
}
}
}
- //! Flatten block list into a list of <block, port_index>
- // For example block list {b0[0,1] ,b1[0,1]} (i.e block 0 with 2 port 0 and 1,etc ..)
- // this will return {<b0,0> <b1,0> <b0,1> <b1,1>}
+
+ //! Flatten and sort a block list into a list of <block, port_index>
+ // For a block list {b0 ,b1} where each block has ports {p0, p1}, this will
+ // return {<b0,p0> <b0,p1> <b1,p0> <b1,p1>}
std::vector<source_port_t> _flatten_blocks_by_n_ports(source_block_list_t block_list)
{
std::vector<source_port_t> result;
@@ -891,25 +893,12 @@ private: // methods
result.push_back(std::make_pair(block, port));
}
}
- // assign to block prior ports
- size_t port = 0;
- size_t i = 0;
- for (size_t j = 0; j < result.size(); j++) {
- auto block = block_list[j % block_list.size()];
- UHD_ASSERT_THROW(port < block->get_output_ports().size());
- if (i == block_list.size()) {
- i = 0;
- port++;
- }
- result[j] = std::make_pair(block, port);
- i++;
- }
return result;
}
- //! Flatten block list into a list of <block, port_index>
- // For example block list {b0[0,1] ,b1[0,1]} (i.e block 0 with 2 port 0 and 1,etc ..)
- // this will return {<b0,0> <b1,0> <b0,1> <b1,1>}
+ //! Flatten and sort a block list into a list of <block, port_index>
+ // For a block list {b0 ,b1} where each block has ports {p0, p1}, this will
+ // return {<b0,p0> <b0,p1> <b1,p0> <b1,p1>}
std::vector<sink_port_t> _flatten_blocks_by_n_ports(sink_block_list_t block_list)
{
std::vector<sink_port_t> result;
@@ -918,19 +907,19 @@ private: // methods
result.push_back(std::make_pair(block, port));
}
}
- // assign to block prior ports
- size_t port = 0;
- size_t i = 0;
- for (size_t j = 0; j < result.size(); j++) {
- auto block = block_list[j % block_list.size()];
- UHD_ASSERT_THROW(port < block->get_input_ports().size());
- if (i == block_list.size()) {
- i = 0;
- port++;
- }
- result[j] = std::make_pair(block, port);
- i++;
- }
+ return result;
+ }
+
+ template <typename T>
+ std::vector<std::pair<T, size_t>> _filter_flattened_blocks(
+ const std::vector<std::pair<T, size_t>>& blocks, size_t device_number)
+ {
+ const auto pred = [&device_number](const std::pair<T, size_t>& block) {
+ return (block.first->get_block_id().get_device_no() == device_number);
+ };
+
+ std::vector<std::pair<T, size_t>> result;
+ std::copy_if(blocks.begin(), blocks.end(), std::back_inserter(result), pred);
return result;
}
@@ -950,86 +939,108 @@ private: // methods
*/
void connect_blocks()
{
- _graph = _device->create_graph("legacy");
const size_t rx_bpp = _rx_spp * BYTES_PER_SAMPLE + MAX_BYTES_PER_HEADER;
const size_t tx_bpp = _tx_spp * BYTES_PER_SAMPLE + MAX_BYTES_PER_HEADER;
+ _graph = _device->create_graph("legacy");
+
block_name_to_block_map_t legacy_block_map = get_legacy_blocks(_device);
- size_t index = 0, sram_fifo_index = 0, dma_fifo_index = 0;
- auto ddc_snk_flat =
+ // create a list of all devices in the legacy block map
+ std::set<size_t> device_numbers;
+ for (const auto& block_list : legacy_block_map) {
+ for (const auto& block : block_list.second.first) {
+ device_numbers.insert(block->get_block_id().get_device_no());
+ }
+ for (const auto& block : block_list.second.second) {
+ device_numbers.insert(block->get_block_id().get_device_no());
+ }
+ }
+
+ // Generate lists of all available ports in the graph
+ // RX connections:
+ // Radio => DDC
+ auto radio_sources =
+ _flatten_blocks_by_n_ports(legacy_block_map[RADIO_BLOCK_NAME].first);
+ auto ddc_sinks =
_flatten_blocks_by_n_ports(legacy_block_map[DDC_BLOCK_NAME].second);
- auto duc_src_flat =
+
+ // TX connections:
+ // DUC => Radio
+ auto duc_sources =
_flatten_blocks_by_n_ports(legacy_block_map[DUC_BLOCK_NAME].first);
- auto duc_snk_flat =
- _flatten_blocks_by_n_ports(legacy_block_map[DUC_BLOCK_NAME].second);
- auto radio_src_flat =
- _flatten_blocks_by_n_ports(legacy_block_map[RADIO_BLOCK_NAME].first);
- auto radio_snk_flat =
+ auto radio_sinks =
_flatten_blocks_by_n_ports(legacy_block_map[RADIO_BLOCK_NAME].second);
- auto sfifo_src_flat =
+
+ // FIFO (SRAM or DMA) => DUC
+ auto duc_sinks =
+ _flatten_blocks_by_n_ports(legacy_block_map[DUC_BLOCK_NAME].second);
+ auto sfifo_sources =
_flatten_blocks_by_n_ports(legacy_block_map[SFIFO_BLOCK_NAME].first);
- auto dfifo_src_flat =
+ auto dfifo_sources =
_flatten_blocks_by_n_ports(legacy_block_map[DFIFO_BLOCK_NAME].first);
- for (auto each_src_radio_block : radio_src_flat) {
- auto radio_block = each_src_radio_block.first->get_block_id();
+
+ for (const auto& device_number : device_numbers) {
+ // for RX, if there are DDCs, connect them to the radios
if (_has_ddcs) {
- UHD_ASSERT_THROW(index < ddc_snk_flat.size());
- auto ddc_block = ddc_snk_flat[index].first->get_block_id();
- _graph->connect(radio_block,
- each_src_radio_block.second,
- ddc_block,
- ddc_snk_flat[index].second,
- rx_bpp);
+ auto filtered_radio_sources =
+ _filter_flattened_blocks(radio_sources, device_number);
+ auto filtered_ddc_sinks =
+ _filter_flattened_blocks(ddc_sinks, device_number);
+ UHD_ASSERT_THROW(
+ filtered_radio_sources.size() <= filtered_ddc_sinks.size());
+
+ for (size_t i = 0; i < filtered_radio_sources.size(); ++i) {
+ _graph->connect(filtered_radio_sources[i].first->get_block_id(),
+ filtered_radio_sources[i].second,
+ filtered_ddc_sinks[i].first->get_block_id(),
+ filtered_ddc_sinks[i].second,
+ rx_bpp);
+ }
}
- index++;
- }
- index = 0;
- for (auto each_snk_radio_block : radio_snk_flat) {
- auto radio_block = each_snk_radio_block.first->get_block_id();
- auto down_stream_block = radio_block;
- auto down_stream_port = each_snk_radio_block.second;
+
+ // for TX, if there are DUCs, connect them to the radios
if (_has_ducs) {
- UHD_ASSERT_THROW(index < duc_snk_flat.size());
- UHD_ASSERT_THROW(index < duc_src_flat.size());
- auto duc_snk_block = duc_snk_flat[index].first->get_block_id();
- auto duc_src_block = duc_src_flat[index].first->get_block_id();
- _graph->connect(duc_src_block,
- duc_src_flat[index].second,
- radio_block,
- each_snk_radio_block.second,
- tx_bpp);
- down_stream_block = duc_snk_block;
- down_stream_port = duc_snk_flat[index].second;
- }
- if (_has_sramfifo) {
- if (sram_fifo_index < sfifo_src_flat.size()) {
- auto sfifo_block =
- sfifo_src_flat[sram_fifo_index].first->get_block_id();
- _graph->connect(sfifo_block,
- sfifo_src_flat[sram_fifo_index].second,
- down_stream_block,
- down_stream_port,
+ auto filtered_duc_sources =
+ _filter_flattened_blocks(duc_sources, device_number);
+ auto filtered_radio_sinks =
+ _filter_flattened_blocks(radio_sinks, device_number);
+
+ UHD_ASSERT_THROW(
+ filtered_duc_sources.size() <= filtered_radio_sinks.size());
+
+ for (size_t i = 0; i < filtered_duc_sources.size(); ++i) {
+ _graph->connect(filtered_duc_sources[i].first->get_block_id(),
+ filtered_duc_sources[i].second,
+ filtered_radio_sinks[i].first->get_block_id(),
+ filtered_radio_sinks[i].second,
tx_bpp);
- sram_fifo_index++;
- } else {
- UHD_LOGGER_WARNING("RFNOC")
- << "[legacy compat] Running out of SRAM FIFO ports to connect.";
}
- } else if (_has_dmafifo) {
- if (dma_fifo_index < dfifo_src_flat.size()) {
- auto dfifo_block =
- dfifo_src_flat[dma_fifo_index].first->get_block_id();
- _graph->connect(dfifo_block,
- dfifo_src_flat[dma_fifo_index].second,
- down_stream_block,
- down_stream_port,
+ }
+
+ // for TX, if there are SRAM or DMA fifos, connect them to the DUCs
+ // (or radios, if there are no DUCs)
+ if (_has_sramfifo or _has_dmafifo) {
+ auto filtered_sources =
+ (_has_sramfifo)
+ ? _filter_flattened_blocks(sfifo_sources, device_number)
+ : _filter_flattened_blocks(dfifo_sources, device_number);
+ auto filtered_sinks =
+ (_has_ducs) ? _filter_flattened_blocks(duc_sinks, device_number)
+ : _filter_flattened_blocks(radio_sinks, device_number);
+
+ if (filtered_sources.size() < filtered_sinks.size()) {
+ UHD_LOG_WARNING("RFNOC",
+ "[legacy compat] Not enough FIFO ports to connect, not all TX "
+ "sinks will be connected");
+ }
+
+ for (size_t i = 0; i < filtered_sources.size(); ++i) {
+ _graph->connect(filtered_sources[i].first->get_block_id(),
+ filtered_sources[i].second,
+ filtered_sinks[i].first->get_block_id(),
+ filtered_sinks[i].second,
tx_bpp);
- dma_fifo_index++;
- } else {
- UHD_LOGGER_WARNING("RFNOC")
- << "[legacy compat] Running out of DRAM FIFO ports to connect.";
}
}
- index++;
}
}