blob: 06b5e4f2be3749de2dca2fe8dd5fe462f6674891 (
plain)
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
|
#ifdef _WIN32
#include "dinput.hpp"
#include <QDebug>
std::atomic<int> dinput_handle::refcnt;
std::atomic_flag dinput_handle::init_lock = ATOMIC_FLAG_INIT;
dinput_handle::di_t dinput_handle::handle(dinput_handle::make_di());
LPDIRECTINPUT8& dinput_handle::init_di()
{
HRESULT hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
if (FAILED(hr))
qDebug() << "dinput: failed CoInitializeEx" << hr << GetLastError();
static LPDIRECTINPUT8 di_ = nullptr;
if (di_ == nullptr)
{
if (!SUCCEEDED(DirectInput8Create(GetModuleHandle(NULL), DIRECTINPUT_VERSION, IID_IDirectInput8, (void**)&di_, NULL)))
{
di_ = nullptr;
}
}
return di_;
}
dinput_handle::di_t dinput_handle::make_di()
{
while (init_lock.test_and_set()) { /* busy loop */ }
LPDIRECTINPUT8& ret = init_di();
init_lock.clear();
return di_t(ret);
}
#endif
dinput_handle::di_t::di_t(LPDIRECTINPUT8& handle) : handle(handle)
{
while (init_lock.test_and_set()) { /* busy loop */ }
refcnt++;
init_lock.clear();
}
void dinput_handle::di_t::free_di()
{
if (handle)
handle->Release();
handle = nullptr;
}
dinput_handle::di_t::~di_t()
{
while (init_lock.test_and_set()) { /* busy loop */ }
int refcnt_ = refcnt.fetch_sub(1);
if (refcnt_ == 1)
{
qDebug() << "exit: deleting di handle";
free_di();
}
init_lock.clear();
}
|