aboutsummaryrefslogtreecommitdiffstats
path: root/host/include/uhd/rfnoc/node_ctrl_base.ipp
blob: 266df445a6365772bbc57b1431488cf929ee02f5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
//
// Copyright 2014 Ettus Research LLC
// Copyright 2018 Ettus Research, a National Instruments Company
//
// SPDX-License-Identifier: GPL-3.0-or-later
//

// Implements templated functions from node_ctrl_base.hpp

#ifndef INCLUDED_LIBUHD_NODE_CTRL_BASE_IPP
#define INCLUDED_LIBUHD_NODE_CTRL_BASE_IPP

#include <uhd/exception.hpp>

#include <boost/shared_ptr.hpp>
#include <vector>

namespace uhd {
    namespace rfnoc {

    template <typename T, bool downstream>
    std::vector< boost::shared_ptr<T> > node_ctrl_base::_find_child_node(bool active_only)
    {
        typedef boost::shared_ptr<T> T_sptr;
        static const size_t MAX_ITER = 20;
        size_t iters = 0;
        // List of return values:
        std::set< T_sptr > results_s;
        // To avoid cycles:
        std::set < sptr > explored;
        // Initialize our search queue with ourself:
        std::set < node_map_pair_t > search_q;
        // FIXME:  Port is initialized to ANY_PORT, but it should really be
        // passed in by the caller.
        search_q.insert(node_map_pair_t(ANY_PORT, shared_from_this()));
        std::set < node_map_pair_t > next_q;

        while (iters++ < MAX_ITER) {
            next_q.clear();
            BOOST_FOREACH(const node_map_pair_t node_pair, search_q) {
                sptr node = node_pair.second.lock();
                if (not node)
                {
                    continue;
                }
                size_t our_port = node_pair.first;
                // Add this node to the list of explored nodes
                explored.insert(node);
                // Create set of all child nodes of this_node that are not in explored:
                std::set< node_map_pair_t > next_nodes;
                {
                    node_map_t all_next_nodes = downstream ?
                        node->list_downstream_nodes() :
                        node->list_upstream_nodes();
                    for (
                        node_map_t::iterator it = all_next_nodes.begin();
                        it != all_next_nodes.end();
                        ++it
                    ) {
                        size_t connected_port = it->first;
                        // If port is given, limit traversal to only that port.
                        if (our_port != ANY_PORT and our_port != connected_port)
                        {
                            continue;
                        }
                        if (active_only
                            and not (downstream ? _tx_streamer_active[our_port] : _rx_streamer_active[our_port] )) {
                            continue;
                        }
                        sptr one_next_node = it->second.lock();
                        if (not one_next_node or explored.count(one_next_node)) {
                            continue;
                        }
                        T_sptr next_node_sptr = boost::dynamic_pointer_cast<T>(one_next_node);
                        if (next_node_sptr) {
                            results_s.insert(next_node_sptr);
                        } else {
                            size_t next_port = ANY_PORT;
                            // FIXME:  Need proper mapping from input port
                            // to output port.
                            // The code below assumes that blocks with the same
                            // number of connected upstream and downstream nodes
                            // map each input port directly to the same output
                            // port.  This limits the graph traversal to prevent
                            // finding nodes that are not part of this chain.
                            if (one_next_node->list_upstream_nodes().size() ==
                                one_next_node->list_downstream_nodes().size())
                            {
                                next_port = (downstream ?
                                    node->get_downstream_port(connected_port) :
                                    node->get_upstream_port(connected_port));
                            }
                            next_nodes.insert(node_map_pair_t(next_port, it->second));
                        }
                    }
                }
                // Add all of these nodes to the next search queue
                next_q.insert(next_nodes.begin(), next_nodes.end());
            }
            // If next_q is empty, we've exhausted our graph
            if (next_q.empty()) {
                break;
            }
            // Re-init the search queue
            search_q = next_q;
        }

        std::vector< T_sptr > results(results_s.begin(), results_s.end());
        return results;
    }

    template <typename T, typename value_type, bool downstream>
    value_type node_ctrl_base::_find_unique_property(
            boost::function<value_type(boost::shared_ptr<T>, size_t)> get_property,
            value_type NULL_VALUE,
            const std::set< boost::shared_ptr<T> > &exclude_nodes
    ) {
        std::vector< boost::shared_ptr<T> > descendant_rate_nodes = _find_child_node<T, downstream>();
        value_type ret_val = NULL_VALUE;
        std::string first_node_id;
        BOOST_FOREACH(const boost::shared_ptr<T> &node, descendant_rate_nodes) {
            if (exclude_nodes.count(node)) {
                continue;
            }
            // FIXME we need to know the port!!!
            size_t port = ANY_PORT; // NOOO! this is wrong!!!! FIXME
            value_type this_property = get_property(node, port);
            if (this_property == NULL_VALUE) {
                continue;
            }
            // We use the first property we find as reference
            if (ret_val == NULL_VALUE) {
                ret_val = this_property;
                first_node_id = node->unique_id();
                continue;
            }
            // In all subsequent finds, we make sure the property is equal to the reference
            if (this_property != ret_val) {
                throw uhd::runtime_error(
                    str(
                        boost::format("Node %1% specifies %2%, node %3% specifies %4%")
                            % first_node_id % ret_val % node->unique_id() % this_property
                    )
                );
            }
        }
        return ret_val;
    }

}} /* namespace uhd::rfnoc */

#endif /* INCLUDED_LIBUHD_NODE_CTRL_BASE_IPP */
// vim: sw=4 et: