DSPatch v.10.2.4
Loading...
Searching...
No Matches
CircuitThread.h
1/******************************************************************************
2DSPatch - The Refreshingly Simple C++ Dataflow Framework
3Copyright (c) 2025, Marcus Tomlinson
4
5BSD 2-Clause License
6
7Redistribution and use in source and binary forms, with or without
8modification, are permitted provided that the following conditions are met:
9
101. Redistributions of source code must retain the above copyright notice, this
11 list of conditions and the following disclaimer.
12
132. Redistributions in binary form must reproduce the above copyright notice,
14 this list of conditions and the following disclaimer in the documentation
15 and/or other materials provided with the distribution.
16
17THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
21FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
23SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
24CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
25OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27******************************************************************************/
28
29#pragma once
30
31#include <dspatch/Component.h>
32
33#ifdef _WIN32
34#define WIN32_LEAN_AND_MEAN
35#include <windows.h>
36#undef WIN32_LEAN_AND_MEAN
37#endif
38
39#include <condition_variable>
40#include <thread>
41
42namespace DSPatch
43{
44namespace internal
45{
46
48
63class CircuitThread final
64{
65public:
66 NONCOPYABLE( CircuitThread );
67
68 inline CircuitThread() = default;
69
70 // cppcheck-suppress missingMemberCopy
72 {
73 }
74
75 inline ~CircuitThread()
76 {
77 Stop();
78 }
79
80 inline void Start( std::vector<DSPatch::Component*>* components, int bufferNo )
81 {
82 _components = components;
83 _bufferNo = bufferNo;
84
85 _stop = false;
86 _gotSync = false;
87
88 _thread = std::thread( &CircuitThread::_Run, this );
89 }
90
91 inline void Stop()
92 {
93 _stop = true;
94
95 Resume();
96
97 if ( _thread.joinable() )
98 {
99 _thread.join();
100 }
101 }
102
103 inline void Sync()
104 {
105 std::unique_lock<std::mutex> lock( _syncMutex );
106
107 if ( !_gotSync ) // if haven't already got sync
108 {
109 _syncCondt.wait( lock ); // wait for sync
110 }
111 }
112
113 inline void Resume()
114 {
115 _gotSync = false; // reset the sync flag
116 _resumeCondt.notify_all();
117 }
118
119 inline void SyncAndResume()
120 {
121 Sync();
122 Resume();
123 }
124
125private:
126 inline void _Run()
127 {
128#ifdef _WIN32
129 SetThreadPriority( GetCurrentThread(), THREAD_PRIORITY_HIGHEST );
130#else
131 sched_param sch_params;
132 sch_params.sched_priority = sched_get_priority_max( SCHED_RR );
133 pthread_setschedparam( pthread_self(), SCHED_RR, &sch_params );
134#endif
135
136 if ( _components )
137 {
138 while ( !_stop )
139 {
140 {
141 std::unique_lock<std::mutex> lock( _syncMutex );
142
143 _gotSync = true; // set the sync flag
144 _syncCondt.notify_all();
145 _resumeCondt.wait( lock ); // wait for resume
146 }
147
148 // cppcheck-suppress knownConditionTrueFalse
149 if ( !_stop )
150 {
151 // You might be thinking: Can't we have each thread start on a different component?
152
153 // Well no. Because bufferNo == bufferNo, in order to maintain synchronisation
154 // within the circuit, when a component wants to process its buffers in-order, it
155 // requires that every other in-order component in the system has not only
156 // processed its buffers in the same order, but has processed the same number of
157 // buffers too.
158
159 // E.g. 1,2,3 and 1,2,3. Not 1,2,3 and 2,3,1,2,3.
160
161 for ( auto component : *_components )
162 {
163 component->TickSeries( _bufferNo );
164 }
165 }
166 }
167 }
168 }
169
170 std::thread _thread;
171 std::vector<DSPatch::Component*>* _components = nullptr;
172 int _bufferNo = 0;
173 bool _stop = false;
174 bool _gotSync = false;
175 std::mutex _syncMutex;
176 std::condition_variable _resumeCondt, _syncCondt;
177};
178
179} // namespace internal
180} // namespace DSPatch
Thread class for asynchronously ticking circuit components.