diff options
author | Brent Stapleton <brent.stapleton@ettus.com> | 2019-10-09 11:41:38 -0700 |
---|---|---|
committer | Martin Braun <martin.braun@ettus.com> | 2019-11-26 12:21:32 -0800 |
commit | 2da99fad97a7123cb08b429e93a557327582ade9 (patch) | |
tree | 53620e52cdfa9fc46c4652d7d671320f414db8ab /host/lib/utils/graph_utils.cpp | |
parent | c9126ae31d315127e66966b5c63a12cf8c66d44d (diff) | |
download | uhd-2da99fad97a7123cb08b429e93a557327582ade9.tar.gz uhd-2da99fad97a7123cb08b429e93a557327582ade9.tar.bz2 uhd-2da99fad97a7123cb08b429e93a557327582ade9.zip |
rfnoc: Adding rfnoc_graph utilities
Adding graph_utils to keep rfnoc_graph utilities to contain helper
function and commonly used algorithms for the rfnoc_graph. These
functions aren't core to the rfnoc_graph's functionality, so we'll keep
them out of its API.
Diffstat (limited to 'host/lib/utils/graph_utils.cpp')
-rw-r--r-- | host/lib/utils/graph_utils.cpp | 133 |
1 files changed, 133 insertions, 0 deletions
diff --git a/host/lib/utils/graph_utils.cpp b/host/lib/utils/graph_utils.cpp new file mode 100644 index 000000000..d6e82f546 --- /dev/null +++ b/host/lib/utils/graph_utils.cpp @@ -0,0 +1,133 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Branch +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#include <uhd/rfnoc/block_id.hpp> +#include <uhd/rfnoc/defaults.hpp> +#include <uhd/rfnoc/graph_edge.hpp> +#include <uhd/rfnoc/noc_block_base.hpp> +#include <uhd/utils/graph_utils.hpp> +#include <uhd/utils/log.hpp> +#include <boost/format.hpp> +#include <numeric> +#include <utility> + + +namespace uhd { namespace rfnoc { + + +//! Returns whether or not a block (and port) is know to to terminate data paths +bool check_terminator_block(const block_id_t blk_id, const size_t port) +{ + const std::string blk_id_str = blk_id.get_block_name(); + for (auto term_block : TERMINATOR_BLOCKS) { + auto optional_port = std::get<1>(term_block); + if (blk_id_str == std::get<0>(term_block) + && (!optional_port || port == optional_port.get())) { + return true; + } + } + return false; +} + + +std::vector<graph_edge_t> get_block_chain(const rfnoc_graph::sptr graph, + const block_id_t start_block, + const size_t port, + const bool source_chain) +{ + // Enumerate blocks in the chain + auto edges = graph->enumerate_static_connections(); + + std::vector<graph_edge_t> block_chain; + std::string current_block = start_block.to_string(); + size_t current_port = port; + while (true) { + UHD_LOG_TRACE("GRAPH_UTILS", + "Looking for current block " << current_block << ", port " << current_port); + bool next_found = false; + for (auto& edge : edges) { + if ((source_chain) + ? (edge.src_blockid == current_block && edge.src_port == current_port) + : (edge.dst_blockid == current_block + && edge.dst_port == current_port)) { + // If the current block is the edge's source, make the edge's + // destination the current block + next_found = true; + UHD_LOG_TRACE("GRAPH_UTILS", "Found next block: " + edge.dst_blockid); + + block_chain.push_back(edge); + current_block = (source_chain) ? edge.dst_blockid : edge.src_blockid; + current_port = (source_chain) ? edge.dst_port : edge.src_port; + // Compare our current block and port + if (check_terminator_block(current_block, current_port)) { + // If we've found a terminating block, stop iterating through the + // edges + break; + } + } + } + if (not next_found) { + UHD_LOG_TRACE( + "GRAPH_UTILS", "Failed to find current block in static connections"); + break; + } + if (check_terminator_block(current_block, current_port)) { + // If we've found a terminating block, stop iterating through the edges + break; + } + } + return block_chain; +} + + +void connect_through_blocks(rfnoc_graph::sptr graph, + const block_id_t src_blk, + const size_t src_port, + const block_id_t dst_blk, + const size_t dst_port) +{ + // First, create a chain from the source block to a stream endpoint + auto block_chain = get_block_chain(graph, src_blk, src_port, true); + UHD_LOG_TRACE("GRAPH_UTILS", "Found source chain for " + src_blk.to_string()); + // See if dst_blk is in our block_chain already + const bool dst_found = std::accumulate(block_chain.begin(), + block_chain.end(), + false, + [dst_blk, dst_port](bool dst_found, const graph_edge_t edge) { + // This is our "accumulator" function that checks if the current_blk's ID and + // input port match what we're looking for + return dst_found + || (dst_blk.to_string() == edge.dst_blockid + && dst_port == edge.dst_port); + }); + // If our dst_blk is in the chain already, make sure its the last element and continue + if (dst_found) { + UHD_LOG_TRACE( + "GRAPH_UTILS", "Found dst_blk (" + dst_blk.to_string() + ") in source chain"); + while (dst_blk.to_string() == block_chain.back().dst_blockid + && dst_port == block_chain.back().dst_port) { + UHD_LOG_TRACE("GRAPH_UTILS", + boost::format( + "Last block (%s:%d) doesn't match dst_blk (%s:%d); removing.") + % block_chain.back().dst_blockid % block_chain.back().dst_port + % dst_blk.to_string() % dst_port); + block_chain.pop_back(); + } + } else { + // If we hadn't found dst_blk, find it now, then merge the two chain + auto dest_chain = get_block_chain(graph, dst_blk, dst_port, false); + block_chain.insert(block_chain.end(), dest_chain.begin(), dest_chain.end()); + UHD_LOG_TRACE( + "GRAPH_UTILS", "Found destination chain for " + dst_blk.to_string()); + } + + // Finally, make all of the connections in our chain + for (auto edge : block_chain) { + graph->connect(edge.src_blockid, edge.src_port, edge.dst_blockid, edge.dst_port); + } +} + +}} // namespace uhd::rfnoc |