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
|
c@94
|
53 class PluginStub : public Vamp::Plugin
|
c@94
|
54 {
|
c@94
|
55 enum State {
|
cannam@167
|
56 Loaded, Configured, Finished, Failed
|
c@94
|
57 };
|
c@94
|
58
|
c@94
|
59 public:
|
c@94
|
60 PluginStub(PluginClient *client,
|
c@94
|
61 std::string pluginKey,
|
c@94
|
62 float inputSampleRate,
|
c@94
|
63 int adapterFlags,
|
c@97
|
64 PluginStaticData psd,
|
c@97
|
65 PluginConfiguration defaultConfig) :
|
c@94
|
66 Plugin(inputSampleRate),
|
c@94
|
67 m_client(client),
|
c@94
|
68 m_key(pluginKey),
|
c@94
|
69 m_adapterFlags(adapterFlags),
|
c@94
|
70 m_state(Loaded),
|
c@94
|
71 m_psd(psd),
|
c@94
|
72 m_defaultConfig(defaultConfig),
|
c@94
|
73 m_config(defaultConfig)
|
c@94
|
74 { }
|
c@94
|
75
|
c@94
|
76 virtual ~PluginStub() {
|
cannam@167
|
77 if (m_state != Finished && m_state != Failed) {
|
cannam@167
|
78 try {
|
cannam@167
|
79 (void)m_client->finish(this);
|
cannam@167
|
80 } catch (const std::exception &e) {
|
cannam@167
|
81 // Finish can throw, but our destructor must not
|
cannam@167
|
82 std::cerr << "WARNING: PluginStub::~PluginStub: caught exception from finish(): " << e.what() << std::endl;
|
cannam@167
|
83 }
|
c@94
|
84 }
|
c@94
|
85 }
|
c@94
|
86
|
c@94
|
87 virtual std::string getIdentifier() const {
|
c@94
|
88 return m_psd.basic.identifier;
|
c@94
|
89 }
|
c@94
|
90
|
c@94
|
91 virtual std::string getName() const {
|
c@94
|
92 return m_psd.basic.name;
|
c@94
|
93 }
|
c@94
|
94
|
c@94
|
95 virtual std::string getDescription() const {
|
c@94
|
96 return m_psd.basic.description;
|
c@94
|
97 }
|
c@94
|
98
|
c@94
|
99 virtual std::string getMaker() const {
|
c@94
|
100 return m_psd.maker;
|
c@94
|
101 }
|
c@94
|
102
|
c@94
|
103 virtual std::string getCopyright() const {
|
c@94
|
104 return m_psd.copyright;
|
c@94
|
105 }
|
c@94
|
106
|
c@94
|
107 virtual int getPluginVersion() const {
|
c@94
|
108 return m_psd.pluginVersion;
|
c@94
|
109 }
|
c@94
|
110
|
c@94
|
111 virtual ParameterList getParameterDescriptors() const {
|
c@94
|
112 return m_psd.parameters;
|
c@94
|
113 }
|
c@94
|
114
|
c@94
|
115 virtual float getParameter(std::string name) const {
|
c@94
|
116 if (m_config.parameterValues.find(name) != m_config.parameterValues.end()) {
|
c@94
|
117 return m_config.parameterValues.at(name);
|
c@94
|
118 } else {
|
c@94
|
119 return 0.f;
|
c@94
|
120 }
|
c@94
|
121 }
|
c@94
|
122
|
c@94
|
123 virtual void setParameter(std::string name, float value) {
|
cannam@167
|
124 if (m_state == Failed) {
|
cannam@167
|
125 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
126 }
|
c@94
|
127 if (m_state != Loaded) {
|
cannam@167
|
128 m_state = Failed;
|
c@94
|
129 throw std::logic_error("Can't set parameter after plugin initialised");
|
c@94
|
130 }
|
c@94
|
131 m_config.parameterValues[name] = value;
|
c@94
|
132 }
|
c@94
|
133
|
c@94
|
134 virtual ProgramList getPrograms() const {
|
c@94
|
135 return m_psd.programs;
|
c@94
|
136 }
|
c@94
|
137
|
c@94
|
138 virtual std::string getCurrentProgram() const {
|
c@94
|
139 return m_config.currentProgram;
|
c@94
|
140 }
|
c@94
|
141
|
c@94
|
142 virtual void selectProgram(std::string program) {
|
cannam@167
|
143 if (m_state == Failed) {
|
cannam@167
|
144 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
145 }
|
c@94
|
146 if (m_state != Loaded) {
|
cannam@167
|
147 m_state = Failed;
|
c@94
|
148 throw std::logic_error("Can't select program after plugin initialised");
|
c@94
|
149 }
|
c@94
|
150 m_config.currentProgram = program;
|
c@94
|
151 }
|
c@94
|
152
|
c@94
|
153 virtual bool initialise(size_t inputChannels,
|
c@94
|
154 size_t stepSize,
|
c@94
|
155 size_t blockSize) {
|
c@94
|
156
|
cannam@167
|
157 if (m_state == Failed) {
|
cannam@167
|
158 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
159 }
|
c@94
|
160 if (m_state != Loaded) {
|
cannam@167
|
161 m_state = Failed;
|
c@94
|
162 throw std::logic_error("Plugin has already been initialised");
|
c@94
|
163 }
|
c@94
|
164
|
c@102
|
165 m_config.channelCount = int(inputChannels);
|
c@102
|
166 m_config.stepSize = int(stepSize);
|
c@102
|
167 m_config.blockSize = int(blockSize);
|
c@94
|
168
|
cannam@169
|
169 try {
|
cannam@169
|
170 m_outputs = m_client->configure(this, m_config);
|
cannam@169
|
171 } catch (const std::exception &e) {
|
cannam@169
|
172 m_state = Failed;
|
cannam@169
|
173 throw;
|
cannam@169
|
174 }
|
c@94
|
175
|
c@94
|
176 if (!m_outputs.empty()) {
|
c@94
|
177 m_state = Configured;
|
c@94
|
178 return true;
|
c@94
|
179 } else {
|
c@94
|
180 return false;
|
c@94
|
181 }
|
c@94
|
182 }
|
c@94
|
183
|
c@94
|
184 virtual void reset() {
|
c@94
|
185
|
cannam@167
|
186 if (m_state == Failed) {
|
cannam@167
|
187 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
188 }
|
c@94
|
189 if (m_state == Loaded) {
|
c@94
|
190 // reset is a no-op if the plugin hasn't been initialised yet
|
c@94
|
191 return;
|
c@94
|
192 }
|
cannam@169
|
193
|
cannam@169
|
194 try {
|
cannam@169
|
195 m_client->reset(this, m_config);
|
cannam@169
|
196 } catch (const std::exception &e) {
|
cannam@169
|
197 m_state = Failed;
|
cannam@169
|
198 throw;
|
cannam@169
|
199 }
|
c@94
|
200
|
c@94
|
201 m_state = Configured;
|
c@94
|
202 }
|
c@94
|
203
|
c@94
|
204 virtual InputDomain getInputDomain() const {
|
c@94
|
205 return m_psd.inputDomain;
|
c@94
|
206 }
|
c@94
|
207
|
c@94
|
208 virtual size_t getPreferredBlockSize() const {
|
c@94
|
209 return m_defaultConfig.blockSize;
|
c@94
|
210 }
|
c@94
|
211
|
c@94
|
212 virtual size_t getPreferredStepSize() const {
|
c@94
|
213 return m_defaultConfig.stepSize;
|
c@94
|
214 }
|
c@94
|
215
|
c@94
|
216 virtual size_t getMinChannelCount() const {
|
c@94
|
217 return m_psd.minChannelCount;
|
c@94
|
218 }
|
c@94
|
219
|
c@94
|
220 virtual size_t getMaxChannelCount() const {
|
c@94
|
221 return m_psd.maxChannelCount;
|
c@94
|
222 }
|
c@94
|
223
|
c@94
|
224 virtual OutputList getOutputDescriptors() const {
|
cannam@167
|
225
|
cannam@167
|
226 if (m_state == Failed) {
|
cannam@167
|
227 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
228 }
|
c@94
|
229 if (m_state == Configured) {
|
c@94
|
230 return m_outputs;
|
c@94
|
231 }
|
c@94
|
232
|
c@94
|
233 //!!! todo: figure out for which hosts (and adapters?) it may
|
c@94
|
234 //!!! be a problem that the output descriptors are incomplete
|
c@94
|
235 //!!! here. Any such hosts/adapters are broken, but I bet they
|
c@94
|
236 //!!! exist
|
c@94
|
237
|
c@94
|
238 OutputList staticOutputs;
|
c@94
|
239 for (const auto &o: m_psd.basicOutputInfo) {
|
c@94
|
240 OutputDescriptor od;
|
c@94
|
241 od.identifier = o.identifier;
|
c@94
|
242 od.name = o.name;
|
c@94
|
243 od.description = o.description;
|
c@94
|
244 staticOutputs.push_back(od);
|
c@94
|
245 }
|
c@94
|
246 return staticOutputs;
|
c@94
|
247 }
|
c@94
|
248
|
c@94
|
249 virtual FeatureSet process(const float *const *inputBuffers,
|
c@118
|
250 Vamp::RealTime timestamp) {
|
c@94
|
251
|
cannam@167
|
252 if (m_state == Failed) {
|
cannam@167
|
253 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
254 }
|
c@94
|
255 if (m_state == Loaded) {
|
cannam@167
|
256 m_state = Failed;
|
c@94
|
257 throw std::logic_error("Plugin has not been initialised");
|
c@94
|
258 }
|
c@94
|
259 if (m_state == Finished) {
|
cannam@167
|
260 m_state = Failed;
|
c@94
|
261 throw std::logic_error("Plugin has already been disposed of");
|
c@94
|
262 }
|
c@94
|
263
|
c@94
|
264 std::vector<std::vector<float> > vecbuf;
|
c@94
|
265 for (int c = 0; c < m_config.channelCount; ++c) {
|
c@94
|
266 vecbuf.push_back(std::vector<float>
|
c@94
|
267 (inputBuffers[c],
|
c@94
|
268 inputBuffers[c] + m_config.blockSize));
|
c@94
|
269 }
|
cannam@169
|
270
|
cannam@169
|
271 try {
|
cannam@169
|
272 return m_client->process(this, vecbuf, timestamp);
|
cannam@169
|
273 } catch (const std::exception &e) {
|
cannam@169
|
274 m_state = Failed;
|
cannam@169
|
275 throw;
|
cannam@169
|
276 }
|
c@94
|
277 }
|
c@94
|
278
|
c@94
|
279 virtual FeatureSet getRemainingFeatures() {
|
c@94
|
280
|
cannam@167
|
281 if (m_state == Failed) {
|
cannam@167
|
282 throw std::logic_error("Plugin is in failed state");
|
cannam@167
|
283 }
|
c@94
|
284 if (m_state == Loaded) {
|
cannam@167
|
285 m_state = Failed;
|
c@94
|
286 throw std::logic_error("Plugin has not been configured");
|
c@94
|
287 }
|
c@94
|
288 if (m_state == Finished) {
|
cannam@167
|
289 m_state = Failed;
|
c@94
|
290 throw std::logic_error("Plugin has already been disposed of");
|
c@94
|
291 }
|
c@94
|
292
|
c@94
|
293 m_state = Finished;
|
c@94
|
294
|
cannam@169
|
295 try {
|
cannam@169
|
296 return m_client->finish(this);
|
cannam@169
|
297 } catch (const std::exception &e) {
|
cannam@169
|
298 m_state = Failed;
|
cannam@169
|
299 throw;
|
cannam@169
|
300 }
|
c@94
|
301 }
|
c@94
|
302
|
c@94
|
303 // Not Plugin methods, but needed by the PluginClient to support reloads:
|
c@94
|
304
|
c@94
|
305 virtual float getInputSampleRate() const {
|
c@94
|
306 return m_inputSampleRate;
|
c@94
|
307 }
|
c@94
|
308
|
c@94
|
309 virtual std::string getPluginKey() const {
|
c@94
|
310 return m_key;
|
c@94
|
311 }
|
c@94
|
312
|
c@94
|
313 virtual int getAdapterFlags() const {
|
c@94
|
314 return m_adapterFlags;
|
c@94
|
315 }
|
c@94
|
316
|
c@94
|
317 private:
|
c@94
|
318 PluginClient *m_client;
|
c@94
|
319 std::string m_key;
|
c@94
|
320 int m_adapterFlags;
|
c@94
|
321 State m_state;
|
c@97
|
322 PluginStaticData m_psd;
|
c@94
|
323 OutputList m_outputs;
|
c@97
|
324 PluginConfiguration m_defaultConfig;
|
c@97
|
325 PluginConfiguration m_config;
|
c@94
|
326 };
|
c@94
|
327
|
c@94
|
328 }
|
c@94
|
329 }
|
c@94
|
330
|
c@94
|
331 #endif
|