annotate vamp-capnp/VampnProto.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@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@127 498
c@127 499 static void
c@127 500 buildListRequest(piper::ListRequest::Builder &r,
c@127 501 const ListRequest &resp) {
c@75 502
c@127 503 auto p = r.initFrom(unsigned(resp.from.size()));
c@127 504 for (int i = 0; i < int(resp.from.size()); ++i) {
c@127 505 p.set(i, resp.from[i]);
c@127 506 }
c@127 507 }
c@127 508
c@127 509 static void
c@127 510 readListRequest(ListRequest &lr,
c@127 511 const piper::ListRequest::Reader &r) {
c@127 512
c@127 513 lr.from.clear();
c@127 514 for (auto f: r.getFrom()) {
c@127 515 lr.from.push_back(f);
c@127 516 }
c@127 517 }
c@127 518
c@127 519 static void
c@127 520 buildListResponse(piper::ListResponse::Builder &r,
c@127 521 const ListResponse &resp) {
c@127 522
c@127 523 auto p = r.initAvailable(unsigned(resp.available.size()));
c@127 524 for (int i = 0; i < int(resp.available.size()); ++i) {
c@127 525 auto pd = p[i];
c@127 526 buildExtractorStaticData(pd, resp.available[i]);
c@127 527 }
c@127 528 }
c@127 529
c@75 530 static void
c@97 531 readListResponse(ListResponse &lr,
c@97 532 const piper::ListResponse::Reader &r) {
c@95 533
c@95 534 lr.available.clear();
c@95 535 auto pp = r.getAvailable();
c@95 536 for (const auto &p: pp) {
c@97 537 PluginStaticData psd;
c@95 538 readExtractorStaticData(psd, p);
c@95 539 lr.available.push_back(psd);
c@95 540 }
c@95 541 }
c@95 542
c@95 543 static void
c@97 544 buildLoadRequest(piper::LoadRequest::Builder &r,
c@97 545 const LoadRequest &req) {
c@75 546
c@75 547 r.setKey(req.pluginKey);
c@75 548 r.setInputSampleRate(req.inputSampleRate);
c@75 549
c@97 550 std::vector<piper::AdapterFlag> flags;
c@75 551 if (req.adapterFlags & Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN) {
c@97 552 flags.push_back(piper::AdapterFlag::ADAPT_INPUT_DOMAIN);
c@75 553 }
c@75 554 if (req.adapterFlags & Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT) {
c@97 555 flags.push_back(piper::AdapterFlag::ADAPT_CHANNEL_COUNT);
c@75 556 }
c@75 557 if (req.adapterFlags & Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE) {
c@97 558 flags.push_back(piper::AdapterFlag::ADAPT_BUFFER_SIZE);
c@75 559 }
c@75 560
c@102 561 auto f = r.initAdapterFlags(unsigned(flags.size()));
c@102 562 for (int i = 0; i < int(flags.size()); ++i) {
c@75 563 f.set(i, flags[i]);
c@75 564 }
c@75 565 }
c@75 566
c@75 567 static void
c@97 568 readLoadRequest(LoadRequest &req,
c@97 569 const piper::LoadRequest::Reader &r) {
c@75 570
c@75 571 req.pluginKey = r.getKey();
c@75 572 req.inputSampleRate = r.getInputSampleRate();
c@75 573
c@75 574 int flags = 0;
c@75 575 auto aa = r.getAdapterFlags();
c@75 576 for (auto a: aa) {
c@97 577 if (a == piper::AdapterFlag::ADAPT_INPUT_DOMAIN) {
c@75 578 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN;
c@75 579 }
c@97 580 if (a == piper::AdapterFlag::ADAPT_CHANNEL_COUNT) {
c@75 581 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT;
c@75 582 }
c@97 583 if (a == piper::AdapterFlag::ADAPT_BUFFER_SIZE) {
c@75 584 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE;
c@75 585 }
c@75 586 }
c@75 587 req.adapterFlags = flags;
c@75 588 }
c@75 589
c@75 590 static void
c@97 591 buildLoadResponse(piper::LoadResponse::Builder &b,
c@97 592 const LoadResponse &resp,
c@75 593 const PluginHandleMapper &pmapper) {
c@75 594
c@75 595 b.setHandle(pmapper.pluginToHandle(resp.plugin));
c@75 596 auto sd = b.initStaticData();
c@75 597 buildExtractorStaticData(sd, resp.staticData);
c@75 598 auto conf = b.initDefaultConfiguration();
c@75 599 buildConfiguration(conf, resp.defaultConfiguration);
c@75 600 }
c@75 601
c@75 602 static void
c@97 603 readLoadResponse(LoadResponse &resp,
c@97 604 const piper::LoadResponse::Reader &r,
c@75 605 const PluginHandleMapper &pmapper) {
c@75 606
c@75 607 resp.plugin = pmapper.handleToPlugin(r.getHandle());
c@75 608 readExtractorStaticData(resp.staticData, r.getStaticData());
c@75 609 readConfiguration(resp.defaultConfiguration,
c@80 610 r.getDefaultConfiguration());
c@75 611 }
c@75 612
c@75 613 static void
c@97 614 buildConfigurationRequest(piper::ConfigurationRequest::Builder &b,
c@97 615 const ConfigurationRequest &cr,
c@75 616 const PluginHandleMapper &pmapper) {
c@75 617
c@75 618 b.setHandle(pmapper.pluginToHandle(cr.plugin));
c@75 619 auto c = b.initConfiguration();
c@75 620 buildConfiguration(c, cr.configuration);
c@75 621 }
c@75 622
c@75 623 static void
c@97 624 readConfigurationRequest(ConfigurationRequest &cr,
c@97 625 const piper::ConfigurationRequest::Reader &r,
c@75 626 const PluginHandleMapper &pmapper) {
c@75 627
c@75 628 auto h = r.getHandle();
c@75 629 cr.plugin = pmapper.handleToPlugin(h);
c@75 630 auto c = r.getConfiguration();
c@75 631 readConfiguration(cr.configuration, c);
c@75 632 }
c@75 633
c@75 634 static void
c@97 635 buildConfigurationResponse(piper::ConfigurationResponse::Builder &b,
c@97 636 const ConfigurationResponse &cr,
c@75 637 const PluginHandleMapper &pmapper) {
c@75 638
c@75 639 b.setHandle(pmapper.pluginToHandle(cr.plugin));
c@102 640 auto olist = b.initOutputs(unsigned(cr.outputs.size()));
c@102 641 for (int i = 0; i < int(cr.outputs.size()); ++i) {
c@75 642 auto od = olist[i];
c@75 643 buildOutputDescriptor(od, cr.outputs[i]);
c@75 644 }
c@75 645 }
c@75 646
c@75 647 static void
c@97 648 readConfigurationResponse(ConfigurationResponse &cr,
c@97 649 const piper::ConfigurationResponse::Reader &r,
c@75 650 const PluginHandleMapper &pmapper) {
c@75 651
c@75 652 cr.plugin = pmapper.handleToPlugin(r.getHandle());
c@75 653 cr.outputs.clear();
c@75 654 auto oo = r.getOutputs();
c@75 655 for (const auto &o: oo) {
c@75 656 Vamp::Plugin::OutputDescriptor desc;
c@75 657 readOutputDescriptor(desc, o);
c@75 658 cr.outputs.push_back(desc);
c@75 659 }
c@75 660 }
c@75 661
c@75 662 static void
c@97 663 buildProcessInput(piper::ProcessInput::Builder &b,
c@75 664 Vamp::RealTime timestamp,
c@75 665 const std::vector<std::vector<float> > &buffers) {
c@75 666
c@75 667 auto t = b.initTimestamp();
c@75 668 buildRealTime(t, timestamp);
c@102 669 auto vv = b.initInputBuffers(unsigned(buffers.size()));
c@102 670 for (int ch = 0; ch < int(buffers.size()); ++ch) {
c@75 671 const int n = int(buffers[ch].size());
c@75 672 vv.init(ch, n);
c@75 673 auto v = vv[ch];
c@75 674 for (int i = 0; i < n; ++i) {
c@75 675 v.set(i, buffers[ch][i]);
c@75 676 }
c@75 677 }
c@75 678 }
c@75 679
c@75 680 static void
c@75 681 readProcessInput(Vamp::RealTime &timestamp,
c@75 682 std::vector<std::vector<float> > &buffers,
c@97 683 const piper::ProcessInput::Reader &b) {
c@75 684
c@75 685 readRealTime(timestamp, b.getTimestamp());
c@75 686 buffers.clear();
c@75 687 auto vv = b.getInputBuffers();
c@75 688 for (const auto &v: vv) {
c@75 689 std::vector<float> buf;
c@75 690 for (auto x: v) {
c@75 691 buf.push_back(x);
c@75 692 }
c@75 693 buffers.push_back(buf);
c@75 694 }
c@75 695 }
c@75 696
c@75 697 static void
c@97 698 buildProcessRequest(piper::ProcessRequest::Builder &b,
c@97 699 const ProcessRequest &pr,
c@75 700 const PluginHandleMapper &pmapper) {
c@75 701
c@75 702 b.setHandle(pmapper.pluginToHandle(pr.plugin));
c@75 703 auto input = b.initProcessInput();
c@75 704 buildProcessInput(input, pr.timestamp, pr.inputBuffers);
c@75 705 }
c@75 706
c@75 707 static void
c@97 708 readProcessRequest(ProcessRequest &pr,
c@97 709 const piper::ProcessRequest::Reader &r,
c@75 710 const PluginHandleMapper &pmapper) {
c@75 711
c@75 712 auto h = r.getHandle();
c@75 713 pr.plugin = pmapper.handleToPlugin(h);
c@75 714 readProcessInput(pr.timestamp, pr.inputBuffers, r.getProcessInput());
c@75 715 }
c@75 716
c@75 717 static void
c@97 718 buildProcessResponse(piper::ProcessResponse::Builder &b,
c@97 719 const ProcessResponse &pr,
c@75 720 const PluginHandleMapper &pmapper) {
c@75 721
c@75 722 b.setHandle(pmapper.pluginToHandle(pr.plugin));
c@75 723 auto f = b.initFeatures();
c@75 724 buildFeatureSet(f, pr.features,
c@75 725 *pmapper.pluginToOutputIdMapper(pr.plugin));
c@75 726 }
c@75 727
c@75 728 static void
c@97 729 readProcessResponse(ProcessResponse &pr,
c@97 730 const piper::ProcessResponse::Reader &r,
c@75 731 const PluginHandleMapper &pmapper) {
c@75 732
c@75 733 auto h = r.getHandle();
c@75 734 pr.plugin = pmapper.handleToPlugin(h);
c@75 735 readFeatureSet(pr.features, r.getFeatures(),
c@75 736 *pmapper.handleToOutputIdMapper(r.getHandle()));
c@75 737 }
c@75 738
c@75 739 static void
c@97 740 buildFinishResponse(piper::FinishResponse::Builder &b,
c@97 741 const FinishResponse &pr,
c@75 742 const PluginHandleMapper &pmapper) {
c@75 743
c@75 744 b.setHandle(pmapper.pluginToHandle(pr.plugin));
c@75 745 auto f = b.initFeatures();
c@75 746 buildFeatureSet(f, pr.features,
c@75 747 *pmapper.pluginToOutputIdMapper(pr.plugin));
c@75 748 }
c@75 749
c@75 750 static void
c@97 751 readFinishResponse(FinishResponse &pr,
c@97 752 const piper::FinishResponse::Reader &r,
c@75 753 const PluginHandleMapper &pmapper) {
c@75 754
c@75 755 auto h = r.getHandle();
c@75 756 pr.plugin = pmapper.handleToPlugin(h);
c@75 757 readFeatureSet(pr.features, r.getFeatures(),
c@75 758 *pmapper.handleToOutputIdMapper(r.getHandle()));
c@75 759 }
c@75 760
c@75 761 static void
c@127 762 buildRpcRequest_List(piper::RpcRequest::Builder &b,
c@127 763 const ListRequest &req) {
c@127 764 auto r = b.getRequest().initList();
c@127 765 buildListRequest(r, req);
c@75 766 }
c@75 767
c@75 768 static void
c@97 769 buildRpcResponse_List(piper::RpcResponse::Builder &b,
c@97 770 const ListResponse &resp) {
c@75 771
c@75 772 auto r = b.getResponse().initList();
c@127 773 buildListResponse(r, resp);
c@75 774 }
c@75 775
c@75 776 static void
c@97 777 buildRpcRequest_Load(piper::RpcRequest::Builder &b,
c@97 778 const LoadRequest &req) {
c@75 779 auto u = b.getRequest().initLoad();
c@75 780 buildLoadRequest(u, req);
c@75 781 }
c@75 782
c@75 783 static void
c@97 784 buildRpcResponse_Load(piper::RpcResponse::Builder &b,
c@97 785 const LoadResponse &resp,
c@80 786 const PluginHandleMapper &pmapper) {
c@75 787
c@75 788 if (resp.plugin) {
c@75 789 auto u = b.getResponse().initLoad();
c@75 790 buildLoadResponse(u, resp, pmapper);
c@75 791 } else {
c@75 792 buildRpcResponse_Error(b, "Failed to load plugin", RRType::Load);
c@75 793 }
c@75 794 }
c@75 795
c@75 796 static void
c@97 797 buildRpcRequest_Configure(piper::RpcRequest::Builder &b,
c@97 798 const ConfigurationRequest &cr,
c@80 799 const PluginHandleMapper &pmapper) {
c@75 800 auto u = b.getRequest().initConfigure();
c@75 801 buildConfigurationRequest(u, cr, pmapper);
c@75 802 }
c@75 803
c@75 804 static void
c@97 805 buildRpcResponse_Configure(piper::RpcResponse::Builder &b,
c@97 806 const ConfigurationResponse &cr,
c@80 807 const PluginHandleMapper &pmapper) {
c@75 808
c@75 809 if (!cr.outputs.empty()) {
c@75 810 auto u = b.getResponse().initConfigure();
c@75 811 buildConfigurationResponse(u, cr, pmapper);
c@75 812 } else {
c@75 813 buildRpcResponse_Error(b, "Failed to configure plugin",
c@75 814 RRType::Configure);
c@75 815 }
c@75 816 }
c@75 817
c@75 818 static void
c@97 819 buildRpcRequest_Process(piper::RpcRequest::Builder &b,
c@97 820 const ProcessRequest &pr,
c@80 821 const PluginHandleMapper &pmapper) {
c@75 822 auto u = b.getRequest().initProcess();
c@75 823 buildProcessRequest(u, pr, pmapper);
c@75 824 }
c@75 825
c@75 826 static void
c@97 827 buildRpcResponse_Process(piper::RpcResponse::Builder &b,
c@97 828 const ProcessResponse &pr,
c@80 829 const PluginHandleMapper &pmapper) {
c@75 830
c@75 831 auto u = b.getResponse().initProcess();
c@75 832 buildProcessResponse(u, pr, pmapper);
c@75 833 }
c@75 834
c@75 835 static void
c@97 836 buildRpcRequest_Finish(piper::RpcRequest::Builder &b,
c@97 837 const FinishRequest &req,
c@80 838 const PluginHandleMapper &pmapper) {
c@75 839
c@75 840 auto u = b.getRequest().initFinish();
c@75 841 u.setHandle(pmapper.pluginToHandle(req.plugin));
c@75 842 }
c@75 843
c@75 844 static void
c@97 845 buildRpcResponse_Finish(piper::RpcResponse::Builder &b,
c@97 846 const FinishResponse &pr,
c@80 847 const PluginHandleMapper &pmapper) {
c@75 848
c@75 849 auto u = b.getResponse().initFinish();
c@75 850 buildFinishResponse(u, pr, pmapper);
c@75 851 }
c@75 852
c@75 853 static void
c@97 854 buildRpcResponse_Error(piper::RpcResponse::Builder &b,
c@80 855 const std::string &errorText,
c@80 856 RRType responseType)
c@75 857 {
c@75 858 std::string type;
c@75 859
c@75 860 auto e = b.getResponse().initError();
c@75 861
c@75 862 if (responseType == RRType::List) {
c@75 863 type = "list";
c@75 864 } else if (responseType == RRType::Load) {
c@75 865 type = "load";
c@75 866 } else if (responseType == RRType::Configure) {
c@75 867 type = "configure";
c@75 868 } else if (responseType == RRType::Process) {
c@75 869 type = "process";
c@75 870 } else if (responseType == RRType::Finish) {
c@75 871 type = "finish";
c@75 872 } else {
c@75 873 type = "invalid";
c@75 874 }
c@75 875
c@75 876 e.setCode(0);
cannam@158 877
cannam@158 878 if (responseType == RRType::NotValid) {
cannam@158 879 e.setMessage(errorText);
cannam@158 880 } else {
cannam@158 881 e.setMessage
cannam@158 882 (std::string("error in ") + type + " request: " + errorText);
cannam@158 883 }
c@75 884 }
c@75 885
c@75 886 static void
c@97 887 buildRpcResponse_Exception(piper::RpcResponse::Builder &b,
c@80 888 const std::exception &e,
c@80 889 RRType responseType)
c@75 890 {
c@75 891 return buildRpcResponse_Error(b, e.what(), responseType);
c@75 892 }
c@75 893
c@75 894 static RRType
c@97 895 getRequestResponseType(const piper::RpcRequest::Reader &r) {
c@75 896 switch (r.getRequest().which()) {
c@97 897 case piper::RpcRequest::Request::Which::LIST:
c@75 898 return RRType::List;
c@97 899 case piper::RpcRequest::Request::Which::LOAD:
c@75 900 return RRType::Load;
c@97 901 case piper::RpcRequest::Request::Which::CONFIGURE:
c@75 902 return RRType::Configure;
c@97 903 case piper::RpcRequest::Request::Which::PROCESS:
c@75 904 return RRType::Process;
c@97 905 case piper::RpcRequest::Request::Which::FINISH:
c@75 906 return RRType::Finish;
c@75 907 }
c@75 908 return RRType::NotValid;
c@75 909 }
c@75 910
c@75 911 static RRType
c@97 912 getRequestResponseType(const piper::RpcResponse::Reader &r) {
c@75 913 switch (r.getResponse().which()) {
c@97 914 case piper::RpcResponse::Response::Which::ERROR:
cannam@170 915 return RRType::NotValid;
c@97 916 case piper::RpcResponse::Response::Which::LIST:
c@75 917 return RRType::List;
c@97 918 case piper::RpcResponse::Response::Which::LOAD:
c@75 919 return RRType::Load;
c@97 920 case piper::RpcResponse::Response::Which::CONFIGURE:
c@75 921 return RRType::Configure;
c@97 922 case piper::RpcResponse::Response::Which::PROCESS:
c@75 923 return RRType::Process;
c@97 924 case piper::RpcResponse::Response::Which::FINISH:
c@75 925 return RRType::Finish;
c@75 926 }
c@75 927 return RRType::NotValid;
c@75 928 }
c@75 929
c@75 930 static void
c@75 931 readRpcResponse_Error(int &code,
c@75 932 std::string &message,
c@97 933 const piper::RpcResponse::Reader &r) {
c@75 934 if (getRequestResponseType(r) != RRType::NotValid) {
c@75 935 throw std::logic_error("not an error response");
c@75 936 }
c@75 937 code = r.getResponse().getError().getCode();
c@75 938 message = r.getResponse().getError().getMessage();
c@75 939 }
c@75 940
c@75 941 static void
c@127 942 readRpcRequest_List(ListRequest &req,
c@127 943 const piper::RpcRequest::Reader &r) {
c@75 944 if (getRequestResponseType(r) != RRType::List) {
c@75 945 throw std::logic_error("not a list request");
c@75 946 }
c@127 947 readListRequest(req, r.getRequest().getList());
c@75 948 }
c@75 949
c@75 950 static void
c@97 951 readRpcResponse_List(ListResponse &resp,
c@97 952 const piper::RpcResponse::Reader &r) {
c@75 953 if (getRequestResponseType(r) != RRType::List) {
c@75 954 throw std::logic_error("not a list response");
c@75 955 }
c@95 956 readListResponse(resp, r.getResponse().getList());
c@75 957 }
c@75 958
c@75 959 static void
c@97 960 readRpcRequest_Load(LoadRequest &req,
c@97 961 const piper::RpcRequest::Reader &r) {
c@75 962 if (getRequestResponseType(r) != RRType::Load) {
c@75 963 throw std::logic_error("not a load request");
c@75 964 }
c@75 965 readLoadRequest(req, r.getRequest().getLoad());
c@75 966 }
c@75 967
c@75 968 static void
c@97 969 readRpcResponse_Load(LoadResponse &resp,
c@97 970 const piper::RpcResponse::Reader &r,
c@75 971 const PluginHandleMapper &pmapper) {
c@75 972 if (getRequestResponseType(r) != RRType::Load) {
c@75 973 throw std::logic_error("not a load response");
c@75 974 }
c@75 975 resp = {};
c@75 976 readLoadResponse(resp, r.getResponse().getLoad(), pmapper);
c@75 977 }
c@75 978
c@75 979 static void
c@97 980 readRpcRequest_Configure(ConfigurationRequest &req,
c@97 981 const piper::RpcRequest::Reader &r,
c@80 982 const PluginHandleMapper &pmapper) {
c@75 983 if (getRequestResponseType(r) != RRType::Configure) {
c@75 984 throw std::logic_error("not a configuration request");
c@75 985 }
c@75 986 readConfigurationRequest(req, r.getRequest().getConfigure(), pmapper);
c@75 987 }
c@75 988
c@75 989 static void
c@97 990 readRpcResponse_Configure(ConfigurationResponse &resp,
c@97 991 const piper::RpcResponse::Reader &r,
c@80 992 const PluginHandleMapper &pmapper) {
c@75 993 if (getRequestResponseType(r) != RRType::Configure) {
c@75 994 throw std::logic_error("not a configuration response");
c@75 995 }
c@75 996 resp = {};
c@75 997 readConfigurationResponse(resp,
c@75 998 r.getResponse().getConfigure(),
c@75 999 pmapper);
c@75 1000 }
c@75 1001
c@75 1002 static void
c@97 1003 readRpcRequest_Process(ProcessRequest &req,
c@97 1004 const piper::RpcRequest::Reader &r,
c@80 1005 const PluginHandleMapper &pmapper) {
c@75 1006 if (getRequestResponseType(r) != RRType::Process) {
c@75 1007 throw std::logic_error("not a process request");
c@75 1008 }
c@75 1009 readProcessRequest(req, r.getRequest().getProcess(), pmapper);
c@75 1010 }
c@75 1011
c@75 1012 static void
c@97 1013 readRpcResponse_Process(ProcessResponse &resp,
c@97 1014 const piper::RpcResponse::Reader &r,
c@80 1015 const PluginHandleMapper &pmapper) {
c@75 1016 if (getRequestResponseType(r) != RRType::Process) {
c@75 1017 throw std::logic_error("not a process response");
c@75 1018 }
c@75 1019 resp = {};
c@75 1020 readProcessResponse(resp, r.getResponse().getProcess(), pmapper);
c@75 1021 }
c@75 1022
c@75 1023 static void
c@97 1024 readRpcRequest_Finish(FinishRequest &req,
c@97 1025 const piper::RpcRequest::Reader &r,
c@80 1026 const PluginHandleMapper &pmapper) {
c@75 1027 if (getRequestResponseType(r) != RRType::Finish) {
c@75 1028 throw std::logic_error("not a finish request");
c@75 1029 }
c@75 1030 req.plugin = pmapper.handleToPlugin
c@75 1031 (r.getRequest().getFinish().getHandle());
c@75 1032 }
c@75 1033
c@75 1034 static void
c@97 1035 readRpcResponse_Finish(FinishResponse &resp,
c@97 1036 const piper::RpcResponse::Reader &r,
c@80 1037 const PluginHandleMapper &pmapper) {
c@75 1038 if (getRequestResponseType(r) != RRType::Finish) {
c@75 1039 throw std::logic_error("not a finish response");
c@75 1040 }
c@75 1041 resp = {};
c@75 1042 readFinishResponse(resp, r.getResponse().getFinish(), pmapper);
c@75 1043 }
c@75 1044 };
c@75 1045
c@75 1046 }
c@75 1047
c@75 1048