/** * @file mingw.thread.h * @brief std::thread 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 WIN32STDTHREAD_H #define WIN32STDTHREAD_H #include #include #include #include #include #include #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. \ It is likely that you will get class redefinition errors below, and unfortunately \ 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 //instead of INVALID_HANDLE_VALUE _beginthreadex returns 0 #define _STD_THREAD_INVALID_HANDLE 0 namespace std { class thread { public: class id { DWORD mId; void clear() {mId = 0;} friend class thread; public: id(DWORD aId=0):mId(aId){} bool operator==(const id& other) const {return mId == other.mId;} }; protected: HANDLE mHandle; id mThreadId; public: typedef HANDLE native_handle_type; id get_id() const noexcept {return mThreadId;} native_handle_type native_handle() const {return mHandle;} thread(): mHandle(_STD_THREAD_INVALID_HANDLE){} thread(thread&& other) :mHandle(other.mHandle), mThreadId(other.mThreadId) { other.mHandle = _STD_THREAD_INVALID_HANDLE; other.mThreadId.clear(); } thread(const thread &other)=delete; template explicit thread(Function&& f, Args&&... args) { typedef decltype(std::bind(f, args...)) Call; Call* call = new Call(std::bind(f, args...)); mHandle = (HANDLE)_beginthreadex(NULL, 0, threadfunc, (LPVOID)call, 0, (unsigned*)&(mThreadId.mId)); } template static unsigned int __stdcall threadfunc(void* arg) { std::unique_ptr upCall(static_cast(arg)); (*upCall)(); return (unsigned long)0; } bool joinable() const {return mHandle != _STD_THREAD_INVALID_HANDLE;} void join() { if (get_id() == GetCurrentThreadId()) throw system_error(EDEADLK, generic_category()); if (mHandle == _STD_THREAD_INVALID_HANDLE) throw system_error(ESRCH, generic_category()); if (!joinable()) throw system_error(EINVAL, generic_category()); WaitForSingleObject(mHandle, INFINITE); CloseHandle(mHandle); mHandle = _STD_THREAD_INVALID_HANDLE; mThreadId.clear(); } ~thread() { if (joinable()) std::terminate(); } thread& operator=(const thread&) = delete; thread& operator=(thread&& other) noexcept { if (joinable()) std::terminate(); swap(std::forward(other)); return *this; } void swap(thread&& other) noexcept { std::swap(mHandle, other.mHandle); std::swap(mThreadId.mId, other.mThreadId.mId); } static unsigned int hardware_concurrency() noexcept { static int ncpus = -1; if (ncpus == -1) { SYSTEM_INFO sysinfo; GetSystemInfo(&sysinfo); ncpus = sysinfo.dwNumberOfProcessors; } return ncpus; } void detach() { if (!joinable()) throw system_error(); if (mHandle != _STD_THREAD_INVALID_HANDLE) { CloseHandle(mHandle); mHandle = _STD_THREAD_INVALID_HANDLE; } mThreadId.clear(); } }; namespace this_thread { inline thread::id get_id() {return thread::id(GetCurrentThreadId());} inline void yield() {Sleep(0);} template< class Rep, class Period > void sleep_for( const std::chrono::duration& sleep_duration) { Sleep(chrono::duration_cast(sleep_duration).count()); } template void sleep_until(const std::chrono::time_point& sleep_time) { sleep_for(sleep_time-Clock::now()); } } } #endif // WIN32STDTHREAD_H