annotate vamp-client/PluginStub.h @ 151:255403a4b321

We'll need the piper dir
author Chris Cannam <cannam@all-day-breakfast.com>
date Fri, 20 Jan 2017 17:46:56 +0000
parents ff3fd8d1b2dc
children 4a37daf5f8b4
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
c@94 45 #include <cstdint>
c@94 46
c@94 47 #include "PluginClient.h"
c@94 48
c@97 49 namespace piper_vamp {
c@97 50 namespace client {
c@94 51
c@94 52 class PluginStub : public Vamp::Plugin
c@94 53 {
c@94 54 enum State {
c@94 55 Loaded, Configured, Finished
c@94 56 };
c@94 57
c@94 58 public:
c@94 59 PluginStub(PluginClient *client,
c@94 60 std::string pluginKey,
c@94 61 float inputSampleRate,
c@94 62 int adapterFlags,
c@97 63 PluginStaticData psd,
c@97 64 PluginConfiguration defaultConfig) :
c@94 65 Plugin(inputSampleRate),
c@94 66 m_client(client),
c@94 67 m_key(pluginKey),
c@94 68 m_adapterFlags(adapterFlags),
c@94 69 m_state(Loaded),
c@94 70 m_psd(psd),
c@94 71 m_defaultConfig(defaultConfig),
c@94 72 m_config(defaultConfig)
c@94 73 { }
c@94 74
c@94 75 virtual ~PluginStub() {
c@94 76 if (m_state != Finished) {
c@118 77 (void)m_client->finish(this);
c@94 78 }
c@94 79 }
c@94 80
c@94 81 virtual std::string getIdentifier() const {
c@94 82 return m_psd.basic.identifier;
c@94 83 }
c@94 84
c@94 85 virtual std::string getName() const {
c@94 86 return m_psd.basic.name;
c@94 87 }
c@94 88
c@94 89 virtual std::string getDescription() const {
c@94 90 return m_psd.basic.description;
c@94 91 }
c@94 92
c@94 93 virtual std::string getMaker() const {
c@94 94 return m_psd.maker;
c@94 95 }
c@94 96
c@94 97 virtual std::string getCopyright() const {
c@94 98 return m_psd.copyright;
c@94 99 }
c@94 100
c@94 101 virtual int getPluginVersion() const {
c@94 102 return m_psd.pluginVersion;
c@94 103 }
c@94 104
c@94 105 virtual ParameterList getParameterDescriptors() const {
c@94 106 return m_psd.parameters;
c@94 107 }
c@94 108
c@94 109 virtual float getParameter(std::string name) const {
c@94 110 if (m_config.parameterValues.find(name) != m_config.parameterValues.end()) {
c@94 111 return m_config.parameterValues.at(name);
c@94 112 } else {
c@94 113 return 0.f;
c@94 114 }
c@94 115 }
c@94 116
c@94 117 virtual void setParameter(std::string name, float value) {
c@94 118 if (m_state != Loaded) {
c@94 119 throw std::logic_error("Can't set parameter after plugin initialised");
c@94 120 }
c@94 121 m_config.parameterValues[name] = value;
c@94 122 }
c@94 123
c@94 124 virtual ProgramList getPrograms() const {
c@94 125 return m_psd.programs;
c@94 126 }
c@94 127
c@94 128 virtual std::string getCurrentProgram() const {
c@94 129 return m_config.currentProgram;
c@94 130 }
c@94 131
c@94 132 virtual void selectProgram(std::string program) {
c@94 133 if (m_state != Loaded) {
c@94 134 throw std::logic_error("Can't select program after plugin initialised");
c@94 135 }
c@94 136 m_config.currentProgram = program;
c@94 137 }
c@94 138
c@94 139 virtual bool initialise(size_t inputChannels,
c@94 140 size_t stepSize,
c@94 141 size_t blockSize) {
c@94 142
c@94 143 if (m_state != Loaded) {
c@94 144 throw std::logic_error("Plugin has already been initialised");
c@94 145 }
c@94 146
c@102 147 m_config.channelCount = int(inputChannels);
c@102 148 m_config.stepSize = int(stepSize);
c@102 149 m_config.blockSize = int(blockSize);
c@94 150
c@94 151 m_outputs = m_client->configure(this, m_config);
c@94 152
c@94 153 if (!m_outputs.empty()) {
c@94 154 m_state = Configured;
c@94 155 return true;
c@94 156 } else {
c@94 157 return false;
c@94 158 }
c@94 159 }
c@94 160
c@94 161 virtual void reset() {
c@94 162
c@94 163 if (m_state == Loaded) {
c@94 164 // reset is a no-op if the plugin hasn't been initialised yet
c@94 165 return;
c@94 166 }
c@94 167
c@94 168 m_client->reset(this, m_config);
c@94 169
c@94 170 m_state = Configured;
c@94 171 }
c@94 172
c@94 173 virtual InputDomain getInputDomain() const {
c@94 174 return m_psd.inputDomain;
c@94 175 }
c@94 176
c@94 177 virtual size_t getPreferredBlockSize() const {
c@94 178 return m_defaultConfig.blockSize;
c@94 179 }
c@94 180
c@94 181 virtual size_t getPreferredStepSize() const {
c@94 182 return m_defaultConfig.stepSize;
c@94 183 }
c@94 184
c@94 185 virtual size_t getMinChannelCount() const {
c@94 186 return m_psd.minChannelCount;
c@94 187 }
c@94 188
c@94 189 virtual size_t getMaxChannelCount() const {
c@94 190 return m_psd.maxChannelCount;
c@94 191 }
c@94 192
c@94 193 virtual OutputList getOutputDescriptors() const {
c@94 194 if (m_state == Configured) {
c@94 195 return m_outputs;
c@94 196 }
c@94 197
c@94 198 //!!! todo: figure out for which hosts (and adapters?) it may
c@94 199 //!!! be a problem that the output descriptors are incomplete
c@94 200 //!!! here. Any such hosts/adapters are broken, but I bet they
c@94 201 //!!! exist
c@94 202
c@94 203 OutputList staticOutputs;
c@94 204 for (const auto &o: m_psd.basicOutputInfo) {
c@94 205 OutputDescriptor od;
c@94 206 od.identifier = o.identifier;
c@94 207 od.name = o.name;
c@94 208 od.description = o.description;
c@94 209 staticOutputs.push_back(od);
c@94 210 }
c@94 211 return staticOutputs;
c@94 212 }
c@94 213
c@94 214 virtual FeatureSet process(const float *const *inputBuffers,
c@118 215 Vamp::RealTime timestamp) {
c@94 216
c@94 217 if (m_state == Loaded) {
c@94 218 throw std::logic_error("Plugin has not been initialised");
c@94 219 }
c@94 220 if (m_state == Finished) {
c@94 221 throw std::logic_error("Plugin has already been disposed of");
c@94 222 }
c@94 223
c@94 224 //!!! ew
c@94 225 std::vector<std::vector<float> > vecbuf;
c@94 226 for (int c = 0; c < m_config.channelCount; ++c) {
c@94 227 vecbuf.push_back(std::vector<float>
c@94 228 (inputBuffers[c],
c@94 229 inputBuffers[c] + m_config.blockSize));
c@94 230 }
c@94 231
c@94 232 return m_client->process(this, vecbuf, timestamp);
c@94 233 }
c@94 234
c@94 235 virtual FeatureSet getRemainingFeatures() {
c@94 236
c@94 237 if (m_state == Loaded) {
c@94 238 throw std::logic_error("Plugin has not been configured");
c@94 239 }
c@94 240 if (m_state == Finished) {
c@94 241 throw std::logic_error("Plugin has already been disposed of");
c@94 242 }
c@94 243
c@94 244 m_state = Finished;
c@94 245
c@94 246 return m_client->finish(this);
c@94 247 }
c@94 248
c@94 249 // Not Plugin methods, but needed by the PluginClient to support reloads:
c@94 250
c@94 251 virtual float getInputSampleRate() const {
c@94 252 return m_inputSampleRate;
c@94 253 }
c@94 254
c@94 255 virtual std::string getPluginKey() const {
c@94 256 return m_key;
c@94 257 }
c@94 258
c@94 259 virtual int getAdapterFlags() const {
c@94 260 return m_adapterFlags;
c@94 261 }
c@94 262
c@94 263 private:
c@94 264 PluginClient *m_client;
c@94 265 std::string m_key;
c@94 266 int m_adapterFlags;
c@94 267 State m_state;
c@97 268 PluginStaticData m_psd;
c@94 269 OutputList m_outputs;
c@97 270 PluginConfiguration m_defaultConfig;
c@97 271 PluginConfiguration m_config;
c@94 272 };
c@94 273
c@94 274 }
c@94 275 }
c@94 276
c@94 277 #endif