c@118
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
c@118
|
2 /*
|
c@118
|
3 Piper C++
|
c@118
|
4
|
c@118
|
5 An API for audio analysis and feature extraction plugins.
|
c@118
|
6
|
c@118
|
7 Centre for Digital Music, Queen Mary, University of London.
|
c@118
|
8 Copyright 2006-2016 Chris Cannam and QMUL.
|
c@118
|
9
|
c@118
|
10 Permission is hereby granted, free of charge, to any person
|
c@118
|
11 obtaining a copy of this software and associated documentation
|
c@118
|
12 files (the "Software"), to deal in the Software without
|
c@118
|
13 restriction, including without limitation the rights to use, copy,
|
c@118
|
14 modify, merge, publish, distribute, sublicense, and/or sell copies
|
c@118
|
15 of the Software, and to permit persons to whom the Software is
|
c@118
|
16 furnished to do so, subject to the following conditions:
|
c@118
|
17
|
c@118
|
18 The above copyright notice and this permission notice shall be
|
c@118
|
19 included in all copies or substantial portions of the Software.
|
c@118
|
20
|
c@118
|
21 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
c@118
|
22 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
c@118
|
23 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
c@118
|
24 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
|
c@118
|
25 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
c@118
|
26 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
c@118
|
27 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
c@118
|
28
|
c@118
|
29 Except as contained in this notice, the names of the Centre for
|
c@118
|
30 Digital Music; Queen Mary, University of London; and Chris Cannam
|
c@118
|
31 shall not be used in advertising or otherwise to promote the sale,
|
c@118
|
32 use or other dealings in this Software without prior written
|
c@118
|
33 authorization.
|
c@118
|
34 */
|
c@94
|
35
|
c@94
|
36 #ifndef PIPER_PLUGIN_STUB_H
|
c@94
|
37 #define PIPER_PLUGIN_STUB_H
|
c@94
|
38
|
c@94
|
39 #include <vamp-hostsdk/Plugin.h>
|
c@94
|
40 #include <vamp-hostsdk/PluginLoader.h>
|
c@97
|
41
|
c@97
|
42 #include "vamp-support/PluginStaticData.h"
|
c@97
|
43 #include "vamp-support/PluginConfiguration.h"
|
c@94
|
44
|
cannam@167
|
45 #include "PluginClient.h"
|
cannam@167
|
46
|
c@94
|
47 #include <cstdint>
|
cannam@167
|
48 #include <iostream>
|
c@94
|
49
|
c@97
|
50 namespace piper_vamp {
|
c@97
|
51 namespace client {
|
c@94
|
52
|
cannam@187
|
53 /**
|
cannam@187
|
54 * PluginStub presents a Piper feature extractor in the form of a Vamp plugin.
|
cannam@187
|
55 */
|
c@94
|
56 class PluginStub : public Vamp::Plugin
|
c@94
|
57 {
|
c@94
|
58 enum State {
|
cannam@186
|
59 /**
|
cannam@186
|
60 * The plugin's corresponding Piper feature extractor has been
|
cannam@186
|
61 * loaded but no subsequent state change has happened. This is
|
cannam@186
|
62 * the initial state of PluginStub on construction, since it
|
cannam@186
|
63 * is associated with a pre-loaded handle.
|
cannam@186
|
64 */
|
cannam@186
|
65 Loaded,
|
cannam@186
|
66
|
cannam@186
|
67 /**
|
cannam@186
|
68 * The plugin has been configured, and the step and block size
|
cannam@186
|
69 * received from the host in its last call to initialise()
|
cannam@186
|
70 * match those that were returned in the configuration
|
cannam@186
|
71 * response (i.e. the server's desired step and block
|
cannam@186
|
72 * size). Our m_config record reflects these correct
|
cannam@186
|
73 * values. The plugin is ready to process.
|
cannam@186
|
74 */
|
cannam@186
|
75 Configured,
|
cannam@186
|
76
|
cannam@186
|
77 /**
|
cannam@186
|
78 * The plugin has been configured, but the step and block size
|
cannam@186
|
79 * received from the host in its last call to initialise()
|
cannam@186
|
80 * differ from those returned by the server in the
|
cannam@186
|
81 * configuration response. Our initialise() call therefore
|
cannam@186
|
82 * returned false, and the plugin cannot be used until the
|
cannam@186
|
83 * host calls initialise() again with the "correct" step and
|
cannam@186
|
84 * block size. Our m_config record reflects these correct
|
cannam@186
|
85 * values, so the host can retrieve them through
|
cannam@186
|
86 * getPreferredStepSize and getPreferredBlockSize.
|
cannam@186
|
87 */
|
cannam@186
|
88 Misconfigured,
|
cannam@186
|
89
|
cannam@186
|
90 /**
|
cannam@186
|
91 * The finish() function has been called and the plugin
|
cannam@186
|
92 * unloaded. No further plugin activity is possible.
|
cannam@186
|
93 */
|
cannam@186
|
94 Finished,
|
cannam@186
|
95
|
cannam@186
|
96 /**
|
cannam@186
|
97 * A call has failed unrecoverably. No further plugin activity
|
cannam@186
|
98 * is possible.
|
cannam@186
|
99 */
|
cannam@186
|
100 Failed
|
c@94
|
101 };
|
c@94
|
102
|
c@94
|
103 public:
|
c@94
|
104 PluginStub(PluginClient *client,
|
c@94
|
105 std::string pluginKey,
|
c@94
|
106 float inputSampleRate,
|
c@94
|
107 int adapterFlags,
|
c@97
|
108 PluginStaticData psd,
|
c@97
|
109 PluginConfiguration defaultConfig) :
|
c@94
|
110 Plugin(inputSampleRate),
|
c@94
|
111 m_client(client),
|
c@94
|
112 m_key(pluginKey),
|
c@94
|
113 m_adapterFlags(adapterFlags),
|
c@94
|
114 m_state(Loaded),
|
c@94
|
115 m_psd(psd),
|
c@94
|
116 m_defaultConfig(defaultConfig),
|
c@94
|
117 m_config(defaultConfig)
|
c@94
|
118 { }
|
c@94
|
119
|
c@94
|
120 virtual ~PluginStub() {
|
cannam@167
|
121 if (m_state != Finished && m_state != Failed) {
|
cannam@167
|
122 try {
|
cannam@167
|
123 (void)m_client->finish(this);
|
cannam@167
|
124 } catch (const std::exception &e) {
|
cannam@167
|
125 // Finish can throw, but our destructor must not
|
cannam@167
|
126 std::cerr << "WARNING: PluginStub::~PluginStub: caught exception from finish(): " << e.what() << std::endl;
|
cannam@167
|
127 }
|
c@94
|
128 }
|
c@94
|
129 }
|
c@94
|
130
|
c@94
|
131 virtual std::string getIdentifier() const {
|
c@94
|
132 return m_psd.basic.identifier;
|
c@94
|
133 }
|
c@94
|
134
|
c@94
|
135 virtual std::string getName() const {
|
c@94
|
136 return m_psd.basic.name;
|
c@94
|
137 }
|
c@94
|
138
|
c@94
|
139 virtual std::string getDescription() const {
|
c@94
|
140 return m_psd.basic.description;
|
c@94
|
141 }
|
c@94
|
142
|
c@94
|
143 virtual std::string getMaker() const {
|
c@94
|
144 return m_psd.maker;
|
c@94
|
145 }
|
c@94
|
146
|
c@94
|
147 virtual std::string getCopyright() const {
|
c@94
|
148 return m_psd.copyright;
|
c@94
|
149 }
|
c@94
|
150
|
c@94
|
151 virtual int getPluginVersion() const {
|
c@94
|
152 return m_psd.pluginVersion;
|
c@94
|
153 }
|
c@94
|
154
|
c@94
|
155 virtual ParameterList getParameterDescriptors() const {
|
c@94
|
156 return m_psd.parameters;
|
c@94
|
157 }
|
c@94
|
158
|
c@94
|
159 virtual float getParameter(std::string name) const {
|
c@94
|
160 if (m_config.parameterValues.find(name) != m_config.parameterValues.end()) {
|
c@94
|
161 return m_config.parameterValues.at(name);
|
c@94
|
162 } else {
|
c@94
|
163 return 0.f;
|
c@94
|
164 }
|
c@94
|
165 }
|
c@94
|
166
|
c@94
|
167 virtual void setParameter(std::string name, float value) {
|
cannam@167
|
168 if (m_state == Failed) {
|
cannam@167
|
169 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
170 }
|
c@94
|
171 if (m_state != Loaded) {
|
cannam@167
|
172 m_state = Failed;
|
c@94
|
173 throw std::logic_error("Can't set parameter after plugin initialised");
|
c@94
|
174 }
|
c@94
|
175 m_config.parameterValues[name] = value;
|
c@94
|
176 }
|
c@94
|
177
|
c@94
|
178 virtual ProgramList getPrograms() const {
|
c@94
|
179 return m_psd.programs;
|
c@94
|
180 }
|
c@94
|
181
|
c@94
|
182 virtual std::string getCurrentProgram() const {
|
c@94
|
183 return m_config.currentProgram;
|
c@94
|
184 }
|
c@94
|
185
|
c@94
|
186 virtual void selectProgram(std::string program) {
|
cannam@167
|
187 if (m_state == Failed) {
|
cannam@167
|
188 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
189 }
|
c@94
|
190 if (m_state != Loaded) {
|
cannam@167
|
191 m_state = Failed;
|
c@94
|
192 throw std::logic_error("Can't select program after plugin initialised");
|
c@94
|
193 }
|
c@94
|
194 m_config.currentProgram = program;
|
c@94
|
195 }
|
c@94
|
196
|
c@94
|
197 virtual bool initialise(size_t inputChannels,
|
c@94
|
198 size_t stepSize,
|
c@94
|
199 size_t blockSize) {
|
c@94
|
200
|
cannam@167
|
201 if (m_state == Failed) {
|
cannam@167
|
202 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
203 }
|
cannam@186
|
204
|
cannam@186
|
205 if (m_state == Misconfigured) {
|
cannam@186
|
206 if (int(stepSize) == m_config.framing.stepSize &&
|
cannam@186
|
207 int(blockSize) == m_config.framing.blockSize) {
|
cannam@186
|
208 m_state = Configured;
|
cannam@186
|
209 return true;
|
cannam@186
|
210 } else {
|
cannam@186
|
211 return false;
|
cannam@186
|
212 }
|
cannam@186
|
213 }
|
cannam@186
|
214
|
c@94
|
215 if (m_state != Loaded) {
|
cannam@167
|
216 m_state = Failed;
|
c@94
|
217 throw std::logic_error("Plugin has already been initialised");
|
c@94
|
218 }
|
c@94
|
219
|
c@102
|
220 m_config.channelCount = int(inputChannels);
|
cannam@185
|
221 m_config.framing.stepSize = int(stepSize);
|
cannam@185
|
222 m_config.framing.blockSize = int(blockSize);
|
c@94
|
223
|
cannam@169
|
224 try {
|
cannam@185
|
225 auto response = m_client->configure(this, m_config);
|
cannam@185
|
226 m_outputs = response.outputs;
|
cannam@186
|
227
|
cannam@185
|
228 // Update with the new preferred step and block size now
|
cannam@185
|
229 // that the plugin has taken into account its parameter
|
cannam@185
|
230 // settings. If the values passed in to initialise()
|
cannam@185
|
231 // weren't suitable, then this ensures that a subsequent
|
cannam@185
|
232 // call to getPreferredStepSize/BlockSize on this plugin
|
cannam@185
|
233 // object will at least get acceptable values from now on
|
cannam@185
|
234 m_config.framing = response.framing;
|
cannam@186
|
235
|
cannam@186
|
236 // And if they didn't match up with the passed-in ones,
|
cannam@186
|
237 // lodge ourselves in Misconfigured state and report
|
cannam@186
|
238 // failure so as to provoke the host to call initialise()
|
cannam@186
|
239 // again before any processing.
|
cannam@186
|
240 if (m_config.framing.stepSize != int(stepSize) ||
|
cannam@186
|
241 m_config.framing.blockSize != int(blockSize)) {
|
cannam@186
|
242 m_state = Misconfigured;
|
cannam@186
|
243 return false;
|
cannam@186
|
244 }
|
cannam@185
|
245
|
cannam@169
|
246 } catch (const std::exception &e) {
|
cannam@169
|
247 m_state = Failed;
|
cannam@169
|
248 throw;
|
cannam@169
|
249 }
|
c@94
|
250
|
c@94
|
251 if (!m_outputs.empty()) {
|
c@94
|
252 m_state = Configured;
|
c@94
|
253 return true;
|
c@94
|
254 } else {
|
c@94
|
255 return false;
|
c@94
|
256 }
|
c@94
|
257 }
|
c@94
|
258
|
c@94
|
259 virtual void reset() {
|
c@94
|
260
|
cannam@167
|
261 if (m_state == Failed) {
|
cannam@167
|
262 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
263 }
|
cannam@186
|
264 if (m_state == Loaded || m_state == Misconfigured) {
|
c@94
|
265 // reset is a no-op if the plugin hasn't been initialised yet
|
c@94
|
266 return;
|
c@94
|
267 }
|
cannam@169
|
268
|
cannam@169
|
269 try {
|
cannam@169
|
270 m_client->reset(this, m_config);
|
cannam@169
|
271 } catch (const std::exception &e) {
|
cannam@169
|
272 m_state = Failed;
|
cannam@169
|
273 throw;
|
cannam@169
|
274 }
|
c@94
|
275
|
c@94
|
276 m_state = Configured;
|
c@94
|
277 }
|
c@94
|
278
|
c@94
|
279 virtual InputDomain getInputDomain() const {
|
c@94
|
280 return m_psd.inputDomain;
|
c@94
|
281 }
|
c@94
|
282
|
c@94
|
283 virtual size_t getPreferredBlockSize() const {
|
cannam@185
|
284 // Return this from m_config instead of m_defaultConfig, so
|
cannam@185
|
285 // that it gets updated in the event of an initialise() call
|
cannam@185
|
286 // that fails for the wrong value
|
cannam@185
|
287 return m_config.framing.blockSize;
|
c@94
|
288 }
|
c@94
|
289
|
c@94
|
290 virtual size_t getPreferredStepSize() const {
|
cannam@185
|
291 // Return this from m_config instead of m_defaultConfig, so
|
cannam@185
|
292 // that it gets updated in the event of an initialise() call
|
cannam@185
|
293 // that fails for the wrong value
|
cannam@185
|
294 return m_config.framing.stepSize;
|
c@94
|
295 }
|
c@94
|
296
|
c@94
|
297 virtual size_t getMinChannelCount() const {
|
c@94
|
298 return m_psd.minChannelCount;
|
c@94
|
299 }
|
c@94
|
300
|
c@94
|
301 virtual size_t getMaxChannelCount() const {
|
c@94
|
302 return m_psd.maxChannelCount;
|
c@94
|
303 }
|
c@94
|
304
|
c@94
|
305 virtual OutputList getOutputDescriptors() const {
|
cannam@167
|
306
|
cannam@167
|
307 if (m_state == Failed) {
|
cannam@167
|
308 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
309 }
|
c@94
|
310 if (m_state == Configured) {
|
c@94
|
311 return m_outputs;
|
c@94
|
312 }
|
c@94
|
313
|
c@94
|
314 //!!! todo: figure out for which hosts (and adapters?) it may
|
c@94
|
315 //!!! be a problem that the output descriptors are incomplete
|
c@94
|
316 //!!! here. Any such hosts/adapters are broken, but I bet they
|
c@94
|
317 //!!! exist
|
c@94
|
318
|
c@94
|
319 OutputList staticOutputs;
|
c@94
|
320 for (const auto &o: m_psd.basicOutputInfo) {
|
c@94
|
321 OutputDescriptor od;
|
c@94
|
322 od.identifier = o.identifier;
|
c@94
|
323 od.name = o.name;
|
c@94
|
324 od.description = o.description;
|
c@94
|
325 staticOutputs.push_back(od);
|
c@94
|
326 }
|
c@94
|
327 return staticOutputs;
|
c@94
|
328 }
|
c@94
|
329
|
c@94
|
330 virtual FeatureSet process(const float *const *inputBuffers,
|
c@118
|
331 Vamp::RealTime timestamp) {
|
c@94
|
332
|
cannam@167
|
333 if (m_state == Failed) {
|
cannam@167
|
334 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
335 }
|
cannam@186
|
336 if (m_state == Loaded || m_state == Misconfigured) {
|
cannam@167
|
337 m_state = Failed;
|
c@94
|
338 throw std::logic_error("Plugin has not been initialised");
|
c@94
|
339 }
|
c@94
|
340 if (m_state == Finished) {
|
cannam@167
|
341 m_state = Failed;
|
c@94
|
342 throw std::logic_error("Plugin has already been disposed of");
|
c@94
|
343 }
|
c@94
|
344
|
c@94
|
345 std::vector<std::vector<float> > vecbuf;
|
c@94
|
346 for (int c = 0; c < m_config.channelCount; ++c) {
|
c@94
|
347 vecbuf.push_back(std::vector<float>
|
c@94
|
348 (inputBuffers[c],
|
cannam@185
|
349 inputBuffers[c] + m_config.framing.blockSize));
|
c@94
|
350 }
|
cannam@169
|
351
|
cannam@169
|
352 try {
|
cannam@169
|
353 return m_client->process(this, vecbuf, timestamp);
|
cannam@169
|
354 } catch (const std::exception &e) {
|
cannam@169
|
355 m_state = Failed;
|
cannam@169
|
356 throw;
|
cannam@169
|
357 }
|
c@94
|
358 }
|
c@94
|
359
|
c@94
|
360 virtual FeatureSet getRemainingFeatures() {
|
c@94
|
361
|
cannam@167
|
362 if (m_state == Failed) {
|
cannam@167
|
363 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
364 }
|
cannam@186
|
365 if (m_state == Loaded || m_state == Misconfigured) {
|
cannam@167
|
366 m_state = Failed;
|
c@94
|
367 throw std::logic_error("Plugin has not been configured");
|
c@94
|
368 }
|
c@94
|
369 if (m_state == Finished) {
|
cannam@167
|
370 m_state = Failed;
|
c@94
|
371 throw std::logic_error("Plugin has already been disposed of");
|
c@94
|
372 }
|
c@94
|
373
|
c@94
|
374 m_state = Finished;
|
c@94
|
375
|
cannam@169
|
376 try {
|
cannam@169
|
377 return m_client->finish(this);
|
cannam@169
|
378 } catch (const std::exception &e) {
|
cannam@169
|
379 m_state = Failed;
|
cannam@169
|
380 throw;
|
cannam@169
|
381 }
|
c@94
|
382 }
|
c@94
|
383
|
c@94
|
384 // Not Plugin methods, but needed by the PluginClient to support reloads:
|
c@94
|
385
|
c@94
|
386 virtual float getInputSampleRate() const {
|
c@94
|
387 return m_inputSampleRate;
|
c@94
|
388 }
|
c@94
|
389
|
c@94
|
390 virtual std::string getPluginKey() const {
|
c@94
|
391 return m_key;
|
c@94
|
392 }
|
c@94
|
393
|
c@94
|
394 virtual int getAdapterFlags() const {
|
c@94
|
395 return m_adapterFlags;
|
c@94
|
396 }
|
c@94
|
397
|
c@94
|
398 private:
|
c@94
|
399 PluginClient *m_client;
|
c@94
|
400 std::string m_key;
|
c@94
|
401 int m_adapterFlags;
|
c@94
|
402 State m_state;
|
c@97
|
403 PluginStaticData m_psd;
|
c@94
|
404 OutputList m_outputs;
|
c@97
|
405 PluginConfiguration m_defaultConfig;
|
c@97
|
406 PluginConfiguration m_config;
|
c@94
|
407 };
|
c@94
|
408
|
c@94
|
409 }
|
c@94
|
410 }
|
c@94
|
411
|
c@94
|
412 #endif
|