| // Copyright 2016 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 <windows.h> | 
 |  | 
 | #include <vector> | 
 |  | 
 | #include "base/win/base_win_buildflags.h" | 
 | #include "base/win/current_module.h" | 
 | #include "base/win/scoped_handle.h" | 
 | #include "base/win/scoped_handle_verifier.h" | 
 |  | 
 | namespace base { | 
 | namespace win { | 
 | namespace testing { | 
 |  | 
 | extern "C" bool __declspec(dllexport) RunTest(); | 
 |  | 
 | namespace { | 
 |  | 
 | struct ThreadParams { | 
 |   HANDLE ready_event; | 
 |   HANDLE start_event; | 
 | }; | 
 |  | 
 | // Note, this must use all native functions to avoid instantiating the | 
 | // ActiveVerifier. e.g. can't use base::Thread or even base::PlatformThread. | 
 | DWORD __stdcall ThreadFunc(void* params) { | 
 |   ThreadParams* thread_params = reinterpret_cast<ThreadParams*>(params); | 
 |   HANDLE handle = ::CreateMutex(nullptr, false, nullptr); | 
 |  | 
 |   ::SetEvent(thread_params->ready_event); | 
 |   ::WaitForSingleObject(thread_params->start_event, INFINITE); | 
 |   ScopedHandle handle_holder(handle); | 
 |   return 0; | 
 | } | 
 |  | 
 | bool InternalRunThreadTest() { | 
 |   std::vector<HANDLE> threads_; | 
 |   // From manual testing, the bug fixed by crrev.com/678736a starts reliably | 
 |   // causing handle verifier asserts to trigger at around 100 threads, so make | 
 |   // it 200 to be sure to detect any future regressions. | 
 |   const size_t kNumThreads = 200; | 
 |  | 
 |   // bManualReset is set to true to allow signalling multiple threads. | 
 |   HANDLE start_event = ::CreateEvent(nullptr, true, false, nullptr); | 
 |   if (!start_event) | 
 |     return false; | 
 |  | 
 |   HANDLE ready_event = CreateEvent(nullptr, false, false, nullptr); | 
 |   if (!ready_event) | 
 |     return false; | 
 |  | 
 |   ThreadParams thread_params = { ready_event, start_event }; | 
 |  | 
 |   for (size_t i = 0; i < kNumThreads; i++) { | 
 |     HANDLE thread_handle = | 
 |         ::CreateThread(nullptr, 0, ThreadFunc, | 
 |                        reinterpret_cast<void*>(&thread_params), 0, nullptr); | 
 |     if (!thread_handle) | 
 |       break; | 
 |     ::WaitForSingleObject(ready_event, INFINITE); | 
 |     threads_.push_back(thread_handle); | 
 |   } | 
 |  | 
 |   ::CloseHandle(ready_event); | 
 |  | 
 |   if (threads_.size() != kNumThreads) { | 
 |     for (auto* thread : threads_) | 
 |       ::CloseHandle(thread); | 
 |     ::CloseHandle(start_event); | 
 |     return false; | 
 |   } | 
 |  | 
 |   ::SetEvent(start_event); | 
 |   ::CloseHandle(start_event); | 
 |   for (auto* thread : threads_) { | 
 |     ::WaitForSingleObject(thread, INFINITE); | 
 |     ::CloseHandle(thread); | 
 |   } | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | bool InternalRunLocationTest() { | 
 |   // Create a new handle and then set LastError again. | 
 |   HANDLE handle = ::CreateMutex(nullptr, false, nullptr); | 
 |   if (!handle) | 
 |     return false; | 
 |   ScopedHandle handle_holder(handle); | 
 |  | 
 |   HMODULE verifier_module = | 
 |       base::win::internal::GetHandleVerifierModuleForTesting(); | 
 |   if (!verifier_module) | 
 |     return false; | 
 |  | 
 |   // Get my module | 
 |   HMODULE my_module = CURRENT_MODULE(); | 
 |   if (!my_module) | 
 |     return false; | 
 |  | 
 |   HMODULE main_module = ::GetModuleHandle(NULL); | 
 |  | 
 |   // In a non-component build, ActiveVerifier should always be created in the | 
 |   // version of base linked with the main executable. | 
 |   if (verifier_module == my_module || verifier_module != main_module) | 
 |     return false; | 
 |   return true; | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | bool RunTest() { | 
 |   return InternalRunThreadTest() && InternalRunLocationTest(); | 
 | } | 
 |  | 
 | }  // testing | 
 | }  // win | 
 | }  // base |