Telegram-Android/TMessagesProj/jni/voip/webrtc/base/native_library_win.cc
2020-09-30 16:48:47 +03:00

258 lines
9.1 KiB
C++

// Copyright (c) 2011 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/native_library.h"
#include <windows.h>
#include "base/files/file_util.h"
#include "base/metrics/histogram_macros.h"
#include "base/path_service.h"
#include "base/scoped_native_library.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "base/threading/scoped_blocking_call.h"
namespace base {
namespace {
// forward declare
HMODULE AddDllDirectory(PCWSTR new_directory);
// This enum is used to back an UMA histogram, and should therefore be treated
// as append-only.
enum LoadLibraryResult {
// LoadLibraryExW API/flags are available and the call succeeds.
SUCCEED = 0,
// LoadLibraryExW API/flags are availabe to use but the call fails, then
// LoadLibraryW is used and succeeds.
FAIL_AND_SUCCEED,
// LoadLibraryExW API/flags are availabe to use but the call fails, then
// LoadLibraryW is used but fails as well.
FAIL_AND_FAIL,
// LoadLibraryExW API/flags are unavailabe to use, then LoadLibraryW is used
// and succeeds.
UNAVAILABLE_AND_SUCCEED,
// LoadLibraryExW API/flags are unavailabe to use, then LoadLibraryW is used
// but fails.
UNAVAILABLE_AND_FAIL,
// Add new items before this one, always keep this one at the end.
END
};
// A helper method to log library loading result to UMA.
void LogLibrarayLoadResultToUMA(LoadLibraryResult result) {
UMA_HISTOGRAM_ENUMERATION("LibraryLoader.LoadNativeLibraryWindows", result,
LoadLibraryResult::END);
}
// A helper method to check if AddDllDirectory method is available, thus
// LOAD_LIBRARY_SEARCH_* flags are available on systems.
bool AreSearchFlagsAvailable() {
// The LOAD_LIBRARY_SEARCH_* flags are available on systems that have
// KB2533623 installed. To determine whether the flags are available, use
// GetProcAddress to get the address of the AddDllDirectory,
// RemoveDllDirectory, or SetDefaultDllDirectories function. If GetProcAddress
// succeeds, the LOAD_LIBRARY_SEARCH_* flags can be used with LoadLibraryEx.
// https://msdn.microsoft.com/en-us/library/windows/desktop/ms684179(v=vs.85).aspx
// The LOAD_LIBRARY_SEARCH_* flags are used in the LoadNativeLibraryHelper
// method.
static const auto add_dll_dir_func =
reinterpret_cast<decltype(AddDllDirectory)*>(
GetProcAddress(GetModuleHandle(L"kernel32.dll"), "AddDllDirectory"));
return !!add_dll_dir_func;
}
// A helper method to encode the library loading result to enum
// LoadLibraryResult.
LoadLibraryResult GetLoadLibraryResult(bool are_search_flags_available,
bool has_load_library_succeeded) {
LoadLibraryResult result;
if (are_search_flags_available) {
if (has_load_library_succeeded)
result = LoadLibraryResult::FAIL_AND_SUCCEED;
else
result = LoadLibraryResult::FAIL_AND_FAIL;
} else if (has_load_library_succeeded) {
result = LoadLibraryResult::UNAVAILABLE_AND_SUCCEED;
} else {
result = LoadLibraryResult::UNAVAILABLE_AND_FAIL;
}
return result;
}
NativeLibrary LoadNativeLibraryHelper(const FilePath& library_path,
NativeLibraryLoadError* error) {
// LoadLibrary() opens the file off disk and acquires the LoaderLock, hence
// must not be called from DllMain.
ScopedBlockingCall scoped_blocking_call(FROM_HERE, BlockingType::MAY_BLOCK);
HMODULE module = nullptr;
// This variable records the library loading result.
LoadLibraryResult load_library_result = LoadLibraryResult::SUCCEED;
bool are_search_flags_available = AreSearchFlagsAvailable();
if (are_search_flags_available) {
// LOAD_LIBRARY_SEARCH_DLL_LOAD_DIR flag is needed to search the library
// directory as the library may have dependencies on DLLs in this
// directory.
module = ::LoadLibraryExW(
library_path.value().c_str(), nullptr,
LOAD_LIBRARY_SEARCH_DLL_LOAD_DIR | LOAD_LIBRARY_SEARCH_DEFAULT_DIRS);
// If LoadLibraryExW succeeds, log this metric and return.
if (module) {
LogLibrarayLoadResultToUMA(load_library_result);
return module;
}
// GetLastError() needs to be called immediately after
// LoadLibraryExW call.
if (error)
error->code = ::GetLastError();
}
// If LoadLibraryExW API/flags are unavailable or API call fails, try
// LoadLibraryW API. From UMA, this fallback is necessary for many users.
// Switch the current directory to the library directory as the library
// may have dependencies on DLLs in this directory.
bool restore_directory = false;
FilePath current_directory;
if (GetCurrentDirectory(&current_directory)) {
FilePath plugin_path = library_path.DirName();
if (!plugin_path.empty()) {
SetCurrentDirectory(plugin_path);
restore_directory = true;
}
}
module = ::LoadLibraryW(library_path.value().c_str());
// GetLastError() needs to be called immediately after LoadLibraryW call.
if (!module && error)
error->code = ::GetLastError();
if (restore_directory)
SetCurrentDirectory(current_directory);
// Get the library loading result and log it to UMA.
LogLibrarayLoadResultToUMA(
GetLoadLibraryResult(are_search_flags_available, !!module));
return module;
}
NativeLibrary LoadSystemLibraryHelper(const FilePath& library_path,
NativeLibraryLoadError* error) {
// GetModuleHandleEx and subsequently LoadLibraryEx acquire the LoaderLock,
// hence must not be called from Dllmain.
ScopedBlockingCall scoped_blocking_call(FROM_HERE, BlockingType::MAY_BLOCK);
NativeLibrary module;
BOOL module_found =
::GetModuleHandleExW(0, library_path.value().c_str(), &module);
if (!module_found) {
bool are_search_flags_available = AreSearchFlagsAvailable();
// Prefer LOAD_LIBRARY_SEARCH_SYSTEM32 to avoid DLL preloading attacks.
DWORD flags = are_search_flags_available ? LOAD_LIBRARY_SEARCH_SYSTEM32
: LOAD_WITH_ALTERED_SEARCH_PATH;
module = ::LoadLibraryExW(library_path.value().c_str(), nullptr, flags);
if (!module && error)
error->code = ::GetLastError();
LogLibrarayLoadResultToUMA(
GetLoadLibraryResult(are_search_flags_available, !!module));
}
return module;
}
FilePath GetSystemLibraryName(FilePath::StringPieceType name) {
FilePath library_path;
// Use an absolute path to load the DLL to avoid DLL preloading attacks.
if (PathService::Get(DIR_SYSTEM, &library_path))
library_path = library_path.Append(name);
return library_path;
}
} // namespace
std::string NativeLibraryLoadError::ToString() const {
return StringPrintf("%lu", code);
}
NativeLibrary LoadNativeLibraryWithOptions(const FilePath& library_path,
const NativeLibraryOptions& options,
NativeLibraryLoadError* error) {
return LoadNativeLibraryHelper(library_path, error);
}
void UnloadNativeLibrary(NativeLibrary library) {
FreeLibrary(library);
}
void* GetFunctionPointerFromNativeLibrary(NativeLibrary library,
StringPiece name) {
return reinterpret_cast<void*>(GetProcAddress(library, name.data()));
}
std::string GetNativeLibraryName(StringPiece name) {
DCHECK(IsStringASCII(name));
return name.as_string() + ".dll";
}
std::string GetLoadableModuleName(StringPiece name) {
return GetNativeLibraryName(name);
}
NativeLibrary LoadSystemLibrary(FilePath::StringPieceType name,
NativeLibraryLoadError* error) {
FilePath library_path = GetSystemLibraryName(name);
if (library_path.empty()) {
if (error)
error->code = ERROR_NOT_FOUND;
return nullptr;
}
return LoadSystemLibraryHelper(library_path, error);
}
NativeLibrary PinSystemLibrary(FilePath::StringPieceType name,
NativeLibraryLoadError* error) {
FilePath library_path = GetSystemLibraryName(name);
if (library_path.empty()) {
if (error)
error->code = ERROR_NOT_FOUND;
return nullptr;
}
// GetModuleHandleEx acquires the LoaderLock, hence must not be called from
// Dllmain.
ScopedBlockingCall scoped_blocking_call(FROM_HERE, BlockingType::MAY_BLOCK);
ScopedNativeLibrary module;
if (::GetModuleHandleExW(GET_MODULE_HANDLE_EX_FLAG_PIN,
library_path.value().c_str(),
ScopedNativeLibrary::Receiver(module).get())) {
return module.release();
}
// Load and pin the library since it wasn't already loaded.
module = ScopedNativeLibrary(LoadSystemLibraryHelper(library_path, error));
if (!module.is_valid())
return nullptr;
ScopedNativeLibrary temp;
if (::GetModuleHandleExW(GET_MODULE_HANDLE_EX_FLAG_PIN,
library_path.value().c_str(),
ScopedNativeLibrary::Receiver(temp).get())) {
return module.release();
}
if (error)
error->code = ::GetLastError();
// Return nullptr since we failed to pin the module.
return nullptr;
}
} // namespace base