forked from eriksvedang/cakelisp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDynamicLoader.cpp
172 lines (150 loc) · 4.62 KB
/
DynamicLoader.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
#include "DynamicLoader.hpp"
#include <stdio.h>
#include <string>
#include <unordered_map>
#include "FileUtilities.hpp"
#include "Utilities.hpp"
#if defined(UNIX) || defined(MACOS)
#include <dlfcn.h>
#elif WINDOWS
#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#else
#error Platform support is needed for dynamic loading
#endif
struct DynamicLibrary
{
DynamicLibHandle handle;
};
typedef std::unordered_map<std::string, DynamicLibrary> DynamicLibraryMap;
static DynamicLibraryMap dynamicLibraries;
DynamicLibHandle loadDynamicLibrary(const char* libraryPath)
{
void* libHandle = nullptr;
#if defined(UNIX) || defined(MACOS)
// Clear error
dlerror();
// RTLD_LAZY: Don't look up symbols the shared library needs until it encounters them
// RTLD_GLOBAL: Allow subsequently loaded libraries to resolve from this library (mainly for
// compile-time function execution)
// Note that this requires linking with -Wl,-rpath,. in order to turn up relative path .so files
libHandle = dlopen(libraryPath, RTLD_LAZY | RTLD_GLOBAL);
const char* error = dlerror();
if (!libHandle || error)
{
Logf("DynamicLoader Error:\n%s\n", error);
return nullptr;
}
#elif WINDOWS
// TODO Clean this up! Only the cakelispBin is necessary I think (need to double check that)
// TODO Clear added dirs after? (RemoveDllDirectory())
const char* absoluteLibPath =
makeAbsolutePath_Allocated(/*fromDirectory=*/nullptr, libraryPath);
char convertedPath[MAX_PATH_LENGTH] = {0};
// TODO Remove, redundant with makeAbsolutePath_Allocated()
makeBackslashFilename(convertedPath, sizeof(convertedPath), absoluteLibPath);
char dllDirectory[MAX_PATH_LENGTH] = {0};
getDirectoryFromPath(convertedPath, dllDirectory, sizeof(dllDirectory));
{
int wchars_num = MultiByteToWideChar(CP_UTF8, 0, dllDirectory, -1, nullptr, 0);
wchar_t* wstrDllDirectory = new wchar_t[wchars_num];
MultiByteToWideChar(CP_UTF8, 0, dllDirectory, -1, wstrDllDirectory, wchars_num);
AddDllDirectory(wstrDllDirectory);
delete[] wstrDllDirectory;
}
// When loading cakelisp.lib, it will actually need to find cakelisp.exe for the symbols
{
const char* cakelispBinDirectory =
makeAbsolutePath_Allocated(/*fromDirectory=*/nullptr, "bin");
int wchars_num = MultiByteToWideChar(CP_UTF8, 0, cakelispBinDirectory, -1, nullptr, 0);
wchar_t* wstrDllDirectory = new wchar_t[wchars_num];
MultiByteToWideChar(CP_UTF8, 0, cakelispBinDirectory, -1, wstrDllDirectory, wchars_num);
AddDllDirectory(wstrDllDirectory);
free((void*)cakelispBinDirectory);
delete[] wstrDllDirectory;
}
libHandle = LoadLibraryEx(convertedPath, nullptr,
LOAD_LIBRARY_SEARCH_USER_DIRS | LOAD_LIBRARY_SEARCH_DEFAULT_DIRS);
if (!libHandle)
{
Logf("DynamicLoader Error: Failed to load %s with code %d\n", convertedPath,
GetLastError());
free((void*)absoluteLibPath);
return nullptr;
}
free((void*)absoluteLibPath);
#endif
dynamicLibraries[libraryPath] = {libHandle};
return libHandle;
}
void* getSymbolFromDynamicLibrary(DynamicLibHandle library, const char* symbolName)
{
if (!library)
{
Log("DynamicLoader Error: Received empty library handle\n");
return nullptr;
}
#if defined(UNIX) || defined(MACOS)
// Clear any existing error before running dlsym
char* error = dlerror();
if (error != nullptr)
{
Logf("DynamicLoader Error:\n%s\n", error);
return nullptr;
}
void* symbol = dlsym(library, symbolName);
error = dlerror();
if (error != nullptr)
{
Logf("DynamicLoader Error:\n%s\n", error);
return nullptr;
}
return symbol;
#elif WINDOWS
void* procedure = (void*)GetProcAddress((HINSTANCE)library, symbolName);
if (!procedure)
{
Logf("DynamicLoader Error:\n%d\n", GetLastError());
return nullptr;
}
return procedure;
#else
return nullptr;
#endif
}
void closeAllDynamicLibraries()
{
for (std::pair<const std::string, DynamicLibrary>& libraryPair : dynamicLibraries)
{
#if defined(UNIX) || defined(MACOS)
dlclose(libraryPair.second.handle);
#elif WINDOWS
FreeLibrary((HMODULE)libraryPair.second.handle);
#endif
}
dynamicLibraries.clear();
}
void closeDynamicLibrary(DynamicLibHandle handleToClose)
{
DynamicLibHandle libHandle = nullptr;
for (DynamicLibraryMap::iterator libraryIt = dynamicLibraries.begin();
libraryIt != dynamicLibraries.end(); ++libraryIt)
{
if (handleToClose == libraryIt->second.handle)
{
libHandle = libraryIt->second.handle;
dynamicLibraries.erase(libraryIt);
break;
}
}
if (!libHandle)
{
Log("warning: closing library which wasn't in the list of loaded libraries\n");
libHandle = handleToClose;
}
#if defined(UNIX) || defined(MACOS)
dlclose(libHandle);
#elif WINDOWS
FreeLibrary((HMODULE)libHandle);
#endif
}