OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
sound.hpp
1#pragma once
2#include <ossia/audio/fade.hpp>
3#include <ossia/dataflow/audio_stretch_mode.hpp>
4#include <ossia/dataflow/node_process.hpp>
5#include <ossia/dataflow/nodes/media.hpp>
6#include <ossia/dataflow/nodes/timestretch/raw_stretcher.hpp>
7#include <ossia/dataflow/nodes/timestretch/repitch_stretcher.hpp>
8#include <ossia/dataflow/nodes/timestretch/rubberband_stretcher.hpp>
9#include <ossia/dataflow/port.hpp>
10#include <ossia/dataflow/sample_to_float.hpp>
11#include <ossia/detail/pod_vector.hpp>
12#include <ossia/detail/variant.hpp>
13
14namespace ossia
15{
16namespace snd
17{
18struct sample_read_info
19{
20 int64_t samples_to_read{};
21 int64_t samples_to_write{};
22};
23
24inline auto
25sample_info(int64_t bufferSize, double durationRatio, const ossia::token_request& t)
26{
27 sample_read_info _;
28 if(t.paused())
29 return _;
30
31 if(t.speed == 0.0)
32 return _;
33
34 _.samples_to_read = t.physical_read_duration(durationRatio);
35 _.samples_to_write = std::min(
36 t.physical_write_duration(durationRatio),
37 t.safe_physical_write_duration(durationRatio, bufferSize));
38
39 return _;
40}
41
42inline void
43perform_upmix(const std::size_t upmix, const std::size_t chan, ossia::audio_port& ap)
44{
45 // Upmix
46 if(upmix != 0)
47 {
48 if(upmix < chan)
49 {
50 /* TODO
51 // Downmix
52 switch(upmix)
53 {
54 case 1:
55 {
56 for(std::size_t i = 1; i < chan; i++)
57 {
58 if(ap.channel(0).size() < ap.channel(i).size())
59 ap.channel(0).resize(ap.channel(i).size());
60
61 for(std::size_t j = 0; j < ap.channel(i).size(); j++)
62 ap.channel(0)[j] += ap.channel(i)[j];
63 }
64 }
65 default:
66 // TODO
67 break;
68 }
69 */
70 }
71 else if(upmix > chan)
72 {
73 switch(chan)
74 {
75 case 1: {
76 for(std::size_t chan = 1; chan < upmix; ++chan)
77 ap.channel(chan).assign(ap.channel(0).begin(), ap.channel(0).end());
78 break;
79 }
80 default:
81 // TODO
82 break;
83 }
84 }
85 }
86}
87
88inline void perform_start_offset(const std::size_t start, ossia::audio_port& ap)
89{
90 if(start != 0)
91 {
92 ap.get().insert(ap.get().begin(), start, ossia::audio_channel{});
93 }
94}
95}
96
97template <typename T>
98struct at_end
99{
100 T func;
101 at_end(T t)
102 : func{t}
103 {
104 }
105 ~at_end() { func(); }
106};
107
108struct resampler
109{
110 enum
111 {
112 RawStretcher = 0,
113 RubberbandStretcher = 1,
114 RepitchStretcher = 2
115 };
116 [[nodiscard]] int64_t next_sample_to_read() const noexcept
117 {
118 return ossia::visit(
119 [](auto& stretcher) noexcept { return stretcher.next_sample_to_read; },
120 m_stretch);
121 }
122
123 void transport(int64_t date)
124 {
125 ossia::visit(
126 [=](auto& stretcher) noexcept { return stretcher.transport(date); }, m_stretch);
127 }
128
129 void reset(
130 int64_t date, ossia::audio_stretch_mode mode, std::size_t channels,
131 std::size_t fileSampleRate)
132 {
133 // TODO use the date parameter to buffer ! else transport won't work
134 switch(mode)
135 {
136 default:
137 case ossia::audio_stretch_mode::None: {
138 if(auto s = ossia::get_if<RawStretcher>(&m_stretch))
139 {
140 s->transport(date);
141 }
142 else
143 {
144 m_stretch.emplace<RawStretcher>(date);
145 }
146 break;
147 }
148
149#if defined(OSSIA_ENABLE_RUBBERBAND)
150 case ossia::audio_stretch_mode::RubberBandStandard:
151 case ossia::audio_stretch_mode::RubberBandPercussive:
152 case ossia::audio_stretch_mode::RubberBandStandardHQ:
153 case ossia::audio_stretch_mode::RubberBandPercussiveHQ: {
154 const auto preset = get_rubberband_preset(mode);
155 if(auto s = ossia::get_if<RubberbandStretcher>(&m_stretch);
156 s && s->options == preset)
157 {
158 s->transport(date);
159 }
160 else
161 {
162 m_stretch.emplace<rubberband_stretcher>(
163 preset, channels, fileSampleRate, date);
164 }
165 break;
166 }
167#endif
168
169#if defined(OSSIA_ENABLE_LIBSAMPLERATE)
170 case ossia::audio_stretch_mode::Repitch:
171 case ossia::audio_stretch_mode::RepitchMediumQ:
172 case ossia::audio_stretch_mode::RepitchFastestQ: {
173 const auto preset = get_samplerate_preset(mode);
174 if(auto s = ossia::get_if<RepitchStretcher>(&m_stretch);
175 s && s->repitchers.size() == channels && s->preset == preset)
176 {
177 s->transport(date);
178 }
179 else
180 {
181 // FIXME why 1024 here ?!
182 m_stretch.emplace<repitch_stretcher>(preset, channels, 1024, date);
183 }
184 break;
185 }
186#endif
187 }
188 }
189
190 template <typename T>
191 void
192 run(T& audio_fetcher, const ossia::token_request& t, ossia::exec_state_facade e,
193 double tempo_ratio, std::size_t chan, std::size_t len, int64_t samples_to_read,
194 int64_t samples_to_write, int64_t samples_offset,
195 const ossia::mutable_audio_span<double>& ap)
196 {
197 ossia::visit(
198 [&](auto& stretcher) {
199 stretcher.run(
200 audio_fetcher, t, e, tempo_ratio, chan, len, samples_to_read, samples_to_write,
201 samples_offset, ap);
202 },
203 m_stretch);
204 }
205
206 [[nodiscard]] bool stretch() const noexcept { return m_stretch.index() != 0; }
207
208private:
209 ossia::variant<
210 raw_stretcher
211#if defined(OSSIA_ENABLE_RUBBERBAND)
212 ,
213 rubberband_stretcher
214#endif
215#if defined(OSSIA_ENABLE_LIBSAMPLERATE)
216 ,
217 repitch_stretcher
218#endif
219 >
220 m_stretch;
221};
222
223struct sound_processing_info
224{
225 time_value m_prev_date{time_value::infinite_min};
226
227 time_value m_loop_duration{};
228 time_value m_start_offset{};
229
230 double tempo{};
231
232 int64_t m_loop_duration_samples{};
233 int64_t m_start_offset_samples{};
234
235 ossia::resampler m_resampler{};
236
237 bool m_loops{};
238
239 void set_loop_info(
240 ossia::time_value loop_duration, ossia::time_value start_offset, bool loops)
241 {
242 m_loop_duration = loop_duration;
243 m_start_offset = start_offset;
244 m_loops = loops;
245 }
246
247 void set_resampler(ossia::resampler&& r)
248 {
249 auto date = m_resampler.next_sample_to_read();
250 m_resampler = std::move(r);
251 m_resampler.transport(date);
252 }
253
254 void set_native_tempo(double v) { tempo = v; }
255
256 double update_stretch(
257 const ossia::token_request& t, const ossia::exec_state_facade& e) noexcept
258 {
259 double stretch_ratio = 1.;
260 double model_ratio = 1.;
261 if(tempo != 0.)
262 {
263 if(m_resampler.stretch())
264 {
265 model_ratio = ossia::root_tempo / this->tempo;
266 stretch_ratio = this->tempo / t.tempo;
267 }
268 else
269 {
270 model_ratio = ossia::root_tempo / t.tempo;
271 }
272 }
273
274 m_loop_duration_samples = m_loop_duration.impl * e.modelToSamples() * model_ratio;
275 m_start_offset_samples = m_start_offset.impl * e.modelToSamples() * model_ratio;
276 return stretch_ratio;
277 }
278};
279
280class sound_node
281 : public ossia::nonowning_graph_node
282 , public sound_processing_info
283{
284public:
285 virtual void transport(time_value date) = 0;
286};
287
288class dummy_sound_node final : public sound_node
289{
290public:
291 ossia::audio_outlet audio_out;
292 dummy_sound_node()
293 {
294 // Add a dummy outlet so that interval can connect propagation to it
295 m_outlets.push_back(&audio_out);
296 }
297
298 std::string label() const noexcept override { return "dummy_sound_node"; }
299
300 void transport(time_value date) override { }
301
302 void run(const ossia::token_request& t, ossia::exec_state_facade e) noexcept override
303 {
304 }
305};
306
307#if defined(OSSIA_SCENARIO_DATAFLOW)
308class sound_process final : public ossia::node_process
309{
310public:
311 using ossia::node_process::node_process;
312
313protected:
314 void state(const ossia::token_request& req) override
315 {
316 // TODO here we should also pass the execution state so that we can
317 // leverage the timing info & transform loop_duration / start_offset in
318 // samples right here...
319 static_cast<sound_node&>(*this->node)
320 .set_loop_info(m_loop_duration, m_start_offset, m_loops);
321
322 // Start offset and looping are done manually inside the sound nodes
323 // since it is much more efficient in this case
324 // (see fetch_audio)
325 node->request(req);
326 }
327
328 void offset_impl(time_value date) override
329 {
330 static_cast<sound_node&>(*this->node).transport(date);
331 }
332 void transport_impl(time_value date) override
333 {
334 static_cast<sound_node&>(*this->node).transport(date);
335 }
336};
337#endif
338
339}
Definition git_info.h:7
The time_value class.
Definition ossia/editor/scenario/time_value.hpp:30