|  | //= llvm/Support/Win32/Mutex.inc - Win32 Reader/Writer Mutual Exclusion Lock  =// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This file implements the Win32 specific (non-pthread) RWMutex class. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | //===----------------------------------------------------------------------===// | 
|  | //=== WARNING: Implementation here must contain only generic Win32 code that | 
|  | //===          is guaranteed to work on *all* Win32 variants. | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "Windows.h" | 
|  |  | 
|  | namespace llvm { | 
|  | using namespace sys; | 
|  |  | 
|  | // Windows has slim read-writer lock support on Vista and higher, so we | 
|  | // will attempt to load the APIs.  If they exist, we will use them, and | 
|  | // if not, we will fall back on critical sections.  When we drop support | 
|  | // for XP, we can stop lazy-loading these APIs and just use them directly. | 
|  | #if defined(__MINGW32__) | 
|  | // Taken from WinNT.h | 
|  | typedef struct _RTL_SRWLOCK { | 
|  | PVOID Ptr; | 
|  | } RTL_SRWLOCK, *PRTL_SRWLOCK; | 
|  |  | 
|  | // Taken from WinBase.h | 
|  | typedef RTL_SRWLOCK SRWLOCK, *PSRWLOCK; | 
|  | #endif | 
|  |  | 
|  | static VOID (WINAPI *fpInitializeSRWLock)(PSRWLOCK lock) = NULL; | 
|  | static VOID (WINAPI *fpAcquireSRWLockExclusive)(PSRWLOCK lock) = NULL; | 
|  | static VOID (WINAPI *fpAcquireSRWLockShared)(PSRWLOCK lock) = NULL; | 
|  | static VOID (WINAPI *fpReleaseSRWLockExclusive)(PSRWLOCK lock) = NULL; | 
|  | static VOID (WINAPI *fpReleaseSRWLockShared)(PSRWLOCK lock) = NULL; | 
|  |  | 
|  | static bool sHasSRW = false; | 
|  |  | 
|  | static bool loadSRW() { | 
|  | static bool sChecked = false; | 
|  | if (!sChecked) { | 
|  | sChecked = true; | 
|  |  | 
|  | HMODULE hLib = ::LoadLibrary(TEXT("Kernel32")); | 
|  | if (hLib) { | 
|  | fpInitializeSRWLock = | 
|  | (VOID (WINAPI *)(PSRWLOCK))::GetProcAddress(hLib, | 
|  | "InitializeSRWLock"); | 
|  | fpAcquireSRWLockExclusive = | 
|  | (VOID (WINAPI *)(PSRWLOCK))::GetProcAddress(hLib, | 
|  | "AcquireSRWLockExclusive"); | 
|  | fpAcquireSRWLockShared = | 
|  | (VOID (WINAPI *)(PSRWLOCK))::GetProcAddress(hLib, | 
|  | "AcquireSRWLockShared"); | 
|  | fpReleaseSRWLockExclusive = | 
|  | (VOID (WINAPI *)(PSRWLOCK))::GetProcAddress(hLib, | 
|  | "ReleaseSRWLockExclusive"); | 
|  | fpReleaseSRWLockShared = | 
|  | (VOID (WINAPI *)(PSRWLOCK))::GetProcAddress(hLib, | 
|  | "ReleaseSRWLockShared"); | 
|  | ::FreeLibrary(hLib); | 
|  |  | 
|  | if (fpInitializeSRWLock != NULL) { | 
|  | sHasSRW = true; | 
|  | } | 
|  | } | 
|  | } | 
|  | return sHasSRW; | 
|  | } | 
|  |  | 
|  | RWMutexImpl::RWMutexImpl() { | 
|  | if (loadSRW()) { | 
|  | data_ = calloc(1, sizeof(SRWLOCK)); | 
|  | fpInitializeSRWLock(static_cast<PSRWLOCK>(data_)); | 
|  | } else { | 
|  | data_ = calloc(1, sizeof(CRITICAL_SECTION)); | 
|  | InitializeCriticalSection(static_cast<LPCRITICAL_SECTION>(data_)); | 
|  | } | 
|  | } | 
|  |  | 
|  | RWMutexImpl::~RWMutexImpl() { | 
|  | if (sHasSRW) { | 
|  | // Nothing to do in the case of slim reader/writers | 
|  | } else { | 
|  | DeleteCriticalSection(static_cast<LPCRITICAL_SECTION>(data_)); | 
|  | free(data_); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool RWMutexImpl::reader_acquire() { | 
|  | if (sHasSRW) { | 
|  | fpAcquireSRWLockShared(static_cast<PSRWLOCK>(data_)); | 
|  | } else { | 
|  | EnterCriticalSection(static_cast<LPCRITICAL_SECTION>(data_)); | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool RWMutexImpl::reader_release() { | 
|  | if (sHasSRW) { | 
|  | fpReleaseSRWLockShared(static_cast<PSRWLOCK>(data_)); | 
|  | } else { | 
|  | LeaveCriticalSection(static_cast<LPCRITICAL_SECTION>(data_)); | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool RWMutexImpl::writer_acquire() { | 
|  | if (sHasSRW) { | 
|  | fpAcquireSRWLockExclusive(static_cast<PSRWLOCK>(data_)); | 
|  | } else { | 
|  | EnterCriticalSection(static_cast<LPCRITICAL_SECTION>(data_)); | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool RWMutexImpl::writer_release() { | 
|  | if (sHasSRW) { | 
|  | fpReleaseSRWLockExclusive(static_cast<PSRWLOCK>(data_)); | 
|  | } else { | 
|  | LeaveCriticalSection(static_cast<LPCRITICAL_SECTION>(data_)); | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  |  | 
|  | } |