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
|