annotate plugincandidates.cpp @ 2:2288c1d05c28

Simple Qt-based class to invoke the helper
author Chris Cannam
date Tue, 12 Apr 2016 17:38:57 +0100
parents
children 3bae396cf8e0
rev   line source
Chris@2 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@2 2
Chris@2 3 #include "plugincandidates.h"
Chris@2 4
Chris@2 5 #include <set>
Chris@2 6 #include <stdexcept>
Chris@2 7 #include <iostream>
Chris@2 8
Chris@2 9 #include <QProcess>
Chris@2 10 #include <QDir>
Chris@2 11
Chris@2 12 #ifdef _WIN32
Chris@2 13 #define PLUGIN_GLOB "*.dll"
Chris@2 14 #define PATH_SEPARATOR ';'
Chris@2 15 #else
Chris@2 16 #define PATH_SEPARATOR ':'
Chris@2 17 #ifdef __APPLE__
Chris@2 18 #define PLUGIN_GLOB "*.dylib *.so"
Chris@2 19 #else
Chris@2 20 #define PLUGIN_GLOB "*.so"
Chris@2 21 #endif
Chris@2 22 #endif
Chris@2 23
Chris@2 24 using namespace std;
Chris@2 25
Chris@2 26 PluginCandidates::PluginCandidates(string helperExecutableName) :
Chris@2 27 m_helper(helperExecutableName)
Chris@2 28 {
Chris@2 29 }
Chris@2 30
Chris@2 31 vector<string>
Chris@2 32 PluginCandidates::getCandidateLibrariesFor(string tag)
Chris@2 33 {
Chris@2 34 return m_candidates[tag];
Chris@2 35 }
Chris@2 36
Chris@2 37 vector<PluginCandidates::FailureRec>
Chris@2 38 PluginCandidates::getFailedLibrariesFor(string tag)
Chris@2 39 {
Chris@2 40 return m_failures[tag];
Chris@2 41 }
Chris@2 42
Chris@2 43 vector<string>
Chris@2 44 PluginCandidates::getLibrariesInPath(vector<string> path)
Chris@2 45 {
Chris@2 46 vector<string> candidates;
Chris@2 47
Chris@2 48 for (string dirname: path) {
Chris@2 49
Chris@2 50 //#ifdef DEBUG_PLUGIN_SCAN_AND_INSTANTIATE
Chris@2 51 cerr << "getLibrariesInPath: scanning directory " << dirname << endl;
Chris@2 52 //#endif
Chris@2 53
Chris@2 54 QDir dir(dirname.c_str(), PLUGIN_GLOB,
Chris@2 55 QDir::Name | QDir::IgnoreCase,
Chris@2 56 QDir::Files | QDir::Readable);
Chris@2 57
Chris@2 58 for (unsigned int i = 0; i < dir.count(); ++i) {
Chris@2 59 QString soname = dir.filePath(dir[i]);
Chris@2 60 candidates.push_back(soname.toStdString());
Chris@2 61 }
Chris@2 62 }
Chris@2 63
Chris@2 64 return candidates;
Chris@2 65 }
Chris@2 66
Chris@2 67 void
Chris@2 68 PluginCandidates::scan(string tag,
Chris@2 69 vector<string> pluginPath,
Chris@2 70 string descriptorFunctionName)
Chris@2 71 {
Chris@2 72 vector<string> libraries = getLibrariesInPath(pluginPath);
Chris@2 73 vector<string> remaining = libraries;
Chris@2 74
Chris@2 75 int runlimit = 20;
Chris@2 76 int runcount = 0;
Chris@2 77
Chris@2 78 vector<string> result;
Chris@2 79
Chris@2 80 while (result.size() < libraries.size() && runcount < runlimit) {
Chris@2 81 vector<string> output = runHelper(remaining, descriptorFunctionName);
Chris@2 82 result.insert(result.end(), output.begin(), output.end());
Chris@2 83 int shortfall = int(remaining.size()) - int(output.size());
Chris@2 84 if (shortfall > 0) {
Chris@2 85 // Helper bailed out for some reason presumably associated
Chris@2 86 // with the plugin following the last one it reported
Chris@2 87 // on. Add a null entry for that one and continue with the
Chris@2 88 // following ones.
Chris@2 89 result.push_back("");
Chris@2 90 if (shortfall == 1) {
Chris@2 91 remaining = vector<string>();
Chris@2 92 } else {
Chris@2 93 remaining = vector<string>
Chris@2 94 (remaining.rbegin(), remaining.rbegin() + shortfall - 1);
Chris@2 95 }
Chris@2 96 }
Chris@2 97 ++runcount;
Chris@2 98 }
Chris@2 99
Chris@2 100 recordResult(tag, result);
Chris@2 101 }
Chris@2 102
Chris@2 103 vector<string>
Chris@2 104 PluginCandidates::runHelper(vector<string> libraries, string descriptor)
Chris@2 105 {
Chris@2 106 vector<string> output;
Chris@2 107 cerr << "running helper with following library list:" << endl;
Chris@2 108 for (auto &lib: libraries) cerr << lib << endl;
Chris@2 109
Chris@2 110 QProcess process;
Chris@2 111 process.setReadChannel(QProcess::StandardOutput);
Chris@2 112 process.start(m_helper.c_str(), { descriptor.c_str() });
Chris@2 113 if (!process.waitForStarted()) {
Chris@2 114 cerr << "helper failed to start" << endl;
Chris@2 115 throw runtime_error("plugin load helper failed to start");
Chris@2 116 }
Chris@2 117 for (auto &lib: libraries) {
Chris@2 118 process.write(lib.c_str(), lib.size());
Chris@2 119 process.write("\n", 1);
Chris@2 120 }
Chris@2 121
Chris@2 122 int buflen = 4096;
Chris@2 123 while (process.waitForReadyRead()) {
Chris@2 124 char buf[buflen];
Chris@2 125 qint64 linelen = process.readLine(buf, buflen);
Chris@2 126 // cerr << "read line: " << buf;
Chris@2 127 if (linelen < 0) {
Chris@2 128 cerr << "read failed from plugin load helper" << endl;
Chris@2 129 return output;
Chris@2 130 }
Chris@2 131 output.push_back(buf);
Chris@2 132 if (output.size() == libraries.size()) {
Chris@2 133 process.close();
Chris@2 134 process.waitForFinished();
Chris@2 135 break;
Chris@2 136 }
Chris@2 137 }
Chris@2 138
Chris@2 139 return output;
Chris@2 140 }
Chris@2 141
Chris@2 142 void
Chris@2 143 PluginCandidates::recordResult(string tag, vector<string> result)
Chris@2 144 {
Chris@2 145 cerr << "recordResult: not yet implemented, but result was:" << endl;
Chris@2 146 for (auto &r: result) cerr << r;
Chris@2 147 cerr << "(ends)" << endl;
Chris@2 148 }
Chris@2 149
Chris@2 150 int main(int argc, char **argv)
Chris@2 151 {
Chris@2 152 //!!! just a test
Chris@2 153 PluginCandidates candidates("./helper");
Chris@2 154 candidates.scan("vamp",
Chris@2 155 { "/usr/lib/vamp", "/usr/local/lib/vamp" },
Chris@2 156 "vampGetPluginDescriptor");
Chris@2 157 }
Chris@2 158