OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
graph_utils.hpp
1#pragma once
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>
19
20#include <boost/graph/adjacency_list.hpp>
21#include <boost/predef.h>
22// broken due to dynamic_property_map requiring rtti...
23// #include <boost/graph/graphviz.hpp>
24#include <boost/graph/topological_sort.hpp>
25
26#include <ankerl/unordered_dense.h>
27
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
31#endif
32
33class DataflowTest;
34namespace ossia
35{
36using graph_t = boost::adjacency_list<
37 boost::smallvecS, boost::smallvecS, boost::directedS, node_ptr,
38 std::shared_ptr<graph_edge>>;
39}
40namespace boost::detail
41{
42
43template <>
44class stored_edge_property<unsigned long, std::shared_ptr<ossia::graph_edge>>
45 : public stored_edge<unsigned long>
46{
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;
51
52public:
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))
58 {
59 }
60 stored_edge_property(self&& x) noexcept
61 : Base(static_cast<Base&&>(x))
62 , m_property(std::move(x.m_property))
63 {
64 }
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))
68 {
69 }
70 self& operator=(self&& x)
71 {
72 // NOTE: avoid 'Base::operator=(x);' broken on SGI MIPSpro (bug
73 // 55771 of Mozilla).
74 static_cast<Base&>(*this) = static_cast<Base&&>(x);
75 m_property = std::move(x.m_property);
76 return *this;
77 }
78 self& operator=(self const& x)
79 {
80 // NOTE: avoid 'Base::operator=(x);' broken on SGI MIPSpro (bug
81 // 55771 of Mozilla).
82 static_cast<Base&>(*this) = static_cast<Base const&>(x);
83 m_property = std::move(const_cast<self&>(x).m_property);
84 return *this;
85 }
86 inline Property& get_property() noexcept { return m_property; }
87 inline const Property& get_property() const noexcept { return m_property; }
88
89protected:
90 Property m_property;
91};
92
93}
94
95namespace boost
96{
97namespace detail
98{
99template <typename G>
100using DFSVertexInfo = std::pair<
101 typename graph_traits<G>::vertex_descriptor,
102 std::pair<
103 boost::optional<typename graph_traits<G>::edge_descriptor>,
104 std::pair<
105 typename graph_traits<G>::out_edge_iterator,
106 typename graph_traits<G>::out_edge_iterator>>>;
107
108template <class IncidenceGraph, 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)
112{
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>>>
124 VertexInfo;
125
126 boost::optional<Edge> src_e;
127 Iter ei, ei_end;
128
129 // Possible optimization for vector
130 stack.clear();
131 stack.reserve(num_vertices(g));
132
133 put(color, u, Color::gray());
134 vis.discover_vertex(u, g);
135 boost::tie(ei, ei_end) = out_edges(u, g);
136 if(func(u, g))
137 {
138 // If this vertex terminates the search, we push empty range
139 stack.push_back(std::make_pair(
140 u, std::make_pair(boost::optional<Edge>(), std::make_pair(ei_end, ei_end))));
141 }
142 else
143 {
144 stack.push_back(std::make_pair(
145 u, std::make_pair(boost::optional<Edge>(), std::make_pair(ei, ei_end))));
146 }
147 while(!stack.empty())
148 {
149 VertexInfo& back = stack.back();
150 u = back.first;
151 src_e = back.second.first;
152 boost::tie(ei, ei_end) = back.second.second;
153 stack.pop_back();
154 // finish_edge has to be called here, not after the
155 // loop. Think of the pop as the return from a recursive call.
156 if(src_e)
157 {
158 call_finish_edge(vis, src_e.get(), g);
159 }
160 while(ei != ei_end)
161 {
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())
166 {
167 vis.tree_edge(*ei, g);
168 src_e = *ei;
169 stack.push_back(
170 std::make_pair(u, std::make_pair(src_e, std::make_pair(++ei, ei_end))));
171 u = v;
172 put(color, u, Color::gray());
173 vis.discover_vertex(u, g);
174 boost::tie(ei, ei_end) = out_edges(u, g);
175 if(func(u, g))
176 {
177 ei = ei_end;
178 }
179 }
180 else
181 {
182 if(v_color == Color::gray())
183 {
184 vis.back_edge(*ei, g);
185 }
186 else
187 {
188 vis.forward_or_cross_edge(*ei, g);
189 }
190 call_finish_edge(vis, *ei, g);
191 ++ei;
192 }
193 }
194 put(color, u, Color::black());
195 vis.finish_vertex(u, g);
196 }
197}
198}
199
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)
205{
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;
210
211 typename graph_traits<VertexListGraph>::vertex_iterator ui, ui_end;
212 for(boost::tie(ui, ui_end) = vertices(g); ui != ui_end; ++ui)
213 {
214 Vertex u = implicit_cast<Vertex>(*ui);
215 put(color, u, Color::white());
216 vis.initialize_vertex(u, g);
217 }
218
219 if(start_vertex != detail::get_default_starting_vertex(g))
220 {
221 vis.start_vertex(start_vertex, g);
222 detail::custom_depth_first_visit_impl(g, start_vertex, vis, color, stack);
223 }
224
225 for(boost::tie(ui, ui_end) = vertices(g); ui != ui_end; ++ui)
226 {
227 Vertex u = implicit_cast<Vertex>(*ui);
228 ColorValue u_color = get(color, u);
229 if(u_color == Color::white())
230 {
231 vis.start_vertex(u, g);
232 detail::custom_depth_first_visit_impl(g, u, vis, color, stack);
233 }
234 }
235}
236
237}
238namespace ossia
239{
240inline void remove_vertex(typename graph_t::vertex_descriptor v, graph_t& g)
241{
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());
245}
246
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)
252{
253 color_map.clear();
254 color_map.resize(boost::num_vertices(g));
255
256 auto map = boost::make_iterator_property_map(
257 color_map.begin(), boost::get(boost::vertex_index, g), color_map[0]);
258
259 boost::custom_depth_first_search(
260 g, boost::topo_sort_visitor<OutputIterator>(result), map,
261 boost::detail::get_default_starting_vertex(g), stack);
262
263 // depth_first_search(g, params.visitor(TopoVisitor(result)));
264}
265
266using graph_vertex_t = graph_t::vertex_descriptor;
267using graph_edge_t = graph_t::edge_descriptor;
268
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>>
275#else
276 ossia::small_vector<std::pair<std::shared_ptr<T>, V>, 1024>
277#endif
278 >;
279#else
280template <typename T, typename V>
281using dense_shared_ptr_map
282 = ossia::hash_map<std::shared_ptr<T>, V, egur_hash, pointer_equal>;
283#endif
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>;
286
287using node_flat_set = ossia::flat_set<graph_node*>;
288enum class node_ordering
289{
290 topological,
291 temporal,
292 hierarchical
293};
294
295template <typename T>
296auto apply_con(const T& visitor, ossia::connection& con)
297{
298 auto tgt = con.target();
299 switch(con.which().index())
300 {
301 case ossia::connection::index_of<immediate_glutton_connection>().index():
302 return visitor(*reinterpret_cast<immediate_glutton_connection*>(tgt));
303 break;
304 case ossia::connection::index_of<immediate_strict_connection>().index():
305 return visitor(*reinterpret_cast<immediate_strict_connection*>(tgt));
306 break;
307 case ossia::connection::index_of<delayed_glutton_connection>().index():
308 return visitor(*reinterpret_cast<delayed_glutton_connection*>(tgt));
309 break;
310 case ossia::connection::index_of<delayed_strict_connection>().index():
311 return visitor(*reinterpret_cast<delayed_strict_connection*>(tgt));
312 break;
313 case ossia::connection::index_of<dependency_connection>().index():
314 return visitor(*reinterpret_cast<dependency_connection*>(tgt));
315 break;
316 default:
317 return visitor();
318 break;
319 }
320}
321template <typename T>
322auto apply_con(const T& visitor, const ossia::connection& con)
323{
324 auto tgt = con.target();
325 switch(con.which().index())
326 {
327 case ossia::connection::index_of<immediate_glutton_connection>().index():
328 return visitor(*reinterpret_cast<const immediate_glutton_connection*>(tgt));
329 break;
330 case ossia::connection::index_of<immediate_strict_connection>().index():
331 return visitor(*reinterpret_cast<const immediate_strict_connection*>(tgt));
332 break;
333 case ossia::connection::index_of<delayed_glutton_connection>().index():
334 return visitor(*reinterpret_cast<const delayed_glutton_connection*>(tgt));
335 break;
336 case ossia::connection::index_of<delayed_strict_connection>().index():
337 return visitor(*reinterpret_cast<const delayed_strict_connection*>(tgt));
338 break;
339 case ossia::connection::index_of<dependency_connection>().index():
340 return visitor(*reinterpret_cast<const dependency_connection*>(tgt));
341 break;
342 default:
343 return visitor();
344 break;
345 }
346}
347
348template <typename Graph_T, typename IO>
349void print_graph(Graph_T& g, IO& stream)
350{
351#if 0
352 std::stringstream s;
353 boost::write_graphviz(
354 s, g,
355 [&](auto& out, auto v) {
356 if(g[v] && !g[v]->label().empty())
357 out << "[label=\"" << g[v]->label() << "\"]";
358 else
359 out << "[]";
360 },
361 [](auto&&...) {});
362
363 stream << s.str() << "\n";
364#endif
365}
366
367struct OSSIA_EXPORT graph_util
368{
369 static void pull_from_parameter(inlet& in, execution_state& e)
370 {
371 apply_to_destination(
372 in.address, e.exec_devices(),
373 [&](ossia::net::parameter_base* addr, bool) {
374 if(in.scope & port::scope_t::local)
375 {
376 e.find_and_copy(*addr, in);
377 }
378 else if(in.scope & port::scope_t::global)
379 {
380 e.copy_from_global(*addr, in);
381 }
382 },
383 [&](ossia::net::node_base* node, bool) { e.copy_from_global_node(*node, in); });
384 }
385
387 static void init_outlet(outlet& out, execution_state&)
388 {
389 out.visit(clear_data{});
390
391 out.pre_process();
392 }
393
394 static void init_inlet(inlet& in, execution_state& e)
395 {
396 bool must_copy = in.sources.empty();
397
398 for(const graph_edge* edge : in.sources)
399 {
400 must_copy |= ossia::apply_con(init_must_copy_visitor{*edge}, edge->con);
401 }
402
403 if(must_copy)
404 pull_from_parameter(in, e);
405
406 for(auto edge : in.sources)
407 {
408 ossia::apply_con(init_node_visitor{in, *edge, e}, edge->con);
409 }
410
411 in.pre_process();
412 }
413
414 static void init_node(graph_node& n, execution_state& e)
415 {
416 if(n.not_fp_safe())
417 {
418 [[unlikely]];
419 disable_fpe();
420 }
421
422 // Clear the outputs of the node
423 for_each_outlet(n, [&](auto& port) { init_outlet(port, e); });
424
425 // Copy from environment and previous ports to inputs
426 for_each_inlet(n, [&](auto& port) { init_inlet(port, e); });
427 }
428
430 static void teardown_outlet(outlet& out, execution_state& e)
431 {
432 out.post_process();
433 bool must_copy = out.targets.empty();
434
435 // If some following glutton nodes aren't enabled, then we copy to the
436 // env.
437 for(const auto& tgt : out.targets)
438 {
439 must_copy |= ossia::apply_con(env_writer{out, *tgt}, tgt->con);
440 }
441
442 // if there are two outgoing glutton connections, one active, the other
443 // inactive then we want to copy through cable for the first one, and
444 // through env for the second one
445 if(must_copy)
446 out.write(e);
447 }
448
449 static void teardown_inlet(inlet& in, execution_state&)
450 {
451 in.post_process();
452 in.visit(clear_data{});
453 }
454
455 static void teardown_node(const graph_node& n, execution_state& e)
456 {
457 if(n.not_fp_safe())
458 {
459 [[unlikely]];
460 disable_fpe();
461 }
462
463 // Copy from output ports to environment
464 // Clear the outputs of the node
465 for_each_outlet(n, [&](auto& port) { teardown_outlet(port, e); });
466
467 // Copy from environment and previous ports to inputs
468 for_each_inlet(n, [&](auto& port) { teardown_inlet(port, e); });
469 }
470
471 /*
472 static void disable_strict_nodes_bfs(const graph_t& graph)
473 {
474 // while(Find a non-marked disabled node)
475 // Do a BFS from it
476 ossia::flat_map<graph_vertex_t, boost::two_bit_color_type> mark;
477 struct disable_visitor : public boost::default_bfs_visitor
478 {
479 void discover_vertex(graph_vertex_t vtx, graph_t& g) const
480 {
481 auto ptr = g[vtx].get();
482
483 }
484 };
485 }
486 */
487
488 static bool disable_strict_nodes(const graph_node* node)
489 {
490 if(node->muted())
491 return true;
492
493 auto test_disable_inlet = [&](const ossia::inlet& inlet) {
494 for(const auto& edge : inlet.sources)
495 {
496 assert(edge);
497 assert(edge->out_node);
498
499 if(immediate_strict_connection* sc
500 = edge->con.target<immediate_strict_connection>())
501 {
502 if((sc->required_sides
503 & immediate_strict_connection::required_sides_t::outbound)
504 && node->enabled() && !edge->out_node->enabled())
505 {
506 return true;
507 }
508 }
509 else if(
510 delayed_strict_connection* delay
511 = edge->con.target<delayed_strict_connection>())
512 {
513 const auto n = ossia::apply(data_size{}, delay->buffer);
514 if(n == 0 || delay->pos >= n)
515 {
516 return true;
517 }
518 }
519 }
520 return false;
521 };
522
523 auto test_disable_outlet = [&](const ossia::outlet& outlet) {
524 for(const auto& edge : outlet.targets)
525 {
526 assert(edge->in_node);
527
528 if(auto sc = edge->con.target<immediate_strict_connection>())
529 {
530 if((sc->required_sides
531 & immediate_strict_connection::required_sides_t::inbound)
532 && node->enabled() && !edge->in_node->enabled())
533 {
534 return true;
535 }
536 }
537 }
538 return false;
539 };
540
541 if(any_of_inlet(*node, test_disable_inlet))
542 return true;
543 if(any_of_outlet(*node, test_disable_outlet))
544 return true;
545
546 return false;
547 }
548
549 static void
550 disable_strict_nodes(const node_flat_set& enabled_nodes, node_flat_set& ret)
551 {
552 for(graph_node* node : enabled_nodes)
553 {
554 if(disable_strict_nodes(node))
555 ret.insert(node);
556 }
557 }
558
559 static void disable_strict_nodes_rec(
560 node_flat_set& cur_enabled_node, node_flat_set& disabled_cache)
561 {
562 do
563 {
564 disabled_cache.clear();
565 disable_strict_nodes(cur_enabled_node, disabled_cache);
566 for(ossia::graph_node* n : disabled_cache)
567 {
568 n->disable();
569
570 cur_enabled_node.erase(n);
571 }
572
573 } while(!disabled_cache.empty());
574 }
575
576 static void log_inputs(const graph_node&, ossia::logger_type& logger);
577 static void log_outputs(const graph_node&, ossia::logger_type& logger);
578
579 static void check_inputs(const graph_node&, execution_state& e);
580 static void
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);
583
584#if !defined(NDEBUG)
585#define OSSIA_DEBUG_MISBEHAVING_NODES 1
586#endif
587 static void exec_node(graph_node& first_node, execution_state& e)
588 {
589 init_node(first_node, e);
590
591#if defined(OSSIA_DEBUG_MISBEHAVING_NODES)
592 check_inputs(first_node, e);
593#endif
594
595 if(!first_node.requested_tokens.empty())
596 {
597 if(first_node.start_discontinuous())
598 {
599 first_node.requested_tokens.front().start_discontinuous = true;
600 first_node.set_start_discontinuous(false);
601 }
602 if(first_node.end_discontinuous())
603 {
604 first_node.requested_tokens.front().end_discontinuous = true;
605 first_node.set_end_discontinuous(false);
606 }
607 }
608
609 for(const auto& request : first_node.requested_tokens)
610 {
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);
615#endif
616 }
617
618 first_node.set_executed(true);
619 teardown_node(first_node, e);
620 }
621
622 static void
623 exec_node(graph_node& first_node, execution_state& e, ossia::logger_type& logger)
624 {
625 init_node(first_node, e);
626
627#if defined(OSSIA_DEBUG_MISBEHAVING_NODES)
628 check_inputs(first_node, e);
629#endif
630
631 if(!first_node.requested_tokens.empty())
632 {
633 if(first_node.start_discontinuous())
634 {
635 first_node.requested_tokens.front().start_discontinuous = true;
636 first_node.set_start_discontinuous(false);
637 }
638 if(first_node.end_discontinuous())
639 {
640 first_node.requested_tokens.front().end_discontinuous = true;
641 first_node.set_end_discontinuous(false);
642 }
643 }
644
645 log_inputs(first_node, logger);
646 for(const auto& request : first_node.requested_tokens)
647 {
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);
652#endif
653 }
654 log_outputs(first_node, logger);
655
656 first_node.set_executed(true);
657 teardown_node(first_node, e);
658 }
659
660 // These methods are only accessed by ossia::graph
661 static bool can_execute(graph_node& node, const execution_state&)
662 {
663 return all_of_inlet(node, [](const auto& inlet) {
664 // A port can execute if : - it has source ports and all its source
665 // ports have executed
666
667 // if there was a strict connection, this node would have been
668 // disabled so we can just do the following check.
669 bool previous_nodes_executed = ossia::all_of(inlet.sources, [&](graph_edge* edge) {
670 return edge->out_node->executed()
671 || (!edge->out_node->enabled() /* && bool(inlet->address) */
672 /* TODO check that it's in scope */);
673 });
674
675 // it does not have source ports ; we have to check for variables :
676 // find if address available in local / global scope
677 return !inlet.sources.empty() ? previous_nodes_executed : true; // TODO
678 });
679 }
680
681 static void finish_nodes(const node_map& nodes)
682 {
683 for(auto& node : nodes)
684 {
685 ossia::graph_node& n = *node.first;
686 n.set_executed(false);
687 n.disable();
688 }
689 }
690
691 template <typename DevicesT>
692 static auto find_address_connection(
693 ossia::graph_node& source, ossia::graph_node& sink, const DevicesT& devices)
694 {
695 return any_of_outlet(source, [&](auto& outlet) {
696 return any_of_inlet(sink, [&](auto& inlet) {
697 bool ok = false;
698 apply_to_destination(
699 outlet.address, devices,
700 [&](ossia::net::parameter_base* p1, bool) {
701 apply_to_destination(
702 inlet.address, devices,
703 [&](ossia::net::parameter_base* p2, bool) {
704 if(p1 == p2)
705 ok = true;
706 },
707 do_nothing_for_nodes{});
708 },
709 do_nothing_for_nodes{});
710 return ok;
711 });
712 });
713 }
714};
715
716struct OSSIA_EXPORT graph_base : graph_interface
717{
718 graph_base() noexcept
719 {
720#if !defined(OSSIA_FREESTANDING)
721 m_nodes.reserve(1024);
722 m_node_list.reserve(1024);
723 m_edges.reserve(1024);
724#endif
725 }
726 [[nodiscard]] tcb::span<ossia::graph_node* const>
727 get_nodes() const noexcept final override
728 {
729 return tcb::span{m_node_list};
730 }
731
732 void recompute_maps()
733 {
734 // TODO we should instead mark it for cleaning and do it once per tick ?
735 m_nodes.clear();
736 m_edges.clear();
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)
740 {
741 graph_vertex_t k = *it;
742 node_ptr n = m_graph[k];
743 assert(n);
744
745 m_nodes.insert({n, k});
746 }
747
748 // https://svn.boost.org/trac10/ticket/5706#no1
749#if !defined(__clang__)
750#pragma GCC diagnostic push
751#pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
752#endif
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)
756 {
757 graph_edge_t k = *it;
758 edge_ptr n = m_graph[k];
759 assert(n);
760
761 m_edges.insert({n, k});
762 }
763#if !defined(__clang__)
764#pragma GCC diagnostic pop
765#endif
766 }
767
768 auto add_node_impl(node_ptr n)
769 {
770 // auto& bench = *ossia::bench_ptr();
771 // bench[n.get()];
772
773 auto vtx = boost::add_vertex(n, m_graph);
774 // m_nodes.insert({std::move(n), vtx});
775 m_node_list.push_back(n.get());
776 m_dirty = true;
777 recompute_maps();
778 return vtx;
779 }
780
781 void add_node(node_ptr n) final override
782 {
783 if(m_nodes.find(n) == m_nodes.end())
784 {
785 add_node_impl(std::move(n));
786 }
787 }
788
789 void remove_node(const node_ptr& n) final override
790 {
791 for_each_inlet(*n, [&](auto& port) {
792 auto s = port.sources;
793 for(auto edge : s)
794 disconnect(edge);
795 });
796 for_each_outlet(*n, [&](auto& port) {
797 auto s = port.targets;
798 for(auto edge : s)
799 disconnect(edge);
800 });
801
802 auto it = m_nodes.find(n);
803 if(it != m_nodes.end())
804 {
805 auto vtx = boost::vertices(m_graph);
806 if(std::find(vtx.first, vtx.second, it->second) != vtx.second)
807 {
808 boost::clear_vertex(it->second, m_graph);
809 remove_vertex(it->second, m_graph);
810
811 recompute_maps();
812 }
813 // no need to erase it since it won't be here after recompute_maps
814 }
815 ossia::remove_one(m_node_list, n.get());
816 m_dirty = true;
817 }
818
819 void connect(std::shared_ptr<graph_edge> edge) final override
820 {
821 if(edge)
822 {
823 edge->init();
824
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);
829 else
830 in_vtx = it1->second;
831
832 auto it2 = m_nodes.find(edge->out_node);
833 if(it2 == m_nodes.end())
834 out_vtx = add_node_impl(edge->out_node);
835 else
836 out_vtx = it2->second;
837
838 // TODO check that two edges can be added
839 boost::add_edge(in_vtx, out_vtx, edge, m_graph);
840 recompute_maps();
841 m_dirty = true;
842 }
843 }
844
845 void disconnect(const std::shared_ptr<graph_edge>& edge) final override
846 {
847 disconnect(edge.get());
848 }
849
850 void disconnect(graph_edge* edge) final override
851 {
852 if(edge)
853 {
854 edge->clear();
855 auto it = m_edges.find(edge);
856 if(it != m_edges.end())
857 {
858 auto edg = boost::edges(m_graph);
859 if(std::find(edg.first, edg.second, it->second) != edg.second)
860 {
861 boost::remove_edge(it->second, m_graph);
862 recompute_maps();
863 }
864 m_dirty = true;
865 // no need to erase it since it won't be here after recompute_maps
866 }
867 }
868 }
869
870 void clear() final override
871 {
872 // TODO clear all the connections, ports, etc, to ensure that there is no
873 // shared_ptr loop
874 for(auto& edge : m_edges)
875 {
876 edge.first->clear();
877 }
878 for(auto& node : m_nodes)
879 {
880 node.first->clear();
881 }
882 m_dirty = true;
883 m_nodes.clear();
884 m_node_list.clear();
885 m_edges.clear();
886 m_graph.clear();
887 }
888
889 void mark_dirty() final override { m_dirty = true; }
890
891 ~graph_base() override { clear(); }
892
893 node_map m_nodes;
894 edge_map m_edges;
895#if defined(OSSIA_FREESTANDING)
896 ossia::small_vector<ossia::graph_node*, 16> m_node_list;
897#else
898 ossia::small_vector<ossia::graph_node*, 1024> m_node_list;
899#endif
900
901 graph_t m_graph;
902
903 bool m_dirty{};
904};
905}
The node_base class.
Definition node.hpp:48
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
Definition git_info.h:7
spdlog::logger & logger() noexcept
Where the errors will be logged. Default is stderr.
Definition context.cpp:118