annotate vamp-client/PluginStub.h @ 187:ad6025dc0b04

Documentation
author Chris Cannam <cannam@all-day-breakfast.com>
date Wed, 08 Feb 2017 10:09:51 +0000
parents 52322dde68ea
children
rev   line source
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