annotate transform/TransformFactory.cpp @ 1865:7b6e18380e8f startup-timing

Make it possible to check whether the transforms have been populated yet
author Chris Cannam
date Wed, 03 Jun 2020 13:57:50 +0100
parents 6f626cfdba51
children 652c5360e682
rev   line source
Chris@330 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@330 2
Chris@330 3 /*
Chris@330 4 Sonic Visualiser
Chris@330 5 An audio file viewer and annotation editor.
Chris@330 6 Centre for Digital Music, Queen Mary, University of London.
Chris@330 7 This file copyright 2006 Chris Cannam and QMUL.
Chris@330 8
Chris@330 9 This program is free software; you can redistribute it and/or
Chris@330 10 modify it under the terms of the GNU General Public License as
Chris@330 11 published by the Free Software Foundation; either version 2 of the
Chris@330 12 License, or (at your option) any later version. See the file
Chris@330 13 COPYING included with this distribution for more information.
Chris@330 14 */
Chris@330 15
Chris@330 16 #include "TransformFactory.h"
Chris@330 17
Chris@330 18 #include "plugin/FeatureExtractionPluginFactory.h"
Chris@1225 19
Chris@330 20 #include "plugin/RealTimePluginFactory.h"
Chris@332 21 #include "plugin/RealTimePluginInstance.h"
Chris@330 22 #include "plugin/PluginXml.h"
Chris@330 23
Chris@475 24 #include <vamp-hostsdk/Plugin.h>
Chris@475 25 #include <vamp-hostsdk/PluginHostAdapter.h>
Chris@475 26 #include <vamp-hostsdk/PluginWrapper.h>
Chris@330 27
Chris@457 28 #include "rdf/PluginRDFIndexer.h"
Chris@457 29 #include "rdf/PluginRDFDescription.h"
Chris@457 30
Chris@446 31 #include "base/XmlExportable.h"
Chris@446 32
Chris@330 33 #include <iostream>
Chris@330 34 #include <set>
Chris@1546 35 #include <functional>
Chris@330 36
Chris@330 37 #include <QRegExp>
Chris@350 38 #include <QTextStream>
Chris@330 39
Chris@460 40 #include "base/Thread.h"
Chris@460 41
Chris@810 42 //#define DEBUG_TRANSFORM_FACTORY 1
Chris@810 43
Chris@330 44 TransformFactory *
Chris@330 45 TransformFactory::m_instance = new TransformFactory;
Chris@330 46
Chris@330 47 TransformFactory *
Chris@330 48 TransformFactory::getInstance()
Chris@330 49 {
Chris@330 50 return m_instance;
Chris@330 51 }
Chris@330 52
Chris@574 53 void
Chris@574 54 TransformFactory::deleteInstance()
Chris@574 55 {
Chris@690 56 SVDEBUG << "TransformFactory::deleteInstance called" << endl;
Chris@574 57 delete m_instance;
Chris@1582 58 m_instance = nullptr;
Chris@574 59 }
Chris@574 60
Chris@457 61 TransformFactory::TransformFactory() :
Chris@457 62 m_transformsPopulated(false),
Chris@477 63 m_uninstalledTransformsPopulated(false),
Chris@1582 64 m_thread(nullptr),
Chris@976 65 m_exiting(false),
Chris@976 66 m_populatingSlowly(false)
Chris@457 67 {
Chris@457 68 }
Chris@457 69
Chris@330 70 TransformFactory::~TransformFactory()
Chris@330 71 {
Chris@574 72 m_exiting = true;
Chris@574 73 if (m_thread) {
Chris@600 74 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@690 75 SVDEBUG << "TransformFactory::~TransformFactory: waiting on thread" << endl;
Chris@600 76 #endif
Chris@574 77 m_thread->wait();
Chris@574 78 delete m_thread;
Chris@600 79 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@690 80 SVDEBUG << "TransformFactory::~TransformFactory: waited and done" << endl;
Chris@600 81 #endif
Chris@574 82 }
Chris@330 83 }
Chris@330 84
Chris@477 85 void
Chris@477 86 TransformFactory::startPopulationThread()
Chris@477 87 {
Chris@482 88 m_uninstalledTransformsMutex.lock();
Chris@477 89
Chris@482 90 if (m_thread) {
Chris@482 91 m_uninstalledTransformsMutex.unlock();
Chris@482 92 return;
Chris@482 93 }
Chris@482 94 m_thread = new UninstalledTransformsPopulateThread(this);
Chris@477 95
Chris@482 96 m_uninstalledTransformsMutex.unlock();
Chris@482 97
Chris@477 98 m_thread->start();
Chris@477 99 }
Chris@477 100
Chris@481 101 void
Chris@481 102 TransformFactory::UninstalledTransformsPopulateThread::run()
Chris@481 103 {
Chris@481 104 m_factory->m_populatingSlowly = true;
Chris@1865 105 while (!m_factory->havePopulated()) {
Chris@1865 106 sleep(1);
Chris@1865 107 }
Chris@481 108 m_factory->populateUninstalledTransforms();
Chris@481 109 }
Chris@481 110
Chris@330 111 TransformList
Chris@350 112 TransformFactory::getAllTransformDescriptions()
Chris@330 113 {
Chris@460 114 populateTransforms();
Chris@330 115
Chris@330 116 std::set<TransformDescription> dset;
Chris@1546 117 for (auto i = m_transforms.begin(); i != m_transforms.end(); ++i) {
Chris@600 118 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@1546 119 cerr << "inserting transform into set: id = " << i->second.identifier << " (" << i->second.name << ")" << endl;
Chris@600 120 #endif
Chris@1429 121 dset.insert(i->second);
Chris@330 122 }
Chris@330 123
Chris@330 124 TransformList list;
Chris@1546 125 for (auto i = dset.begin(); i != dset.end(); ++i) {
Chris@600 126 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@1546 127 cerr << "inserting transform into list: id = " << i->identifier << " (" << i->name << ")" << endl;
Chris@600 128 #endif
Chris@1429 129 list.push_back(*i);
Chris@330 130 }
Chris@330 131
Chris@330 132 return list;
Chris@330 133 }
Chris@330 134
Chris@350 135 TransformDescription
Chris@350 136 TransformFactory::getTransformDescription(TransformId id)
Chris@350 137 {
Chris@460 138 populateTransforms();
Chris@350 139
Chris@350 140 if (m_transforms.find(id) == m_transforms.end()) {
Chris@350 141 return TransformDescription();
Chris@350 142 }
Chris@350 143
Chris@350 144 return m_transforms[id];
Chris@350 145 }
Chris@350 146
Chris@485 147 bool
Chris@485 148 TransformFactory::haveInstalledTransforms()
Chris@485 149 {
Chris@485 150 populateTransforms();
Chris@485 151 return !m_transforms.empty();
Chris@485 152 }
Chris@485 153
Chris@457 154 TransformList
Chris@457 155 TransformFactory::getUninstalledTransformDescriptions()
Chris@457 156 {
Chris@479 157 m_populatingSlowly = false;
Chris@460 158 populateUninstalledTransforms();
Chris@457 159
Chris@457 160 std::set<TransformDescription> dset;
Chris@1546 161 for (auto i = m_uninstalledTransforms.begin();
Chris@1429 162 i != m_uninstalledTransforms.end(); ++i) {
Chris@600 163 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@810 164 cerr << "inserting transform into set: id = " << i->second.identifier << endl;
Chris@600 165 #endif
Chris@1429 166 dset.insert(i->second);
Chris@457 167 }
Chris@457 168
Chris@457 169 TransformList list;
Chris@1546 170 for (auto i = dset.begin(); i != dset.end(); ++i) {
Chris@600 171 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@810 172 cerr << "inserting transform into uninstalled list: id = " << i->identifier << endl;
Chris@600 173 #endif
Chris@1429 174 list.push_back(*i);
Chris@457 175 }
Chris@457 176
Chris@457 177 return list;
Chris@457 178 }
Chris@457 179
Chris@457 180 TransformDescription
Chris@457 181 TransformFactory::getUninstalledTransformDescription(TransformId id)
Chris@457 182 {
Chris@479 183 m_populatingSlowly = false;
Chris@460 184 populateUninstalledTransforms();
Chris@457 185
Chris@457 186 if (m_uninstalledTransforms.find(id) == m_uninstalledTransforms.end()) {
Chris@457 187 return TransformDescription();
Chris@457 188 }
Chris@457 189
Chris@457 190 return m_uninstalledTransforms[id];
Chris@457 191 }
Chris@457 192
Chris@485 193 bool
Chris@485 194 TransformFactory::haveUninstalledTransforms(bool waitForCheckToComplete)
Chris@485 195 {
Chris@485 196 if (waitForCheckToComplete) {
Chris@485 197 populateUninstalledTransforms();
Chris@485 198 } else {
Chris@485 199 if (!m_uninstalledTransformsMutex.tryLock()) {
Chris@485 200 return false;
Chris@485 201 }
Chris@485 202 if (!m_uninstalledTransformsPopulated) {
Chris@485 203 m_uninstalledTransformsMutex.unlock();
Chris@485 204 return false;
Chris@485 205 }
Chris@485 206 m_uninstalledTransformsMutex.unlock();
Chris@485 207 }
Chris@485 208
Chris@485 209 return !m_uninstalledTransforms.empty();
Chris@485 210 }
Chris@485 211
Chris@457 212 TransformFactory::TransformInstallStatus
Chris@457 213 TransformFactory::getTransformInstallStatus(TransformId id)
Chris@457 214 {
Chris@460 215 populateTransforms();
Chris@457 216
Chris@457 217 if (m_transforms.find(id) != m_transforms.end()) {
Chris@457 218 return TransformInstalled;
Chris@457 219 }
Chris@473 220
Chris@473 221 if (!m_uninstalledTransformsMutex.tryLock()) {
Chris@473 222 // uninstalled transforms are being populated; this may take some time,
Chris@473 223 // and they aren't critical
Chris@473 224 return TransformUnknown;
Chris@473 225 }
Chris@473 226
Chris@473 227 if (!m_uninstalledTransformsPopulated) {
Chris@473 228 m_uninstalledTransformsMutex.unlock();
Chris@479 229 m_populatingSlowly = false;
Chris@473 230 populateUninstalledTransforms();
Chris@473 231 m_uninstalledTransformsMutex.lock();
Chris@473 232 }
Chris@473 233
Chris@457 234 if (m_uninstalledTransforms.find(id) != m_uninstalledTransforms.end()) {
Chris@482 235 m_uninstalledTransformsMutex.unlock();
Chris@457 236 return TransformNotInstalled;
Chris@457 237 }
Chris@473 238
Chris@473 239 m_uninstalledTransformsMutex.unlock();
Chris@457 240 return TransformUnknown;
Chris@457 241 }
Chris@457 242
Chris@457 243
Chris@487 244 std::vector<TransformDescription::Type>
Chris@330 245 TransformFactory::getAllTransformTypes()
Chris@330 246 {
Chris@460 247 populateTransforms();
Chris@330 248
Chris@487 249 std::set<TransformDescription::Type> types;
Chris@1546 250 for (auto i = m_transforms.begin(); i != m_transforms.end(); ++i) {
Chris@330 251 types.insert(i->second.type);
Chris@330 252 }
Chris@330 253
Chris@487 254 std::vector<TransformDescription::Type> rv;
Chris@1546 255 for (auto i = types.begin(); i != types.end(); ++i) {
Chris@330 256 rv.push_back(*i);
Chris@330 257 }
Chris@330 258
Chris@330 259 return rv;
Chris@330 260 }
Chris@330 261
Chris@330 262 std::vector<QString>
Chris@487 263 TransformFactory::getTransformCategories(TransformDescription::Type transformType)
Chris@330 264 {
Chris@460 265 populateTransforms();
Chris@330 266
Chris@1546 267 std::set<QString, std::function<bool(QString, QString)>>
Chris@1546 268 categories(TransformDescription::compareUserStrings);
Chris@1546 269
Chris@1546 270 for (auto i = m_transforms.begin(); i != m_transforms.end(); ++i) {
Chris@330 271 if (i->second.type == transformType) {
Chris@330 272 categories.insert(i->second.category);
Chris@330 273 }
Chris@330 274 }
Chris@330 275
Chris@330 276 bool haveEmpty = false;
Chris@330 277
Chris@330 278 std::vector<QString> rv;
Chris@1546 279 for (auto i = categories.begin(); i != categories.end(); ++i) {
Chris@330 280 if (*i != "") rv.push_back(*i);
Chris@330 281 else haveEmpty = true;
Chris@330 282 }
Chris@330 283
Chris@330 284 if (haveEmpty) rv.push_back(""); // make sure empty category sorts last
Chris@330 285
Chris@330 286 return rv;
Chris@330 287 }
Chris@330 288
Chris@330 289 std::vector<QString>
Chris@487 290 TransformFactory::getTransformMakers(TransformDescription::Type transformType)
Chris@330 291 {
Chris@460 292 populateTransforms();
Chris@330 293
Chris@1546 294 std::set<QString, std::function<bool(QString, QString)>>
Chris@1546 295 makers(TransformDescription::compareUserStrings);
Chris@1546 296
Chris@1546 297 for (auto i = m_transforms.begin(); i != m_transforms.end(); ++i) {
Chris@330 298 if (i->second.type == transformType) {
Chris@330 299 makers.insert(i->second.maker);
Chris@330 300 }
Chris@330 301 }
Chris@330 302
Chris@330 303 bool haveEmpty = false;
Chris@330 304
Chris@330 305 std::vector<QString> rv;
Chris@1546 306 for (auto i = makers.begin(); i != makers.end(); ++i) {
Chris@330 307 if (*i != "") rv.push_back(*i);
Chris@330 308 else haveEmpty = true;
Chris@330 309 }
Chris@330 310
Chris@330 311 if (haveEmpty) rv.push_back(""); // make sure empty category sorts last
Chris@330 312
Chris@330 313 return rv;
Chris@330 314 }
Chris@330 315
Chris@487 316 QString
Chris@487 317 TransformFactory::getTransformTypeName(TransformDescription::Type type) const
Chris@487 318 {
Chris@487 319 switch (type) {
Chris@487 320 case TransformDescription::Analysis: return tr("Analysis");
Chris@487 321 case TransformDescription::Effects: return tr("Effects");
Chris@487 322 case TransformDescription::EffectsData: return tr("Effects Data");
Chris@487 323 case TransformDescription::Generator: return tr("Generator");
Chris@487 324 case TransformDescription::UnknownType: return tr("Other");
Chris@487 325 }
Chris@489 326 return tr("Other");
Chris@487 327 }
Chris@487 328
Chris@1865 329 bool
Chris@1865 330 TransformFactory::havePopulated()
Chris@1865 331 {
Chris@1865 332 MutexLocker locker(&m_transformsMutex, "TransformFactory::havePopulated");
Chris@1865 333 return m_transformsPopulated;
Chris@1865 334 }
Chris@1865 335
Chris@330 336 void
Chris@330 337 TransformFactory::populateTransforms()
Chris@330 338 {
Chris@460 339 MutexLocker locker(&m_transformsMutex,
Chris@460 340 "TransformFactory::populateTransforms");
Chris@460 341 if (m_transformsPopulated) {
Chris@460 342 return;
Chris@460 343 }
Chris@460 344
Chris@330 345 TransformDescriptionMap transforms;
Chris@330 346
Chris@330 347 populateFeatureExtractionPlugins(transforms);
Chris@576 348 if (m_exiting) return;
Chris@330 349 populateRealTimePlugins(transforms);
Chris@576 350 if (m_exiting) return;
Chris@330 351
Chris@330 352 // disambiguate plugins with similar names
Chris@330 353
Chris@330 354 std::map<QString, int> names;
Chris@330 355 std::map<QString, QString> pluginSources;
Chris@330 356 std::map<QString, QString> pluginMakers;
Chris@330 357
Chris@330 358 for (TransformDescriptionMap::iterator i = transforms.begin();
Chris@330 359 i != transforms.end(); ++i) {
Chris@330 360
Chris@330 361 TransformDescription desc = i->second;
Chris@330 362
Chris@330 363 QString td = desc.name;
Chris@330 364 QString tn = td.section(": ", 0, 0);
Chris@330 365 QString pn = desc.identifier.section(":", 1, 1);
Chris@330 366
Chris@330 367 if (pluginSources.find(tn) != pluginSources.end()) {
Chris@330 368 if (pluginSources[tn] != pn && pluginMakers[tn] != desc.maker) {
Chris@330 369 ++names[tn];
Chris@330 370 }
Chris@330 371 } else {
Chris@330 372 ++names[tn];
Chris@330 373 pluginSources[tn] = pn;
Chris@330 374 pluginMakers[tn] = desc.maker;
Chris@330 375 }
Chris@330 376 }
Chris@330 377
Chris@330 378 std::map<QString, int> counts;
Chris@330 379 m_transforms.clear();
Chris@330 380
Chris@330 381 for (TransformDescriptionMap::iterator i = transforms.begin();
Chris@330 382 i != transforms.end(); ++i) {
Chris@330 383
Chris@330 384 TransformDescription desc = i->second;
Chris@1429 385 QString identifier = desc.identifier;
Chris@330 386 QString maker = desc.maker;
Chris@330 387
Chris@330 388 QString td = desc.name;
Chris@330 389 QString tn = td.section(": ", 0, 0);
Chris@330 390 QString to = td.section(": ", 1);
Chris@330 391
Chris@1429 392 if (names[tn] > 1) {
Chris@330 393 maker.replace(QRegExp(tr(" [\\(<].*$")), "");
Chris@1429 394 tn = QString("%1 [%2]").arg(tn).arg(maker);
Chris@1429 395 }
Chris@330 396
Chris@330 397 if (to != "") {
Chris@330 398 desc.name = QString("%1: %2").arg(tn).arg(to);
Chris@330 399 } else {
Chris@330 400 desc.name = tn;
Chris@330 401 }
Chris@330 402
Chris@1429 403 m_transforms[identifier] = desc;
Chris@1429 404 }
Chris@457 405
Chris@457 406 m_transformsPopulated = true;
Chris@1865 407
Chris@1865 408 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@1865 409 SVCERR << "populateTransforms exiting" << endl;
Chris@1865 410 #endif
Chris@1865 411
Chris@1865 412 emit transformsPopulated();
Chris@330 413 }
Chris@330 414
Chris@330 415 void
Chris@330 416 TransformFactory::populateFeatureExtractionPlugins(TransformDescriptionMap &transforms)
Chris@330 417 {
Chris@1225 418 FeatureExtractionPluginFactory *factory =
Chris@1225 419 FeatureExtractionPluginFactory::instance();
Chris@1227 420
Chris@1227 421 QString errorMessage;
Chris@1227 422 std::vector<QString> plugs = factory->getPluginIdentifiers(errorMessage);
Chris@1227 423 if (errorMessage != "") {
Chris@1227 424 m_errorString = tr("Failed to list Vamp plugins: %1").arg(errorMessage);
Chris@1227 425 }
Chris@1225 426
Chris@576 427 if (m_exiting) return;
Chris@330 428
Chris@1546 429 for (int i = 0; in_range_for(plugs, i); ++i) {
Chris@330 430
Chris@1429 431 QString pluginId = plugs[i];
Chris@330 432
Chris@1223 433 piper_vamp::PluginStaticData psd = factory->getPluginStaticData(pluginId);
Chris@330 434
Chris@1223 435 if (psd.pluginKey == "") {
Chris@1223 436 cerr << "WARNING: TransformFactory::populateTransforms: No plugin static data available for instance " << pluginId << endl;
Chris@1223 437 continue;
Chris@1223 438 }
Chris@1223 439
Chris@1223 440 QString pluginName = QString::fromStdString(psd.basic.name);
Chris@330 441 QString category = factory->getPluginCategory(pluginId);
Chris@1223 442
Chris@1223 443 const auto &basicOutputs = psd.basicOutputInfo;
Chris@330 444
Chris@1223 445 for (const auto &o: basicOutputs) {
Chris@330 446
Chris@1223 447 QString outputName = QString::fromStdString(o.name);
Chris@330 448
Chris@1429 449 QString transformId = QString("%1:%2")
Chris@1223 450 .arg(pluginId).arg(QString::fromStdString(o.identifier));
Chris@330 451
Chris@1429 452 QString userName;
Chris@330 453 QString friendlyName;
Chris@1223 454 //!!! return to this QString units = outputs[j].unit.c_str();
Chris@1223 455 QString description = QString::fromStdString(psd.basic.description);
Chris@1223 456 QString maker = QString::fromStdString(psd.maker);
Chris@330 457 if (maker == "") maker = tr("<unknown maker>");
Chris@330 458
Chris@443 459 QString longDescription = description;
Chris@443 460
Chris@443 461 if (longDescription == "") {
Chris@1223 462 if (basicOutputs.size() == 1) {
Chris@443 463 longDescription = tr("Extract features using \"%1\" plugin (from %2)")
Chris@330 464 .arg(pluginName).arg(maker);
Chris@330 465 } else {
Chris@443 466 longDescription = tr("Extract features using \"%1\" output of \"%2\" plugin (from %3)")
Chris@1223 467 .arg(outputName).arg(pluginName).arg(maker);
Chris@330 468 }
Chris@330 469 } else {
Chris@1223 470 if (basicOutputs.size() == 1) {
Chris@443 471 longDescription = tr("%1 using \"%2\" plugin (from %3)")
Chris@443 472 .arg(longDescription).arg(pluginName).arg(maker);
Chris@330 473 } else {
Chris@443 474 longDescription = tr("%1 using \"%2\" output of \"%3\" plugin (from %4)")
Chris@1223 475 .arg(longDescription).arg(outputName).arg(pluginName).arg(maker);
Chris@330 476 }
Chris@330 477 }
Chris@330 478
Chris@1429 479 if (basicOutputs.size() == 1) {
Chris@1429 480 userName = pluginName;
Chris@330 481 friendlyName = pluginName;
Chris@1429 482 } else {
Chris@1429 483 userName = QString("%1: %2").arg(pluginName).arg(outputName);
Chris@1223 484 friendlyName = outputName;
Chris@1429 485 }
Chris@330 486
Chris@1223 487 bool configurable = (!psd.programs.empty() ||
Chris@1223 488 !psd.parameters.empty());
Chris@330 489
Chris@600 490 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@686 491 cerr << "Feature extraction plugin transform: " << transformId << " friendly name: " << friendlyName << endl;
Chris@600 492 #endif
Chris@330 493
Chris@1429 494 transforms[transformId] =
Chris@487 495 TransformDescription(TransformDescription::Analysis,
Chris@332 496 category,
Chris@332 497 transformId,
Chris@332 498 userName,
Chris@332 499 friendlyName,
Chris@332 500 description,
Chris@443 501 longDescription,
Chris@332 502 maker,
Chris@1223 503 //!!! units,
Chris@1223 504 "",
Chris@332 505 configurable);
Chris@1429 506 }
Chris@330 507 }
Chris@330 508 }
Chris@330 509
Chris@330 510 void
Chris@330 511 TransformFactory::populateRealTimePlugins(TransformDescriptionMap &transforms)
Chris@330 512 {
Chris@330 513 std::vector<QString> plugs =
Chris@1429 514 RealTimePluginFactory::getAllPluginIdentifiers();
Chris@576 515 if (m_exiting) return;
Chris@330 516
Chris@330 517 static QRegExp unitRE("[\\[\\(]([A-Za-z0-9/]+)[\\)\\]]$");
Chris@330 518
Chris@1546 519 for (int i = 0; in_range_for(plugs, i); ++i) {
Chris@330 520
Chris@1429 521 QString pluginId = plugs[i];
Chris@330 522
Chris@330 523 RealTimePluginFactory *factory =
Chris@330 524 RealTimePluginFactory::instanceFor(pluginId);
Chris@330 525
Chris@1429 526 if (!factory) {
Chris@1429 527 cerr << "WARNING: TransformFactory::populateTransforms: No real time plugin factory for instance " << pluginId << endl;
Chris@1429 528 continue;
Chris@1429 529 }
Chris@330 530
Chris@1830 531 RealTimePluginDescriptor descriptor =
Chris@330 532 factory->getPluginDescriptor(pluginId);
Chris@330 533
Chris@1830 534 if (descriptor.name == "") {
Chris@1429 535 cerr << "WARNING: TransformFactory::populateTransforms: Failed to query plugin " << pluginId << endl;
Chris@1429 536 continue;
Chris@1429 537 }
Chris@1429 538
Chris@1830 539 //!!! if (descriptor.controlOutputPortCount == 0 ||
Chris@1830 540 // descriptor.audioInputPortCount == 0) continue;
Chris@330 541
Chris@1830 542 // cout << "TransformFactory::populateRealTimePlugins: plugin " << pluginId << " has " << descriptor.controlOutputPortCount << " control output ports, " << descriptor.audioOutputPortCount << " audio outputs, " << descriptor.audioInputPortCount << " audio inputs" << endl;
Chris@1429 543
Chris@1830 544 QString pluginName = descriptor.name.c_str();
Chris@330 545 QString category = factory->getPluginCategory(pluginId);
Chris@1830 546 bool configurable = (descriptor.parameterCount > 0);
Chris@1830 547 QString maker = descriptor.maker.c_str();
Chris@330 548 if (maker == "") maker = tr("<unknown maker>");
Chris@330 549
Chris@1830 550 if (descriptor.audioInputPortCount > 0) {
Chris@330 551
Chris@1830 552 for (int j = 0; j < (int)descriptor.controlOutputPortCount; ++j) {
Chris@330 553
Chris@330 554 QString transformId = QString("%1:%2").arg(pluginId).arg(j);
Chris@330 555 QString userName;
Chris@330 556 QString units;
Chris@330 557 QString portName;
Chris@330 558
Chris@1830 559 if (j < (int)descriptor.controlOutputPortNames.size() &&
Chris@1830 560 descriptor.controlOutputPortNames[j] != "") {
Chris@330 561
Chris@1830 562 portName = descriptor.controlOutputPortNames[j].c_str();
Chris@330 563
Chris@330 564 userName = tr("%1: %2")
Chris@330 565 .arg(pluginName)
Chris@330 566 .arg(portName);
Chris@330 567
Chris@330 568 if (unitRE.indexIn(portName) >= 0) {
Chris@330 569 units = unitRE.cap(1);
Chris@330 570 }
Chris@330 571
Chris@1830 572 } else if (descriptor.controlOutputPortCount > 1) {
Chris@330 573
Chris@330 574 userName = tr("%1: Output %2")
Chris@330 575 .arg(pluginName)
Chris@330 576 .arg(j + 1);
Chris@330 577
Chris@330 578 } else {
Chris@330 579
Chris@330 580 userName = pluginName;
Chris@330 581 }
Chris@330 582
Chris@330 583 QString description;
Chris@330 584
Chris@330 585 if (portName != "") {
Chris@330 586 description = tr("Extract \"%1\" data output from \"%2\" effect plugin (from %3)")
Chris@330 587 .arg(portName)
Chris@330 588 .arg(pluginName)
Chris@330 589 .arg(maker);
Chris@330 590 } else {
Chris@330 591 description = tr("Extract data output %1 from \"%2\" effect plugin (from %3)")
Chris@330 592 .arg(j + 1)
Chris@330 593 .arg(pluginName)
Chris@330 594 .arg(maker);
Chris@330 595 }
Chris@330 596
Chris@330 597 transforms[transformId] =
Chris@487 598 TransformDescription(TransformDescription::EffectsData,
Chris@332 599 category,
Chris@332 600 transformId,
Chris@332 601 userName,
Chris@332 602 userName,
Chris@443 603 "",
Chris@332 604 description,
Chris@332 605 maker,
Chris@332 606 units,
Chris@332 607 configurable);
Chris@330 608 }
Chris@330 609 }
Chris@330 610
Chris@1830 611 if (!descriptor.isSynth || descriptor.audioInputPortCount > 0) {
Chris@330 612
Chris@1830 613 if (descriptor.audioOutputPortCount > 0) {
Chris@330 614
Chris@330 615 QString transformId = QString("%1:A").arg(pluginId);
Chris@487 616 TransformDescription::Type type = TransformDescription::Effects;
Chris@330 617
Chris@330 618 QString description = tr("Transform audio signal with \"%1\" effect plugin (from %2)")
Chris@330 619 .arg(pluginName)
Chris@330 620 .arg(maker);
Chris@330 621
Chris@1830 622 if (descriptor.audioInputPortCount == 0) {
Chris@487 623 type = TransformDescription::Generator;
Chris@330 624 QString description = tr("Generate audio signal using \"%1\" plugin (from %2)")
Chris@330 625 .arg(pluginName)
Chris@330 626 .arg(maker);
Chris@330 627 }
Chris@330 628
Chris@330 629 transforms[transformId] =
Chris@330 630 TransformDescription(type,
Chris@332 631 category,
Chris@332 632 transformId,
Chris@332 633 pluginName,
Chris@332 634 pluginName,
Chris@443 635 "",
Chris@332 636 description,
Chris@332 637 maker,
Chris@332 638 "",
Chris@332 639 configurable);
Chris@330 640 }
Chris@330 641 }
Chris@330 642 }
Chris@330 643 }
Chris@330 644
Chris@457 645 void
Chris@457 646 TransformFactory::populateUninstalledTransforms()
Chris@457 647 {
Chris@576 648 if (m_exiting) return;
Chris@576 649
Chris@460 650 populateTransforms();
Chris@576 651 if (m_exiting) return;
Chris@460 652
Chris@460 653 MutexLocker locker(&m_uninstalledTransformsMutex,
Chris@460 654 "TransformFactory::populateUninstalledTransforms");
Chris@460 655 if (m_uninstalledTransformsPopulated) return;
Chris@460 656
Chris@461 657 PluginRDFIndexer::getInstance()->indexConfiguredURLs();
Chris@576 658 if (m_exiting) return;
Chris@457 659
Chris@1844 660 PluginRDFIndexer::getInstance()->performConsistencyChecks();
Chris@1844 661
Chris@457 662 //!!! This will be amazingly slow
Chris@457 663
Chris@457 664 QStringList ids = PluginRDFIndexer::getInstance()->getIndexedPluginIds();
Chris@457 665
Chris@457 666 for (QStringList::const_iterator i = ids.begin(); i != ids.end(); ++i) {
Chris@457 667
Chris@457 668 PluginRDFDescription desc(*i);
Chris@457 669
Chris@457 670 QString name = desc.getPluginName();
Chris@600 671 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@600 672 if (name == "") {
Chris@1844 673 SVCERR << "TransformFactory::populateUninstalledTransforms: "
Chris@810 674 << "No name available for plugin " << *i
Chris@810 675 << ", skipping" << endl;
Chris@600 676 continue;
Chris@600 677 }
Chris@600 678 #endif
Chris@457 679
Chris@457 680 QString description = desc.getPluginDescription();
Chris@457 681 QString maker = desc.getPluginMaker();
Chris@1845 682 Provider provider = desc.getPluginProvider();
Chris@457 683
Chris@457 684 QStringList oids = desc.getOutputIds();
Chris@457 685
Chris@457 686 for (QStringList::const_iterator j = oids.begin(); j != oids.end(); ++j) {
Chris@457 687
Chris@457 688 TransformId tid = Transform::getIdentifierForPluginOutput(*i, *j);
Chris@457 689
Chris@457 690 if (m_transforms.find(tid) != m_transforms.end()) {
Chris@600 691 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@1844 692 SVCERR << "TransformFactory::populateUninstalledTransforms: "
Chris@1845 693 << tid << " is installed; adding provider if present, skipping rest" << endl;
Chris@600 694 #endif
Chris@1845 695 if (provider != Provider()) {
Chris@1845 696 if (m_transforms[tid].provider == Provider()) {
Chris@1845 697 m_transforms[tid].provider = provider;
Chris@468 698 }
Chris@468 699 }
Chris@457 700 continue;
Chris@457 701 }
Chris@457 702
Chris@600 703 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@1844 704 SVCERR << "TransformFactory::populateUninstalledTransforms: "
Chris@1844 705 << "adding " << tid << endl;
Chris@600 706 #endif
Chris@457 707
Chris@457 708 QString oname = desc.getOutputName(*j);
Chris@457 709 if (oname == "") oname = *j;
Chris@457 710
Chris@457 711 TransformDescription td;
Chris@487 712 td.type = TransformDescription::Analysis;
Chris@457 713 td.category = "";
Chris@457 714 td.identifier = tid;
Chris@457 715
Chris@457 716 if (oids.size() == 1) {
Chris@457 717 td.name = name;
Chris@457 718 } else if (name != "") {
Chris@457 719 td.name = tr("%1: %2").arg(name).arg(oname);
Chris@457 720 }
Chris@457 721
Chris@462 722 QString longDescription = description;
Chris@462 723 //!!! basically duplicated from above
Chris@462 724 if (longDescription == "") {
Chris@462 725 if (oids.size() == 1) {
Chris@462 726 longDescription = tr("Extract features using \"%1\" plugin (from %2)")
Chris@462 727 .arg(name).arg(maker);
Chris@462 728 } else {
Chris@462 729 longDescription = tr("Extract features using \"%1\" output of \"%2\" plugin (from %3)")
Chris@462 730 .arg(oname).arg(name).arg(maker);
Chris@462 731 }
Chris@462 732 } else {
Chris@462 733 if (oids.size() == 1) {
Chris@462 734 longDescription = tr("%1 using \"%2\" plugin (from %3)")
Chris@462 735 .arg(longDescription).arg(name).arg(maker);
Chris@462 736 } else {
Chris@462 737 longDescription = tr("%1 using \"%2\" output of \"%3\" plugin (from %4)")
Chris@462 738 .arg(longDescription).arg(oname).arg(name).arg(maker);
Chris@462 739 }
Chris@462 740 }
Chris@462 741
Chris@457 742 td.friendlyName = name; //!!!???
Chris@457 743 td.description = description;
Chris@462 744 td.longDescription = longDescription;
Chris@457 745 td.maker = maker;
Chris@1845 746 td.provider = provider;
Chris@457 747 td.units = "";
Chris@457 748 td.configurable = false;
Chris@457 749
Chris@457 750 m_uninstalledTransforms[tid] = td;
Chris@457 751 }
Chris@574 752
Chris@576 753 if (m_exiting) return;
Chris@457 754 }
Chris@457 755
Chris@457 756 m_uninstalledTransformsPopulated = true;
Chris@460 757
Chris@600 758 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@1844 759 SVCERR << "populateUninstalledTransforms exiting" << endl;
Chris@600 760 #endif
Chris@1865 761
Chris@1865 762 emit uninstalledTransformsPopulated();
Chris@457 763 }
Chris@350 764
Chris@350 765 Transform
Chris@1047 766 TransformFactory::getDefaultTransformFor(TransformId id, sv_samplerate_t rate)
Chris@350 767 {
Chris@350 768 Transform t;
Chris@350 769 t.setIdentifier(id);
Chris@1047 770 if (rate != 0) t.setSampleRate(rate);
Chris@350 771
Chris@1264 772 SVDEBUG << "TransformFactory::getDefaultTransformFor: identifier \""
Chris@1264 773 << id << "\"" << endl;
Chris@1264 774
Chris@1830 775 std::shared_ptr<Vamp::PluginBase> plugin = instantiateDefaultPluginFor(id, rate);
Chris@350 776
Chris@350 777 if (plugin) {
Chris@366 778 t.setPluginVersion(QString("%1").arg(plugin->getPluginVersion()));
Chris@350 779 setParametersFromPlugin(t, plugin);
Chris@350 780 makeContextConsistentWithPlugin(t, plugin);
Chris@350 781 }
Chris@350 782
Chris@350 783 return t;
Chris@350 784 }
Chris@350 785
Chris@1830 786 std::shared_ptr<Vamp::PluginBase>
Chris@351 787 TransformFactory::instantiatePluginFor(const Transform &transform)
Chris@351 788 {
Chris@1264 789 SVDEBUG << "TransformFactory::instantiatePluginFor: identifier \""
Chris@1264 790 << transform.getIdentifier() << "\"" << endl;
Chris@1264 791
Chris@1830 792 std::shared_ptr<Vamp::PluginBase> plugin = instantiateDefaultPluginFor
Chris@1047 793 (transform.getIdentifier(), transform.getSampleRate());
Chris@508 794
Chris@351 795 if (plugin) {
Chris@351 796 setPluginParameters(transform, plugin);
Chris@351 797 }
Chris@508 798
Chris@351 799 return plugin;
Chris@351 800 }
Chris@351 801
Chris@1830 802 std::shared_ptr<Vamp::PluginBase>
Chris@1047 803 TransformFactory::instantiateDefaultPluginFor(TransformId identifier,
Chris@1047 804 sv_samplerate_t rate)
Chris@350 805 {
Chris@350 806 Transform t;
Chris@350 807 t.setIdentifier(identifier);
Chris@1047 808 if (rate == 0) rate = 44100.0;
Chris@350 809 QString pluginId = t.getPluginIdentifier();
Chris@350 810
Chris@1830 811 std::shared_ptr<Vamp::PluginBase> plugin = nullptr;
Chris@350 812
Chris@350 813 if (t.getType() == Transform::FeatureExtraction) {
Chris@350 814
Chris@1264 815 SVDEBUG << "TransformFactory::instantiateDefaultPluginFor: identifier \""
Chris@1264 816 << identifier << "\" is a feature extraction transform" << endl;
Chris@1139 817
Chris@1225 818 FeatureExtractionPluginFactory *factory =
Chris@1225 819 FeatureExtractionPluginFactory::instance();
Chris@350 820
Chris@439 821 if (factory) {
Chris@1047 822 plugin = factory->instantiatePlugin(pluginId, rate);
Chris@439 823 }
Chris@350 824
Chris@1139 825 } else if (t.getType() == Transform::RealTimeEffect) {
Chris@1139 826
Chris@1264 827 SVDEBUG << "TransformFactory::instantiateDefaultPluginFor: identifier \""
Chris@1264 828 << identifier << "\" is a real-time transform" << endl;
Chris@350 829
Chris@350 830 RealTimePluginFactory *factory =
Chris@350 831 RealTimePluginFactory::instanceFor(pluginId);
Chris@439 832
Chris@439 833 if (factory) {
Chris@439 834 plugin = factory->instantiatePlugin(pluginId, 0, 0, rate, 1024, 1);
Chris@439 835 }
Chris@1139 836
Chris@1139 837 } else {
Chris@1264 838 SVDEBUG << "TransformFactory: ERROR: transform id \""
Chris@1264 839 << identifier << "\" is of unknown type" << endl;
Chris@350 840 }
Chris@350 841
Chris@350 842 return plugin;
Chris@350 843 }
Chris@350 844
Chris@330 845 bool
Chris@330 846 TransformFactory::haveTransform(TransformId identifier)
Chris@330 847 {
Chris@460 848 populateTransforms();
Chris@330 849 return (m_transforms.find(identifier) != m_transforms.end());
Chris@330 850 }
Chris@330 851
Chris@330 852 QString
Chris@330 853 TransformFactory::getTransformName(TransformId identifier)
Chris@330 854 {
Chris@330 855 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@1429 856 return m_transforms[identifier].name;
Chris@330 857 } else return "";
Chris@330 858 }
Chris@330 859
Chris@330 860 QString
Chris@330 861 TransformFactory::getTransformFriendlyName(TransformId identifier)
Chris@330 862 {
Chris@330 863 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@1429 864 return m_transforms[identifier].friendlyName;
Chris@330 865 } else return "";
Chris@330 866 }
Chris@330 867
Chris@330 868 QString
Chris@330 869 TransformFactory::getTransformUnits(TransformId identifier)
Chris@330 870 {
Chris@330 871 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@1429 872 return m_transforms[identifier].units;
Chris@330 873 } else return "";
Chris@330 874 }
Chris@330 875
Chris@1845 876 Provider
Chris@1845 877 TransformFactory::getTransformProvider(TransformId identifier)
Chris@472 878 {
Chris@472 879 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@1845 880 return m_transforms[identifier].provider;
Chris@1845 881 } else return {};
Chris@472 882 }
Chris@472 883
Chris@350 884 Vamp::Plugin::InputDomain
Chris@350 885 TransformFactory::getTransformInputDomain(TransformId identifier)
Chris@350 886 {
Chris@350 887 Transform transform;
Chris@350 888 transform.setIdentifier(identifier);
Chris@350 889
Chris@1264 890 SVDEBUG << "TransformFactory::getTransformInputDomain: identifier \""
Chris@1264 891 << identifier << "\"" << endl;
Chris@1264 892
Chris@350 893 if (transform.getType() != Transform::FeatureExtraction) {
Chris@350 894 return Vamp::Plugin::TimeDomain;
Chris@350 895 }
Chris@350 896
Chris@1830 897 std::shared_ptr<Vamp::Plugin> plugin =
Chris@1830 898 std::dynamic_pointer_cast<Vamp::Plugin>
Chris@1830 899 (instantiateDefaultPluginFor(identifier, 0));
Chris@350 900
Chris@350 901 if (plugin) {
Chris@350 902 Vamp::Plugin::InputDomain d = plugin->getInputDomain();
Chris@350 903 return d;
Chris@350 904 }
Chris@350 905
Chris@350 906 return Vamp::Plugin::TimeDomain;
Chris@350 907 }
Chris@350 908
Chris@330 909 bool
Chris@330 910 TransformFactory::isTransformConfigurable(TransformId identifier)
Chris@330 911 {
Chris@330 912 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@1429 913 return m_transforms[identifier].configurable;
Chris@330 914 } else return false;
Chris@330 915 }
Chris@330 916
Chris@330 917 bool
Chris@330 918 TransformFactory::getTransformChannelRange(TransformId identifier,
Chris@330 919 int &min, int &max)
Chris@330 920 {
Chris@330 921 QString id = identifier.section(':', 0, 2);
Chris@330 922
Chris@1225 923 if (RealTimePluginFactory::instanceFor(id)) {
Chris@350 924
Chris@1830 925 RealTimePluginDescriptor descriptor =
Chris@330 926 RealTimePluginFactory::instanceFor(id)->
Chris@330 927 getPluginDescriptor(id);
Chris@1830 928 if (descriptor.name == "") {
Chris@1830 929 return false;
Chris@1830 930 }
Chris@330 931
Chris@1830 932 min = descriptor.audioInputPortCount;
Chris@1830 933 max = descriptor.audioInputPortCount;
Chris@330 934
Chris@330 935 return true;
Chris@1225 936
Chris@1225 937 } else {
Chris@1225 938
Chris@1225 939 auto psd = FeatureExtractionPluginFactory::instance()->
Chris@1225 940 getPluginStaticData(id);
Chris@1225 941 if (psd.pluginKey == "") return false;
Chris@1225 942
Chris@1225 943 min = (int)psd.minChannelCount;
Chris@1225 944 max = (int)psd.maxChannelCount;
Chris@1225 945
Chris@1225 946 return true;
Chris@330 947 }
Chris@330 948
Chris@330 949 return false;
Chris@330 950 }
Chris@332 951
Chris@332 952 void
Chris@332 953 TransformFactory::setParametersFromPlugin(Transform &transform,
Chris@1830 954 std::shared_ptr<Vamp::PluginBase> plugin)
Chris@332 955 {
Chris@332 956 Transform::ParameterMap pmap;
Chris@332 957
Chris@350 958 //!!! record plugin & API version
Chris@350 959
Chris@350 960 //!!! check that this is the right plugin!
Chris@350 961
Chris@332 962 Vamp::PluginBase::ParameterList parameters =
Chris@332 963 plugin->getParameterDescriptors();
Chris@332 964
Chris@332 965 for (Vamp::PluginBase::ParameterList::const_iterator i = parameters.begin();
Chris@332 966 i != parameters.end(); ++i) {
Chris@332 967 pmap[i->identifier.c_str()] = plugin->getParameter(i->identifier);
Chris@810 968 // cerr << "TransformFactory::setParametersFromPlugin: parameter "
Chris@583 969 // << i->identifier << " -> value " <<
Chris@687 970 // pmap[i->identifier.c_str()] << endl;
Chris@332 971 }
Chris@332 972
Chris@332 973 transform.setParameters(pmap);
Chris@332 974
Chris@332 975 if (plugin->getPrograms().empty()) {
Chris@332 976 transform.setProgram("");
Chris@332 977 } else {
Chris@332 978 transform.setProgram(plugin->getCurrentProgram().c_str());
Chris@332 979 }
Chris@332 980
Chris@1830 981 std::shared_ptr<RealTimePluginInstance> rtpi =
Chris@1830 982 std::dynamic_pointer_cast<RealTimePluginInstance>(plugin);
Chris@332 983
Chris@332 984 Transform::ConfigurationMap cmap;
Chris@332 985
Chris@332 986 if (rtpi) {
Chris@332 987
Chris@332 988 RealTimePluginInstance::ConfigurationPairMap configurePairs =
Chris@332 989 rtpi->getConfigurePairs();
Chris@332 990
Chris@332 991 for (RealTimePluginInstance::ConfigurationPairMap::const_iterator i
Chris@332 992 = configurePairs.begin(); i != configurePairs.end(); ++i) {
Chris@332 993 cmap[i->first.c_str()] = i->second.c_str();
Chris@332 994 }
Chris@332 995 }
Chris@332 996
Chris@332 997 transform.setConfiguration(cmap);
Chris@332 998 }
Chris@332 999
Chris@332 1000 void
Chris@350 1001 TransformFactory::setPluginParameters(const Transform &transform,
Chris@1830 1002 std::shared_ptr<Vamp::PluginBase> plugin)
Chris@350 1003 {
Chris@350 1004 //!!! check plugin & API version (see e.g. PluginXml::setParameters)
Chris@350 1005
Chris@350 1006 //!!! check that this is the right plugin!
Chris@350 1007
Chris@1830 1008 std::shared_ptr<RealTimePluginInstance> rtpi =
Chris@1830 1009 std::dynamic_pointer_cast<RealTimePluginInstance>(plugin);
Chris@350 1010
Chris@350 1011 if (rtpi) {
Chris@350 1012 const Transform::ConfigurationMap &cmap = transform.getConfiguration();
Chris@350 1013 for (Transform::ConfigurationMap::const_iterator i = cmap.begin();
Chris@350 1014 i != cmap.end(); ++i) {
Chris@350 1015 rtpi->configure(i->first.toStdString(), i->second.toStdString());
Chris@350 1016 }
Chris@350 1017 }
Chris@350 1018
Chris@350 1019 if (transform.getProgram() != "") {
Chris@350 1020 plugin->selectProgram(transform.getProgram().toStdString());
Chris@350 1021 }
Chris@350 1022
Chris@350 1023 const Transform::ParameterMap &pmap = transform.getParameters();
Chris@350 1024
Chris@350 1025 Vamp::PluginBase::ParameterList parameters =
Chris@350 1026 plugin->getParameterDescriptors();
Chris@350 1027
Chris@350 1028 for (Vamp::PluginBase::ParameterList::const_iterator i = parameters.begin();
Chris@350 1029 i != parameters.end(); ++i) {
Chris@350 1030 QString key = i->identifier.c_str();
Chris@350 1031 Transform::ParameterMap::const_iterator pmi = pmap.find(key);
Chris@350 1032 if (pmi != pmap.end()) {
Chris@350 1033 plugin->setParameter(i->identifier, pmi->second);
Chris@350 1034 }
Chris@350 1035 }
Chris@350 1036 }
Chris@350 1037
Chris@350 1038 void
Chris@332 1039 TransformFactory::makeContextConsistentWithPlugin(Transform &transform,
Chris@1830 1040 std::shared_ptr<Vamp::PluginBase> plugin)
Chris@332 1041 {
Chris@1830 1042 std::shared_ptr<Vamp::Plugin> vp =
Chris@1830 1043 std::dynamic_pointer_cast<Vamp::Plugin>(plugin);
Chris@332 1044
Chris@332 1045 if (!vp) {
Chris@332 1046 // time domain input for real-time effects plugin
Chris@332 1047 if (!transform.getBlockSize()) {
Chris@332 1048 if (!transform.getStepSize()) transform.setStepSize(1024);
Chris@332 1049 transform.setBlockSize(transform.getStepSize());
Chris@332 1050 } else {
Chris@332 1051 transform.setStepSize(transform.getBlockSize());
Chris@332 1052 }
Chris@332 1053 } else {
Chris@332 1054 Vamp::Plugin::InputDomain domain = vp->getInputDomain();
Chris@332 1055 if (!transform.getStepSize()) {
Chris@1039 1056 transform.setStepSize((int)vp->getPreferredStepSize());
Chris@332 1057 }
Chris@332 1058 if (!transform.getBlockSize()) {
Chris@1039 1059 transform.setBlockSize((int)vp->getPreferredBlockSize());
Chris@332 1060 }
Chris@332 1061 if (!transform.getBlockSize()) {
Chris@332 1062 transform.setBlockSize(1024);
Chris@332 1063 }
Chris@332 1064 if (!transform.getStepSize()) {
Chris@332 1065 if (domain == Vamp::Plugin::FrequencyDomain) {
Chris@443 1066 // cerr << "frequency domain, step = " << blockSize/2 << endl;
Chris@332 1067 transform.setStepSize(transform.getBlockSize()/2);
Chris@332 1068 } else {
Chris@443 1069 // cerr << "time domain, step = " << blockSize/2 << endl;
Chris@332 1070 transform.setStepSize(transform.getBlockSize());
Chris@332 1071 }
Chris@332 1072 }
Chris@332 1073 }
Chris@332 1074 }
Chris@332 1075
Chris@350 1076 QString
Chris@350 1077 TransformFactory::getPluginConfigurationXml(const Transform &t)
Chris@332 1078 {
Chris@350 1079 QString xml;
Chris@350 1080
Chris@1264 1081 SVDEBUG << "TransformFactory::getPluginConfigurationXml: identifier \""
Chris@1264 1082 << t.getIdentifier() << "\"" << endl;
Chris@1264 1083
Chris@1830 1084 auto plugin = instantiateDefaultPluginFor(t.getIdentifier(), 0);
Chris@350 1085 if (!plugin) {
Chris@1264 1086 SVDEBUG << "TransformFactory::getPluginConfigurationXml: "
Chris@1264 1087 << "Unable to instantiate plugin for transform \""
Chris@1264 1088 << t.getIdentifier() << "\"" << endl;
Chris@350 1089 return xml;
Chris@332 1090 }
Chris@332 1091
Chris@351 1092 setPluginParameters(t, plugin);
Chris@351 1093
Chris@350 1094 QTextStream out(&xml);
Chris@350 1095 PluginXml(plugin).toXml(out);
Chris@332 1096
Chris@350 1097 return xml;
Chris@350 1098 }
Chris@332 1099
Chris@350 1100 void
Chris@350 1101 TransformFactory::setParametersFromPluginConfigurationXml(Transform &t,
Chris@350 1102 QString xml)
Chris@350 1103 {
Chris@1264 1104 SVDEBUG << "TransformFactory::setParametersFromPluginConfigurationXml: identifier \""
Chris@1264 1105 << t.getIdentifier() << "\"" << endl;
Chris@1264 1106
Chris@1830 1107 auto plugin = instantiateDefaultPluginFor(t.getIdentifier(), 0);
Chris@350 1108 if (!plugin) {
Chris@1264 1109 SVDEBUG << "TransformFactory::setParametersFromPluginConfigurationXml: "
Chris@1264 1110 << "Unable to instantiate plugin for transform \""
Chris@1264 1111 << t.getIdentifier() << "\"" << endl;
Chris@350 1112 return;
Chris@332 1113 }
Chris@332 1114
Chris@350 1115 PluginXml(plugin).setParametersFromXml(xml);
Chris@350 1116 setParametersFromPlugin(t, plugin);
Chris@332 1117 }
Chris@332 1118
Chris@443 1119 TransformFactory::SearchResults
Chris@443 1120 TransformFactory::search(QString keyword)
Chris@443 1121 {
Chris@443 1122 QStringList keywords;
Chris@443 1123 keywords << keyword;
Chris@443 1124 return search(keywords);
Chris@443 1125 }
Chris@443 1126
Chris@443 1127 TransformFactory::SearchResults
Chris@443 1128 TransformFactory::search(QStringList keywords)
Chris@443 1129 {
Chris@460 1130 populateTransforms();
Chris@443 1131
Chris@1842 1132 SearchResults results = searchUnadjusted(keywords);
Chris@1842 1133
Chris@447 1134 if (keywords.size() > 1) {
Chris@1842 1135
Chris@1842 1136 // If there are any hits for all keywords in a row, put them
Chris@1842 1137 // in (replacing previous hits for the same transforms) but
Chris@1842 1138 // ensure they score more than any of the others
Chris@1842 1139
Chris@1842 1140 int maxScore = 0;
Chris@1842 1141 for (auto r: results) {
Chris@1842 1142 if (r.second.score > maxScore) {
Chris@1842 1143 maxScore = r.second.score;
Chris@1842 1144 }
Chris@1842 1145 }
Chris@1842 1146
Chris@1842 1147 QStringList oneBigKeyword;
Chris@1842 1148 oneBigKeyword << keywords.join(" ");
Chris@1842 1149 SearchResults oneBigKeywordResults = searchUnadjusted(oneBigKeyword);
Chris@1842 1150 for (auto r: oneBigKeywordResults) {
Chris@1842 1151 results[r.first] = r.second;
Chris@1842 1152 results[r.first].score += maxScore;
Chris@1842 1153 }
Chris@447 1154 }
Chris@447 1155
Chris@1842 1156 return results;
Chris@1842 1157 }
Chris@1842 1158
Chris@1842 1159 TransformFactory::SearchResults
Chris@1842 1160 TransformFactory::searchUnadjusted(QStringList keywords)
Chris@1842 1161 {
Chris@443 1162 SearchResults results;
Chris@457 1163 TextMatcher matcher;
Chris@443 1164
Chris@443 1165 for (TransformDescriptionMap::const_iterator i = m_transforms.begin();
Chris@443 1166 i != m_transforms.end(); ++i) {
Chris@443 1167
Chris@457 1168 TextMatcher::Match match;
Chris@443 1169
Chris@457 1170 match.key = i->first;
Chris@443 1171
Chris@487 1172 matcher.test(match, keywords,
Chris@487 1173 getTransformTypeName(i->second.type),
Chris@487 1174 tr("Plugin type"), 5);
Chris@487 1175
Chris@457 1176 matcher.test(match, keywords, i->second.category, tr("Category"), 20);
Chris@457 1177 matcher.test(match, keywords, i->second.identifier, tr("System Identifier"), 6);
Chris@457 1178 matcher.test(match, keywords, i->second.name, tr("Name"), 30);
Chris@457 1179 matcher.test(match, keywords, i->second.description, tr("Description"), 20);
Chris@457 1180 matcher.test(match, keywords, i->second.maker, tr("Maker"), 10);
Chris@457 1181 matcher.test(match, keywords, i->second.units, tr("Units"), 10);
Chris@457 1182
Chris@457 1183 if (match.score > 0) results[i->first] = match;
Chris@457 1184 }
Chris@457 1185
Chris@460 1186 if (!m_uninstalledTransformsMutex.tryLock()) {
Chris@460 1187 // uninstalled transforms are being populated; this may take some time,
Chris@484 1188 // and they aren't critical, but we will speed them up if necessary
Chris@1264 1189 SVDEBUG << "TransformFactory::search: Uninstalled transforms mutex is held, skipping" << endl;
Chris@484 1190 m_populatingSlowly = false;
Chris@460 1191 return results;
Chris@460 1192 }
Chris@460 1193
Chris@460 1194 if (!m_uninstalledTransformsPopulated) {
Chris@1264 1195 SVDEBUG << "WARNING: TransformFactory::search: Uninstalled transforms are not populated yet" << endl
Chris@1264 1196 << "and are not being populated either -- was the thread not started correctly?" << endl;
Chris@460 1197 m_uninstalledTransformsMutex.unlock();
Chris@460 1198 return results;
Chris@460 1199 }
Chris@460 1200
Chris@460 1201 m_uninstalledTransformsMutex.unlock();
Chris@457 1202
Chris@457 1203 for (TransformDescriptionMap::const_iterator i = m_uninstalledTransforms.begin();
Chris@457 1204 i != m_uninstalledTransforms.end(); ++i) {
Chris@457 1205
Chris@457 1206 TextMatcher::Match match;
Chris@457 1207
Chris@457 1208 match.key = i->first;
Chris@457 1209
Chris@487 1210 matcher.test(match, keywords,
Chris@487 1211 getTransformTypeName(i->second.type),
Chris@487 1212 tr("Plugin type"), 2);
Chris@487 1213
Chris@457 1214 matcher.test(match, keywords, i->second.category, tr("Category"), 10);
Chris@457 1215 matcher.test(match, keywords, i->second.identifier, tr("System Identifier"), 3);
Chris@457 1216 matcher.test(match, keywords, i->second.name, tr("Name"), 15);
Chris@457 1217 matcher.test(match, keywords, i->second.description, tr("Description"), 10);
Chris@457 1218 matcher.test(match, keywords, i->second.maker, tr("Maker"), 5);
Chris@457 1219 matcher.test(match, keywords, i->second.units, tr("Units"), 5);
Chris@443 1220
Chris@443 1221 if (match.score > 0) results[i->first] = match;
Chris@443 1222 }
Chris@443 1223
Chris@1842 1224 #ifdef DEBUG_TRANSFORM_FACTORY
Chris@1842 1225 SVCERR << "TransformFactory::search: keywords are: " << keywords.join(", ")
Chris@1842 1226 << endl;
Chris@1842 1227 int n = int(results.size()), i = 1;
Chris@1842 1228 SVCERR << "TransformFactory::search: results (" << n << "):" << endl;
Chris@1842 1229
Chris@1842 1230 for (const auto &r: results) {
Chris@1842 1231 QStringList frags;
Chris@1842 1232 for (const auto &f: r.second.fragments) {
Chris@1842 1233 frags << QString("{\"%1\": \"%2\"}").arg(f.first).arg(f.second);
Chris@1842 1234 }
Chris@1842 1235 SVCERR << "[" << i << "/" << n << "] id " << r.first
Chris@1842 1236 << ": score " << r.second.score
Chris@1842 1237 << ", key " << r.second.key << ", fragments "
Chris@1842 1238 << frags.join(";") << endl;
Chris@1842 1239 ++i;
Chris@1842 1240 }
Chris@1842 1241 SVCERR << endl;
Chris@1842 1242 #endif
Chris@1842 1243
Chris@443 1244 return results;
Chris@443 1245 }
Chris@443 1246