diff options
author | Junio C Hamano <gitster@pobox.com> | 2017-09-28 14:47:57 +0900 |
---|---|---|
committer | Junio C Hamano <gitster@pobox.com> | 2017-09-28 14:47:57 +0900 |
commit | 3b6e73a3b18336cf123993b191e3fab6442bd5a3 (patch) | |
tree | 4e796e2257d10df5e5551d2d8c170976934d238b | |
parent | 4da3e234f5a3989328c4dc8368cf5906f8679b30 (diff) | |
parent | db2f7c48cb30f8fdcb2933aa7d9890b34e0c4e94 (diff) | |
download | git-3b6e73a3b18336cf123993b191e3fab6442bd5a3.tar.gz git-3b6e73a3b18336cf123993b191e3fab6442bd5a3.tar.xz |
Merge branch 'js/win32-lazyload-dll'
Add a helper in anticipation for its need in a future topic RSN.
* js/win32-lazyload-dll:
Win32: simplify loading of DLL functions
-rw-r--r-- | compat/win32/lazyload.h | 57 |
1 files changed, 57 insertions, 0 deletions
diff --git a/compat/win32/lazyload.h b/compat/win32/lazyload.h new file mode 100644 index 000000000..9e631c859 --- /dev/null +++ b/compat/win32/lazyload.h @@ -0,0 +1,57 @@ +#ifndef LAZYLOAD_H +#define LAZYLOAD_H + +/* + * A pair of macros to simplify loading of DLL functions. Example: + * + * DECLARE_PROC_ADDR(kernel32.dll, BOOL, CreateHardLinkW, + * LPCWSTR, LPCWSTR, LPSECURITY_ATTRIBUTES); + * + * if (!INIT_PROC_ADDR(CreateHardLinkW)) + * return error("Could not find CreateHardLinkW() function"; + * + * if (!CreateHardLinkW(source, target, NULL)) + * return error("could not create hardlink from %S to %S", + * source, target); + */ + +struct proc_addr { + const char *const dll; + const char *const function; + FARPROC pfunction; + unsigned initialized : 1; +}; + +/* Declares a function to be loaded dynamically from a DLL. */ +#define DECLARE_PROC_ADDR(dll, rettype, function, ...) \ + static struct proc_addr proc_addr_##function = \ + { #dll, #function, NULL, 0 }; \ + static rettype (WINAPI *function)(__VA_ARGS__) + +/* + * Loads a function from a DLL (once-only). + * Returns non-NULL function pointer on success. + * Returns NULL + errno == ENOSYS on failure. + * This function is not thread-safe. + */ +#define INIT_PROC_ADDR(function) \ + (function = get_proc_addr(&proc_addr_##function)) + +static inline void *get_proc_addr(struct proc_addr *proc) +{ + /* only do this once */ + if (!proc->initialized) { + HANDLE hnd; + proc->initialized = 1; + hnd = LoadLibraryExA(proc->dll, NULL, + LOAD_LIBRARY_SEARCH_SYSTEM32); + if (hnd) + proc->pfunction = GetProcAddress(hnd, proc->function); + } + /* set ENOSYS if DLL or function was not found */ + if (!proc->pfunction) + errno = ENOSYS; + return proc->pfunction; +} + +#endif |