2#include <ossia/dataflow/bench_map.hpp>
3#include <ossia/dataflow/data_copy.hpp>
4#include <ossia/dataflow/dataflow.hpp>
5#include <ossia/dataflow/execution_state.hpp>
6#include <ossia/dataflow/for_each_port.hpp>
7#include <ossia/dataflow/graph/breadth_first_search.hpp>
8#include <ossia/dataflow/graph/graph_interface.hpp>
9#include <ossia/dataflow/graph/graph_ordering.hpp>
10#include <ossia/dataflow/graph/small_graph.hpp>
11#include <ossia/dataflow/graph_edge.hpp>
12#include <ossia/dataflow/graph_node.hpp>
13#include <ossia/dataflow/port.hpp>
15#include <ossia/detail/disable_fpe.hpp>
16#include <ossia/detail/flat_set.hpp>
17#include <ossia/detail/ptr_set.hpp>
20#include <boost/graph/adjacency_list.hpp>
21#include <boost/predef.h>
24#include <boost/graph/topological_sort.hpp>
26#include <ankerl/unordered_dense.h>
28#if BOOST_LIB_STD_GNU >= BOOST_VERSION_NUMBER(13, 0, 0) \
29 && BOOST_VERSION_NUMBER <= BOOST_VERSION_NUMBER(1, 83, 0)
30#define OSSIA_SMALL_VECTOR_ALLOCATOR_REBIND_FAILS 1
36using graph_t = boost::adjacency_list<
37 boost::smallvecS, boost::smallvecS, boost::directedS, node_ptr,
38 std::shared_ptr<graph_edge>>;
40namespace boost::detail
44class stored_edge_property<unsigned long, std::shared_ptr<ossia::graph_edge>>
45 :
public stored_edge<unsigned long>
47 using Vertex =
unsigned long;
48 using Property = std::shared_ptr<ossia::graph_edge>;
49 typedef stored_edge_property self;
50 typedef stored_edge<Vertex> Base;
53 typedef Property property_type;
54 inline stored_edge_property() { }
55 inline stored_edge_property(Vertex target, Property p = {})
56 : stored_edge<Vertex>(target)
57 , m_property(std::move(p))
60 stored_edge_property(self&& x) noexcept
61 : Base(
static_cast<Base&&
>(x))
62 , m_property(std::move(x.m_property))
65 stored_edge_property(self
const& x)
66 : Base(static_cast<Base const&>(x))
67 , m_property(std::move(const_cast<self&>(x).m_property))
70 self& operator=(self&& x)
74 static_cast<Base&
>(*this) =
static_cast<Base&&
>(x);
75 m_property = std::move(x.m_property);
78 self& operator=(self
const& x)
82 static_cast<Base&
>(*this) =
static_cast<Base const&
>(x);
83 m_property = std::move(
const_cast<self&
>(x).m_property);
86 inline Property& get_property() noexcept {
return m_property; }
87 inline const Property& get_property() const noexcept {
return m_property; }
100using DFSVertexInfo = std::pair<
101 typename graph_traits<G>::vertex_descriptor,
103 boost::optional<typename graph_traits<G>::edge_descriptor>,
105 typename graph_traits<G>::out_edge_iterator,
106 typename graph_traits<G>::out_edge_iterator>>>;
108template <
class Inc
idenceGraph,
class DFSVisitor,
class ColorMap>
109void custom_depth_first_visit_impl(
110 const IncidenceGraph& g,
typename graph_traits<IncidenceGraph>::vertex_descriptor u,
111 DFSVisitor& vis, ColorMap color, std::vector<DFSVertexInfo<IncidenceGraph>>& stack)
113 constexpr detail::nontruth2 func;
114 BOOST_CONCEPT_ASSERT((IncidenceGraphConcept<IncidenceGraph>));
115 BOOST_CONCEPT_ASSERT((DFSVisitorConcept<DFSVisitor, IncidenceGraph>));
116 typedef typename graph_traits<IncidenceGraph>::vertex_descriptor Vertex;
117 typedef typename graph_traits<IncidenceGraph>::edge_descriptor Edge;
118 BOOST_CONCEPT_ASSERT((ReadWritePropertyMapConcept<ColorMap, Vertex>));
119 typedef typename property_traits<ColorMap>::value_type ColorValue;
120 BOOST_CONCEPT_ASSERT((ColorValueConcept<ColorValue>));
121 typedef color_traits<ColorValue> Color;
122 typedef typename graph_traits<IncidenceGraph>::out_edge_iterator Iter;
123 typedef std::pair<Vertex, std::pair<boost::optional<Edge>, std::pair<Iter, Iter>>>
126 boost::optional<Edge> src_e;
131 stack.reserve(num_vertices(g));
133 put(color, u, Color::gray());
134 vis.discover_vertex(u, g);
135 boost::tie(ei, ei_end) = out_edges(u, g);
139 stack.push_back(std::make_pair(
140 u, std::make_pair(boost::optional<Edge>(), std::make_pair(ei_end, ei_end))));
144 stack.push_back(std::make_pair(
145 u, std::make_pair(boost::optional<Edge>(), std::make_pair(ei, ei_end))));
147 while(!stack.empty())
149 VertexInfo& back = stack.back();
151 src_e = back.second.first;
152 boost::tie(ei, ei_end) = back.second.second;
158 call_finish_edge(vis, src_e.get(), g);
162 Vertex v = target(*ei, g);
163 vis.examine_edge(*ei, g);
164 ColorValue v_color = get(color, v);
165 if(v_color == Color::white())
167 vis.tree_edge(*ei, g);
170 std::make_pair(u, std::make_pair(src_e, std::make_pair(++ei, ei_end))));
172 put(color, u, Color::gray());
173 vis.discover_vertex(u, g);
174 boost::tie(ei, ei_end) = out_edges(u, g);
182 if(v_color == Color::gray())
184 vis.back_edge(*ei, g);
188 vis.forward_or_cross_edge(*ei, g);
190 call_finish_edge(vis, *ei, g);
194 put(color, u, Color::black());
195 vis.finish_vertex(u, g);
200template <
class VertexListGraph,
class DFSVisitor,
class ColorMap>
201void custom_depth_first_search(
202 const VertexListGraph& g, DFSVisitor vis, ColorMap color,
203 typename graph_traits<VertexListGraph>::vertex_descriptor start_vertex,
204 std::vector<boost::detail::DFSVertexInfo<VertexListGraph>>& stack)
206 typedef typename graph_traits<VertexListGraph>::vertex_descriptor Vertex;
207 BOOST_CONCEPT_ASSERT((DFSVisitorConcept<DFSVisitor, VertexListGraph>));
208 typedef typename property_traits<ColorMap>::value_type ColorValue;
209 typedef color_traits<ColorValue> Color;
211 typename graph_traits<VertexListGraph>::vertex_iterator ui, ui_end;
212 for(boost::tie(ui, ui_end) = vertices(g); ui != ui_end; ++ui)
214 Vertex u = implicit_cast<Vertex>(*ui);
215 put(color, u, Color::white());
216 vis.initialize_vertex(u, g);
219 if(start_vertex != detail::get_default_starting_vertex(g))
221 vis.start_vertex(start_vertex, g);
222 detail::custom_depth_first_visit_impl(g, start_vertex, vis, color, stack);
225 for(boost::tie(ui, ui_end) = vertices(g); ui != ui_end; ++ui)
227 Vertex u = implicit_cast<Vertex>(*ui);
228 ColorValue u_color = get(color, u);
229 if(u_color == Color::white())
231 vis.start_vertex(u, g);
232 detail::custom_depth_first_visit_impl(g, u, vis, color, stack);
240inline void remove_vertex(
typename graph_t::vertex_descriptor v, graph_t& g)
242 typedef typename graph_t::directed_category Cat;
243 g.removing_vertex(v, boost::graph_detail::iterator_stability(g.m_vertices));
244 boost::detail::remove_vertex_dispatch(g, v, Cat());
247template <
typename VertexListGraph,
typename OutputIterator>
248void custom_topological_sort(
249 VertexListGraph& g, OutputIterator result,
250 std::vector<boost::default_color_type>& color_map,
251 std::vector<boost::detail::DFSVertexInfo<VertexListGraph>>& stack)
254 color_map.resize(boost::num_vertices(g));
256 auto map = boost::make_iterator_property_map(
257 color_map.begin(), boost::get(boost::vertex_index, g), color_map[0]);
259 boost::custom_depth_first_search(
260 g, boost::topo_sort_visitor<OutputIterator>(result), map,
261 boost::detail::get_default_starting_vertex(g), stack);
266using graph_vertex_t = graph_t::vertex_descriptor;
267using graph_edge_t = graph_t::edge_descriptor;
269#if !defined(OSSIA_NO_FAST_CONTAINERS)
270template <
typename T,
typename V>
271using dense_shared_ptr_map = ankerl::unordered_dense::map<
272 std::shared_ptr<T>, V, egur_hash, pointer_equal,
273#if defined(OSSIA_SMALL_VECTOR_ALLOCATOR_REBIND_FAILS)
274 std::vector<std::pair<std::shared_ptr<T>, V>>
276 ossia::small_vector<std::pair<std::shared_ptr<T>, V>, 1024>
280template <
typename T,
typename V>
281using dense_shared_ptr_map
282 = ossia::hash_map<std::shared_ptr<T>, V, egur_hash, pointer_equal>;
284using node_map = ossia::dense_shared_ptr_map<ossia::graph_node, graph_vertex_t>;
285using edge_map = ossia::dense_shared_ptr_map<ossia::graph_edge, graph_edge_t>;
287using node_flat_set = ossia::flat_set<graph_node*>;
288enum class node_ordering
296auto apply_con(
const T& visitor, ossia::connection& con)
298 auto tgt = con.target();
299 switch(con.which().index())
301 case ossia::connection::index_of<immediate_glutton_connection>().index():
302 return visitor(*
reinterpret_cast<immediate_glutton_connection*
>(tgt));
304 case ossia::connection::index_of<immediate_strict_connection>().index():
305 return visitor(*
reinterpret_cast<immediate_strict_connection*
>(tgt));
307 case ossia::connection::index_of<delayed_glutton_connection>().index():
308 return visitor(*
reinterpret_cast<delayed_glutton_connection*
>(tgt));
310 case ossia::connection::index_of<delayed_strict_connection>().index():
311 return visitor(*
reinterpret_cast<delayed_strict_connection*
>(tgt));
313 case ossia::connection::index_of<dependency_connection>().index():
314 return visitor(*
reinterpret_cast<dependency_connection*
>(tgt));
322auto apply_con(
const T& visitor,
const ossia::connection& con)
324 auto tgt = con.target();
325 switch(con.which().index())
327 case ossia::connection::index_of<immediate_glutton_connection>().index():
328 return visitor(*
reinterpret_cast<const immediate_glutton_connection*
>(tgt));
330 case ossia::connection::index_of<immediate_strict_connection>().index():
331 return visitor(*
reinterpret_cast<const immediate_strict_connection*
>(tgt));
333 case ossia::connection::index_of<delayed_glutton_connection>().index():
334 return visitor(*
reinterpret_cast<const delayed_glutton_connection*
>(tgt));
336 case ossia::connection::index_of<delayed_strict_connection>().index():
337 return visitor(*
reinterpret_cast<const delayed_strict_connection*
>(tgt));
339 case ossia::connection::index_of<dependency_connection>().index():
340 return visitor(*
reinterpret_cast<const dependency_connection*
>(tgt));
348template <
typename Graph_T,
typename IO>
349void print_graph(Graph_T& g, IO& stream)
353 boost::write_graphviz(
355 [&](
auto& out,
auto v) {
356 if(g[v] && !g[v]->label().empty())
357 out <<
"[label=\"" << g[v]->label() <<
"\"]";
363 stream << s.str() <<
"\n";
367struct OSSIA_EXPORT graph_util
369 static void pull_from_parameter(inlet& in, execution_state& e)
371 apply_to_destination(
372 in.address, e.exec_devices(),
374 if(in.scope & port::scope_t::local)
376 e.find_and_copy(*addr, in);
378 else if(in.scope & port::scope_t::global)
380 e.copy_from_global(*addr, in);
387 static void init_outlet(outlet& out, execution_state&)
389 out.visit(clear_data{});
394 static void init_inlet(inlet& in, execution_state& e)
396 bool must_copy = in.sources.empty();
398 for(
const graph_edge* edge : in.sources)
400 must_copy |= ossia::apply_con(init_must_copy_visitor{*edge}, edge->con);
404 pull_from_parameter(in, e);
406 for(
auto edge : in.sources)
408 ossia::apply_con(init_node_visitor{in, *edge, e}, edge->con);
414 static void init_node(graph_node& n, execution_state& e)
423 for_each_outlet(n, [&](
auto& port) { init_outlet(port, e); });
426 for_each_inlet(n, [&](
auto& port) { init_inlet(port, e); });
430 static void teardown_outlet(outlet& out, execution_state& e)
433 bool must_copy = out.targets.empty();
437 for(
const auto& tgt : out.targets)
439 must_copy |= ossia::apply_con(env_writer{out, *tgt}, tgt->con);
449 static void teardown_inlet(inlet& in, execution_state&)
452 in.visit(clear_data{});
455 static void teardown_node(
const graph_node& n, execution_state& e)
465 for_each_outlet(n, [&](
auto& port) { teardown_outlet(port, e); });
468 for_each_inlet(n, [&](
auto& port) { teardown_inlet(port, e); });
488 static bool disable_strict_nodes(
const graph_node* node)
493 auto test_disable_inlet = [&](
const ossia::inlet& inlet) {
494 for(
const auto& edge : inlet.sources)
497 assert(edge->out_node);
499 if(immediate_strict_connection* sc
500 = edge->con.target<immediate_strict_connection>())
502 if((sc->required_sides
503 & immediate_strict_connection::required_sides_t::outbound)
504 && node->enabled() && !edge->out_node->enabled())
510 delayed_strict_connection* delay
511 = edge->con.target<delayed_strict_connection>())
513 const auto n = ossia::apply(data_size{}, delay->buffer);
514 if(n == 0 || delay->pos >= n)
523 auto test_disable_outlet = [&](
const ossia::outlet& outlet) {
524 for(
const auto& edge : outlet.targets)
526 assert(edge->in_node);
528 if(
auto sc = edge->con.target<immediate_strict_connection>())
530 if((sc->required_sides
531 & immediate_strict_connection::required_sides_t::inbound)
532 && node->enabled() && !edge->in_node->enabled())
541 if(any_of_inlet(*node, test_disable_inlet))
543 if(any_of_outlet(*node, test_disable_outlet))
550 disable_strict_nodes(
const node_flat_set& enabled_nodes, node_flat_set& ret)
552 for(graph_node* node : enabled_nodes)
554 if(disable_strict_nodes(node))
559 static void disable_strict_nodes_rec(
560 node_flat_set& cur_enabled_node, node_flat_set& disabled_cache)
564 disabled_cache.clear();
565 disable_strict_nodes(cur_enabled_node, disabled_cache);
566 for(ossia::graph_node* n : disabled_cache)
570 cur_enabled_node.erase(n);
573 }
while(!disabled_cache.empty());
576 static void log_inputs(
const graph_node&, ossia::logger_type&
logger);
577 static void log_outputs(
const graph_node&, ossia::logger_type&
logger);
579 static void check_inputs(
const graph_node&, execution_state& e);
581 check_outputs(
const graph_node&, execution_state& e,
const ossia::token_request& req);
582 static void run_scaled(graph_node& first_node, execution_state& e);
585#define OSSIA_DEBUG_MISBEHAVING_NODES 1
587 static void exec_node(graph_node& first_node, execution_state& e)
589 init_node(first_node, e);
591#if defined(OSSIA_DEBUG_MISBEHAVING_NODES)
592 check_inputs(first_node, e);
595 if(!first_node.requested_tokens.empty())
597 if(first_node.start_discontinuous())
599 first_node.requested_tokens.front().start_discontinuous =
true;
600 first_node.set_start_discontinuous(
false);
602 if(first_node.end_discontinuous())
604 first_node.requested_tokens.front().end_discontinuous =
true;
605 first_node.set_end_discontinuous(
false);
609 for(
const auto& request : first_node.requested_tokens)
611 first_node.run(request, {&e});
612 first_node.process_time(request, e);
613#if defined(OSSIA_DEBUG_MISBEHAVING_NODES)
614 check_outputs(first_node, e, request);
618 first_node.set_executed(
true);
619 teardown_node(first_node, e);
623 exec_node(graph_node& first_node, execution_state& e, ossia::logger_type&
logger)
625 init_node(first_node, e);
627#if defined(OSSIA_DEBUG_MISBEHAVING_NODES)
628 check_inputs(first_node, e);
631 if(!first_node.requested_tokens.empty())
633 if(first_node.start_discontinuous())
635 first_node.requested_tokens.front().start_discontinuous =
true;
636 first_node.set_start_discontinuous(
false);
638 if(first_node.end_discontinuous())
640 first_node.requested_tokens.front().end_discontinuous =
true;
641 first_node.set_end_discontinuous(
false);
645 log_inputs(first_node,
logger);
646 for(
const auto& request : first_node.requested_tokens)
648 first_node.run(request, {&e});
649 first_node.process_time(request, e);
650#if defined(OSSIA_DEBUG_MISBEHAVING_NODES)
651 check_outputs(first_node, e, request);
654 log_outputs(first_node,
logger);
656 first_node.set_executed(
true);
657 teardown_node(first_node, e);
661 static bool can_execute(graph_node& node,
const execution_state&)
663 return all_of_inlet(node, [](
const auto& inlet) {
669 bool previous_nodes_executed = ossia::all_of(inlet.sources, [&](graph_edge* edge) {
670 return edge->out_node->executed()
671 || (!edge->out_node->enabled()
677 return !inlet.sources.empty() ? previous_nodes_executed :
true;
681 static void finish_nodes(
const node_map& nodes)
683 for(
auto& node : nodes)
685 ossia::graph_node& n = *node.first;
686 n.set_executed(
false);
691 template <
typename DevicesT>
692 static auto find_address_connection(
693 ossia::graph_node& source, ossia::graph_node& sink,
const DevicesT& devices)
695 return any_of_outlet(source, [&](
auto& outlet) {
696 return any_of_inlet(sink, [&](
auto& inlet) {
698 apply_to_destination(
699 outlet.address, devices,
701 apply_to_destination(
702 inlet.address, devices,
703 [&](ossia::net::parameter_base* p2, bool) {
707 do_nothing_for_nodes{});
709 do_nothing_for_nodes{});
716struct OSSIA_EXPORT graph_base : graph_interface
718 graph_base() noexcept
720#if !defined(OSSIA_FREESTANDING)
721 m_nodes.reserve(1024);
722 m_node_list.reserve(1024);
723 m_edges.reserve(1024);
726 [[nodiscard]] tcb::span<ossia::graph_node* const>
727 get_nodes() const noexcept final
override
729 return tcb::span{m_node_list};
732 void recompute_maps()
737 auto vertices = boost::vertices(m_graph);
738 m_nodes.reserve(m_graph.m_vertices.size());
739 for(
auto it = vertices.first; it != vertices.second; ++it)
741 graph_vertex_t k = *it;
742 node_ptr n = m_graph[k];
745 m_nodes.insert({n, k});
749#if !defined(__clang__)
750#pragma GCC diagnostic push
751#pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
753 auto edges = boost::edges(m_graph);
754 m_edges.reserve(std::distance(edges.first, edges.second));
755 for(
auto it = edges.first; it != edges.second; ++it)
757 graph_edge_t k = *it;
758 edge_ptr n = m_graph[k];
761 m_edges.insert({n, k});
763#if !defined(__clang__)
764#pragma GCC diagnostic pop
768 auto add_node_impl(node_ptr n)
773 auto vtx = boost::add_vertex(n, m_graph);
775 m_node_list.push_back(n.get());
781 void add_node(node_ptr n)
final override
783 if(m_nodes.find(n) == m_nodes.end())
785 add_node_impl(std::move(n));
789 void remove_node(
const node_ptr& n)
final override
791 for_each_inlet(*n, [&](
auto& port) {
792 auto s = port.sources;
796 for_each_outlet(*n, [&](
auto& port) {
797 auto s = port.targets;
802 auto it = m_nodes.find(n);
803 if(it != m_nodes.end())
805 auto vtx = boost::vertices(m_graph);
806 if(std::find(vtx.first, vtx.second, it->second) != vtx.second)
808 boost::clear_vertex(it->second, m_graph);
809 remove_vertex(it->second, m_graph);
815 ossia::remove_one(m_node_list, n.get());
819 void connect(std::shared_ptr<graph_edge> edge)
final override
825 graph_vertex_t in_vtx, out_vtx;
826 auto it1 = m_nodes.find(edge->in_node);
827 if(it1 == m_nodes.end())
828 in_vtx = add_node_impl(edge->in_node);
830 in_vtx = it1->second;
832 auto it2 = m_nodes.find(edge->out_node);
833 if(it2 == m_nodes.end())
834 out_vtx = add_node_impl(edge->out_node);
836 out_vtx = it2->second;
839 boost::add_edge(in_vtx, out_vtx, edge, m_graph);
845 void disconnect(
const std::shared_ptr<graph_edge>& edge)
final override
847 disconnect(edge.get());
850 void disconnect(graph_edge* edge)
final override
855 auto it = m_edges.find(edge);
856 if(it != m_edges.end())
858 auto edg = boost::edges(m_graph);
859 if(std::find(edg.first, edg.second, it->second) != edg.second)
861 boost::remove_edge(it->second, m_graph);
870 void clear() final
override
874 for(
auto& edge : m_edges)
878 for(
auto& node : m_nodes)
889 void mark_dirty() final
override { m_dirty =
true; }
891 ~graph_base()
override { clear(); }
895#if defined(OSSIA_FREESTANDING)
896 ossia::small_vector<ossia::graph_node*, 16> m_node_list;
898 ossia::small_vector<ossia::graph_node*, 1024> m_node_list;
The node_base class.
Definition node.hpp:48
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
spdlog::logger & logger() noexcept
Where the errors will be logged. Default is stderr.
Definition context.cpp:118