mirror of
				https://github.com/zenorogue/hyperrogue.git
				synced 2025-10-31 14:02:59 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			361 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			361 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**
 | |
| * @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
 | |
| 
 | |
| #if !defined(__cplusplus) || (__cplusplus < 201103L)
 | |
| #error A C++11 compiler is required!
 | |
| #endif
 | |
| 
 | |
| //  Use the standard classes for std::, if available.
 | |
| #include <thread>
 | |
| 
 | |
| #include <cstddef>      //  For std::size_t
 | |
| #include <cerrno>       //  Detect error type.
 | |
| #include <exception>    //  For std::terminate
 | |
| #include <system_error> //  For std::system_error
 | |
| #include <functional>   //  For std::hash
 | |
| #include <tuple>        //  For std::tuple
 | |
| #include <chrono>       //  For sleep timing.
 | |
| #include <memory>       //  For std::unique_ptr
 | |
| #include <iosfwd>       //  Stream output for thread ids.
 | |
| #include <utility>      //  For std::swap, std::forward
 | |
| 
 | |
| #include "mingw.invoke.h"
 | |
| 
 | |
| #if (defined(__MINGW32__) && !defined(__MINGW64_VERSION_MAJOR))
 | |
| #pragma message "The Windows API that MinGW-w32 provides is not fully compatible\
 | |
|  with Microsoft's API. We'll try to work around this, but we can make no\
 | |
|  guarantees. This problem does not exist in MinGW-w64."
 | |
| #include <windows.h>    //  No further granularity can be expected.
 | |
| #else
 | |
| #include <synchapi.h>   //  For WaitForSingleObject
 | |
| #include <handleapi.h>  //  For CloseHandle, etc.
 | |
| #include <sysinfoapi.h> //  For GetNativeSystemInfo
 | |
| #include <processthreadsapi.h>  //  For GetCurrentThreadId
 | |
| #endif
 | |
| #include <process.h>  //  For _beginthreadex
 | |
| 
 | |
| #ifndef NDEBUG
 | |
| #include <cstdio>
 | |
| #endif
 | |
| 
 | |
| #if !defined(_WIN32_WINNT) || (_WIN32_WINNT < 0x0501)
 | |
| #error To use the MinGW-std-threads library, you will need to define the macro _WIN32_WINNT to be 0x0501 (Windows XP) or higher.
 | |
| #endif
 | |
| 
 | |
| //  Instead of INVALID_HANDLE_VALUE, _beginthreadex returns 0.
 | |
| namespace mingw_stdthread
 | |
| {
 | |
| namespace detail
 | |
| {
 | |
|     template<std::size_t...>
 | |
|     struct IntSeq {};
 | |
| 
 | |
|     template<std::size_t N, std::size_t... S>
 | |
|     struct GenIntSeq : GenIntSeq<N-1, N-1, S...> { };
 | |
| 
 | |
|     template<std::size_t... S>
 | |
|     struct GenIntSeq<0, S...> { typedef IntSeq<S...> type; };
 | |
| 
 | |
| //    Use a template specialization to avoid relying on compiler optimization
 | |
| //  when determining the parameter integer sequence.
 | |
|     template<class Func, class T, typename... Args>
 | |
|     class ThreadFuncCall;
 | |
| // We can't define the Call struct in the function - the standard forbids template methods in that case
 | |
|     template<class Func, std::size_t... S, typename... Args>
 | |
|     class ThreadFuncCall<Func, detail::IntSeq<S...>, Args...>
 | |
|     {
 | |
|         static_assert(sizeof...(S) == sizeof...(Args), "Args must match.");
 | |
|         using Tuple = std::tuple<typename std::decay<Args>::type...>;
 | |
|         typename std::decay<Func>::type mFunc;
 | |
|         Tuple mArgs;
 | |
| 
 | |
|     public:
 | |
|         ThreadFuncCall(Func&& aFunc, Args&&... aArgs)
 | |
|           : mFunc(std::forward<Func>(aFunc)),
 | |
|             mArgs(std::forward<Args>(aArgs)...)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         void callFunc()
 | |
|         {
 | |
|             detail::invoke(std::move(mFunc), std::move(std::get<S>(mArgs)) ...);
 | |
|         }
 | |
|     };
 | |
| 
 | |
| //  Allow construction of threads without exposing implementation.
 | |
|     class ThreadIdTool;
 | |
| } //  Namespace "detail"
 | |
| 
 | |
| class thread
 | |
| {
 | |
| public:
 | |
|     class id
 | |
|     {
 | |
|         DWORD mId = 0;
 | |
|         friend class thread;
 | |
|         friend class std::hash<id>;
 | |
|         friend class detail::ThreadIdTool;
 | |
|         explicit id(DWORD aId) noexcept : mId(aId){}
 | |
|     public:
 | |
|         id (void) noexcept = default;
 | |
|         friend bool operator==(id x, id y) noexcept {return x.mId == y.mId; }
 | |
|         friend bool operator!=(id x, id y) noexcept {return x.mId != y.mId; }
 | |
|         friend bool operator< (id x, id y) noexcept {return x.mId <  y.mId; }
 | |
|         friend bool operator<=(id x, id y) noexcept {return x.mId <= y.mId; }
 | |
|         friend bool operator> (id x, id y) noexcept {return x.mId >  y.mId; }
 | |
|         friend bool operator>=(id x, id y) noexcept {return x.mId >= y.mId; }
 | |
| 
 | |
|         template<class _CharT, class _Traits>
 | |
|         friend std::basic_ostream<_CharT, _Traits>&
 | |
|         operator<<(std::basic_ostream<_CharT, _Traits>& __out, id __id)
 | |
|         {
 | |
|             if (__id.mId == 0)
 | |
|             {
 | |
|                 return __out << "(invalid std::thread::id)";
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return __out << __id.mId;
 | |
|             }
 | |
|         }
 | |
|     };
 | |
| private:
 | |
|     static constexpr HANDLE kInvalidHandle = nullptr;
 | |
|     static constexpr DWORD kInfinite = 0xffffffffl;
 | |
|     HANDLE mHandle;
 | |
|     id mThreadId;
 | |
| 
 | |
|     template <class Call>
 | |
|     static unsigned __stdcall threadfunc(void* arg)
 | |
|     {
 | |
|         std::unique_ptr<Call> call(static_cast<Call*>(arg));
 | |
|         call->callFunc();
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     static unsigned int _hardware_concurrency_helper() noexcept
 | |
|     {
 | |
|         SYSTEM_INFO sysinfo;
 | |
| //    This is one of the few functions used by the library which has a nearly-
 | |
| //  equivalent function defined in earlier versions of Windows. Include the
 | |
| //  workaround, just as a reminder that it does exist.
 | |
| #if defined(_WIN32_WINNT) && (_WIN32_WINNT >= 0x0501)
 | |
|         ::GetNativeSystemInfo(&sysinfo);
 | |
| #else
 | |
|         ::GetSystemInfo(&sysinfo);
 | |
| #endif
 | |
|         return sysinfo.dwNumberOfProcessors;
 | |
|     }
 | |
| public:
 | |
|     typedef HANDLE native_handle_type;
 | |
|     id get_id() const noexcept {return mThreadId;}
 | |
|     native_handle_type native_handle() const {return mHandle;}
 | |
|     thread(): mHandle(kInvalidHandle), mThreadId(){}
 | |
| 
 | |
|     thread(thread&& other)
 | |
|     :mHandle(other.mHandle), mThreadId(other.mThreadId)
 | |
|     {
 | |
|         other.mHandle = kInvalidHandle;
 | |
|         other.mThreadId = id{};
 | |
|     }
 | |
| 
 | |
|     thread(const thread &other)=delete;
 | |
| 
 | |
|     template<class Func, typename... Args>
 | |
|     explicit thread(Func&& func, Args&&... args) : mHandle(), mThreadId()
 | |
|     {
 | |
|         using ArgSequence = typename detail::GenIntSeq<sizeof...(Args)>::type;
 | |
|         using Call = detail::ThreadFuncCall<Func, ArgSequence, Args...>;
 | |
|         auto call = new Call(
 | |
|             std::forward<Func>(func), std::forward<Args>(args)...);
 | |
|         unsigned id_receiver;
 | |
|         auto int_handle = _beginthreadex(NULL, 0, threadfunc<Call>,
 | |
|             static_cast<LPVOID>(call), 0, &id_receiver);
 | |
|         if (int_handle == 0)
 | |
|         {
 | |
|             mHandle = kInvalidHandle;
 | |
|             int errnum = errno;
 | |
|             delete call;
 | |
| //  Note: Should only throw EINVAL, EAGAIN, EACCES
 | |
|             throw std::system_error(errnum, std::generic_category());
 | |
|         } else {
 | |
|             mThreadId.mId = id_receiver;
 | |
|             mHandle = reinterpret_cast<HANDLE>(int_handle);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bool joinable() const {return mHandle != kInvalidHandle;}
 | |
| 
 | |
| //    Note: Due to lack of synchronization, this function has a race condition
 | |
| //  if called concurrently, which leads to undefined behavior. The same applies
 | |
| //  to all other member functions of this class, but this one is mentioned
 | |
| //  explicitly.
 | |
|     void join()
 | |
|     {
 | |
|         using namespace std;
 | |
|         if (get_id() == id(GetCurrentThreadId()))
 | |
|             throw system_error(make_error_code(errc::resource_deadlock_would_occur));
 | |
|         if (mHandle == kInvalidHandle)
 | |
|             throw system_error(make_error_code(errc::no_such_process));
 | |
|         if (!joinable())
 | |
|             throw system_error(make_error_code(errc::invalid_argument));
 | |
|         WaitForSingleObject(mHandle, kInfinite);
 | |
|         CloseHandle(mHandle);
 | |
|         mHandle = kInvalidHandle;
 | |
|         mThreadId = id{};
 | |
|     }
 | |
| 
 | |
|     ~thread()
 | |
|     {
 | |
|         if (joinable())
 | |
|         {
 | |
| #ifndef NDEBUG
 | |
|             std::printf("Error: Must join() or detach() a thread before \
 | |
| destroying it.\n");
 | |
| #endif
 | |
|             std::terminate();
 | |
|         }
 | |
|     }
 | |
|     thread& operator=(const thread&) = delete;
 | |
|     thread& operator=(thread&& other) noexcept
 | |
|     {
 | |
|         if (joinable())
 | |
|         {
 | |
| #ifndef NDEBUG
 | |
|             std::printf("Error: Must join() or detach() a thread before \
 | |
| moving another thread to it.\n");
 | |
| #endif
 | |
|             std::terminate();
 | |
|         }
 | |
|         swap(std::forward<thread>(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 unsigned int cached = _hardware_concurrency_helper();
 | |
|         return cached;
 | |
|     }
 | |
| 
 | |
|     void detach()
 | |
|     {
 | |
|         if (!joinable())
 | |
|         {
 | |
|             using namespace std;
 | |
|             throw system_error(make_error_code(errc::invalid_argument));
 | |
|         }
 | |
|         if (mHandle != kInvalidHandle)
 | |
|         {
 | |
|             CloseHandle(mHandle);
 | |
|             mHandle = kInvalidHandle;
 | |
|         }
 | |
|         mThreadId = id{};
 | |
|     }
 | |
| };
 | |
| 
 | |
| namespace detail
 | |
| {
 | |
|     class ThreadIdTool
 | |
|     {
 | |
|     public:
 | |
|         static thread::id make_id (DWORD base_id) noexcept
 | |
|         {
 | |
|             return thread::id(base_id);
 | |
|         }
 | |
|     };
 | |
| } //  Namespace "detail"
 | |
| 
 | |
| namespace this_thread
 | |
| {
 | |
|     inline thread::id get_id() noexcept
 | |
|     {
 | |
|         return detail::ThreadIdTool::make_id(GetCurrentThreadId());
 | |
|     }
 | |
|     inline void yield() noexcept {Sleep(0);}
 | |
|     template< class Rep, class Period >
 | |
|     void sleep_for( const std::chrono::duration<Rep,Period>& sleep_duration)
 | |
|     {
 | |
|         static constexpr DWORD kInfinite = 0xffffffffl;
 | |
|         using namespace std::chrono;
 | |
|         using rep = milliseconds::rep;
 | |
|         rep ms = duration_cast<milliseconds>(sleep_duration).count();
 | |
|         while (ms > 0)
 | |
|         {
 | |
|             constexpr rep kMaxRep = static_cast<rep>(kInfinite - 1);
 | |
|             auto sleepTime = (ms < kMaxRep) ? ms : kMaxRep;
 | |
|             Sleep(static_cast<DWORD>(sleepTime));
 | |
|             ms -= sleepTime;
 | |
|         }
 | |
|     }
 | |
|     template <class Clock, class Duration>
 | |
|     void sleep_until(const std::chrono::time_point<Clock,Duration>& sleep_time)
 | |
|     {
 | |
|         sleep_for(sleep_time-Clock::now());
 | |
|     }
 | |
| }
 | |
| } //  Namespace mingw_stdthread
 | |
| 
 | |
| namespace std
 | |
| {
 | |
| //    Because of quirks of the compiler, the common "using namespace std;"
 | |
| //  directive would flatten the namespaces and introduce ambiguity where there
 | |
| //  was none. Direct specification (std::), however, would be unaffected.
 | |
| //    Take the safe option, and include only in the presence of MinGW's win32
 | |
| //  implementation.
 | |
| #if defined(__MINGW32__ ) && !defined(_GLIBCXX_HAS_GTHREADS)
 | |
| using mingw_stdthread::thread;
 | |
| //    Remove ambiguity immediately, to avoid problems arising from the above.
 | |
| //using std::thread;
 | |
| namespace this_thread
 | |
| {
 | |
| using namespace mingw_stdthread::this_thread;
 | |
| }
 | |
| #elif !defined(MINGW_STDTHREAD_REDUNDANCY_WARNING)  //  Skip repetition
 | |
| #define MINGW_STDTHREAD_REDUNDANCY_WARNING
 | |
| #pragma message "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. These classes, found in namespace std, are not overridden\
 | |
|  by the mingw-std-thread library. If you would still like to use this\
 | |
|  implementation (as it is more lightweight), use the classes provided in\
 | |
|  namespace mingw_stdthread."
 | |
| #endif
 | |
| 
 | |
| //    Specialize hash for this implementation's thread::id, even if the
 | |
| //  std::thread::id already has a hash.
 | |
| template<>
 | |
| struct hash<mingw_stdthread::thread::id>
 | |
| {
 | |
|     typedef mingw_stdthread::thread::id argument_type;
 | |
|     typedef size_t result_type;
 | |
|     size_t operator() (const argument_type & i) const noexcept
 | |
|     {
 | |
|         return i.mId;
 | |
|     }
 | |
| };
 | |
| }
 | |
| #endif // WIN32STDTHREAD_H
 | 
