| /* |
| * This file is part of FFmpeg. |
| * |
| * FFmpeg is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * FFmpeg is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with FFmpeg; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| */ |
| |
| #ifndef COMPAT_W32DLFCN_H |
| #define COMPAT_W32DLFCN_H |
| |
| #ifdef _WIN32 |
| #include <stdint.h> |
| |
| #include <windows.h> |
| |
| #include "config.h" |
| #include "libavutil/macros.h" |
| #include "libavutil/mem.h" |
| #include "libavutil/wchar_filename.h" |
| |
| static inline wchar_t *get_module_filename(HMODULE module) |
| { |
| wchar_t *path = NULL, *new_path; |
| DWORD path_size = 0, path_len; |
| |
| do { |
| path_size = path_size ? FFMIN(2 * path_size, INT16_MAX + 1) : MAX_PATH; |
| new_path = av_realloc_array(path, path_size, sizeof *path); |
| if (!new_path) { |
| av_free(path); |
| return NULL; |
| } |
| path = new_path; |
| // Returns path_size in case of insufficient buffer. |
| // Whether the error is set or not and whether the output |
| // is null-terminated or not depends on the version of Windows. |
| path_len = GetModuleFileNameW(module, path, path_size); |
| } while (path_len && path_size <= INT16_MAX && path_size <= path_len); |
| |
| if (!path_len) { |
| av_free(path); |
| return NULL; |
| } |
| return path; |
| } |
| |
| /** |
| * Safe function used to open dynamic libs. This attempts to improve program security |
| * by removing the current directory from the dll search path. Only dll's found in the |
| * executable or system directory are allowed to be loaded. |
| * @param name The dynamic lib name. |
| * @return A handle to the opened lib. |
| */ |
| static inline HMODULE win32_dlopen(const char *name) |
| { |
| wchar_t *name_w; |
| HMODULE module = NULL; |
| if (utf8towchar(name, &name_w)) |
| name_w = NULL; |
| #if _WIN32_WINNT < 0x0602 |
| // On Win7 and earlier we check if KB2533623 is available |
| if (!GetProcAddress(GetModuleHandleW(L"kernel32.dll"), "SetDefaultDllDirectories")) { |
| wchar_t *path = NULL, *new_path; |
| DWORD pathlen, pathsize, namelen; |
| if (!name_w) |
| goto exit; |
| namelen = wcslen(name_w); |
| // Try local directory first |
| path = get_module_filename(NULL); |
| if (!path) |
| goto exit; |
| new_path = wcsrchr(path, '\\'); |
| if (!new_path) |
| goto exit; |
| pathlen = new_path - path; |
| pathsize = pathlen + namelen + 2; |
| new_path = av_realloc_array(path, pathsize, sizeof *path); |
| if (!new_path) |
| goto exit; |
| path = new_path; |
| wcscpy(path + pathlen + 1, name_w); |
| module = LoadLibraryExW(path, NULL, LOAD_WITH_ALTERED_SEARCH_PATH); |
| if (module == NULL) { |
| // Next try System32 directory |
| pathlen = GetSystemDirectoryW(path, pathsize); |
| if (!pathlen) |
| goto exit; |
| // Buffer is not enough in two cases: |
| // 1. system directory + \ + module name |
| // 2. system directory even without the module name. |
| if (pathlen + namelen + 2 > pathsize) { |
| pathsize = pathlen + namelen + 2; |
| new_path = av_realloc_array(path, pathsize, sizeof *path); |
| if (!new_path) |
| goto exit; |
| path = new_path; |
| // Query again to handle the case #2. |
| pathlen = GetSystemDirectoryW(path, pathsize); |
| if (!pathlen) |
| goto exit; |
| } |
| path[pathlen] = L'\\'; |
| wcscpy(path + pathlen + 1, name_w); |
| module = LoadLibraryExW(path, NULL, LOAD_WITH_ALTERED_SEARCH_PATH); |
| } |
| exit: |
| av_free(path); |
| av_free(name_w); |
| return module; |
| } |
| #endif |
| #ifndef LOAD_LIBRARY_SEARCH_APPLICATION_DIR |
| # define LOAD_LIBRARY_SEARCH_APPLICATION_DIR 0x00000200 |
| #endif |
| #ifndef LOAD_LIBRARY_SEARCH_SYSTEM32 |
| # define LOAD_LIBRARY_SEARCH_SYSTEM32 0x00000800 |
| #endif |
| #if HAVE_WINRT |
| if (!name_w) |
| return NULL; |
| module = LoadPackagedLibrary(name_w, 0); |
| #else |
| #define LOAD_FLAGS (LOAD_LIBRARY_SEARCH_APPLICATION_DIR | LOAD_LIBRARY_SEARCH_SYSTEM32) |
| /* filename may be be in CP_ACP */ |
| if (!name_w) |
| return LoadLibraryExA(name, NULL, LOAD_FLAGS); |
| module = LoadLibraryExW(name_w, NULL, LOAD_FLAGS); |
| #undef LOAD_FLAGS |
| #endif |
| av_free(name_w); |
| return module; |
| } |
| #define dlopen(name, flags) win32_dlopen(name) |
| #define dlclose FreeLibrary |
| #define dlsym GetProcAddress |
| #else |
| #include <dlfcn.h> |
| #endif |
| |
| #endif /* COMPAT_W32DLFCN_H */ |