1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
|
// file : odb/details/win32/thread.cxx
// copyright : Copyright (c) 2009-2015 Code Synthesis Tools CC
// license : GNU GPL v2; see accompanying LICENSE file
#include <odb/details/win32/windows.hxx>
#include <process.h> // _beginthreadex, _endthreadex
#include <odb/details/unique-ptr.hxx>
#include <odb/details/win32/thread.hxx>
#include <odb/details/win32/exceptions.hxx>
unsigned int __stdcall
odb_thread_thunk (void* arg)
{
odb::details::thread::thread_thunk (arg);
_endthreadex (0);
return 0;
}
namespace odb
{
namespace details
{
void thread::
thread_thunk (void* arg)
{
data* d (static_cast<data*> (arg));
d->ret = d->func (d->arg);
d->mutex.lock ();
unsigned char count = --d->count;
d->mutex.unlock ();
if (count == 0)
delete d;
}
thread::
~thread ()
{
if (handle_ != 0)
{
CloseHandle (handle_);
// Win32 mutex implementation does not throw.
//
data_->mutex.lock ();
unsigned char count = --data_->count;
data_->mutex.unlock ();
if (count == 0)
delete data_;
}
}
thread::
thread (void* (*func) (void*), void* arg)
{
unique_ptr<data> d (new data);
d->func = func;
d->arg = arg;
d->count = 2; // One for the thread and one for us.
handle_ = (HANDLE)_beginthreadex (
0, 0, &odb_thread_thunk, d.get (), 0, 0);
if (handle_ == 0)
throw win32_exception ();
data_ = d.release ();
}
void* thread::
join ()
{
void* r;
if (WaitForSingleObject (handle_, INFINITE) != 0)
throw win32_exception ();
r = data_->ret;
CloseHandle (handle_);
delete data_;
handle_ = 0;
data_ = 0;
return r;
}
}
}
|