Chris@2
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
Chris@0
|
2
|
Chris@0
|
3 /**
|
Chris@1
|
4 * Plugin Load Checker Helper
|
Chris@0
|
5 *
|
Chris@1
|
6 * This program accepts the name of a descriptor function as its only
|
Chris@1
|
7 * command-line argument. It then reads a list of plugin library paths
|
Chris@1
|
8 * from stdin, one per line. For each path read, it attempts to load
|
Chris@1
|
9 * that library and retrieve the named descriptor function, printing a
|
Chris@1
|
10 * line to stdout reporting whether this was successful or not and
|
Chris@1
|
11 * then flushing stdout. The output line format is described
|
Chris@1
|
12 * below. The program exits with code 0 if all libraries were loaded
|
Chris@1
|
13 * successfully and non-zero otherwise.
|
Chris@0
|
14 *
|
Chris@0
|
15 * Note that library paths must be ready to pass to dlopen() or
|
Chris@0
|
16 * equivalent; this usually means they should be absolute paths.
|
Chris@0
|
17 *
|
Chris@0
|
18 * Output line for successful load of library libname.so:
|
Chris@0
|
19 * SUCCESS|/path/to/libname.so|
|
Chris@0
|
20 *
|
Chris@0
|
21 * Output line for failed load of library libname.so:
|
Chris@0
|
22 * FAILURE|/path/to/libname.so|Reason for failure if available
|
Chris@0
|
23 *
|
Chris@2
|
24 * Sometimes plugins will crash completely on load, bringing down this
|
Chris@2
|
25 * program with them. If the program exits before all listed plugins
|
Chris@2
|
26 * have been checked, this means that the plugin following the last
|
Chris@2
|
27 * reported one has crashed. Typically the caller may want to run it
|
Chris@2
|
28 * again, omitting that plugin.
|
Chris@0
|
29 */
|
Chris@0
|
30
|
Chris@0
|
31 #ifdef _WIN32
|
Chris@0
|
32 #include <windows.h>
|
Chris@0
|
33 #include <process.h>
|
Chris@0
|
34 #define DLOPEN(a,b) LoadLibrary((a).toStdWString().c_str())
|
Chris@0
|
35 #define DLSYM(a,b) GetProcAddress((HINSTANCE)(a),(b))
|
Chris@0
|
36 #define DLCLOSE(a) (!FreeLibrary((HINSTANCE)(a)))
|
Chris@0
|
37 #define DLERROR() ""
|
Chris@0
|
38 #else
|
Chris@0
|
39 #include <dlfcn.h>
|
Chris@0
|
40 #define DLOPEN(a,b) dlopen((a).c_str(),(b))
|
Chris@0
|
41 #define DLSYM(a,b) dlsym((a),(b).c_str())
|
Chris@0
|
42 #define DLCLOSE(a) dlclose((a))
|
Chris@0
|
43 #define DLERROR() dlerror()
|
Chris@0
|
44 #endif
|
Chris@0
|
45
|
Chris@0
|
46 #include <string>
|
Chris@0
|
47 #include <iostream>
|
Chris@0
|
48
|
Chris@0
|
49 using namespace std;
|
Chris@0
|
50
|
Chris@0
|
51 string error()
|
Chris@0
|
52 {
|
Chris@0
|
53 string e = dlerror();
|
Chris@0
|
54 if (e == "") return "(unknown error)";
|
Chris@0
|
55 else return e;
|
Chris@0
|
56 }
|
Chris@0
|
57
|
Chris@1
|
58 string check(string soname, string descriptor)
|
Chris@0
|
59 {
|
Chris@0
|
60 void *handle = DLOPEN(soname, RTLD_NOW | RTLD_LOCAL);
|
Chris@0
|
61 if (!handle) {
|
Chris@0
|
62 return "Unable to open plugin library: " + error();
|
Chris@0
|
63 }
|
Chris@0
|
64
|
Chris@0
|
65 void *fn = DLSYM(handle, descriptor);
|
Chris@0
|
66 if (!fn) {
|
Chris@0
|
67 return "Failed to find plugin descriptor " + descriptor +
|
Chris@0
|
68 " in library: " + error();
|
Chris@0
|
69 }
|
Chris@0
|
70
|
Chris@0
|
71 return "";
|
Chris@0
|
72 }
|
Chris@0
|
73
|
Chris@0
|
74 int main(int argc, char **argv)
|
Chris@0
|
75 {
|
Chris@0
|
76 bool allGood = true;
|
Chris@0
|
77 string soname;
|
Chris@0
|
78
|
Chris@1
|
79 if (argc != 2) {
|
Chris@1
|
80 cerr << "\nUsage:\n " << argv[0] << " descriptorname\n"
|
Chris@1
|
81 "\nwhere descriptorname is the name of a plugin descriptor function to be sought\n"
|
Chris@1
|
82 "in each library (e.g. vampGetPluginDescriptor for Vamp plugins). The list of\n"
|
Chris@1
|
83 "candidate plugin library filenames is read from stdin.\n" << endl;
|
Chris@1
|
84 return 2;
|
Chris@1
|
85 }
|
Chris@1
|
86
|
Chris@1
|
87 string descriptor = argv[1];
|
Chris@1
|
88
|
Chris@0
|
89 while (getline(cin, soname)) {
|
Chris@1
|
90 string report = check(soname, descriptor);
|
Chris@0
|
91 if (report != "") {
|
Chris@0
|
92 cout << "FAILURE|" << soname << "|" << report << endl;
|
Chris@0
|
93 allGood = false;
|
Chris@0
|
94 } else {
|
Chris@0
|
95 cout << "SUCCESS|" << soname << "|" << endl;
|
Chris@0
|
96 }
|
Chris@0
|
97 }
|
Chris@0
|
98
|
Chris@0
|
99 return allGood ? 0 : 1;
|
Chris@0
|
100 }
|