Chris@0
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
Chris@0
|
2
|
Chris@26
|
3 /*
|
Chris@26
|
4 VampyHost
|
Chris@26
|
5
|
Chris@26
|
6 Use Vamp audio analysis plugins in Python
|
Chris@26
|
7
|
Chris@26
|
8 Gyorgy Fazekas and Chris Cannam
|
Chris@26
|
9 Centre for Digital Music, Queen Mary, University of London
|
Chris@26
|
10 Copyright 2008-2014 Queen Mary, University of London
|
Chris@26
|
11
|
Chris@26
|
12 Permission is hereby granted, free of charge, to any person
|
Chris@26
|
13 obtaining a copy of this software and associated documentation
|
Chris@26
|
14 files (the "Software"), to deal in the Software without
|
Chris@26
|
15 restriction, including without limitation the rights to use, copy,
|
Chris@26
|
16 modify, merge, publish, distribute, sublicense, and/or sell copies
|
Chris@26
|
17 of the Software, and to permit persons to whom the Software is
|
Chris@26
|
18 furnished to do so, subject to the following conditions:
|
Chris@26
|
19
|
Chris@26
|
20 The above copyright notice and this permission notice shall be
|
Chris@26
|
21 included in all copies or substantial portions of the Software.
|
Chris@26
|
22
|
Chris@26
|
23 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
Chris@26
|
24 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
Chris@26
|
25 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
Chris@26
|
26 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
|
Chris@26
|
27 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
Chris@26
|
28 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
Chris@26
|
29 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
Chris@26
|
30
|
Chris@26
|
31 Except as contained in this notice, the names of the Centre for
|
Chris@26
|
32 Digital Music; Queen Mary, University of London; and the authors
|
Chris@26
|
33 shall not be used in advertising or otherwise to promote the sale,
|
Chris@26
|
34 use or other dealings in this Software without prior written
|
Chris@26
|
35 authorization.
|
Chris@26
|
36 */
|
Chris@26
|
37
|
Chris@31
|
38 #include "PyPluginObject.h"
|
Chris@14
|
39
|
Chris@14
|
40 // define a unique API pointer
|
Chris@27
|
41 #define PY_ARRAY_UNIQUE_SYMBOL VAMPYHOST_ARRAY_API
|
Chris@14
|
42 #define NPY_NO_DEPRECATED_API NPY_1_7_API_VERSION
|
Chris@31
|
43 #define NO_IMPORT_ARRAY
|
Chris@14
|
44 #include "numpy/arrayobject.h"
|
Chris@14
|
45
|
Chris@33
|
46 #include "structmember.h"
|
Chris@33
|
47
|
Chris@29
|
48 #include "VectorConversion.h"
|
Chris@16
|
49 #include "PyRealTime.h"
|
Chris@0
|
50
|
Chris@0
|
51 #include <string>
|
Chris@31
|
52 #include <vector>
|
Chris@33
|
53 #include <cstddef>
|
Chris@0
|
54
|
Chris@0
|
55 using namespace std;
|
Chris@0
|
56 using namespace Vamp;
|
Chris@0
|
57
|
Chris@2
|
58 PyDoc_STRVAR(xx_foo_doc, "Some description"); //!!!
|
Chris@0
|
59
|
Chris@28
|
60 //!!! todo: conv errors
|
Chris@28
|
61
|
Chris@31
|
62 static
|
Chris@21
|
63 PyPluginObject *
|
Chris@21
|
64 getPluginObject(PyObject *pyPluginHandle)
|
Chris@21
|
65 {
|
Chris@21
|
66 cerr << "getPluginObject" << endl;
|
Chris@0
|
67
|
Chris@21
|
68 PyPluginObject *pd = 0;
|
Chris@21
|
69 if (PyPlugin_Check(pyPluginHandle)) {
|
Chris@21
|
70 pd = (PyPluginObject *)pyPluginHandle;
|
Chris@16
|
71 }
|
Chris@16
|
72 if (!pd || !pd->plugin) {
|
Chris@16
|
73 PyErr_SetString(PyExc_AttributeError,
|
Chris@16
|
74 "Invalid or already deleted plugin handle.");
|
Chris@16
|
75 return 0;
|
Chris@0
|
76 } else {
|
Chris@16
|
77 return pd;
|
Chris@0
|
78 }
|
Chris@0
|
79 }
|
Chris@0
|
80
|
Chris@34
|
81 static
|
Chris@34
|
82 PyObject *
|
Chris@34
|
83 pystr(const string &s)
|
Chris@34
|
84 {
|
Chris@34
|
85 return PyString_FromString(s.c_str());
|
Chris@34
|
86 }
|
Chris@34
|
87
|
Chris@31
|
88 PyObject *
|
Chris@31
|
89 PyPluginObject_From_Plugin(Plugin *plugin)
|
Chris@0
|
90 {
|
Chris@31
|
91 PyPluginObject *pd =
|
Chris@31
|
92 (PyPluginObject *)PyType_GenericAlloc(&Plugin_Type, 0);
|
Chris@21
|
93 pd->plugin = plugin;
|
Chris@21
|
94 pd->isInitialised = false;
|
Chris@21
|
95 pd->channels = 0;
|
Chris@21
|
96 pd->blockSize = 0;
|
Chris@21
|
97 pd->stepSize = 0;
|
Chris@34
|
98
|
Chris@34
|
99 PyObject *infodict = PyDict_New();
|
Chris@34
|
100 PyDict_SetItemString
|
Chris@34
|
101 (infodict, "apiVersion", PyInt_FromLong(plugin->getVampApiVersion()));
|
Chris@34
|
102 PyDict_SetItemString
|
Chris@34
|
103 (infodict, "pluginVersion", PyInt_FromLong(plugin->getPluginVersion()));
|
Chris@34
|
104 PyDict_SetItemString
|
Chris@34
|
105 (infodict, "identifier", pystr(plugin->getIdentifier()));
|
Chris@34
|
106 PyDict_SetItemString
|
Chris@34
|
107 (infodict, "name", pystr(plugin->getName()));
|
Chris@34
|
108 PyDict_SetItemString
|
Chris@34
|
109 (infodict, "description", pystr(plugin->getDescription()));
|
Chris@34
|
110 PyDict_SetItemString
|
Chris@34
|
111 (infodict, "maker", pystr(plugin->getMaker()));
|
Chris@34
|
112 PyDict_SetItemString
|
Chris@34
|
113 (infodict, "copyright", pystr(plugin->getCopyright()));
|
Chris@34
|
114 pd->info = infodict;
|
Chris@34
|
115
|
Chris@34
|
116 pd->inputDomain = plugin->getInputDomain();
|
Chris@34
|
117
|
Chris@35
|
118 VectorConversion conv;
|
Chris@35
|
119
|
Chris@34
|
120 Plugin::ParameterList pl = plugin->getParameterDescriptors();
|
Chris@34
|
121 PyObject *params = PyList_New(pl.size());
|
Chris@34
|
122
|
Chris@34
|
123 for (int i = 0; i < (int)pl.size(); ++i) {
|
Chris@34
|
124 PyObject *paramdict = PyDict_New();
|
Chris@34
|
125 PyDict_SetItemString
|
Chris@34
|
126 (paramdict, "identifier", pystr(pl[i].identifier));
|
Chris@34
|
127 PyDict_SetItemString
|
Chris@34
|
128 (paramdict, "name", pystr(pl[i].name));
|
Chris@34
|
129 PyDict_SetItemString
|
Chris@34
|
130 (paramdict, "description", pystr(pl[i].description));
|
Chris@34
|
131 PyDict_SetItemString
|
Chris@34
|
132 (paramdict, "unit", pystr(pl[i].unit));
|
Chris@34
|
133 PyDict_SetItemString
|
Chris@34
|
134 (paramdict, "minValue", PyFloat_FromDouble(pl[i].minValue));
|
Chris@34
|
135 PyDict_SetItemString
|
Chris@34
|
136 (paramdict, "maxValue", PyFloat_FromDouble(pl[i].maxValue));
|
Chris@34
|
137 PyDict_SetItemString
|
Chris@34
|
138 (paramdict, "defaultValue", PyFloat_FromDouble(pl[i].defaultValue));
|
Chris@34
|
139 if (pl[i].isQuantized) {
|
Chris@34
|
140 PyDict_SetItemString
|
Chris@34
|
141 (paramdict, "isQuantized", Py_True);
|
Chris@34
|
142 PyDict_SetItemString
|
Chris@34
|
143 (paramdict, "quantizeStep", PyFloat_FromDouble(pl[i].quantizeStep));
|
Chris@34
|
144 if (!pl[i].valueNames.empty()) {
|
Chris@34
|
145 PyDict_SetItemString
|
Chris@34
|
146 (paramdict, "valueNames", conv.PyValue_From_StringVector(pl[i].valueNames));
|
Chris@34
|
147 }
|
Chris@34
|
148 } else {
|
Chris@34
|
149 PyDict_SetItemString
|
Chris@34
|
150 (paramdict, "isQuantized", Py_False);
|
Chris@34
|
151 }
|
Chris@34
|
152
|
Chris@34
|
153 PyList_SET_ITEM(params, i, paramdict);
|
Chris@34
|
154 }
|
Chris@34
|
155
|
Chris@34
|
156 pd->parameters = params;
|
Chris@35
|
157
|
Chris@35
|
158 Plugin::OutputList ol = plugin->getOutputDescriptors();
|
Chris@35
|
159 PyObject *outputs = PyList_New(ol.size());
|
Chris@35
|
160
|
Chris@35
|
161 for (int i = 0; i < (int)ol.size(); ++i) {
|
Chris@35
|
162 PyObject *outdict = PyDict_New();
|
Chris@35
|
163 PyDict_SetItemString
|
Chris@35
|
164 (outdict, "identifier", pystr(ol[i].identifier));
|
Chris@35
|
165 PyDict_SetItemString
|
Chris@35
|
166 (outdict, "name", pystr(ol[i].name));
|
Chris@35
|
167 PyDict_SetItemString
|
Chris@35
|
168 (outdict, "description", pystr(ol[i].description));
|
Chris@35
|
169 PyDict_SetItemString
|
Chris@35
|
170 (outdict, "binCount", PyInt_FromLong(ol[i].binCount));
|
Chris@35
|
171 if (ol[i].binCount > 0) {
|
Chris@35
|
172 if (ol[i].hasKnownExtents) {
|
Chris@35
|
173 PyDict_SetItemString
|
Chris@35
|
174 (outdict, "hasKnownExtents", Py_True);
|
Chris@35
|
175 PyDict_SetItemString
|
Chris@35
|
176 (outdict, "minValue", PyFloat_FromDouble(ol[i].minValue));
|
Chris@35
|
177 PyDict_SetItemString
|
Chris@35
|
178 (outdict, "maxValue", PyFloat_FromDouble(ol[i].maxValue));
|
Chris@35
|
179 } else {
|
Chris@35
|
180 PyDict_SetItemString
|
Chris@35
|
181 (outdict, "hasKnownExtents", Py_False);
|
Chris@35
|
182 }
|
Chris@35
|
183 if (ol[i].isQuantized) {
|
Chris@35
|
184 PyDict_SetItemString
|
Chris@35
|
185 (outdict, "isQuantized", Py_True);
|
Chris@35
|
186 PyDict_SetItemString
|
Chris@35
|
187 (outdict, "quantizeStep", PyFloat_FromDouble(ol[i].quantizeStep));
|
Chris@35
|
188 } else {
|
Chris@35
|
189 PyDict_SetItemString
|
Chris@35
|
190 (outdict, "isQuantized", Py_False);
|
Chris@35
|
191 }
|
Chris@35
|
192 }
|
Chris@35
|
193 PyDict_SetItemString
|
Chris@35
|
194 (outdict, "sampleType", PyInt_FromLong((int)ol[i].sampleType));
|
Chris@35
|
195 PyDict_SetItemString
|
Chris@35
|
196 (outdict, "sampleRate", PyFloat_FromDouble(ol[i].sampleRate));
|
Chris@35
|
197 PyDict_SetItemString
|
Chris@35
|
198 (outdict, "hasDuration", ol[i].hasDuration ? Py_True : Py_False);
|
Chris@35
|
199
|
Chris@35
|
200 PyList_SET_ITEM(outputs, i, outdict);
|
Chris@35
|
201 }
|
Chris@35
|
202
|
Chris@35
|
203 pd->outputs = outputs;
|
Chris@34
|
204
|
Chris@21
|
205 return (PyObject *)pd;
|
Chris@0
|
206 }
|
Chris@0
|
207
|
Chris@33
|
208 static void
|
Chris@33
|
209 PyPluginObject_dealloc(PyPluginObject *self)
|
Chris@33
|
210 {
|
Chris@33
|
211 cerr << "PyPluginObject_dealloc" << endl;
|
Chris@33
|
212 delete self->plugin;
|
Chris@33
|
213 PyObject_Del(self);
|
Chris@33
|
214 }
|
Chris@33
|
215
|
Chris@0
|
216 static PyObject *
|
Chris@0
|
217 vampyhost_initialise(PyObject *self, PyObject *args)
|
Chris@0
|
218 {
|
Chris@21
|
219 cerr << "vampyhost_initialise" << endl;
|
Chris@21
|
220
|
luis@7
|
221 size_t channels, blockSize, stepSize;
|
Chris@0
|
222
|
Chris@23
|
223 if (!PyArg_ParseTuple (args, "nnn",
|
luis@7
|
224 (size_t) &channels,
|
luis@7
|
225 (size_t) &stepSize,
|
Chris@23
|
226 (size_t) &blockSize)) {
|
Chris@0
|
227 PyErr_SetString(PyExc_TypeError,
|
Chris@23
|
228 "initialise() takes channel count, step size, and block size arguments");
|
Chris@16
|
229 return 0;
|
Chris@0
|
230 }
|
Chris@0
|
231
|
Chris@23
|
232 PyPluginObject *pd = getPluginObject(self);
|
Chris@16
|
233 if (!pd) return 0;
|
Chris@0
|
234
|
Chris@16
|
235 pd->channels = channels;
|
Chris@16
|
236 pd->stepSize = stepSize;
|
Chris@16
|
237 pd->blockSize = blockSize;
|
Chris@0
|
238
|
Chris@16
|
239 if (!pd->plugin->initialise(channels, stepSize, blockSize)) {
|
Chris@17
|
240 cerr << "Failed to initialise native plugin adapter with channels = " << channels << ", stepSize = " << stepSize << ", blockSize = " << blockSize << " and ADAPT_ALL_SAFE set" << endl;
|
Chris@0
|
241 PyErr_SetString(PyExc_TypeError,
|
Chris@17
|
242 "Plugin initialization failed");
|
Chris@16
|
243 return 0;
|
Chris@6
|
244 }
|
Chris@0
|
245
|
Chris@16
|
246 pd->isInitialised = true;
|
luis@7
|
247
|
Chris@0
|
248 return Py_True;
|
Chris@0
|
249 }
|
Chris@0
|
250
|
Chris@0
|
251 static PyObject *
|
Chris@23
|
252 vampyhost_reset(PyObject *self, PyObject *)
|
Chris@18
|
253 {
|
Chris@21
|
254 cerr << "vampyhost_reset" << endl;
|
Chris@21
|
255
|
Chris@23
|
256 PyPluginObject *pd = getPluginObject(self);
|
Chris@18
|
257 if (!pd) return 0;
|
Chris@18
|
258
|
Chris@18
|
259 if (!pd->isInitialised) {
|
Chris@18
|
260 PyErr_SetString(PyExc_StandardError,
|
Chris@18
|
261 "Plugin has not been initialised");
|
Chris@18
|
262 return 0;
|
Chris@18
|
263 }
|
Chris@18
|
264
|
Chris@18
|
265 pd->plugin->reset();
|
Chris@18
|
266 return Py_True;
|
Chris@18
|
267 }
|
Chris@18
|
268
|
Chris@18
|
269 static PyObject *
|
Chris@20
|
270 vampyhost_getParameter(PyObject *self, PyObject *args)
|
Chris@20
|
271 {
|
Chris@21
|
272 cerr << "vampyhost_getParameter" << endl;
|
Chris@21
|
273
|
Chris@20
|
274 PyObject *pyParam;
|
Chris@20
|
275
|
Chris@23
|
276 if (!PyArg_ParseTuple(args, "S", &pyParam)) {
|
Chris@20
|
277 PyErr_SetString(PyExc_TypeError,
|
Chris@23
|
278 "getParameter() takes parameter id (string) argument");
|
Chris@20
|
279 return 0; }
|
Chris@20
|
280
|
Chris@23
|
281 PyPluginObject *pd = getPluginObject(self);
|
Chris@20
|
282 if (!pd) return 0;
|
Chris@20
|
283
|
Chris@20
|
284 float value = pd->plugin->getParameter(PyString_AS_STRING(pyParam));
|
Chris@20
|
285 return PyFloat_FromDouble(double(value));
|
Chris@20
|
286 }
|
Chris@20
|
287
|
Chris@20
|
288 static PyObject *
|
Chris@20
|
289 vampyhost_setParameter(PyObject *self, PyObject *args)
|
Chris@20
|
290 {
|
Chris@21
|
291 cerr << "vampyhost_setParameter" << endl;
|
Chris@21
|
292
|
Chris@20
|
293 PyObject *pyParam;
|
Chris@20
|
294 float value;
|
Chris@20
|
295
|
Chris@23
|
296 if (!PyArg_ParseTuple(args, "Sf", &pyParam, &value)) {
|
Chris@20
|
297 PyErr_SetString(PyExc_TypeError,
|
Chris@23
|
298 "setParameter() takes parameter id (string), and value (float) arguments");
|
Chris@20
|
299 return 0; }
|
Chris@20
|
300
|
Chris@23
|
301 PyPluginObject *pd = getPluginObject(self);
|
Chris@20
|
302 if (!pd) return 0;
|
Chris@20
|
303
|
Chris@20
|
304 pd->plugin->setParameter(PyString_AS_STRING(pyParam), value);
|
Chris@20
|
305 return Py_True;
|
Chris@20
|
306 }
|
Chris@20
|
307
|
Chris@35
|
308 static
|
Chris@35
|
309 PyObject *
|
Chris@35
|
310 convertFeatureSet(const Plugin::FeatureSet &fs)
|
Chris@35
|
311 {
|
Chris@35
|
312 VectorConversion conv;
|
Chris@35
|
313
|
Chris@35
|
314 PyObject *pyFs = PyDict_New();
|
Chris@35
|
315
|
Chris@35
|
316 for (Plugin::FeatureSet::const_iterator fsi = fs.begin();
|
Chris@35
|
317 fsi != fs.end(); ++fsi) {
|
Chris@35
|
318
|
Chris@35
|
319 int fno = fsi->first;
|
Chris@35
|
320 const Plugin::FeatureList &fl = fsi->second;
|
Chris@35
|
321
|
Chris@35
|
322 if (!fl.empty()) {
|
Chris@35
|
323
|
Chris@35
|
324 PyObject *pyFl = PyList_New(fl.size());
|
Chris@35
|
325
|
Chris@35
|
326 for (int fli = 0; fli < (int)fl.size(); ++fli) {
|
Chris@35
|
327
|
Chris@35
|
328 const Plugin::Feature &f = fl[fli];
|
Chris@35
|
329 PyObject *pyF = PyDict_New();
|
Chris@35
|
330
|
Chris@35
|
331 if (f.hasTimestamp) {
|
Chris@35
|
332 PyDict_SetItemString
|
Chris@35
|
333 (pyF, "timestamp", PyRealTime_FromRealTime(f.timestamp));
|
Chris@35
|
334 }
|
Chris@35
|
335 if (f.hasDuration) {
|
Chris@35
|
336 PyDict_SetItemString
|
Chris@35
|
337 (pyF, "duration", PyRealTime_FromRealTime(f.duration));
|
Chris@35
|
338 }
|
Chris@35
|
339
|
Chris@35
|
340 PyDict_SetItemString
|
Chris@35
|
341 (pyF, "label", pystr(f.label));
|
Chris@35
|
342
|
Chris@35
|
343 if (!f.values.empty()) {
|
Chris@35
|
344 PyDict_SetItemString
|
Chris@35
|
345 (pyF, "values", conv.PyArray_From_FloatVector(f.values));
|
Chris@35
|
346 }
|
Chris@35
|
347
|
Chris@35
|
348 PyList_SET_ITEM(pyFl, fli, pyF);
|
Chris@35
|
349 }
|
Chris@35
|
350
|
Chris@35
|
351 PyObject *pyN = PyInt_FromLong(fno);
|
Chris@35
|
352 PyDict_SetItem(pyFs, pyN, pyFl);
|
Chris@35
|
353 }
|
Chris@35
|
354 }
|
Chris@35
|
355
|
Chris@35
|
356 return pyFs;
|
Chris@35
|
357 }
|
Chris@35
|
358
|
Chris@20
|
359 static PyObject *
|
Chris@0
|
360 vampyhost_process(PyObject *self, PyObject *args)
|
Chris@0
|
361 {
|
Chris@21
|
362 cerr << "vampyhost_process" << endl;
|
Chris@21
|
363
|
Chris@0
|
364 PyObject *pyBuffer;
|
Chris@0
|
365 PyObject *pyRealTime;
|
Chris@0
|
366
|
Chris@23
|
367 if (!PyArg_ParseTuple(args, "OO",
|
Chris@0
|
368 &pyBuffer, // Audio data
|
Chris@0
|
369 &pyRealTime)) { // TimeStamp
|
Chris@0
|
370 PyErr_SetString(PyExc_TypeError,
|
Chris@17
|
371 "process() takes plugin handle (object), buffer (2D array of channels * samples floats) and timestamp (RealTime) arguments");
|
Chris@16
|
372 return 0; }
|
Chris@0
|
373
|
Chris@0
|
374 if (!PyRealTime_Check(pyRealTime)) {
|
Chris@0
|
375 PyErr_SetString(PyExc_TypeError,"Valid timestamp required.");
|
Chris@16
|
376 return 0; }
|
Chris@0
|
377
|
Chris@17
|
378 if (!PyList_Check(pyBuffer)) {
|
Chris@17
|
379 PyErr_SetString(PyExc_TypeError, "List of NumPy Array required for process input.");
|
Chris@17
|
380 return 0;
|
Chris@17
|
381 }
|
Chris@17
|
382
|
Chris@23
|
383 PyPluginObject *pd = getPluginObject(self);
|
Chris@16
|
384 if (!pd) return 0;
|
Chris@0
|
385
|
Chris@0
|
386 if (!pd->isInitialised) {
|
Chris@0
|
387 PyErr_SetString(PyExc_StandardError,
|
Chris@0
|
388 "Plugin has not been initialised.");
|
Chris@16
|
389 return 0;
|
Chris@16
|
390 }
|
Chris@0
|
391
|
Chris@12
|
392 int channels = pd->channels;
|
Chris@0
|
393
|
Chris@4
|
394 if (PyList_GET_SIZE(pyBuffer) != channels) {
|
Chris@17
|
395 cerr << "Wrong number of channels: got " << PyList_GET_SIZE(pyBuffer) << ", expected " << channels << endl;
|
Chris@4
|
396 PyErr_SetString(PyExc_TypeError, "Wrong number of channels");
|
Chris@16
|
397 return 0;
|
Chris@4
|
398 }
|
Chris@0
|
399
|
Chris@4
|
400 float **inbuf = new float *[channels];
|
Chris@0
|
401
|
Chris@29
|
402 VectorConversion typeConv;
|
Chris@17
|
403
|
Chris@12
|
404 vector<vector<float> > data;
|
Chris@4
|
405 for (int c = 0; c < channels; ++c) {
|
Chris@4
|
406 PyObject *cbuf = PyList_GET_ITEM(pyBuffer, c);
|
Chris@17
|
407 data.push_back(typeConv.PyValue_To_FloatVector(cbuf));
|
Chris@12
|
408 }
|
Chris@12
|
409
|
Chris@12
|
410 for (int c = 0; c < channels; ++c) {
|
Chris@17
|
411 if (data[c].size() != pd->blockSize) {
|
Chris@17
|
412 cerr << "Wrong number of samples on channel " << c << ": expected " << pd->blockSize << " (plugin's block size), got " << data[c].size() << endl;
|
Chris@17
|
413 PyErr_SetString(PyExc_TypeError, "Wrong number of samples");
|
Chris@17
|
414 return 0;
|
Chris@17
|
415 }
|
Chris@12
|
416 inbuf[c] = &data[c][0];
|
Chris@4
|
417 }
|
Chris@0
|
418
|
Chris@12
|
419 RealTime timeStamp = *PyRealTime_AsRealTime(pyRealTime);
|
Chris@0
|
420
|
Chris@18
|
421 Plugin::FeatureSet fs = pd->plugin->process(inbuf, timeStamp);
|
Chris@0
|
422
|
Chris@4
|
423 delete[] inbuf;
|
Chris@0
|
424
|
Chris@35
|
425 return convertFeatureSet(fs);
|
Chris@35
|
426 }
|
Chris@0
|
427
|
Chris@35
|
428 static PyObject *
|
Chris@35
|
429 vampyhost_getRemainingFeatures(PyObject *self, PyObject *)
|
Chris@35
|
430 {
|
Chris@35
|
431 cerr << "vampyhost_getRemainingFeatures" << endl;
|
Chris@18
|
432
|
Chris@35
|
433 PyPluginObject *pd = getPluginObject(self);
|
Chris@35
|
434 if (!pd) return 0;
|
Chris@18
|
435
|
Chris@35
|
436 if (!pd->isInitialised) {
|
Chris@35
|
437 PyErr_SetString(PyExc_StandardError,
|
Chris@35
|
438 "Plugin has not been initialised.");
|
Chris@35
|
439 return 0;
|
Chris@35
|
440 }
|
Chris@18
|
441
|
Chris@35
|
442 Plugin::FeatureSet fs = pd->plugin->getRemainingFeatures();
|
Chris@18
|
443
|
Chris@35
|
444 return convertFeatureSet(fs);
|
Chris@0
|
445 }
|
Chris@0
|
446
|
Chris@23
|
447 static PyObject *
|
Chris@23
|
448 vampyhost_unload(PyObject *self, PyObject *)
|
Chris@23
|
449 {
|
Chris@23
|
450 cerr << "vampyhost_unloadPlugin" << endl;
|
Chris@23
|
451
|
Chris@23
|
452 PyPluginObject *pd = getPluginObject(self);
|
Chris@23
|
453 if (!pd) return 0;
|
Chris@23
|
454
|
Chris@23
|
455 delete pd->plugin;
|
Chris@32
|
456 pd->plugin = 0; // This is checked by getPluginObject, so we avoid
|
Chris@32
|
457 // blowing up if called repeatedly
|
Chris@23
|
458
|
Chris@23
|
459 return Py_True;
|
Chris@23
|
460 }
|
Chris@23
|
461
|
Chris@33
|
462 static PyMemberDef PyPluginObject_members[] =
|
Chris@33
|
463 {
|
Chris@34
|
464 {(char *)"info", T_OBJECT, offsetof(PyPluginObject, info), READONLY,
|
Chris@34
|
465 xx_foo_doc},
|
Chris@34
|
466
|
Chris@34
|
467 {(char *)"inputDomain", T_INT, offsetof(PyPluginObject, inputDomain), READONLY,
|
Chris@34
|
468 xx_foo_doc},
|
Chris@34
|
469
|
Chris@34
|
470 {(char *)"parameters", T_OBJECT, offsetof(PyPluginObject, parameters), READONLY,
|
Chris@34
|
471 xx_foo_doc},
|
Chris@35
|
472
|
Chris@35
|
473 {(char *)"outputs", T_OBJECT, offsetof(PyPluginObject, outputs), READONLY,
|
Chris@35
|
474 xx_foo_doc},
|
Chris@33
|
475
|
Chris@33
|
476 {0, 0}
|
Chris@33
|
477 };
|
Chris@33
|
478
|
Chris@21
|
479 static PyMethodDef PyPluginObject_methods[] =
|
Chris@21
|
480 {
|
Chris@23
|
481 {"getParameter", vampyhost_getParameter, METH_VARARGS,
|
Chris@23
|
482 xx_foo_doc}, //!!! fix all these!
|
Chris@23
|
483
|
Chris@23
|
484 {"setParameter", vampyhost_setParameter, METH_VARARGS,
|
Chris@23
|
485 xx_foo_doc},
|
Chris@33
|
486
|
Chris@23
|
487 {"initialise", vampyhost_initialise, METH_VARARGS,
|
Chris@23
|
488 xx_foo_doc},
|
Chris@23
|
489
|
Chris@23
|
490 {"reset", vampyhost_reset, METH_NOARGS,
|
Chris@23
|
491 xx_foo_doc},
|
Chris@23
|
492
|
Chris@23
|
493 {"process", vampyhost_process, METH_VARARGS,
|
Chris@23
|
494 xx_foo_doc},
|
Chris@23
|
495
|
Chris@35
|
496 {"getRemainingFeatures", vampyhost_getRemainingFeatures, METH_NOARGS,
|
Chris@35
|
497 xx_foo_doc},
|
Chris@35
|
498
|
Chris@23
|
499 {"unload", vampyhost_unload, METH_NOARGS,
|
Chris@23
|
500 xx_foo_doc},
|
Chris@23
|
501
|
Chris@21
|
502 {0, 0}
|
Chris@21
|
503 };
|
Chris@21
|
504
|
Chris@21
|
505 /* Doc:: 10.3 Type Objects */ /* static */
|
Chris@21
|
506 PyTypeObject Plugin_Type =
|
Chris@21
|
507 {
|
Chris@21
|
508 PyObject_HEAD_INIT(NULL)
|
Chris@21
|
509 0, /*ob_size*/
|
Chris@21
|
510 "vampyhost.Plugin", /*tp_name*/
|
Chris@21
|
511 sizeof(PyPluginObject), /*tp_basicsize*/
|
Chris@21
|
512 0, /*tp_itemsize*/
|
Chris@21
|
513 (destructor)PyPluginObject_dealloc, /*tp_dealloc*/
|
Chris@21
|
514 0, /*tp_print*/
|
Chris@33
|
515 0, /*tp_getattr*/
|
Chris@33
|
516 0, /*tp_setattr*/
|
Chris@21
|
517 0, /*tp_compare*/
|
Chris@21
|
518 0, /*tp_repr*/
|
Chris@21
|
519 0, /*tp_as_number*/
|
Chris@21
|
520 0, /*tp_as_sequence*/
|
Chris@21
|
521 0, /*tp_as_mapping*/
|
Chris@21
|
522 0, /*tp_hash*/
|
Chris@21
|
523 0, /*tp_call*/
|
Chris@21
|
524 0, /*tp_str*/
|
Chris@33
|
525 PyObject_GenericGetAttr, /*tp_getattro*/
|
Chris@33
|
526 PyObject_GenericSetAttr, /*tp_setattro*/
|
Chris@21
|
527 0, /*tp_as_buffer*/
|
Chris@21
|
528 Py_TPFLAGS_DEFAULT, /*tp_flags*/
|
Chris@21
|
529 "Plugin Object", /*tp_doc*/
|
Chris@21
|
530 0, /*tp_traverse*/
|
Chris@21
|
531 0, /*tp_clear*/
|
Chris@21
|
532 0, /*tp_richcompare*/
|
Chris@21
|
533 0, /*tp_weaklistoffset*/
|
Chris@21
|
534 0, /*tp_iter*/
|
Chris@21
|
535 0, /*tp_iternext*/
|
Chris@21
|
536 PyPluginObject_methods, /*tp_methods*/
|
Chris@33
|
537 PyPluginObject_members, /*tp_members*/
|
Chris@21
|
538 0, /*tp_getset*/
|
Chris@21
|
539 0, /*tp_base*/
|
Chris@21
|
540 0, /*tp_dict*/
|
Chris@21
|
541 0, /*tp_descr_get*/
|
Chris@21
|
542 0, /*tp_descr_set*/
|
Chris@21
|
543 0, /*tp_dictoffset*/
|
Chris@21
|
544 0, /*tp_init*/
|
Chris@21
|
545 PyType_GenericAlloc, /*tp_alloc*/
|
Chris@21
|
546 0, /*tp_new*/
|
Chris@21
|
547 PyObject_Del, /*tp_free*/
|
Chris@21
|
548 0, /*tp_is_gc*/
|
Chris@21
|
549 };
|
Chris@0
|
550
|