-
Notifications
You must be signed in to change notification settings - Fork 3
/
rnbo_data_loader.cpp
329 lines (267 loc) · 9.4 KB
/
rnbo_data_loader.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
#include "rnbo_data_loader.h"
#include <atomic>
#include <3rdparty/readerwriterqueue/readerwriterqueue.h>
namespace {
t_class *s_rnbo_data_loader_class = nullptr;
struct loader_audio_data {
size_t channels = 0;
size_t frames = 0;
RNBO::number samplerate = 0.0;
size_t bytes = 0;
char * data = nullptr;
loader_audio_data(size_t cs, size_t fms, RNBO::number sr) : channels(cs), frames(fms), samplerate(sr) {
data = reinterpret_cast<char *>(new float[channels * frames]);
}
~loader_audio_data() {
if (data) {
delete [] data;
}
}
};
t_symbol *ps_close;
t_symbol *ps_error;
t_symbol *ps_open;
const t_fourcc s_types[] = {
FOUR_CHAR_CODE('AIFF'),
FOUR_CHAR_CODE('AIFC'),
FOUR_CHAR_CODE('ULAW'), //next/sun typed by peak and others
FOUR_CHAR_CODE('WAVE'),
FOUR_CHAR_CODE('FLAC'),
FOUR_CHAR_CODE('Sd2f'),
FOUR_CHAR_CODE('QQQQ'), // or whatever IRCAM snd files are saved as...
FOUR_CHAR_CODE('BINA'), // or whatever IRCAM snd files are saved as...
FOUR_CHAR_CODE('NxTS'), //next/sun typed by soundhack
FOUR_CHAR_CODE('Mp3 '),
FOUR_CHAR_CODE('DATA'), // our own 'raw' files
FOUR_CHAR_CODE('M4a '),
FOUR_CHAR_CODE('CAF '),
FOUR_CHAR_CODE('wv64')
};
}
extern "C" {
using moodycamel::ReaderWriterQueue;
// Simplified "loader" object that can load audio data from a file or URL
// without using a Max buffer.
struct _rnbo_data_loader {
t_object obj;
RNBO::DataType::Type _type;
t_symbol *_last_requested;
t_object *_remote_resource;
std::atomic<loader_audio_data *> _newinfo;
loader_audio_data * _activeinfo;
ReaderWriterQueue<loader_audio_data *, 32> * _cleanup;
void * _cleanupqelem;
};
t_rnbo_data_loader *rnbo_data_loader_new(RNBO::DataType::Type type);
void rnbo_data_loader_free(t_rnbo_data_loader *x);
void rnbo_data_loader_drain(t_rnbo_data_loader *x);
t_max_err rnbo_data_loader_notify(t_rnbo_data_loader *loader, t_symbol *s, t_symbol *msg, void *sender, void *data);
static t_max_err rnbo_data_loader_storefile(
t_rnbo_data_loader *loader,
const char *key,
short vol,
const char *filename
) {
t_object *reader = (t_object *) object_new(CLASS_NOBOX, gensym("jsoundfile"));
t_max_err err = (t_max_err) object_method(reader, ps_open, vol, filename, 0, 0);
if (err != MAX_ERR_NONE) {
return err;
}
loader_audio_data * info = new loader_audio_data(
(t_atom_long) object_method(reader, gensym("getchannelcount")),
(t_atom_long) object_method(reader, gensym("getlength")),
(t_atom_long) object_method(reader, gensym("getsr")));
// Get the info
long sampleLength = info->channels * info->frames;
float *fdata = reinterpret_cast<float *>(info->data);
// Read in the data
object_method(reader, gensym("readfloats"), fdata, 0, info->frames, info->channels);
// Close it up
object_method(reader, ps_close);
object_free(reader);
// If we're 64 bit, then copy the data over
if (loader->_type == RNBO::DataType::Float64AudioBuffer) {
double *ddata = new double[info->channels * info->frames];
for (unsigned long i = 0; i < sampleLength; i++) {
ddata[i] = fdata[i];
}
delete [] fdata;
info->data = reinterpret_cast<char *>(ddata);
info->bytes = sampleLength * sizeof(double);
} else {
info->bytes = sampleLength * sizeof(float);
}
info = loader->_newinfo.exchange(info);
if (info != nullptr) {
delete info;
}
return MAX_ERR_NONE;
}
void rnbo_data_loader_register()
{
ps_close = gensym("close");
ps_error = gensym("error");
ps_open = gensym("open");
if (s_rnbo_data_loader_class)
return;
auto c = class_new("rnbo_data_loader", (method)rnbo_data_loader_new, (method)rnbo_data_loader_free, (long)sizeof(t_rnbo_data_loader), 0L, A_CANT, 0);
class_addmethod(c, (method)rnbo_data_loader_notify, "notify", A_CANT, 0);
class_register(CLASS_NOBOX, c);
s_rnbo_data_loader_class = c;
}
t_rnbo_data_loader *rnbo_data_loader_new(RNBO::DataType::Type type)
{
t_rnbo_data_loader *loader = (t_rnbo_data_loader *)object_alloc(s_rnbo_data_loader_class);
if (loader) {
loader->_type = type;
loader->_last_requested = nullptr;
loader->_activeinfo = nullptr;
loader->_newinfo = nullptr;
loader->_cleanup = new ReaderWriterQueue<loader_audio_data *, 32>(32);
loader->_cleanupqelem = qelem_new(loader, (method) rnbo_data_loader_drain);
}
return loader;
}
void rnbo_data_loader_drain(t_rnbo_data_loader *loader) {
if (loader->_cleanup) {
loader_audio_data * info;
while (loader->_cleanup->try_dequeue(info)) {
delete info;
}
}
}
void rnbo_data_loader_free(t_rnbo_data_loader *loader)
{
auto info = loader->_newinfo.exchange(nullptr);
if (info != nullptr) {
delete info;
}
if (loader->_activeinfo != nullptr) {
delete loader->_activeinfo;
}
if (loader->_remote_resource) {
object_free(loader->_remote_resource);
loader->_remote_resource = nullptr;
}
qelem_free(loader->_cleanupqelem);
rnbo_data_loader_drain(loader);
if (loader->_cleanup) {
delete loader->_cleanup;
}
}
static t_max_err rnbo_data_locatefile(const char *in_filename, short *out_path, char *out_filename) {
t_max_err err;
char pathBuffer[MAX_PATH_CHARS];
t_fourcc filetype;
strncpy_zero(pathBuffer, in_filename, MAX_PATH_CHARS);
err = locatefile_extended(pathBuffer, out_path, &filetype, s_types, sizeof(s_types) / sizeof(s_types[0]));
if (err == MAX_ERR_NONE) {
err = path_toabsolutesystempath(*out_path, pathBuffer, pathBuffer);
}
if (err == MAX_ERR_NONE) {
err = path_frompathname(pathBuffer , out_path, out_filename);
}
if (err != MAX_ERR_NONE) {
out_path = 0;
out_filename[0] = '\0';
}
return err;
}
// Attempt to read the audio file from disk and load it into memory, so that
// the contents of the audio file can be passed to RNBO
void rnbo_data_loader_file(t_rnbo_data_loader *loader, const char *filename)
{
// If there were a mechanism to cancel an HTTP request, we could do it here
loader->_last_requested = gensym(filename);
// Locate the file
char out_filename[MAX_PATH_CHARS];
short out_path;
t_max_err err = rnbo_data_locatefile(filename, &out_path, out_filename);
if (err == MAX_ERR_NONE) {
err = rnbo_data_loader_storefile(loader, filename, out_path, out_filename);
} else {
object_error(nullptr, "%s: can't open file", filename);
}
}
// Attempt to download the audio file from a remote URL, then load it so that
// the contents of the audio file can be passed to RNBO
void rnbo_data_loader_url(t_rnbo_data_loader *loader, const char *url)
{
static t_symbol *sym_download = gensym("download");
loader->_last_requested = gensym(url);
if (!loader->_remote_resource) {
loader->_remote_resource = (t_object *) object_new(CLASS_NOBOX, gensym("remote_resource"));
object_attach_byptr_register(loader, loader->_remote_resource, CLASS_NOBOX);
}
object_method(loader->_remote_resource, sym_download, loader->_last_requested);
}
t_max_err rnbo_data_loader_notify(t_rnbo_data_loader *loader, t_symbol *s, t_symbol *msg, void *sender, void *data)
{
static t_symbol *sym_complete = gensym("complete");
static t_symbol *sym_geturl = gensym("geturl");
static t_symbol *sym_filename = gensym("filename");
static t_symbol *sym_vol = gensym("vol");
static t_symbol *sym_remote_resource = gensym("remote_resource");
if (sender && object_classname_compare(sender, sym_remote_resource)) {
t_symbol *url = (t_symbol *) object_method(sender, sym_geturl);
bool showError = false;
// If the url doesn't match _last_requested, it means that RNBO has changed
// its mind about what resource should be loaded into the given data buffer,
// and we can just move on
if (url == loader->_last_requested) {
if (msg == sym_complete) {
t_symbol *filename = (t_symbol *) object_method(sender, sym_filename);
const short vol = (t_atom_long) object_method(sender, sym_vol);
if (filename != NULL && vol != -1) {
rnbo_data_loader_storefile(loader, url->s_name, vol, filename->s_name);
} else {
showError = true;
}
} else if (msg == ps_error) {
showError = true;
}
}
if (showError) object_error(nullptr, "Couldn't download file from %s", url ? url->s_name : "unknown url");
}
return MAX_ERR_NONE;
}
t_symbol *rnbo_data_loader_get_last_requested(t_rnbo_data_loader *loader)
{
return loader->_last_requested;
}
bool rnbo_path_is_url(const char *pathOrURL) {
return strstr(pathOrURL, "://") != nullptr;
}
void rnbo_data_loader_load(t_rnbo_data_loader *x, const char *pathorurl) {
if (rnbo_path_is_url(pathorurl)) {
rnbo_data_loader_url(x, pathorurl);
} else {
rnbo_data_loader_file(x, pathorurl);
}
}
}
namespace RNBO {
void DataLoaderHandoffData(
ExternalDataIndex dataRefIndex,
const ExternalDataRef* ref,
t_rnbo_data_loader *loader,
UpdateRefCallback updateDataRef,
ReleaseRefCallback releaseDataRef
) {
auto info = loader->_newinfo.exchange(nullptr);
if (info != nullptr && info != loader->_activeinfo) {
if (loader->_type == DataType::Float64AudioBuffer) {
Float64AudioBuffer newType(info->channels, info->samplerate);
updateDataRef(dataRefIndex, info->data, info->bytes, newType);
} else {
Float32AudioBuffer newType(info->channels, info->samplerate);
updateDataRef(dataRefIndex, info->data, info->bytes, newType);
}
if (loader->_activeinfo != nullptr) {
loader->_cleanup->try_enqueue(loader->_activeinfo);
qelem_set(loader->_cleanupqelem);
}
loader->_activeinfo = info;
}
}
};