| // Copyright 2011 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/win/scoped_handle.h" |
| |
| #include <windows.h> |
| #include <winternl.h> |
| |
| #include <string> |
| #include <utility> |
| |
| #include "base/command_line.h" |
| #include "base/files/file_path.h" |
| #include "base/scoped_native_library.h" |
| #include "base/test/gmock_expected_support.h" |
| #include "base/test/gtest_util.h" |
| #include "base/test/multiprocess_test.h" |
| #include "base/test/test_timeouts.h" |
| #include "base/types/expected.h" |
| #include "base/win/windows_handle_util.h" |
| #include "build/build_config.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "testing/multiprocess_func_list.h" |
| |
| namespace base { |
| namespace win { |
| |
| namespace { |
| |
| std::string FailureMessage(const std::string& msg) { |
| #if defined(NDEBUG) && defined(OFFICIAL_BUILD) |
| // Official release builds strip all fatal messages for saving binary size, |
| // see base/check.h. |
| return ""; |
| #else |
| return msg; |
| #endif // defined(NDEBUG) && defined(OFFICIAL_BUILD) |
| } |
| |
| } // namespace |
| |
| namespace testing { |
| extern "C" bool __declspec(dllexport) RunTest(); |
| } // namespace testing |
| |
| using ScopedHandleTest = ::testing::Test; |
| using ScopedHandleDeathTest = ::testing::Test; |
| |
| TEST_F(ScopedHandleTest, ScopedHandle) { |
| // Any illegal error code will do. We just need to test that it is preserved |
| // by ScopedHandle to avoid https://crbug.com/528394. |
| const DWORD magic_error = 0x12345678; |
| |
| HANDLE handle = ::CreateMutex(nullptr, false, nullptr); |
| // Call SetLastError after creating the handle. |
| ::SetLastError(magic_error); |
| base::win::ScopedHandle handle_holder(handle); |
| EXPECT_EQ(magic_error, ::GetLastError()); |
| |
| // Create a new handle and then set LastError again. |
| handle = ::CreateMutex(nullptr, false, nullptr); |
| ::SetLastError(magic_error); |
| handle_holder.Set(handle); |
| EXPECT_EQ(magic_error, ::GetLastError()); |
| |
| // Create a new handle and then set LastError again. |
| handle = ::CreateMutex(nullptr, false, nullptr); |
| base::win::ScopedHandle handle_source(handle); |
| ::SetLastError(magic_error); |
| handle_holder = std::move(handle_source); |
| EXPECT_EQ(magic_error, ::GetLastError()); |
| } |
| |
| TEST_F(ScopedHandleTest, InvalidHandles) { |
| base::win::ScopedHandle empty; |
| // Should not get INVALID_HANDLE_VALUE from an empty handle. |
| EXPECT_EQ(empty.get(), nullptr); |
| // Do not allow pseudo handles as scoped handles. |
| base::win::ScopedHandle cur_proc(::GetCurrentProcess()); |
| EXPECT_FALSE(cur_proc.is_valid()); |
| base::win::ScopedHandle cur_thread(::GetCurrentThread()); |
| EXPECT_FALSE(cur_thread.is_valid()); |
| cur_thread.Set(::GetCurrentThread()); |
| EXPECT_FALSE(cur_thread.is_valid()); |
| // Should not get INVALID_HANDLE_VALUE from an invalid handle. |
| EXPECT_EQ(cur_thread.get(), nullptr); |
| // Disallow values that come from uint32_t to ensure casting is working right. |
| base::win::ScopedHandle proc_from_u32(Uint32ToHandle(0xfffffffful)); |
| EXPECT_FALSE(proc_from_u32.is_valid()); |
| proc_from_u32.Set(Uint32ToHandle(0xfffffffful)); |
| EXPECT_FALSE(proc_from_u32.is_valid()); |
| EXPECT_EQ(proc_from_u32.get(), nullptr); |
| base::win::ScopedHandle thread_from_u32(Uint32ToHandle(0xfffffffeul)); |
| EXPECT_FALSE(thread_from_u32.is_valid()); |
| } |
| |
| TEST_F(ScopedHandleDeathTest, HandleVerifierTrackedHasBeenClosed) { |
| HANDLE handle = ::CreateMutex(nullptr, false, nullptr); |
| ASSERT_NE(HANDLE(nullptr), handle); |
| |
| ASSERT_DEATH( |
| { |
| base::win::ScopedHandle handle_holder(handle); |
| ::NtClose(handle); |
| // Destructing a ScopedHandle with an illegally closed handle should |
| // fail. |
| }, |
| FailureMessage("CloseHandle failed")); |
| } |
| |
| TEST_F(ScopedHandleDeathTest, HandleVerifierCloseTrackedHandle) { |
| ASSERT_DEATH( |
| { |
| HANDLE handle = ::CreateMutex(nullptr, false, nullptr); |
| ASSERT_NE(HANDLE(nullptr), handle); |
| |
| // Start tracking the handle so that closes outside of the checker are |
| // caught. |
| base::win::CheckedScopedHandle handle_holder(handle); |
| |
| // Closing a tracked handle using ::CloseHandle should crash due to hook |
| // noticing the illegal close. |
| ::CloseHandle(handle); |
| }, |
| // This test must match the CloseHandleHook causing this failure, because |
| // if the hook doesn't crash and instead the handle is double closed by |
| // the `handle_holder` going out of scope, then there is still a crash, |
| // but a different crash and one we are not explicitly testing here. This |
| // other crash is tested in HandleVerifierTrackedHasBeenClosed above. |
| FailureMessage("CloseHandleHook validation failure")); |
| } |
| |
| TEST_F(ScopedHandleDeathTest, HandleVerifierDoubleTracking) { |
| HANDLE handle = ::CreateMutex(nullptr, false, nullptr); |
| ASSERT_NE(HANDLE(nullptr), handle); |
| |
| base::win::CheckedScopedHandle handle_holder(handle); |
| |
| ASSERT_DEATH( |
| { base::win::CheckedScopedHandle handle_holder2(handle); }, |
| FailureMessage("Handle Already Tracked")); |
| } |
| |
| TEST_F(ScopedHandleDeathTest, HandleVerifierWrongOwner) { |
| HANDLE handle = ::CreateMutex(nullptr, false, nullptr); |
| ASSERT_NE(HANDLE(nullptr), handle); |
| |
| base::win::CheckedScopedHandle handle_holder(handle); |
| ASSERT_DEATH( |
| { |
| base::win::CheckedScopedHandle handle_holder2; |
| handle_holder2.handle_ = handle; |
| }, |
| FailureMessage("Closing a handle owned by something else")); |
| ASSERT_TRUE(handle_holder.is_valid()); |
| handle_holder.Close(); |
| } |
| |
| TEST_F(ScopedHandleDeathTest, HandleVerifierUntrackedHandle) { |
| HANDLE handle = ::CreateMutex(nullptr, false, nullptr); |
| ASSERT_NE(HANDLE(nullptr), handle); |
| |
| ASSERT_DEATH( |
| { |
| base::win::CheckedScopedHandle handle_holder; |
| handle_holder.handle_ = handle; |
| }, |
| FailureMessage("Closing an untracked handle")); |
| |
| ASSERT_TRUE(::CloseHandle(handle)); |
| } |
| |
| TEST(TakeHandleOfTypeTest, NullHandle) { |
| auto handle_or_error = TakeHandleOfType(kNullProcessHandle, L"Process"); |
| ASSERT_FALSE(handle_or_error.has_value()); |
| ASSERT_EQ(handle_or_error.error(), STATUS_INVALID_HANDLE); |
| } |
| |
| TEST(TakeHandleOfTypeTest, InvalidHandle) { |
| auto handle_or_error = TakeHandleOfType(INVALID_HANDLE_VALUE, L"Process"); |
| ASSERT_FALSE(handle_or_error.has_value()); |
| ASSERT_EQ(handle_or_error.error(), STATUS_INVALID_HANDLE); |
| } |
| |
| TEST(TakeHandleOfTypeTest, CurrentProcess) { |
| auto handle_or_error = TakeHandleOfType(::GetCurrentProcess(), L"Process"); |
| ASSERT_FALSE(handle_or_error.has_value()); |
| ASSERT_EQ(handle_or_error.error(), STATUS_INVALID_HANDLE); |
| } |
| |
| TEST(TakeHandleOfTypeDeathTest, CrazyHandle) { |
| EXPECT_DEATH_IF_SUPPORTED( |
| std::ignore = TakeHandleOfType(Uint32ToHandle(0x12345678U), L"Process"), |
| "Received fatal exception 0xc0000008"); |
| } |
| |
| TEST(TakeHandleOfTypeTest, ValidTypeMatch) { |
| Process this_process = Process::Open(GetCurrentProcId()); |
| HANDLE process_handle = this_process.Handle(); |
| ASSERT_OK_AND_ASSIGN(ScopedHandle process, |
| TakeHandleOfType(this_process.Release(), L"Process")); |
| ASSERT_TRUE(process.is_valid()); |
| ASSERT_EQ(process.get(), process_handle); |
| } |
| |
| TEST(TakeHandleOfTypeDeathTest, ValidTypeMismatch) { |
| EXPECT_CHECK_DEATH((void)TakeHandleOfType( |
| Process::Open(GetCurrentProcId()).Release(), L"Section")); |
| } |
| |
| // Under ASan, the multi-process test crashes during process shutdown for |
| // unknown reasons. Disable it for now. http://crbug.com/685262 |
| #if defined(ADDRESS_SANITIZER) |
| #define MAYBE_MultiProcess DISABLED_MultiProcess |
| #else |
| #define MAYBE_MultiProcess MultiProcess |
| #endif |
| |
| TEST_F(ScopedHandleTest, MAYBE_MultiProcess) { |
| CommandLine command_line(base::GetMultiProcessTestChildBaseCommandLine()); |
| |
| base::Process test_child_process = base::SpawnMultiProcessTestChild( |
| "HandleVerifierChildProcess", command_line, LaunchOptions()); |
| |
| int rv = -1; |
| ASSERT_TRUE(test_child_process.WaitForExitWithTimeout( |
| TestTimeouts::action_timeout(), &rv)); |
| EXPECT_EQ(0, rv); |
| } |
| |
| MULTIPROCESS_TEST_MAIN(HandleVerifierChildProcess) { |
| ScopedNativeLibrary module( |
| FilePath(FILE_PATH_LITERAL("scoped_handle_test_dll.dll"))); |
| |
| if (!module.is_valid()) { |
| return 1; |
| } |
| auto run_test_function = reinterpret_cast<decltype(&testing::RunTest)>( |
| module.GetFunctionPointer("RunTest")); |
| if (!run_test_function) { |
| return 1; |
| } |
| if (!run_test_function()) { |
| return 1; |
| } |
| |
| return 0; |
| } |
| |
| } // namespace win |
| } // namespace base |