cannam@150
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
cannam@150
|
2 /*
|
cannam@150
|
3 Piper C++
|
cannam@150
|
4
|
cannam@150
|
5 An API for audio analysis and feature extraction plugins.
|
cannam@150
|
6
|
cannam@150
|
7 Centre for Digital Music, Queen Mary, University of London.
|
cannam@150
|
8 Copyright 2006-2016 Chris Cannam and QMUL.
|
cannam@150
|
9
|
cannam@150
|
10 Permission is hereby granted, free of charge, to any person
|
cannam@150
|
11 obtaining a copy of this software and associated documentation
|
cannam@150
|
12 files (the "Software"), to deal in the Software without
|
cannam@150
|
13 restriction, including without limitation the rights to use, copy,
|
cannam@150
|
14 modify, merge, publish, distribute, sublicense, and/or sell copies
|
cannam@150
|
15 of the Software, and to permit persons to whom the Software is
|
cannam@150
|
16 furnished to do so, subject to the following conditions:
|
cannam@150
|
17
|
cannam@150
|
18 The above copyright notice and this permission notice shall be
|
cannam@150
|
19 included in all copies or substantial portions of the Software.
|
cannam@150
|
20
|
cannam@150
|
21 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
cannam@150
|
22 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
cannam@150
|
23 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
cannam@150
|
24 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
|
cannam@150
|
25 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
cannam@150
|
26 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
cannam@150
|
27 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
cannam@150
|
28
|
cannam@150
|
29 Except as contained in this notice, the names of the Centre for
|
cannam@150
|
30 Digital Music; Queen Mary, University of London; and Chris Cannam
|
cannam@150
|
31 shall not be used in advertising or otherwise to promote the sale,
|
cannam@150
|
32 use or other dealings in this Software without prior written
|
cannam@150
|
33 authorization.
|
cannam@150
|
34 */
|
cannam@150
|
35
|
cannam@150
|
36 #ifndef PIPER_AUTO_PLUGIN_H
|
cannam@150
|
37 #define PIPER_AUTO_PLUGIN_H
|
cannam@150
|
38
|
cannam@150
|
39 #include "ProcessQtTransport.h"
|
cannam@152
|
40 #include "../CapnpRRClient.h"
|
cannam@170
|
41 #include "../Exceptions.h"
|
cannam@150
|
42
|
cannam@150
|
43 #include <cstdint>
|
cannam@150
|
44
|
cannam@150
|
45 namespace piper_vamp {
|
cannam@150
|
46 namespace client {
|
cannam@150
|
47
|
cannam@170
|
48 /**
|
cannam@187
|
49 * AutoPlugin presents a Piper feature extractor in the form of a Vamp
|
cannam@187
|
50 * plugin, managing its own single-use server instance. That is, the
|
cannam@187
|
51 * distinguishing quality of AutoPlugin (in comparison with
|
cannam@187
|
52 * PluginStub) is that it runs and terminates its own Piper server,
|
cannam@187
|
53 * whose lifetime matches that of the plugin.
|
cannam@187
|
54 *
|
cannam@187
|
55 * Example usage:
|
cannam@187
|
56 *
|
cannam@187
|
57 * Vamp::Plugin *plugin =
|
cannam@187
|
58 * new AutoPlugin("piper-server-name.exe",
|
cannam@187
|
59 * "vamp-example-plugins:zerocrossing",
|
cannam@187
|
60 * 44100.0f,
|
cannam@187
|
61 * Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE,
|
cannam@187
|
62 * nullptr);
|
cannam@187
|
63 * plugin->initialise(...);
|
cannam@187
|
64 * plugin->process(...); <-- in the normal way for a Vamp plugin
|
cannam@187
|
65 * delete plugin; <-- causes the server to exit
|
cannam@187
|
66 *
|
cannam@187
|
67 * AutoPlugin makes use of the Loader and PluginClient interfaces,
|
cannam@187
|
68 * providing them its own transport layer object for its single server.
|
cannam@170
|
69 *
|
cannam@170
|
70 * Note that any method may throw ServerCrashed, RequestTimedOut or
|
cannam@170
|
71 * ProtocolError exceptions.
|
cannam@170
|
72 */
|
cannam@150
|
73 class AutoPlugin : public Vamp::Plugin
|
cannam@150
|
74 {
|
cannam@150
|
75 public:
|
cannam@187
|
76 /**
|
cannam@187
|
77 * Construct an AutoPlugin that runs an instance of the Piper
|
cannam@187
|
78 * server with the given server name (executable path), requesting
|
cannam@187
|
79 * the given plugin key from the server.
|
cannam@187
|
80 *
|
cannam@187
|
81 * \param adapterFlags a bitwise OR of the values in the
|
cannam@187
|
82 * Vamp::HostExt::PluginLoader::AdapterFlags enumeration
|
cannam@187
|
83 *
|
cannam@187
|
84 * \param logger an optional callback for log messages. Pass a
|
cannam@187
|
85 * null pointer to use cerr instead.
|
cannam@187
|
86 */
|
cannam@150
|
87 AutoPlugin(std::string serverName,
|
cannam@150
|
88 std::string pluginKey,
|
cannam@150
|
89 float inputSampleRate,
|
cannam@150
|
90 int adapterFlags,
|
cannam@187
|
91 LogCallback *logger) :
|
cannam@150
|
92 Vamp::Plugin(inputSampleRate),
|
cannam@150
|
93 m_logger(logger),
|
cannam@150
|
94 m_transport(serverName, "capnp", logger),
|
cannam@150
|
95 m_client(&m_transport, logger)
|
cannam@150
|
96 {
|
cannam@150
|
97 LoadRequest req;
|
cannam@150
|
98 req.pluginKey = pluginKey;
|
cannam@150
|
99 req.inputSampleRate = inputSampleRate;
|
cannam@150
|
100 req.adapterFlags = adapterFlags;
|
cannam@150
|
101 try {
|
cannam@150
|
102 LoadResponse resp = m_client.loadPlugin(req);
|
cannam@150
|
103 m_plugin = resp.plugin;
|
cannam@150
|
104 } catch (ServerCrashed c) {
|
cannam@150
|
105 log(std::string("AutoPlugin: Server crashed: ") + c.what());
|
cannam@150
|
106 m_plugin = 0;
|
cannam@150
|
107 }
|
cannam@150
|
108 }
|
cannam@150
|
109
|
cannam@150
|
110 virtual ~AutoPlugin() {
|
cannam@150
|
111 delete m_plugin;
|
cannam@187
|
112 // The transport is a plain data member and will be deleted
|
cannam@187
|
113 // here, which will have the effect of terminating the server
|
cannam@150
|
114 }
|
cannam@150
|
115
|
cannam@150
|
116 bool isOK() const {
|
cannam@150
|
117 return (m_plugin != nullptr);
|
cannam@150
|
118 }
|
cannam@150
|
119
|
cannam@150
|
120 virtual std::string getIdentifier() const {
|
cannam@150
|
121 return getPlugin()->getIdentifier();
|
cannam@150
|
122 }
|
cannam@150
|
123
|
cannam@150
|
124 virtual std::string getName() const {
|
cannam@150
|
125 return getPlugin()->getName();
|
cannam@150
|
126 }
|
cannam@150
|
127
|
cannam@150
|
128 virtual std::string getDescription() const {
|
cannam@150
|
129 return getPlugin()->getDescription();
|
cannam@150
|
130 }
|
cannam@150
|
131
|
cannam@150
|
132 virtual std::string getMaker() const {
|
cannam@150
|
133 return getPlugin()->getMaker();
|
cannam@150
|
134 }
|
cannam@150
|
135
|
cannam@150
|
136 virtual std::string getCopyright() const {
|
cannam@150
|
137 return getPlugin()->getCopyright();
|
cannam@150
|
138 }
|
cannam@150
|
139
|
cannam@150
|
140 virtual int getPluginVersion() const {
|
cannam@150
|
141 return getPlugin()->getPluginVersion();
|
cannam@150
|
142 }
|
cannam@150
|
143
|
cannam@150
|
144 virtual ParameterList getParameterDescriptors() const {
|
cannam@150
|
145 return getPlugin()->getParameterDescriptors();
|
cannam@150
|
146 }
|
cannam@150
|
147
|
cannam@150
|
148 virtual float getParameter(std::string name) const {
|
cannam@150
|
149 return getPlugin()->getParameter(name);
|
cannam@150
|
150 }
|
cannam@150
|
151
|
cannam@150
|
152 virtual void setParameter(std::string name, float value) {
|
cannam@150
|
153 getPlugin()->setParameter(name, value);
|
cannam@150
|
154 }
|
cannam@150
|
155
|
cannam@150
|
156 virtual ProgramList getPrograms() const {
|
cannam@150
|
157 return getPlugin()->getPrograms();
|
cannam@150
|
158 }
|
cannam@150
|
159
|
cannam@150
|
160 virtual std::string getCurrentProgram() const {
|
cannam@150
|
161 return getPlugin()->getCurrentProgram();
|
cannam@150
|
162 }
|
cannam@150
|
163
|
cannam@150
|
164 virtual void selectProgram(std::string program) {
|
cannam@150
|
165 getPlugin()->selectProgram(program);
|
cannam@150
|
166 }
|
cannam@150
|
167
|
cannam@150
|
168 virtual bool initialise(size_t inputChannels,
|
cannam@150
|
169 size_t stepSize,
|
cannam@150
|
170 size_t blockSize) {
|
cannam@186
|
171 return getPlugin()->initialise(inputChannels, stepSize, blockSize);
|
cannam@150
|
172 }
|
cannam@150
|
173
|
cannam@150
|
174 virtual void reset() {
|
cannam@150
|
175 getPlugin()->reset();
|
cannam@150
|
176 }
|
cannam@150
|
177
|
cannam@150
|
178 virtual InputDomain getInputDomain() const {
|
cannam@150
|
179 return getPlugin()->getInputDomain();
|
cannam@150
|
180 }
|
cannam@150
|
181
|
cannam@150
|
182 virtual size_t getPreferredBlockSize() const {
|
cannam@150
|
183 return getPlugin()->getPreferredBlockSize();
|
cannam@150
|
184 }
|
cannam@150
|
185
|
cannam@150
|
186 virtual size_t getPreferredStepSize() const {
|
cannam@150
|
187 return getPlugin()->getPreferredStepSize();
|
cannam@150
|
188 }
|
cannam@150
|
189
|
cannam@150
|
190 virtual size_t getMinChannelCount() const {
|
cannam@150
|
191 return getPlugin()->getMinChannelCount();
|
cannam@150
|
192 }
|
cannam@150
|
193
|
cannam@150
|
194 virtual size_t getMaxChannelCount() const {
|
cannam@150
|
195 return getPlugin()->getMaxChannelCount();
|
cannam@150
|
196 }
|
cannam@150
|
197
|
cannam@150
|
198 virtual OutputList getOutputDescriptors() const {
|
cannam@150
|
199 return getPlugin()->getOutputDescriptors();
|
cannam@150
|
200 }
|
cannam@150
|
201
|
cannam@150
|
202 virtual FeatureSet process(const float *const *inputBuffers,
|
cannam@150
|
203 Vamp::RealTime timestamp) {
|
cannam@150
|
204 return getPlugin()->process(inputBuffers, timestamp);
|
cannam@150
|
205 }
|
cannam@150
|
206
|
cannam@150
|
207 virtual FeatureSet getRemainingFeatures() {
|
cannam@150
|
208 return getPlugin()->getRemainingFeatures();
|
cannam@150
|
209 }
|
cannam@150
|
210
|
cannam@150
|
211 private:
|
cannam@150
|
212 LogCallback *m_logger;
|
cannam@150
|
213 ProcessQtTransport m_transport;
|
cannam@150
|
214 CapnpRRClient m_client;
|
cannam@150
|
215 Vamp::Plugin *m_plugin;
|
cannam@150
|
216 Vamp::Plugin *getPlugin() const {
|
cannam@150
|
217 if (!m_plugin) {
|
cannam@150
|
218 log("AutoPlugin: getPlugin() failed (caller should have called AutoPlugin::isOK)");
|
cannam@150
|
219 throw std::logic_error("Plugin load failed");
|
cannam@150
|
220 }
|
cannam@150
|
221 return m_plugin;
|
cannam@150
|
222 }
|
cannam@150
|
223
|
cannam@150
|
224 void log(std::string message) const {
|
cannam@150
|
225 if (m_logger) m_logger->log(message);
|
cannam@150
|
226 else std::cerr << message << std::endl;
|
cannam@150
|
227 }
|
cannam@150
|
228 };
|
cannam@150
|
229
|
cannam@150
|
230 }
|
cannam@150
|
231 }
|
cannam@150
|
232
|
cannam@150
|
233 #endif
|
cannam@150
|
234
|
cannam@150
|
235
|