annotate vamp-client/PluginStub.h @ 170:590b1a1fd955

More work on error and exception handling
author Chris Cannam <cannam@all-day-breakfast.com>
date Tue, 31 Jan 2017 14:53:24 +0000
parents f13dc1db2229
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