aboutsummaryrefslogtreecommitdiffstats
path: root/include/mingw.mutex.h
diff options
context:
space:
mode:
Diffstat (limited to 'include/mingw.mutex.h')
-rw-r--r--include/mingw.mutex.h275
1 files changed, 275 insertions, 0 deletions
diff --git a/include/mingw.mutex.h b/include/mingw.mutex.h
new file mode 100644
index 0000000..d453202
--- /dev/null
+++ b/include/mingw.mutex.h
@@ -0,0 +1,275 @@
+/**
+* @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 <mutex> 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 <windows.h>
+#include <chrono>
+#include <system_error>
+
+#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 B>
+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<B>&) = delete;
+ _NonRecursive& operator= (const _NonRecursive<B>&) = 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<recursive_mutex> 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 <class Rep, class Period>
+ bool try_lock_for(const std::chrono::duration<Rep,Period>& dur)
+ {
+ DWORD timeout = (DWORD)chrono::duration_cast<chrono::milliseconds>(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 <class Clock, class Duration>
+ bool try_lock_until(const std::chrono::time_point<Clock,Duration>& timeout_time)
+ {
+ return try_lock_for(timeout_time - Clock::now());
+ }
+};
+
+class timed_mutex: public _NonRecursive<recursive_timed_mutex>
+{
+protected:
+ typedef _NonRecursive<recursive_timed_mutex> base;
+public:
+ using base::base;
+ timed_mutex(const timed_mutex&) = delete;
+ timed_mutex& operator=(const timed_mutex&) = delete;
+ template <class Rep, class Period>
+ void try_lock_for(const std::chrono::duration<Rep,Period>& dur)
+ {
+ bool ret = base::try_lock_for(dur);
+#ifndef STDMUTEX_NO_RECURSION_CHECKS
+ if (ret)
+ checkSetOwnerAfterLock();
+#endif
+ return ret;
+ }
+public:
+ template <class Clock, class Duration>
+ bool try_lock_until(const std::chrono::time_point<Clock,Duration>& 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 <mutex>
+// In that case, you must include <mutex> 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 M>
+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