-
-
Notifications
You must be signed in to change notification settings - Fork 5.5k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Windows: add wrapper
.exe
to work around lack of RPATH
Windows has no concept of `RPATH`, which makes shipping binaries with relatively-located dependencies quite challenging. Despite investigating many potential avenues of `RPATH` emulation, (including but not limited to Application Manifests, PE file patching, and bundling the true Julia executable as a resource inside of a launcher) the most reliable (And least breaking for external workflows) was found to be to create a launcher executable that invokes the true Julia executable from within the `libexec` directory.
- Loading branch information
1 parent
5d32089
commit 26dc2c9
Showing
4 changed files
with
120 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,81 @@ | ||
#include <windows.h> | ||
#include <tchar.h> | ||
#include <stdio.h> | ||
#include <shlwapi.h> | ||
|
||
/* The maximum path length we'll allow */ | ||
#define MAX_PATH_LEN 32768 | ||
|
||
/* PATH_ENTRIES is our simulated RPATH, usually of the form 'TEXT("../path1"), TEXT("../path2"), ...' */ | ||
#ifndef PATH_ENTRIES | ||
#define PATH_ENTRIES TEXT("") | ||
#endif | ||
|
||
LPWSTR pathEntries[] = { | ||
PATH_ENTRIES | ||
}; | ||
|
||
/* JULIA_EXE_PATH is the relative path to julia.exe */ | ||
#ifndef JULIA_EXE_PATH | ||
#define JULIA_EXE_PATH "../libexec/julia.exe" | ||
#endif | ||
|
||
int wmain(int argc, wchar_t *argv[], wchar_t *envp[]) { | ||
// Determine absolute path to true julia.exe sitting in `libexec/` | ||
WCHAR currFileDir[MAX_PATH_LEN]; | ||
WCHAR juliaPath[MAX_PATH_LEN]; | ||
if (!GetModuleFileName(NULL, currFileDir, MAX_PATH_LEN)) { | ||
fprintf(stderr, "ERROR: GetModuleFileName() failed with code %d\n", GetLastError()); | ||
exit(1); | ||
} | ||
PathRemoveFileSpec(currFileDir); | ||
PathCombine(juliaPath, currFileDir, TEXT(JULIA_EXE_PATH)); | ||
|
||
// On windows, we simulate RPATH by pushing onto PATH | ||
LPWSTR pathVal = (LPWSTR) malloc(MAX_PATH_LEN*sizeof(WCHAR)); | ||
DWORD dwRet = GetEnvironmentVariable(TEXT("PATH"), pathVal, MAX_PATH_LEN); | ||
DWORD numPathEntries = sizeof(pathEntries)/sizeof(pathEntries[0]); | ||
if (dwRet == 0) { | ||
// If we cannot get PATH, then our job is easy! | ||
pathVal[0] = '\0'; | ||
} | ||
else { | ||
// Otherwise, we append, if we have enough space to: | ||
DWORD currFileDirLen = wcslen(currFileDir); | ||
DWORD totalPathLen = dwRet + 1 + currFileDirLen; | ||
int env_idx; | ||
for (env_idx = 0; env_idx < numPathEntries; ++env_idx) { | ||
totalPathLen += 1 + currFileDirLen + 1 + wcslen(pathEntries[env_idx]); | ||
} | ||
if (MAX_PATH_LEN < totalPathLen) { | ||
fprintf(stderr, "ERROR: Cannot append entries to PATH: not enough space in environment block. Reduce size of PATH!"); | ||
exit(1); | ||
} | ||
lstrcat(pathVal, TEXT(";")); | ||
} | ||
// We always add the current directory (e.g. `bin/`) to PATH so that we can find e.g. libjulia.dll | ||
lstrcat(pathVal, currFileDir); | ||
|
||
// For each entry in PATH_ENTRIES, tack it on to the end, relative to the current directory: | ||
int env_idx; | ||
for (env_idx = 0; env_idx < numPathEntries; ++env_idx) { | ||
lstrcat(pathVal, TEXT(";")); | ||
lstrcat(pathVal, currFileDir); | ||
lstrcat(pathVal, TEXT("\\")); | ||
lstrcat(pathVal, pathEntries[env_idx]); | ||
} | ||
SetEnvironmentVariable(TEXT("PATH"), pathVal); | ||
free(pathVal); | ||
|
||
STARTUPINFO info; | ||
PROCESS_INFORMATION processInfo; | ||
DWORD exit_code = 1; | ||
GetStartupInfo(&info); | ||
if (CreateProcess(juliaPath, GetCommandLine(), NULL, NULL, TRUE, 0, NULL, NULL, &info, &processInfo)) { | ||
WaitForSingleObject(processInfo.hProcess, INFINITE); | ||
GetExitCodeProcess(processInfo.hProcess, &exit_code); | ||
CloseHandle(processInfo.hProcess); | ||
CloseHandle(processInfo.hThread); | ||
} | ||
return exit_code; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters