/* * 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 #include #include "config.h" #include "libavutil/macros.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 #endif #endif /* COMPAT_W32DLFCN_H */