/** * @file mingw.mutex.h * @brief std::mutex et al implementation for MinGW ** (c) 2013-2016 by Mega Limited, Auckland, New Zealand * @author Alexander Vassilev * * @copyright Simplified (2-clause) BSD License. * You should have received a copy of the license along with this * program. * * This code 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. * @note * This file may become part of the mingw-w64 runtime package. If/when this happens, * the appropriate license will be added, i.e. this code will become dual-licensed, * and the current BSD 2-clause license will stay. */ #ifndef WIN32STDMUTEX_H #define WIN32STDMUTEX_H #ifdef _GLIBCXX_HAS_GTHREADS #error This version of MinGW seems to include a win32 port of pthreads, and probably \ already has C++11 std threading classes implemented, based on pthreads. \ You are likely to have class redefinition errors below, and unfirtunately this \ implementation can not be used standalone \ and independent of the system header, since it relies on it for \ std::unique_lock and other utility classes. If you would still like to use this \ implementation (as it is more lightweight), you have to edit the \ c++-config.h system header of your MinGW to not define _GLIBCXX_HAS_GTHREADS. \ This will prevent system headers from defining actual threading classes while still \ defining the necessary utility classes. #endif // Recursion checks on non-recursive locks have some performance penalty, so the user // may want to disable the checks in release builds. In that case, make sure they // are always enabled in debug builds. #if defined(STDMUTEX_NO_RECURSION_CHECKS) && !defined(NDEBUG) #undef STDMUTEX_NO_RECURSION_CHECKS #endif #include #include #include #ifndef EPROTO #define EPROTO 134 #endif #ifndef EOWNERDEAD #define EOWNERDEAD 133 #endif namespace std { class recursive_mutex { protected: CRITICAL_SECTION mHandle; public: typedef LPCRITICAL_SECTION native_handle_type; native_handle_type native_handle() {return &mHandle;} recursive_mutex() noexcept { InitializeCriticalSection(&mHandle); } recursive_mutex (const recursive_mutex&) = delete; recursive_mutex& operator=(const recursive_mutex&) = delete; ~recursive_mutex() noexcept { DeleteCriticalSection(&mHandle); } void lock() { EnterCriticalSection(&mHandle); } void unlock() { LeaveCriticalSection(&mHandle); } bool try_lock() { return (TryEnterCriticalSection(&mHandle)!=0); } }; template class _NonRecursive: protected B { protected: typedef B base; DWORD mOwnerThread; public: using base::native_handle_type; using base::native_handle; _NonRecursive() noexcept :base(), mOwnerThread(0) {} _NonRecursive (const _NonRecursive&) = delete; _NonRecursive& operator= (const _NonRecursive&) = delete; void lock() { base::lock(); checkSetOwnerAfterLock(); } protected: void checkSetOwnerAfterLock() { DWORD self = GetCurrentThreadId(); if (mOwnerThread == self) { fprintf(stderr, "FATAL: Recursive locking or non-recursive mutex detected. Throwing sysetm exception\n"); fflush(stderr); throw system_error(EDEADLK, generic_category()); } mOwnerThread = self; } void checkSetOwnerBeforeUnlock() { DWORD self = GetCurrentThreadId(); if (mOwnerThread != self) { fprintf(stderr, "FATAL: Recursive unlocking of non-recursive mutex detected. Throwing system exception\n"); fflush(stderr); throw system_error(EDEADLK, generic_category()); } mOwnerThread = 0; } public: void unlock() { checkSetOwnerBeforeUnlock(); base::unlock(); } bool try_lock() { bool ret = base::try_lock(); if (ret) checkSetOwnerAfterLock(); return ret; } }; #ifndef STDMUTEX_NO_RECURSION_CHECKS typedef _NonRecursive mutex; #else typedef recursive_mutex mutex; #endif class recursive_timed_mutex { protected: HANDLE mHandle; public: typedef HANDLE native_handle_type; native_handle_type native_handle() const {return mHandle;} recursive_timed_mutex(const recursive_timed_mutex&) = delete; recursive_timed_mutex& operator=(const recursive_timed_mutex&) = delete; recursive_timed_mutex(): mHandle(CreateMutex(NULL, FALSE, NULL)){} ~recursive_timed_mutex() { CloseHandle(mHandle); } void lock() { DWORD ret = WaitForSingleObject(mHandle, INFINITE); if (ret != WAIT_OBJECT_0) { if (ret == WAIT_ABANDONED) throw system_error(EOWNERDEAD, generic_category()); else throw system_error(EPROTO, generic_category()); } } void unlock() { if (!ReleaseMutex(mHandle)) throw system_error(EDEADLK, generic_category()); } bool try_lock() { DWORD ret = WaitForSingleObject(mHandle, 0); if (ret == WAIT_TIMEOUT) return false; else if (ret == WAIT_OBJECT_0) return true; else if (ret == WAIT_ABANDONED) throw system_error(EOWNERDEAD, generic_category()); else throw system_error(EPROTO, generic_category()); } template bool try_lock_for(const std::chrono::duration& dur) { DWORD timeout = (DWORD)chrono::duration_cast(dur).count(); DWORD ret = WaitForSingleObject(mHandle, timeout); if (ret == WAIT_TIMEOUT) return false; else if (ret == WAIT_OBJECT_0) return true; else if (ret == WAIT_ABANDONED) throw system_error(EOWNERDEAD, generic_category()); else throw system_error(EPROTO, generic_category()); } template bool try_lock_until(const std::chrono::time_point& timeout_time) { return try_lock_for(timeout_time - Clock::now()); } }; class timed_mutex: public _NonRecursive { protected: typedef _NonRecursive base; public: using base::base; timed_mutex(const timed_mutex&) = delete; timed_mutex& operator=(const timed_mutex&) = delete; template void try_lock_for(const std::chrono::duration& dur) { bool ret = base::try_lock_for(dur); #ifndef STDMUTEX_NO_RECURSION_CHECKS if (ret) checkSetOwnerAfterLock(); #endif return ret; } public: template bool try_lock_until(const std::chrono::time_point& timeout_time) { bool ret = base::try_lock_until(timeout_time); #ifndef STDMUTEX_NO_RECURSION_CHECKS if (ret) checkSetOwnerAfterLock(); #endif return ret; } }; // You can use the scoped locks and other helpers that are still provided by // In that case, you must include before including this file, so that this // file will not try to redefine them #ifndef _GLIBCXX_MUTEX /// Do not acquire ownership of the mutex. struct defer_lock_t { }; /// Try to acquire ownership of the mutex without blocking. struct try_to_lock_t { }; /// Assume the calling thread has already obtained mutex ownership /// and manage it. struct adopt_lock_t { }; constexpr defer_lock_t defer_lock { }; constexpr try_to_lock_t try_to_lock { }; constexpr adopt_lock_t adopt_lock { }; template class lock_guard { protected: M& mMutex; public: typedef M mutex_type; lock_guard(const lock_guard&) = delete; lock_guard& operator=(const lock_guard&) = delete; explicit lock_guard(mutex_type& m): mMutex(m) { mMutex.lock(); } lock_guard(mutex_type& m, std::adopt_lock_t):mMutex(m){} ~lock_guard() { mMutex.unlock(); } }; #endif } #endif // WIN32STDMUTEX_H