aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/include/uhdlib/rfnoc
diff options
context:
space:
mode:
Diffstat (limited to 'host/lib/include/uhdlib/rfnoc')
-rw-r--r--host/lib/include/uhdlib/rfnoc/graph.hpp49
-rw-r--r--host/lib/include/uhdlib/rfnoc/node_accessor.hpp20
2 files changed, 68 insertions, 1 deletions
diff --git a/host/lib/include/uhdlib/rfnoc/graph.hpp b/host/lib/include/uhdlib/rfnoc/graph.hpp
index f9fb7ac41..ec6309bd0 100644
--- a/host/lib/include/uhdlib/rfnoc/graph.hpp
+++ b/host/lib/include/uhdlib/rfnoc/graph.hpp
@@ -7,10 +7,12 @@
#ifndef INCLUDED_LIBUHD_GRAPH_HPP
#define INCLUDED_LIBUHD_GRAPH_HPP
+#include <uhd/rfnoc/actions.hpp>
#include <uhd/rfnoc/node.hpp>
#include <boost/graph/adjacency_list.hpp>
#include <tuple>
#include <memory>
+#include <deque>
namespace uhd { namespace rfnoc {
@@ -192,6 +194,39 @@ private:
void resolve_all_properties();
/**************************************************************************
+ * Action API
+ *************************************************************************/
+ /*! Entrypoint for action delivery
+ *
+ * When a node invokes its node_t::post_action() function, eventually that
+ * call lands here. This function acts as a mailman, that is, it figures out
+ * which edge on which node is supposed to receive this action, and delivers
+ * it via the node_t::receive_action() method.
+ * Note since this is private, nodes can't directly access this functions.
+ * We provide a lambda to nodes for that purpose.
+ *
+ * When an action is posted, that may trigger further actions. In order not
+ * to go into infinite recursion, this function is also responsible for
+ * serializing the actions. Even so, it is possible that, due to
+ * misconfiguration of nodes and their behaviour, a cascade of actions is
+ * posted that never stops. Therefore, another responsibility of this
+ * function is to track the number of follow-up messages sent, and terminate
+ * an infinite cycle of messages.
+ *
+ * \param src_node Reference to the node where the post_action() call is
+ * originating from
+ * \param src_edge The edge on that node where the action is being posted to.
+ * Note that its the edge from the node's point of view, so
+ * if src_edge.type == OUTPUT_EDGE, then the node posted to
+ * its output edge.
+ *
+ * \throws uhd::runtime_error if it has to terminate a infinite cascade of
+ * actions
+ */
+ void enqueue_action(
+ node_ref_t src_node, res_source_info src_edge, action_info::sptr action);
+
+ /**************************************************************************
* Private graph helpers
*************************************************************************/
template <typename VertexContainerType>
@@ -225,7 +260,7 @@ private:
/*! Find the neighbouring node for \p origin based on \p port_info
*
* This function will check port_info to identify the port number and the
- * direction (input or output) from \p port_info. It will then return a
+ * direction (input or output) from \p origin. It will then return a
* reference to the node that is attached to the node \p origin if such a
* node exists, and the edge info.
*
@@ -263,6 +298,18 @@ private:
// descriptor without having to traverse the graph. The rfnoc_graph_t is not
// efficient for lookups of vertices.
node_map_t _node_map;
+
+ using action_tuple_t = std::tuple<node_ref_t, res_source_info, action_info::sptr>;
+
+ //! FIFO for incoming actions
+ std::deque<action_tuple_t> _action_queue;
+
+ //! Flag to ensure serialized handling of actions
+ std::atomic_flag _action_handling_ongoing;
+
+ //! Mutex for to avoid the user from sending one message before another
+ // message is sent
+ std::recursive_mutex _action_mutex;
};
diff --git a/host/lib/include/uhdlib/rfnoc/node_accessor.hpp b/host/lib/include/uhdlib/rfnoc/node_accessor.hpp
index 554cc8f4f..827c87dd2 100644
--- a/host/lib/include/uhdlib/rfnoc/node_accessor.hpp
+++ b/host/lib/include/uhdlib/rfnoc/node_accessor.hpp
@@ -7,7 +7,9 @@
#ifndef INCLUDED_LIBUHD_NODE_ACCESSOR_HPP
#define INCLUDED_LIBUHD_NODE_ACCESSOR_HPP
+#include <uhd/rfnoc/actions.hpp>
#include <uhd/rfnoc/node.hpp>
+#include <uhd/rfnoc/res_source_info.hpp>
#include <functional>
namespace uhd { namespace rfnoc {
@@ -77,6 +79,24 @@ public:
{
dst_node->forward_edge_property(incoming_prop, dst_port);
}
+
+ /*! Set post action callback for the node
+ *
+ * See node_t::set_post_action_callback() for details.
+ */
+ void set_post_action_callback(node_t* node, node_t::action_handler_t&& post_handler)
+ {
+ node->set_post_action_callback(std::move(post_handler));
+ }
+
+ /*! Send an action to \p node
+ *
+ * This will call node_t::receive_action() (see that for details).
+ */
+ void send_action(node_t* node, const res_source_info& port_info, action_info::sptr action)
+ {
+ node->receive_action(port_info, action);
+ }
};