annotate vamp-capnp/VampnProto.h @ 116:d15cb1151d76

Add JSON support directly to the server. Had hoped to avoid this (using Capnp as canonical in the server and then converting externally as necessary) but it's just too useful for debugging purposes when bundled with client app
author Chris Cannam <c.cannam@qmul.ac.uk>
date Thu, 27 Oct 2016 11:39:41 +0100
parents cd438188e3f9
children 5b113c87b6e6
rev   line source
c@75 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
c@75 2
c@75 3 /*
c@75 4 Piper C++
c@75 5
c@75 6 Centre for Digital Music, Queen Mary, University of London.
c@75 7 Copyright 2015-2016 QMUL.
c@75 8
c@75 9 Permission is hereby granted, free of charge, to any person
c@75 10 obtaining a copy of this software and associated documentation
c@75 11 files (the "Software"), to deal in the Software without
c@75 12 restriction, including without limitation the rights to use, copy,
c@75 13 modify, merge, publish, distribute, sublicense, and/or sell copies
c@75 14 of the Software, and to permit persons to whom the Software is
c@75 15 furnished to do so, subject to the following conditions:
c@75 16
c@75 17 The above copyright notice and this permission notice shall be
c@75 18 included in all copies or substantial portions of the Software.
c@75 19
c@75 20 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
c@75 21 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
c@75 22 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
c@75 23 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
c@75 24 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
c@75 25 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
c@75 26 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
c@75 27
c@75 28 Except as contained in this notice, the names of the Centre for
c@75 29 Digital Music; Queen Mary, University of London; and Chris Cannam
c@75 30 shall not be used in advertising or otherwise to promote the sale,
c@75 31 use or other dealings in this Software without prior written
c@75 32 authorization.
c@75 33 */
c@75 34
c@75 35 #include "piper.capnp.h"
c@75 36
c@75 37 #include <capnp/message.h>
c@75 38
c@75 39 #include <vamp-hostsdk/Plugin.h>
c@75 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@97 44 #include "vamp-support/RequestResponse.h"
c@75 45
c@75 46 #include "vamp-support/PluginHandleMapper.h"
c@75 47 #include "vamp-support/PluginOutputIdMapper.h"
c@75 48 #include "vamp-support/RequestResponseType.h"
c@75 49
c@97 50 namespace piper_vamp
c@75 51 {
c@75 52
c@75 53 /**
c@75 54 * Convert the structures laid out in the Vamp SDK classes into Cap'n
c@75 55 * Proto structures (and back again).
c@75 56 *
c@75 57 * At least some of this will be necessary for any implementation
c@75 58 * using Cap'n Proto that uses the C++ Vamp SDK to provide its
c@75 59 * reference structures. An implementation could alternatively use the
c@75 60 * Cap'n Proto structures directly, and interact with Vamp plugins
c@75 61 * using the Vamp C API, without using the C++ Vamp SDK classes at
c@75 62 * all. That would avoid a lot of copying (in Cap'n Proto style).
c@75 63 */
c@75 64 class VampnProto
c@75 65 {
c@75 66 public:
c@75 67 typedef ::capnp::MallocMessageBuilder MsgBuilder;
c@75 68
c@75 69 template <typename T, typename B>
c@75 70 static void buildBasicDescriptor(B &basic, const T &t) {
c@75 71 basic.setIdentifier(t.identifier);
c@75 72 basic.setName(t.name);
c@75 73 basic.setDescription(t.description);
c@75 74 }
c@75 75
c@75 76 template <typename T, typename B>
c@75 77 static void readBasicDescriptor(T &t, const B &basic) {
c@75 78 t.identifier = basic.getIdentifier();
c@75 79 t.name = basic.getName();
c@75 80 t.description = basic.getDescription();
c@75 81 }
c@75 82
c@75 83 template <typename T, typename M>
c@75 84 static void buildValueExtents(M &m, const T &t) {
c@75 85 m.setMinValue(t.minValue);
c@75 86 m.setMaxValue(t.maxValue);
c@75 87 }
c@75 88
c@75 89 template <typename T, typename M>
c@75 90 static void readValueExtents(T &t, const M &m) {
c@75 91 t.minValue = m.getMinValue();
c@75 92 t.maxValue = m.getMaxValue();
c@75 93 }
c@75 94
c@97 95 static void buildRealTime(piper::RealTime::Builder &b,
c@97 96 const Vamp::RealTime &t) {
c@75 97 b.setSec(t.sec);
c@75 98 b.setNsec(t.nsec);
c@75 99 }
c@75 100
c@97 101 static void readRealTime(Vamp::RealTime &t,
c@97 102 const piper::RealTime::Reader &r) {
c@75 103 t.sec = r.getSec();
c@75 104 t.nsec = r.getNsec();
c@75 105 }
c@75 106
c@97 107 static piper::SampleType
c@75 108 fromSampleType(Vamp::Plugin::OutputDescriptor::SampleType t) {
c@75 109 switch (t) {
c@75 110 case Vamp::Plugin::OutputDescriptor::OneSamplePerStep:
c@97 111 return piper::SampleType::ONE_SAMPLE_PER_STEP;
c@75 112 case Vamp::Plugin::OutputDescriptor::FixedSampleRate:
c@97 113 return piper::SampleType::FIXED_SAMPLE_RATE;
c@75 114 case Vamp::Plugin::OutputDescriptor::VariableSampleRate:
c@97 115 return piper::SampleType::VARIABLE_SAMPLE_RATE;
c@75 116 }
c@75 117 throw std::logic_error("unexpected Vamp SampleType enum value");
c@75 118 }
c@75 119
c@75 120 static Vamp::Plugin::OutputDescriptor::SampleType
c@97 121 toSampleType(piper::SampleType t) {
c@75 122 switch (t) {
c@97 123 case piper::SampleType::ONE_SAMPLE_PER_STEP:
c@75 124 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
c@97 125 case piper::SampleType::FIXED_SAMPLE_RATE:
c@75 126 return Vamp::Plugin::OutputDescriptor::FixedSampleRate;
c@97 127 case piper::SampleType::VARIABLE_SAMPLE_RATE:
c@75 128 return Vamp::Plugin::OutputDescriptor::VariableSampleRate;
c@75 129 }
c@75 130 throw std::logic_error("unexpected Capnp SampleType enum value");
c@75 131 }
c@75 132
c@75 133 static void
c@97 134 buildConfiguredOutputDescriptor(piper::ConfiguredOutputDescriptor::Builder &b,
c@75 135 const Vamp::Plugin::OutputDescriptor &od) {
c@75 136
c@75 137 b.setUnit(od.unit);
c@75 138 b.setSampleType(fromSampleType(od.sampleType));
c@75 139 b.setSampleRate(od.sampleRate);
c@75 140 b.setHasDuration(od.hasDuration);
c@75 141
c@75 142 b.setHasFixedBinCount(od.hasFixedBinCount);
c@75 143 if (od.hasFixedBinCount) {
c@102 144 b.setBinCount(int(od.binCount));
c@75 145 if (od.binNames.size() > 0) {
c@102 146 auto binNames = b.initBinNames(unsigned(od.binNames.size()));
c@102 147 for (int i = 0; i < int(od.binNames.size()); ++i) {
c@75 148 binNames.set(i, od.binNames[i]);
c@75 149 }
c@75 150 }
c@75 151 }
c@75 152
c@75 153 b.setHasKnownExtents(od.hasKnownExtents);
c@75 154 if (od.hasKnownExtents) {
c@75 155 buildValueExtents(b, od);
c@75 156 }
c@75 157
c@75 158 b.setIsQuantized(od.isQuantized);
c@75 159 if (od.isQuantized) {
c@75 160 b.setQuantizeStep(od.quantizeStep);
c@75 161 }
c@75 162 }
c@75 163
c@75 164 static void
c@97 165 buildOutputDescriptor(piper::OutputDescriptor::Builder &b,
c@75 166 const Vamp::Plugin::OutputDescriptor &od) {
c@75 167
c@75 168 auto basic = b.initBasic();
c@75 169 buildBasicDescriptor(basic, od);
c@75 170
c@75 171 auto configured = b.initConfigured();
c@75 172 buildConfiguredOutputDescriptor(configured, od);
c@75 173 }
c@75 174
c@75 175 static void
c@75 176 readConfiguredOutputDescriptor(Vamp::Plugin::OutputDescriptor &od,
c@97 177 const piper::ConfiguredOutputDescriptor::Reader &r) {
c@75 178
c@75 179 od.unit = r.getUnit();
c@75 180
c@75 181 od.sampleType = toSampleType(r.getSampleType());
c@75 182 od.sampleRate = r.getSampleRate();
c@75 183 od.hasDuration = r.getHasDuration();
c@75 184
c@75 185 od.hasFixedBinCount = r.getHasFixedBinCount();
c@75 186 if (od.hasFixedBinCount) {
c@75 187 od.binCount = r.getBinCount();
c@75 188 od.binNames.clear();
c@75 189 auto nn = r.getBinNames();
c@75 190 for (const auto &n: nn) {
c@75 191 od.binNames.push_back(n);
c@75 192 }
c@75 193 }
c@75 194
c@75 195 od.hasKnownExtents = r.getHasKnownExtents();
c@75 196 if (od.hasKnownExtents) {
c@75 197 readValueExtents(od, r);
c@75 198 }
c@75 199
c@75 200 od.isQuantized = r.getIsQuantized();
c@75 201 if (od.isQuantized) {
c@75 202 od.quantizeStep = r.getQuantizeStep();
c@75 203 }
c@75 204 }
c@75 205
c@75 206 static void
c@75 207 readOutputDescriptor(Vamp::Plugin::OutputDescriptor &od,
c@97 208 const piper::OutputDescriptor::Reader &r) {
c@75 209
c@75 210 readBasicDescriptor(od, r.getBasic());
c@75 211 readConfiguredOutputDescriptor(od, r.getConfigured());
c@75 212 }
c@75 213
c@75 214 static void
c@97 215 buildParameterDescriptor(piper::ParameterDescriptor::Builder &b,
c@75 216 const Vamp::Plugin::ParameterDescriptor &pd) {
c@75 217
c@75 218 auto basic = b.initBasic();
c@75 219 buildBasicDescriptor(basic, pd);
c@75 220
c@75 221 b.setUnit(pd.unit);
c@75 222
c@75 223 buildValueExtents(b, pd);
c@75 224
c@75 225 b.setDefaultValue(pd.defaultValue);
c@75 226
c@75 227 b.setIsQuantized(pd.isQuantized);
c@75 228 if (pd.isQuantized) {
c@75 229 b.setQuantizeStep(pd.quantizeStep);
c@75 230 }
c@75 231
c@75 232 if (pd.valueNames.size() > 0) {
c@102 233 auto valueNames = b.initValueNames(unsigned(pd.valueNames.size()));
c@102 234 for (int i = 0; i < int(pd.valueNames.size()); ++i) {
c@75 235 valueNames.set(i, pd.valueNames[i]);
c@75 236 }
c@75 237 }
c@75 238 }
c@75 239
c@75 240 static void
c@75 241 readParameterDescriptor(Vamp::Plugin::ParameterDescriptor &pd,
c@97 242 const piper::ParameterDescriptor::Reader &r) {
c@75 243
c@75 244 readBasicDescriptor(pd, r.getBasic());
c@75 245
c@75 246 pd.unit = r.getUnit();
c@75 247
c@75 248 readValueExtents(pd, r);
c@75 249
c@75 250 pd.defaultValue = r.getDefaultValue();
c@75 251
c@75 252 pd.isQuantized = r.getIsQuantized();
c@75 253 if (pd.isQuantized) {
c@75 254 pd.quantizeStep = r.getQuantizeStep();
c@75 255 }
c@75 256
c@75 257 pd.valueNames.clear();
c@75 258 auto nn = r.getValueNames();
c@75 259 for (const auto &n: nn) {
c@75 260 pd.valueNames.push_back(n);
c@75 261 }
c@75 262 }
c@75 263
c@75 264 static void
c@97 265 buildFeature(piper::Feature::Builder &b,
c@75 266 const Vamp::Plugin::Feature &f) {
c@75 267
c@75 268 b.setHasTimestamp(f.hasTimestamp);
c@75 269 if (f.hasTimestamp) {
c@75 270 auto timestamp = b.initTimestamp();
c@75 271 buildRealTime(timestamp, f.timestamp);
c@75 272 }
c@75 273
c@75 274 b.setHasDuration(f.hasDuration);
c@75 275 if (f.hasDuration) {
c@75 276 auto duration = b.initDuration();
c@75 277 buildRealTime(duration, f.duration);
c@75 278 }
c@75 279
c@75 280 b.setLabel(f.label);
c@75 281
c@75 282 if (f.values.size() > 0) {
c@102 283 auto values = b.initFeatureValues(unsigned(f.values.size()));
c@102 284 for (int i = 0; i < int(f.values.size()); ++i) {
c@75 285 values.set(i, f.values[i]);
c@75 286 }
c@75 287 }
c@75 288 }
c@75 289
c@75 290 static void
c@75 291 readFeature(Vamp::Plugin::Feature &f,
c@97 292 const piper::Feature::Reader &r) {
c@75 293
c@75 294 f.hasTimestamp = r.getHasTimestamp();
c@75 295 if (f.hasTimestamp) {
c@75 296 readRealTime(f.timestamp, r.getTimestamp());
c@75 297 }
c@75 298
c@75 299 f.hasDuration = r.getHasDuration();
c@75 300 if (f.hasDuration) {
c@75 301 readRealTime(f.duration, r.getDuration());
c@75 302 }
c@75 303
c@75 304 f.label = r.getLabel();
c@75 305
c@75 306 f.values.clear();
c@75 307 auto vv = r.getFeatureValues();
c@75 308 for (auto v: vv) {
c@75 309 f.values.push_back(v);
c@75 310 }
c@75 311 }
c@75 312
c@75 313 static void
c@97 314 buildFeatureSet(piper::FeatureSet::Builder &b,
c@75 315 const Vamp::Plugin::FeatureSet &fs,
c@75 316 const PluginOutputIdMapper &omapper) {
c@75 317
c@102 318 auto featureset = b.initFeaturePairs(unsigned(fs.size()));
c@75 319 int ix = 0;
c@75 320 for (const auto &fsi : fs) {
c@75 321 auto fspair = featureset[ix];
c@75 322 fspair.setOutput(omapper.indexToId(fsi.first));
c@102 323 auto featurelist = fspair.initFeatures(unsigned(fsi.second.size()));
c@102 324 for (int j = 0; j < int(fsi.second.size()); ++j) {
c@75 325 auto feature = featurelist[j];
c@75 326 buildFeature(feature, fsi.second[j]);
c@75 327 }
c@75 328 ++ix;
c@75 329 }
c@75 330 }
c@75 331
c@75 332 static void
c@75 333 readFeatureSet(Vamp::Plugin::FeatureSet &fs,
c@97 334 const piper::FeatureSet::Reader &r,
c@75 335 const PluginOutputIdMapper &omapper) {
c@75 336
c@75 337 fs.clear();
c@75 338 auto pp = r.getFeaturePairs();
c@75 339 for (const auto &p: pp) {
c@75 340 Vamp::Plugin::FeatureList vfl;
c@75 341 auto ff = p.getFeatures();
c@75 342 for (const auto &f: ff) {
c@75 343 Vamp::Plugin::Feature vf;
c@75 344 readFeature(vf, f);
c@75 345 vfl.push_back(vf);
c@75 346 }
c@75 347 fs[omapper.idToIndex(p.getOutput())] = vfl;
c@75 348 }
c@75 349 }
c@75 350
c@97 351 static piper::InputDomain
c@75 352 fromInputDomain(Vamp::Plugin::InputDomain d) {
c@75 353 switch(d) {
c@75 354 case Vamp::Plugin::TimeDomain:
c@97 355 return piper::InputDomain::TIME_DOMAIN;
c@75 356 case Vamp::Plugin::FrequencyDomain:
c@97 357 return piper::InputDomain::FREQUENCY_DOMAIN;
c@75 358 default:
c@75 359 throw std::logic_error("unexpected Vamp InputDomain enum value");
c@75 360 }
c@75 361 }
c@75 362
c@75 363 static Vamp::Plugin::InputDomain
c@97 364 toInputDomain(piper::InputDomain d) {
c@75 365 switch(d) {
c@97 366 case piper::InputDomain::TIME_DOMAIN:
c@75 367 return Vamp::Plugin::TimeDomain;
c@97 368 case piper::InputDomain::FREQUENCY_DOMAIN:
c@75 369 return Vamp::Plugin::FrequencyDomain;
c@75 370 default:
c@75 371 throw std::logic_error("unexpected Capnp InputDomain enum value");
c@75 372 }
c@75 373 }
c@75 374
c@75 375 static void
c@97 376 buildExtractorStaticData(piper::ExtractorStaticData::Builder &b,
c@97 377 const PluginStaticData &d) {
c@75 378
c@75 379 b.setKey(d.pluginKey);
c@75 380
c@75 381 auto basic = b.initBasic();
c@75 382 buildBasicDescriptor(basic, d.basic);
c@75 383
c@75 384 b.setMaker(d.maker);
c@75 385 b.setCopyright(d.copyright);
c@75 386 b.setVersion(d.pluginVersion);
c@75 387
c@102 388 auto clist = b.initCategory(unsigned(d.category.size()));
c@102 389 for (int i = 0; i < int(d.category.size()); ++i) {
c@75 390 clist.set(i, d.category[i]);
c@75 391 }
c@75 392
c@102 393 b.setMinChannelCount(int(d.minChannelCount));
c@102 394 b.setMaxChannelCount(int(d.maxChannelCount));
c@75 395
c@75 396 const auto &vparams = d.parameters;
c@102 397 auto plist = b.initParameters(unsigned(vparams.size()));
c@102 398 for (int i = 0; i < int(vparams.size()); ++i) {
c@75 399 auto pd = plist[i];
c@75 400 buildParameterDescriptor(pd, vparams[i]);
c@75 401 }
c@75 402
c@75 403 const auto &vprogs = d.programs;
c@102 404 auto pglist = b.initPrograms(unsigned(vprogs.size()));
c@102 405 for (int i = 0; i < int(vprogs.size()); ++i) {
c@75 406 pglist.set(i, vprogs[i]);
c@75 407 }
c@75 408
c@75 409 b.setInputDomain(fromInputDomain(d.inputDomain));
c@75 410
c@75 411 const auto &vouts = d.basicOutputInfo;
c@102 412 auto olist = b.initBasicOutputInfo(unsigned(vouts.size()));
c@102 413 for (int i = 0; i < int(vouts.size()); ++i) {
c@75 414 auto od = olist[i];
c@75 415 buildBasicDescriptor(od, vouts[i]);
c@75 416 }
c@75 417 }
c@75 418
c@75 419 static void
c@97 420 readExtractorStaticData(PluginStaticData &d,
c@97 421 const piper::ExtractorStaticData::Reader &r) {
c@75 422
c@75 423 d.pluginKey = r.getKey();
c@75 424
c@75 425 readBasicDescriptor(d.basic, r.getBasic());
c@75 426
c@75 427 d.maker = r.getMaker();
c@75 428 d.copyright = r.getCopyright();
c@75 429 d.pluginVersion = r.getVersion();
c@75 430
c@75 431 d.category.clear();
c@75 432 auto cc = r.getCategory();
c@75 433 for (auto c: cc) {
c@75 434 d.category.push_back(c);
c@75 435 }
c@75 436
c@75 437 d.minChannelCount = r.getMinChannelCount();
c@75 438 d.maxChannelCount = r.getMaxChannelCount();
c@75 439
c@75 440 d.parameters.clear();
c@75 441 auto pp = r.getParameters();
c@75 442 for (auto p: pp) {
c@75 443 Vamp::Plugin::ParameterDescriptor pd;
c@75 444 readParameterDescriptor(pd, p);
c@75 445 d.parameters.push_back(pd);
c@75 446 }
c@75 447
c@75 448 d.programs.clear();
c@75 449 auto prp = r.getPrograms();
c@75 450 for (auto p: prp) {
c@75 451 d.programs.push_back(p);
c@75 452 }
c@75 453
c@75 454 d.inputDomain = toInputDomain(r.getInputDomain());
c@75 455
c@75 456 d.basicOutputInfo.clear();
c@75 457 auto oo = r.getBasicOutputInfo();
c@75 458 for (auto o: oo) {
c@97 459 PluginStaticData::Basic b;
c@75 460 readBasicDescriptor(b, o);
c@75 461 d.basicOutputInfo.push_back(b);
c@75 462 }
c@75 463 }
c@75 464
c@75 465 static void
c@97 466 buildConfiguration(piper::Configuration::Builder &b,
c@97 467 const PluginConfiguration &c) {
c@75 468
c@75 469 const auto &vparams = c.parameterValues;
c@102 470 auto params = b.initParameterValues(unsigned(vparams.size()));
c@75 471 int i = 0;
c@75 472 for (const auto &pp : vparams) {
c@75 473 auto param = params[i++];
c@75 474 param.setParameter(pp.first);
c@75 475 param.setValue(pp.second);
c@75 476 }
c@75 477
c@75 478 b.setCurrentProgram(c.currentProgram);
c@75 479 b.setChannelCount(c.channelCount);
c@75 480 b.setStepSize(c.stepSize);
c@75 481 b.setBlockSize(c.blockSize);
c@75 482 }
c@75 483
c@75 484 static void
c@97 485 readConfiguration(PluginConfiguration &c,
c@97 486 const piper::Configuration::Reader &r) {
c@75 487
c@75 488 auto pp = r.getParameterValues();
c@75 489 for (const auto &p: pp) {
c@75 490 c.parameterValues[p.getParameter()] = p.getValue();
c@75 491 }
c@75 492
c@75 493 c.currentProgram = r.getCurrentProgram();
c@75 494 c.channelCount = r.getChannelCount();
c@75 495 c.stepSize = r.getStepSize();
c@75 496 c.blockSize = r.getBlockSize();
c@75 497 }
c@75 498
c@75 499 static void
c@97 500 readListResponse(ListResponse &lr,
c@97 501 const piper::ListResponse::Reader &r) {
c@95 502
c@95 503 lr.available.clear();
c@95 504 auto pp = r.getAvailable();
c@95 505 for (const auto &p: pp) {
c@97 506 PluginStaticData psd;
c@95 507 readExtractorStaticData(psd, p);
c@95 508 lr.available.push_back(psd);
c@95 509 }
c@95 510 }
c@95 511
c@95 512 static void
c@97 513 buildLoadRequest(piper::LoadRequest::Builder &r,
c@97 514 const LoadRequest &req) {
c@75 515
c@75 516 r.setKey(req.pluginKey);
c@75 517 r.setInputSampleRate(req.inputSampleRate);
c@75 518
c@97 519 std::vector<piper::AdapterFlag> flags;
c@75 520 if (req.adapterFlags & Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN) {
c@97 521 flags.push_back(piper::AdapterFlag::ADAPT_INPUT_DOMAIN);
c@75 522 }
c@75 523 if (req.adapterFlags & Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT) {
c@97 524 flags.push_back(piper::AdapterFlag::ADAPT_CHANNEL_COUNT);
c@75 525 }
c@75 526 if (req.adapterFlags & Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE) {
c@97 527 flags.push_back(piper::AdapterFlag::ADAPT_BUFFER_SIZE);
c@75 528 }
c@75 529
c@102 530 auto f = r.initAdapterFlags(unsigned(flags.size()));
c@102 531 for (int i = 0; i < int(flags.size()); ++i) {
c@75 532 f.set(i, flags[i]);
c@75 533 }
c@75 534 }
c@75 535
c@75 536 static void
c@97 537 readLoadRequest(LoadRequest &req,
c@97 538 const piper::LoadRequest::Reader &r) {
c@75 539
c@75 540 req.pluginKey = r.getKey();
c@75 541 req.inputSampleRate = r.getInputSampleRate();
c@75 542
c@75 543 int flags = 0;
c@75 544 auto aa = r.getAdapterFlags();
c@75 545 for (auto a: aa) {
c@97 546 if (a == piper::AdapterFlag::ADAPT_INPUT_DOMAIN) {
c@75 547 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN;
c@75 548 }
c@97 549 if (a == piper::AdapterFlag::ADAPT_CHANNEL_COUNT) {
c@75 550 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT;
c@75 551 }
c@97 552 if (a == piper::AdapterFlag::ADAPT_BUFFER_SIZE) {
c@75 553 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE;
c@75 554 }
c@75 555 }
c@75 556 req.adapterFlags = flags;
c@75 557 }
c@75 558
c@75 559 static void
c@97 560 buildLoadResponse(piper::LoadResponse::Builder &b,
c@97 561 const LoadResponse &resp,
c@75 562 const PluginHandleMapper &pmapper) {
c@75 563
c@75 564 b.setHandle(pmapper.pluginToHandle(resp.plugin));
c@75 565 auto sd = b.initStaticData();
c@75 566 buildExtractorStaticData(sd, resp.staticData);
c@75 567 auto conf = b.initDefaultConfiguration();
c@75 568 buildConfiguration(conf, resp.defaultConfiguration);
c@75 569 }
c@75 570
c@75 571 static void
c@97 572 readLoadResponse(LoadResponse &resp,
c@97 573 const piper::LoadResponse::Reader &r,
c@75 574 const PluginHandleMapper &pmapper) {
c@75 575
c@75 576 resp.plugin = pmapper.handleToPlugin(r.getHandle());
c@75 577 readExtractorStaticData(resp.staticData, r.getStaticData());
c@75 578 readConfiguration(resp.defaultConfiguration,
c@80 579 r.getDefaultConfiguration());
c@75 580 }
c@75 581
c@75 582 static void
c@97 583 buildConfigurationRequest(piper::ConfigurationRequest::Builder &b,
c@97 584 const ConfigurationRequest &cr,
c@75 585 const PluginHandleMapper &pmapper) {
c@75 586
c@75 587 b.setHandle(pmapper.pluginToHandle(cr.plugin));
c@75 588 auto c = b.initConfiguration();
c@75 589 buildConfiguration(c, cr.configuration);
c@75 590 }
c@75 591
c@75 592 static void
c@97 593 readConfigurationRequest(ConfigurationRequest &cr,
c@97 594 const piper::ConfigurationRequest::Reader &r,
c@75 595 const PluginHandleMapper &pmapper) {
c@75 596
c@75 597 auto h = r.getHandle();
c@75 598 cr.plugin = pmapper.handleToPlugin(h);
c@75 599 auto c = r.getConfiguration();
c@75 600 readConfiguration(cr.configuration, c);
c@75 601 }
c@75 602
c@75 603 static void
c@97 604 buildConfigurationResponse(piper::ConfigurationResponse::Builder &b,
c@97 605 const ConfigurationResponse &cr,
c@75 606 const PluginHandleMapper &pmapper) {
c@75 607
c@75 608 b.setHandle(pmapper.pluginToHandle(cr.plugin));
c@102 609 auto olist = b.initOutputs(unsigned(cr.outputs.size()));
c@102 610 for (int i = 0; i < int(cr.outputs.size()); ++i) {
c@75 611 auto od = olist[i];
c@75 612 buildOutputDescriptor(od, cr.outputs[i]);
c@75 613 }
c@75 614 }
c@75 615
c@75 616 static void
c@97 617 readConfigurationResponse(ConfigurationResponse &cr,
c@97 618 const piper::ConfigurationResponse::Reader &r,
c@75 619 const PluginHandleMapper &pmapper) {
c@75 620
c@75 621 cr.plugin = pmapper.handleToPlugin(r.getHandle());
c@75 622 cr.outputs.clear();
c@75 623 auto oo = r.getOutputs();
c@75 624 for (const auto &o: oo) {
c@75 625 Vamp::Plugin::OutputDescriptor desc;
c@75 626 readOutputDescriptor(desc, o);
c@75 627 cr.outputs.push_back(desc);
c@75 628 }
c@75 629 }
c@75 630
c@75 631 static void
c@97 632 buildProcessInput(piper::ProcessInput::Builder &b,
c@75 633 Vamp::RealTime timestamp,
c@75 634 const std::vector<std::vector<float> > &buffers) {
c@75 635
c@75 636 auto t = b.initTimestamp();
c@75 637 buildRealTime(t, timestamp);
c@102 638 auto vv = b.initInputBuffers(unsigned(buffers.size()));
c@102 639 for (int ch = 0; ch < int(buffers.size()); ++ch) {
c@75 640 const int n = int(buffers[ch].size());
c@75 641 vv.init(ch, n);
c@75 642 auto v = vv[ch];
c@75 643 for (int i = 0; i < n; ++i) {
c@75 644 v.set(i, buffers[ch][i]);
c@75 645 }
c@75 646 }
c@75 647 }
c@75 648
c@75 649 static void
c@75 650 readProcessInput(Vamp::RealTime &timestamp,
c@75 651 std::vector<std::vector<float> > &buffers,
c@97 652 const piper::ProcessInput::Reader &b) {
c@75 653
c@75 654 readRealTime(timestamp, b.getTimestamp());
c@75 655 buffers.clear();
c@75 656 auto vv = b.getInputBuffers();
c@75 657 for (const auto &v: vv) {
c@75 658 std::vector<float> buf;
c@75 659 for (auto x: v) {
c@75 660 buf.push_back(x);
c@75 661 }
c@75 662 buffers.push_back(buf);
c@75 663 }
c@75 664 }
c@75 665
c@75 666 static void
c@97 667 buildProcessRequest(piper::ProcessRequest::Builder &b,
c@97 668 const ProcessRequest &pr,
c@75 669 const PluginHandleMapper &pmapper) {
c@75 670
c@75 671 b.setHandle(pmapper.pluginToHandle(pr.plugin));
c@75 672 auto input = b.initProcessInput();
c@75 673 buildProcessInput(input, pr.timestamp, pr.inputBuffers);
c@75 674 }
c@75 675
c@75 676 static void
c@97 677 readProcessRequest(ProcessRequest &pr,
c@97 678 const piper::ProcessRequest::Reader &r,
c@75 679 const PluginHandleMapper &pmapper) {
c@75 680
c@75 681 auto h = r.getHandle();
c@75 682 pr.plugin = pmapper.handleToPlugin(h);
c@75 683 readProcessInput(pr.timestamp, pr.inputBuffers, r.getProcessInput());
c@75 684 }
c@75 685
c@75 686 static void
c@97 687 buildProcessResponse(piper::ProcessResponse::Builder &b,
c@97 688 const ProcessResponse &pr,
c@75 689 const PluginHandleMapper &pmapper) {
c@75 690
c@75 691 b.setHandle(pmapper.pluginToHandle(pr.plugin));
c@75 692 auto f = b.initFeatures();
c@75 693 buildFeatureSet(f, pr.features,
c@75 694 *pmapper.pluginToOutputIdMapper(pr.plugin));
c@75 695 }
c@75 696
c@75 697 static void
c@97 698 readProcessResponse(ProcessResponse &pr,
c@97 699 const piper::ProcessResponse::Reader &r,
c@75 700 const PluginHandleMapper &pmapper) {
c@75 701
c@75 702 auto h = r.getHandle();
c@75 703 pr.plugin = pmapper.handleToPlugin(h);
c@75 704 readFeatureSet(pr.features, r.getFeatures(),
c@75 705 *pmapper.handleToOutputIdMapper(r.getHandle()));
c@75 706 }
c@75 707
c@75 708 static void
c@97 709 buildFinishResponse(piper::FinishResponse::Builder &b,
c@97 710 const FinishResponse &pr,
c@75 711 const PluginHandleMapper &pmapper) {
c@75 712
c@75 713 b.setHandle(pmapper.pluginToHandle(pr.plugin));
c@75 714 auto f = b.initFeatures();
c@75 715 buildFeatureSet(f, pr.features,
c@75 716 *pmapper.pluginToOutputIdMapper(pr.plugin));
c@75 717 }
c@75 718
c@75 719 static void
c@97 720 readFinishResponse(FinishResponse &pr,
c@97 721 const piper::FinishResponse::Reader &r,
c@75 722 const PluginHandleMapper &pmapper) {
c@75 723
c@75 724 auto h = r.getHandle();
c@75 725 pr.plugin = pmapper.handleToPlugin(h);
c@75 726 readFeatureSet(pr.features, r.getFeatures(),
c@75 727 *pmapper.handleToOutputIdMapper(r.getHandle()));
c@75 728 }
c@75 729
c@75 730 static void
c@97 731 buildRpcRequest_List(piper::RpcRequest::Builder &b) {
c@75 732 b.getRequest().initList();
c@75 733 }
c@75 734
c@75 735 static void
c@97 736 buildRpcResponse_List(piper::RpcResponse::Builder &b,
c@97 737 const ListResponse &resp) {
c@75 738
c@75 739 auto r = b.getResponse().initList();
c@102 740 auto p = r.initAvailable(unsigned(resp.available.size()));
c@102 741 for (int i = 0; i < int(resp.available.size()); ++i) {
c@75 742 auto pd = p[i];
c@75 743 buildExtractorStaticData(pd, resp.available[i]);
c@75 744 }
c@75 745 }
c@75 746
c@75 747 static void
c@97 748 buildRpcRequest_Load(piper::RpcRequest::Builder &b,
c@97 749 const LoadRequest &req) {
c@75 750 auto u = b.getRequest().initLoad();
c@75 751 buildLoadRequest(u, req);
c@75 752 }
c@75 753
c@75 754 static void
c@97 755 buildRpcResponse_Load(piper::RpcResponse::Builder &b,
c@97 756 const LoadResponse &resp,
c@80 757 const PluginHandleMapper &pmapper) {
c@75 758
c@75 759 if (resp.plugin) {
c@75 760 auto u = b.getResponse().initLoad();
c@75 761 buildLoadResponse(u, resp, pmapper);
c@75 762 } else {
c@75 763 buildRpcResponse_Error(b, "Failed to load plugin", RRType::Load);
c@75 764 }
c@75 765 }
c@75 766
c@75 767 static void
c@97 768 buildRpcRequest_Configure(piper::RpcRequest::Builder &b,
c@97 769 const ConfigurationRequest &cr,
c@80 770 const PluginHandleMapper &pmapper) {
c@75 771 auto u = b.getRequest().initConfigure();
c@75 772 buildConfigurationRequest(u, cr, pmapper);
c@75 773 }
c@75 774
c@75 775 static void
c@97 776 buildRpcResponse_Configure(piper::RpcResponse::Builder &b,
c@97 777 const ConfigurationResponse &cr,
c@80 778 const PluginHandleMapper &pmapper) {
c@75 779
c@75 780 if (!cr.outputs.empty()) {
c@75 781 auto u = b.getResponse().initConfigure();
c@75 782 buildConfigurationResponse(u, cr, pmapper);
c@75 783 } else {
c@75 784 buildRpcResponse_Error(b, "Failed to configure plugin",
c@75 785 RRType::Configure);
c@75 786 }
c@75 787 }
c@75 788
c@75 789 static void
c@97 790 buildRpcRequest_Process(piper::RpcRequest::Builder &b,
c@97 791 const ProcessRequest &pr,
c@80 792 const PluginHandleMapper &pmapper) {
c@75 793 auto u = b.getRequest().initProcess();
c@75 794 buildProcessRequest(u, pr, pmapper);
c@75 795 }
c@75 796
c@75 797 static void
c@97 798 buildRpcResponse_Process(piper::RpcResponse::Builder &b,
c@97 799 const ProcessResponse &pr,
c@80 800 const PluginHandleMapper &pmapper) {
c@75 801
c@75 802 auto u = b.getResponse().initProcess();
c@75 803 buildProcessResponse(u, pr, pmapper);
c@75 804 }
c@75 805
c@75 806 static void
c@97 807 buildRpcRequest_Finish(piper::RpcRequest::Builder &b,
c@97 808 const FinishRequest &req,
c@80 809 const PluginHandleMapper &pmapper) {
c@75 810
c@75 811 auto u = b.getRequest().initFinish();
c@75 812 u.setHandle(pmapper.pluginToHandle(req.plugin));
c@75 813 }
c@75 814
c@75 815 static void
c@97 816 buildRpcResponse_Finish(piper::RpcResponse::Builder &b,
c@97 817 const FinishResponse &pr,
c@80 818 const PluginHandleMapper &pmapper) {
c@75 819
c@75 820 auto u = b.getResponse().initFinish();
c@75 821 buildFinishResponse(u, pr, pmapper);
c@75 822 }
c@75 823
c@75 824 static void
c@97 825 buildRpcResponse_Error(piper::RpcResponse::Builder &b,
c@80 826 const std::string &errorText,
c@80 827 RRType responseType)
c@75 828 {
c@75 829 std::string type;
c@75 830
c@75 831 auto e = b.getResponse().initError();
c@75 832
c@75 833 if (responseType == RRType::List) {
c@75 834 type = "list";
c@75 835 } else if (responseType == RRType::Load) {
c@75 836 type = "load";
c@75 837 } else if (responseType == RRType::Configure) {
c@75 838 type = "configure";
c@75 839 } else if (responseType == RRType::Process) {
c@75 840 type = "process";
c@75 841 } else if (responseType == RRType::Finish) {
c@75 842 type = "finish";
c@75 843 } else {
c@75 844 type = "invalid";
c@75 845 }
c@75 846
c@75 847 e.setCode(0);
c@75 848 e.setMessage(std::string("error in ") + type + " request: " + errorText);
c@75 849 }
c@75 850
c@75 851 static void
c@97 852 buildRpcResponse_Exception(piper::RpcResponse::Builder &b,
c@80 853 const std::exception &e,
c@80 854 RRType responseType)
c@75 855 {
c@75 856 return buildRpcResponse_Error(b, e.what(), responseType);
c@75 857 }
c@75 858
c@75 859 static RRType
c@97 860 getRequestResponseType(const piper::RpcRequest::Reader &r) {
c@75 861 switch (r.getRequest().which()) {
c@97 862 case piper::RpcRequest::Request::Which::LIST:
c@75 863 return RRType::List;
c@97 864 case piper::RpcRequest::Request::Which::LOAD:
c@75 865 return RRType::Load;
c@97 866 case piper::RpcRequest::Request::Which::CONFIGURE:
c@75 867 return RRType::Configure;
c@97 868 case piper::RpcRequest::Request::Which::PROCESS:
c@75 869 return RRType::Process;
c@97 870 case piper::RpcRequest::Request::Which::FINISH:
c@75 871 return RRType::Finish;
c@75 872 }
c@75 873 return RRType::NotValid;
c@75 874 }
c@75 875
c@75 876 static RRType
c@97 877 getRequestResponseType(const piper::RpcResponse::Reader &r) {
c@75 878 switch (r.getResponse().which()) {
c@97 879 case piper::RpcResponse::Response::Which::ERROR:
c@75 880 return RRType::NotValid; //!!! or error type? test this
c@97 881 case piper::RpcResponse::Response::Which::LIST:
c@75 882 return RRType::List;
c@97 883 case piper::RpcResponse::Response::Which::LOAD:
c@75 884 return RRType::Load;
c@97 885 case piper::RpcResponse::Response::Which::CONFIGURE:
c@75 886 return RRType::Configure;
c@97 887 case piper::RpcResponse::Response::Which::PROCESS:
c@75 888 return RRType::Process;
c@97 889 case piper::RpcResponse::Response::Which::FINISH:
c@75 890 return RRType::Finish;
c@75 891 }
c@75 892 return RRType::NotValid;
c@75 893 }
c@75 894
c@75 895 static void
c@75 896 readRpcResponse_Error(int &code,
c@75 897 std::string &message,
c@97 898 const piper::RpcResponse::Reader &r) {
c@75 899 if (getRequestResponseType(r) != RRType::NotValid) {
c@75 900 throw std::logic_error("not an error response");
c@75 901 }
c@75 902 code = r.getResponse().getError().getCode();
c@75 903 message = r.getResponse().getError().getMessage();
c@75 904 }
c@75 905
c@75 906 static void
c@97 907 readRpcRequest_List(const piper::RpcRequest::Reader &r) {
c@75 908 if (getRequestResponseType(r) != RRType::List) {
c@75 909 throw std::logic_error("not a list request");
c@75 910 }
c@75 911 }
c@75 912
c@75 913 static void
c@97 914 readRpcResponse_List(ListResponse &resp,
c@97 915 const piper::RpcResponse::Reader &r) {
c@75 916 if (getRequestResponseType(r) != RRType::List) {
c@75 917 throw std::logic_error("not a list response");
c@75 918 }
c@95 919 readListResponse(resp, r.getResponse().getList());
c@75 920 }
c@75 921
c@75 922 static void
c@97 923 readRpcRequest_Load(LoadRequest &req,
c@97 924 const piper::RpcRequest::Reader &r) {
c@75 925 if (getRequestResponseType(r) != RRType::Load) {
c@75 926 throw std::logic_error("not a load request");
c@75 927 }
c@75 928 readLoadRequest(req, r.getRequest().getLoad());
c@75 929 }
c@75 930
c@75 931 static void
c@97 932 readRpcResponse_Load(LoadResponse &resp,
c@97 933 const piper::RpcResponse::Reader &r,
c@75 934 const PluginHandleMapper &pmapper) {
c@75 935 if (getRequestResponseType(r) != RRType::Load) {
c@75 936 throw std::logic_error("not a load response");
c@75 937 }
c@75 938 resp = {};
c@75 939 readLoadResponse(resp, r.getResponse().getLoad(), pmapper);
c@75 940 }
c@75 941
c@75 942 static void
c@97 943 readRpcRequest_Configure(ConfigurationRequest &req,
c@97 944 const piper::RpcRequest::Reader &r,
c@80 945 const PluginHandleMapper &pmapper) {
c@75 946 if (getRequestResponseType(r) != RRType::Configure) {
c@75 947 throw std::logic_error("not a configuration request");
c@75 948 }
c@75 949 readConfigurationRequest(req, r.getRequest().getConfigure(), pmapper);
c@75 950 }
c@75 951
c@75 952 static void
c@97 953 readRpcResponse_Configure(ConfigurationResponse &resp,
c@97 954 const piper::RpcResponse::Reader &r,
c@80 955 const PluginHandleMapper &pmapper) {
c@75 956 if (getRequestResponseType(r) != RRType::Configure) {
c@75 957 throw std::logic_error("not a configuration response");
c@75 958 }
c@75 959 resp = {};
c@75 960 readConfigurationResponse(resp,
c@75 961 r.getResponse().getConfigure(),
c@75 962 pmapper);
c@75 963 }
c@75 964
c@75 965 static void
c@97 966 readRpcRequest_Process(ProcessRequest &req,
c@97 967 const piper::RpcRequest::Reader &r,
c@80 968 const PluginHandleMapper &pmapper) {
c@75 969 if (getRequestResponseType(r) != RRType::Process) {
c@75 970 throw std::logic_error("not a process request");
c@75 971 }
c@75 972 readProcessRequest(req, r.getRequest().getProcess(), pmapper);
c@75 973 }
c@75 974
c@75 975 static void
c@97 976 readRpcResponse_Process(ProcessResponse &resp,
c@97 977 const piper::RpcResponse::Reader &r,
c@80 978 const PluginHandleMapper &pmapper) {
c@75 979 if (getRequestResponseType(r) != RRType::Process) {
c@75 980 throw std::logic_error("not a process response");
c@75 981 }
c@75 982 resp = {};
c@75 983 readProcessResponse(resp, r.getResponse().getProcess(), pmapper);
c@75 984 }
c@75 985
c@75 986 static void
c@97 987 readRpcRequest_Finish(FinishRequest &req,
c@97 988 const piper::RpcRequest::Reader &r,
c@80 989 const PluginHandleMapper &pmapper) {
c@75 990 if (getRequestResponseType(r) != RRType::Finish) {
c@75 991 throw std::logic_error("not a finish request");
c@75 992 }
c@75 993 req.plugin = pmapper.handleToPlugin
c@75 994 (r.getRequest().getFinish().getHandle());
c@75 995 }
c@75 996
c@75 997 static void
c@97 998 readRpcResponse_Finish(FinishResponse &resp,
c@97 999 const piper::RpcResponse::Reader &r,
c@80 1000 const PluginHandleMapper &pmapper) {
c@75 1001 if (getRequestResponseType(r) != RRType::Finish) {
c@75 1002 throw std::logic_error("not a finish response");
c@75 1003 }
c@75 1004 resp = {};
c@75 1005 readFinishResponse(resp, r.getResponse().getFinish(), pmapper);
c@75 1006 }
c@75 1007 };
c@75 1008
c@75 1009 }
c@75 1010
c@75 1011