OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
sound_mmap.hpp
1#pragma once
2#include <ossia/audio/audio_parameter.hpp>
3#include <ossia/audio/drwav_handle.hpp>
4#include <ossia/dataflow/audio_stretch_mode.hpp>
5#include <ossia/dataflow/graph_node.hpp>
6#include <ossia/dataflow/nodes/media.hpp>
7#include <ossia/dataflow/nodes/sound.hpp>
8#include <ossia/dataflow/port.hpp>
9#include <ossia/detail/pod_vector.hpp>
10
11#include <type_traits>
12
13namespace ossia::nodes
14{
15
16class sound_mmap final : public ossia::sound_node
17{
18public:
19 sound_mmap() { m_outlets.push_back(&audio_out); }
20
21 ~sound_mmap() = default;
22
23 std::string label() const noexcept override { return "sound_mmap"; }
24
25 void set_start(std::size_t v) { start = v; }
26
27 void set_upmix(std::size_t v) { upmix = v; }
28
29 void set_sound(drwav_handle hdl)
30 {
31 using namespace snd;
32 m_handle = std::move(hdl);
33 if(m_handle)
34 {
35 switch(m_handle.translatedFormatTag())
36 {
37 case DR_WAVE_FORMAT_PCM: {
38 switch(m_handle.bitsPerSample())
39 {
40 case 8:
41 m_converter = read_u8;
42 break;
43 case 16:
44 m_converter = read_s16;
45 break;
46 case 24:
47 m_converter = read_s24;
48 break;
49 case 32:
50 m_converter = read_s32;
51 break;
52 }
53 break;
54 }
55 case DR_WAVE_FORMAT_IEEE_FLOAT: {
56 switch(m_handle.bitsPerSample())
57 {
58 case 32:
59 m_converter = read_f32;
60 break;
61 case 64:
62 m_converter = read_f64;
63 break;
64 }
65 break;
66 }
67 default:
68 m_converter = nullptr;
69 break;
70 }
71 }
72 }
73
74 void transport(time_value date) override
75 {
76 m_resampler.transport(to_sample(date, m_handle.sampleRate()));
77 }
78
79 void fetch_audio(
80 int64_t start, int64_t samples_to_write, double** audio_array_base) noexcept
81 {
82 const int channels = this->channels();
83 const int file_duration = this->duration();
84
85 m_resampleBuffer.resize(channels);
86 for(auto& buf : m_resampleBuffer)
87 buf.resize(samples_to_write);
88
89 float** audio_array = (float**)alloca(sizeof(float*) * channels);
90 for(int i = 0; i < channels; i++)
91 {
92 m_resampleBuffer[i].resize(samples_to_write);
93 audio_array[i] = m_resampleBuffer[i].data();
94 }
95
96 ossia::mutable_audio_span<float> source(channels);
97
98 double* frame_data{};
99 if(samples_to_write * channels > 10000)
100 {
101 m_safetyBuffer.resize(samples_to_write * channels);
102 frame_data = m_safetyBuffer.data();
103 // TODO detect if we happen to be in this case often, and if so, garbage
104 // collect at some point
105 }
106 else
107 {
108 frame_data = (double*)alloca(sizeof(double) * samples_to_write * channels);
109 }
110
111 if(m_loops)
112 {
113 for(int k = 0; k < samples_to_write; k++)
114 {
115 // TODO add a special case if [0; samples_to_write] don't loop around
116 int pos = this->m_start_offset_samples
117 + ((start + k) % this->m_loop_duration_samples);
118 if(pos >= file_duration)
119 {
120 for(int i = 0; i < channels; i++)
121 audio_array[i][k] = 0;
122 continue;
123 }
124
125 const bool ok = this->m_handle.seek_to_pcm_frame(pos);
126 if(!ok)
127 {
128 for(int i = 0; i < channels; i++)
129 audio_array[i][k] = 0;
130 continue;
131 }
132
133 const int max = 1;
134 const auto count = this->m_handle.read_pcm_frames(max, frame_data);
135 if(count >= 0)
136 {
137 for(int i = 0; i < channels; i++)
138 source[i] = tcb::span(audio_array[i] + k, count);
139 m_converter(source, frame_data, count);
140 }
141 else
142 {
143 for(int i = 0; i < channels; i++)
144 audio_array[i][k] = 0;
145 }
146 }
147 }
148 else
149 {
150 for(int i = 0; i < channels; i++)
151 {
152 source[i] = tcb::span(audio_array[i], samples_to_write);
153 }
154
155 bool ok = start + m_start_offset_samples < file_duration;
156 if(ok)
157 ok = ok && this->m_handle.seek_to_pcm_frame(start + m_start_offset_samples);
158
159 if(ok)
160 {
161 const auto count = this->m_handle.read_pcm_frames(samples_to_write, frame_data);
162 m_converter(source, frame_data, count);
163 for(int i = 0; i < channels; i++)
164 for(int k = count; k < samples_to_write; k++)
165 audio_array[i][k] = 0;
166 }
167 else
168 {
169 for(int i = 0; i < channels; i++)
170 for(int k = 0; k < samples_to_write; k++)
171 audio_array[i][k] = 0;
172 return;
173 }
174 }
175
176 for(int i = 0; i < channels; i++)
177 std::copy_n(audio_array[i], samples_to_write, audio_array_base[i]);
178 }
179
180 void fetch_audio(int64_t start, int64_t samples_to_write, float** audio_array) noexcept
181 {
182 const int channels = this->channels();
183 const int file_duration = this->duration();
184
185 ossia::mutable_audio_span<float> source(channels);
186
187 double* frame_data{};
188 if(samples_to_write * channels > 10000)
189 {
190 m_safetyBuffer.resize(samples_to_write * channels);
191 frame_data = m_safetyBuffer.data();
192 // TODO detect if we happen to be in this case often, and if so, garbage
193 // collect at some point
194 }
195 else
196 {
197 frame_data = (double*)alloca(sizeof(double) * samples_to_write * channels);
198 }
199
200 if(m_loops)
201 {
202 for(int k = 0; k < samples_to_write; k++)
203 {
204 // TODO add a special case if [0; samples_to_write] don't loop around
205 int pos = this->m_start_offset_samples
206 + ((start + k) % this->m_loop_duration_samples);
207 if(pos >= file_duration)
208 {
209 for(int i = 0; i < channels; i++)
210 audio_array[i][k] = 0;
211 continue;
212 }
213
214 const bool ok = this->m_handle.seek_to_pcm_frame(pos);
215 if(!ok)
216 {
217 for(int i = 0; i < channels; i++)
218 audio_array[i][k] = 0;
219 continue;
220 }
221
222 const int max = 1;
223 const auto count = this->m_handle.read_pcm_frames(max, frame_data);
224 if(count >= 0)
225 {
226 for(int i = 0; i < channels; i++)
227 source[i] = tcb::span(audio_array[i] + k, count);
228 m_converter(source, frame_data, count);
229 }
230 else
231 {
232 for(int i = 0; i < channels; i++)
233 audio_array[i][k] = 0;
234 }
235 }
236 }
237 else
238 {
239 for(int i = 0; i < channels; i++)
240 {
241 source[i] = tcb::span(audio_array[i], samples_to_write);
242 }
243
244 const bool ok = this->m_handle.seek_to_pcm_frame(start + m_start_offset_samples);
245 if(!ok)
246 {
247 for(int i = 0; i < channels; i++)
248 for(int k = 0; k < samples_to_write; k++)
249 audio_array[i][k] = 0;
250 return;
251 }
252
253 const auto count = this->m_handle.read_pcm_frames(samples_to_write, frame_data);
254 m_converter(source, frame_data, count);
255 for(int i = 0; i < channels; i++)
256 for(int k = count; k < samples_to_write; k++)
257 audio_array[i][k] = 0;
258 }
259 }
260
261 void run(const ossia::token_request& t, ossia::exec_state_facade e) noexcept override
262 {
263 if(!m_handle || !m_converter)
264 return;
265
266 // TODO do the backwards play head
267 if(!t.forward())
268 return;
269
270 const auto channels = m_handle.channels();
271 const auto len = m_handle.totalPCMFrameCount();
272
273 ossia::audio_port& ap = *audio_out;
274 ap.set_channels(std::max((std::size_t)upmix, (std::size_t)channels));
275
276 const auto [samples_to_read, samples_to_write]
277 = snd::sample_info(e.bufferSize(), e.modelToSamples(), t);
278 if(samples_to_write <= 0)
279 return;
280
281 assert(samples_to_write > 0);
282
283 const auto samples_offset = t.physical_start(e.modelToSamples());
284 if(t.tempo > 0)
285 {
286 if(t.prev_date < m_prev_date)
287 {
288 // Sentinel: we never played.
289 if(m_prev_date == ossia::time_value{ossia::time_value::infinite_min})
290 {
291 if(t.prev_date != 0_tv)
292 {
293 transport(t.prev_date);
294 }
295 else
296 {
297 // Otherwise we don't need transport, everything is already at 0
298 m_prev_date = 0_tv;
299 }
300 }
301 else
302 {
303 transport(t.prev_date);
304 }
305 }
306
307 for(std::size_t chan = 0; chan < channels; chan++)
308 {
309 ap.channel(chan).resize(e.bufferSize());
310 }
311
312 double stretch_ratio = update_stretch(t, e);
313
314 // Resample
315 m_resampler.run(
316 *this, t, e, stretch_ratio, channels, len, samples_to_read, samples_to_write,
317 samples_offset, ap);
318
319 for(std::size_t chan = 0; chan < channels; chan++)
320 {
321 // fade
322 snd::do_fade(
323 t.start_discontinuous, t.end_discontinuous, ap.channel(chan), samples_offset,
324 samples_to_write);
325 }
326
327 ossia::snd::perform_upmix(this->upmix, channels, ap);
328 ossia::snd::perform_start_offset(this->start, ap);
329
330 m_prev_date = t.date;
331 }
332 else
333 {
334 /* TODO */
335 }
336 }
337
338 [[nodiscard]] std::size_t channels() const
339 {
340 return m_handle ? m_handle.channels() : 0;
341 }
342 [[nodiscard]] std::size_t duration() const
343 {
344 return m_handle ? m_handle.totalPCMFrameCount() : 0;
345 }
346
347private:
348 drwav_handle m_handle{};
349
350 ossia::audio_outlet audio_out;
351
352 std::size_t start{};
353 std::size_t upmix{};
354
355 using read_fn_t
356 = void (*)(ossia::mutable_audio_span<float>& ap, void* data, int64_t samples);
357 read_fn_t m_converter{};
358 std::vector<double> m_safetyBuffer;
359 std::vector<std::vector<float>> m_resampleBuffer;
360};
361
362}
OSSIA_INLINE constexpr auto max(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
max function tailored for values
Definition math.hpp:96
The time_value class.
Definition ossia/editor/scenario/time_value.hpp:30