|
| 1 | +// Copyright (c) Microsoft Corporation. All rights reserved. |
| 2 | +// Licensed under the MIT License. |
| 3 | + |
| 4 | +// == workaround for delay loading of dependencies of onnxruntime.dll == |
| 5 | +// |
| 6 | +// Problem: |
| 7 | +// |
| 8 | +// When onnxruntime.dll uses delay loading for its dependencies, the dependencies are loaded using LoadLibraryEx, |
| 9 | +// which search the directory of process (.exe) instead of this library (onnxruntime.dll). This is a problem for |
| 10 | +// usages of Node.js binding and python binding, because Windows will try to find the dependencies in the directory |
| 11 | +// of node.exe or python.exe, which is not the directory of onnxruntime.dll. |
| 12 | +// |
| 13 | +// Solution: |
| 14 | +// |
| 15 | +// By using the delay load hook `__pfnDliNotifyHook2`, we can intervene the loading procedure by loading from an |
| 16 | +// absolute path. The absolute path is constructed by appending the name of the DLL to load to the directory of |
| 17 | +// onnxruntime.dll. This way, we can ensure that the dependencies are loaded from the same directory as onnxruntime.dll. |
| 18 | +// |
| 19 | +// See also: |
| 20 | +// - https://learn.microsoft.com/en-us/cpp/build/reference/understanding-the-helper-function?view=msvc-170#structure-and-constant-definitions |
| 21 | +// - https://learn.microsoft.com/en-us/windows/win32/dlls/dynamic-link-library-search-order#alternate-search-order-for-unpackaged-apps |
| 22 | +// |
| 23 | +// The DLL DelayLoad hook is only enabled when the compiler is MSVC and at least one of the following is True: |
| 24 | +// - both USE_WEBGPU and BUILD_DAWN_MONOLITHIC_LIBRARY are defined |
| 25 | +// - USE_DML is defined |
| 26 | +// |
| 27 | +#define ORT_DELAY_LOAD_WEBGPU_DAWN_DLL (defined(USE_WEBGPU) && defined(BUILD_DAWN_MONOLITHIC_LIBRARY)) |
| 28 | +#define ORT_DELAY_LOAD_DIRECTML_DLL defined(USE_DML) |
| 29 | +#if defined(_MSC_VER) && (ORT_DELAY_LOAD_WEBGPU_DAWN_DLL || ORT_DELAY_LOAD_DIRECTML_DLL) |
| 30 | + |
| 31 | +#include <Windows.h> |
| 32 | +#include <delayimp.h> |
| 33 | +#include <stdlib.h> |
| 34 | +#include <string> |
| 35 | + |
| 36 | +#include "core/platform/env.h" |
| 37 | + |
| 38 | +namespace { |
| 39 | + |
| 40 | +#define DEFINE_KNOWN_DLL(name) {#name ".dll", L#name L".dll"} |
| 41 | + |
| 42 | +constexpr struct { |
| 43 | + const char* str; |
| 44 | + const wchar_t* wstr; |
| 45 | +} known_dlls[] = { |
| 46 | +#if ORT_DELAY_LOAD_WEBGPU_DAWN_DLL |
| 47 | + DEFINE_KNOWN_DLL(webgpu_dawn), |
| 48 | +#endif |
| 49 | +#if ORT_DELAY_LOAD_DIRECTML_DLL |
| 50 | + DEFINE_KNOWN_DLL(DirectML), |
| 51 | +#endif |
| 52 | +}; |
| 53 | +} // namespace |
| 54 | + |
| 55 | +FARPROC WINAPI delay_load_hook(unsigned dliNotify, PDelayLoadInfo pdli) { |
| 56 | + if (dliNotify == dliNotePreLoadLibrary) { |
| 57 | + for (size_t i = 0; i < _countof(known_dlls); ++i) { |
| 58 | + if (_stricmp(pdli->szDll, known_dlls[i].str) == 0) { |
| 59 | + // Try to load the DLL from the same directory as onnxruntime.dll |
| 60 | + |
| 61 | + // First, get the path to onnxruntime.dll |
| 62 | + auto path = Env::Default().GetRuntimePath(); |
| 63 | + if (path.empty()) { |
| 64 | + // Failed to get the path to onnxruntime.dll. In this case, we will just return NULL and let the system |
| 65 | + // search for the DLL in the default search order. |
| 66 | + return NULL; |
| 67 | + } |
| 68 | + |
| 69 | + // Append the name of the DLL. Now `path` is the absolute path to the DLL to load. |
| 70 | + path.append(known_dlls[i].wstr); |
| 71 | + |
| 72 | + // Load the DLL |
| 73 | + return FARPROC(LoadLibraryExW(path.c_str(), NULL, |
| 74 | + LOAD_LIBRARY_SEARCH_DEFAULT_DIRS | LOAD_LIBRARY_SEARCH_DLL_LOAD_DIR)); |
| 75 | + } |
| 76 | + } |
| 77 | + } |
| 78 | + return NULL; |
| 79 | +} |
| 80 | + |
| 81 | +extern "C" const PfnDliHook __pfnDliNotifyHook2 = delay_load_hook; |
| 82 | + |
| 83 | +#endif |
0 commit comments