annotate vamp-client/PluginStub.h @ 180:bd543e74a9bf

Correct the inspection of the JSON object in successful to look for both error and success objects, writing out an error string if neither are present. Revert error handling in readInput() for JSON.
author Lucas Thompson <dev@lucas.im>
date Fri, 03 Feb 2017 11:12:27 +0000
parents 590b1a1fd955
children 3eb00e5c76c4
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
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