annotate vamp-capnp/VampnProto.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 0876b5e67afe
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