2014-05-03 15:00:21 +04:00
|
|
|
// vim: set sts=8 ts=2 sw=2 tw=99 et:
|
|
|
|
//
|
|
|
|
// Copyright (C) 2013, David Anderson and AlliedModders LLC
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are met:
|
|
|
|
//
|
|
|
|
// * Redistributions of source code must retain the above copyright notice, this
|
|
|
|
// list of conditions and the following disclaimer.
|
|
|
|
// * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
// this list of conditions and the following disclaimer in the documentation
|
|
|
|
// and/or other materials provided with the distribution.
|
|
|
|
// * Neither the name of AlliedModders LLC nor the names of its contributors
|
|
|
|
// may be used to endorse or promote products derived from this software
|
|
|
|
// without specific prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
|
|
|
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
// POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
#ifndef _include_amtl_thread_windows_h_
|
|
|
|
#define _include_amtl_thread_windows_h_
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
namespace ke {
|
|
|
|
|
|
|
|
class CriticalSection : public Lockable
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CriticalSection() {
|
|
|
|
InitializeCriticalSection(&cs_);
|
|
|
|
}
|
|
|
|
~CriticalSection() {
|
|
|
|
DeleteCriticalSection(&cs_);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DoTryLock() KE_OVERRIDE {
|
|
|
|
return !!TryEnterCriticalSection(&cs_);
|
|
|
|
}
|
|
|
|
void DoLock() KE_OVERRIDE {
|
|
|
|
EnterCriticalSection(&cs_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DoUnlock() KE_OVERRIDE {
|
|
|
|
LeaveCriticalSection(&cs_);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
CRITICAL_SECTION cs_;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef CriticalSection Mutex;
|
|
|
|
|
|
|
|
// Currently, this class only supports single-listener CVs.
|
|
|
|
class ConditionVariable : public Lockable
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ConditionVariable() {
|
2014-12-06 14:21:14 +03:00
|
|
|
event_ = CreateEvent(nullptr, FALSE, FALSE, nullptr);
|
2014-05-03 15:00:21 +04:00
|
|
|
}
|
|
|
|
~ConditionVariable() {
|
|
|
|
CloseHandle(event_);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DoTryLock() KE_OVERRIDE {
|
|
|
|
return cs_.DoTryLock();
|
|
|
|
}
|
|
|
|
void DoLock() KE_OVERRIDE {
|
|
|
|
cs_.DoLock();
|
|
|
|
}
|
|
|
|
void DoUnlock() KE_OVERRIDE {
|
|
|
|
cs_.DoUnlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Notify() {
|
|
|
|
AssertCurrentThreadOwns();
|
|
|
|
SetEvent(event_);
|
|
|
|
}
|
|
|
|
|
2014-12-06 14:21:14 +03:00
|
|
|
WaitResult Wait(size_t timeoutMs) {
|
2014-05-03 15:00:21 +04:00
|
|
|
// This will assert if the lock has not been acquired. We don't need to be
|
|
|
|
// atomic here, like pthread_cond_wait, because the event bit will stick
|
|
|
|
// until reset by a wait function.
|
|
|
|
Unlock();
|
2014-12-06 14:21:14 +03:00
|
|
|
DWORD rv = WaitForSingleObject(event_, int(timeoutMs));
|
2014-05-03 15:00:21 +04:00
|
|
|
Lock();
|
|
|
|
|
|
|
|
if (rv == WAIT_TIMEOUT)
|
|
|
|
return Wait_Timeout;
|
|
|
|
if (rv == WAIT_FAILED)
|
|
|
|
return Wait_Error;
|
|
|
|
return Wait_Signaled;
|
|
|
|
}
|
|
|
|
|
|
|
|
WaitResult Wait() {
|
|
|
|
return Wait(INFINITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
CriticalSection cs_;
|
|
|
|
HANDLE event_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Thread
|
|
|
|
{
|
|
|
|
public:
|
2014-12-06 14:21:14 +03:00
|
|
|
Thread(IRunnable *run, const char *name = nullptr) {
|
|
|
|
thread_ = CreateThread(nullptr, 0, Main, run, 0, nullptr);
|
2014-05-03 15:00:21 +04:00
|
|
|
}
|
|
|
|
~Thread() {
|
|
|
|
if (!thread_)
|
|
|
|
return;
|
|
|
|
CloseHandle(thread_);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Succeeded() const {
|
|
|
|
return !!thread_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Join() {
|
|
|
|
if (!Succeeded())
|
|
|
|
return;
|
|
|
|
WaitForSingleObject(thread_, INFINITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
HANDLE handle() const {
|
|
|
|
return thread_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static DWORD WINAPI Main(LPVOID arg) {
|
|
|
|
((IRunnable *)arg)->Run();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma pack(push, 8)
|
|
|
|
struct ThreadNameInfo {
|
|
|
|
DWORD dwType;
|
|
|
|
LPCSTR szName;
|
|
|
|
DWORD dwThreadID;
|
|
|
|
DWORD dwFlags;
|
|
|
|
};
|
|
|
|
#pragma pack(pop)
|
|
|
|
|
|
|
|
private:
|
|
|
|
HANDLE thread_;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace ke
|
|
|
|
|
|
|
|
#endif // _include_amtl_thread_windows_h_
|