Optimize automation-event process splitting
[ardour.git] / libs / ardour / ardour / graph.h
1 /*
2     Copyright (C) 2010 Paul Davis
3     Author: Torben Hohn
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18 */
19
20
21 #ifndef __ardour_graph_h__
22 #define __ardour_graph_h__
23
24 #include <list>
25 #include <set>
26 #include <vector>
27 #include <string>
28
29 #include <boost/shared_ptr.hpp>
30
31 #include <glib.h>
32
33 #include "pbd/semutils.h"
34
35 #include "ardour/libardour_visibility.h"
36 #include "ardour/types.h"
37 #include "ardour/audio_backend.h"
38 #include "ardour/session_handle.h"
39
40 namespace ARDOUR
41 {
42
43 class GraphNode;
44 class Graph;
45
46 class Route;
47 class Session;
48 class GraphEdges;
49
50 typedef boost::shared_ptr<GraphNode> node_ptr_t;
51
52 typedef std::list< node_ptr_t > node_list_t;
53 typedef std::set< node_ptr_t > node_set_t;
54
55 class LIBARDOUR_API Graph : public SessionHandleRef
56 {
57 public:
58         Graph (Session & session);
59
60         void trigger (GraphNode * n);
61         void rechain (boost::shared_ptr<RouteList>, GraphEdges const &);
62
63         void dump (int chain);
64         void dec_ref();
65
66         void helper_thread();
67
68         int process_routes (pframes_t nframes, samplepos_t start_sample, samplepos_t end_sample, bool& need_butler);
69
70         int routes_no_roll (pframes_t nframes, samplepos_t start_sample, samplepos_t end_sample, bool non_rt_pending );
71
72         void process_one_route (Route * route);
73
74         void clear_other_chain ();
75
76         bool in_process_thread () const;
77
78 protected:
79         virtual void session_going_away ();
80
81 private:
82         volatile bool        _threads_active;
83
84         void reset_thread_list ();
85         void drop_threads ();
86         void restart_cycle();
87         bool run_one();
88         void main_thread();
89         void prep();
90
91         node_list_t _nodes_rt[2];
92
93         node_list_t _init_trigger_list[2];
94
95         std::vector<GraphNode *> _trigger_queue;
96         pthread_mutex_t          _trigger_mutex;
97
98         PBD::Semaphore _execution_sem;
99
100         /** Signalled to start a run of the graph for a process callback */
101         PBD::Semaphore _callback_start_sem;
102         PBD::Semaphore _callback_done_sem;
103
104         /** The number of processing threads that are asleep */
105         volatile gint _execution_tokens;
106         /** The number of unprocessed nodes that do not feed any other node; updated during processing */
107         volatile gint _finished_refcount;
108         /** The initial number of nodes that do not feed any other node (for each chain) */
109         volatile gint _init_finished_refcount[2];
110
111         bool _graph_empty;
112
113         // chain swapping
114         Glib::Threads::Mutex  _swap_mutex;
115         Glib::Threads::Cond   _cleanup_cond;
116         volatile int _current_chain;
117         volatile int _pending_chain;
118         volatile int _setup_chain;
119
120         // parameter caches.
121         pframes_t  _process_nframes;
122         samplepos_t _process_start_sample;
123         samplepos_t _process_end_sample;
124         bool       _process_can_record;
125         bool       _process_non_rt_pending;
126
127         bool _process_noroll;
128         int  _process_retval;
129         bool _process_need_butler;
130
131         // enginer / thread connection
132         PBD::ScopedConnectionList engine_connections;
133         void engine_stopped ();
134 };
135
136 } // namespace
137
138 #endif /* __ardour_graph_h__ */