summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--opentrack-dinput/dinput.cpp48
-rw-r--r--opentrack-dinput/dinput.hpp13
2 files changed, 47 insertions, 14 deletions
diff --git a/opentrack-dinput/dinput.cpp b/opentrack-dinput/dinput.cpp
index 217df6a7..ae132e70 100644
--- a/opentrack-dinput/dinput.cpp
+++ b/opentrack-dinput/dinput.cpp
@@ -34,31 +34,46 @@ dinput_handle::di_t dinput_handle::make_di()
return di_t(ret);
}
-#endif
+void dinput_handle::di_t::free_di()
+{
+ if (handle && *handle)
+ (*handle)->Release();
+ *handle = nullptr;
+ handle = nullptr;
+}
-dinput_handle::di_t::di_t(LPDIRECTINPUT8& handle) : handle(handle)
+void dinput_handle::di_t::ref_di()
{
while (init_lock.test_and_set()) { /* busy loop */ }
- refcnt++;
+ const int refcnt_ = refcnt.fetch_add(1) + 1;
+ qDebug() << "start: dinput refcount now" << (refcnt_);
init_lock.clear();
}
-void dinput_handle::di_t::free_di()
+dinput_handle::di_t& dinput_handle::di_t::operator=(const di_t& new_di)
{
if (handle)
- handle->Release();
- handle = nullptr;
+ unref_di();
+
+ handle = new_di.handle;
+
+ if (handle)
+ ref_di();
+
+ return *this;
}
-dinput_handle::di_t::~di_t()
+void dinput_handle::di_t::unref_di()
{
while (init_lock.test_and_set()) { /* busy loop */ }
- int refcnt_ = refcnt.fetch_sub(1);
+ const int refcnt_ = refcnt.fetch_sub(1) - 1;
- if (refcnt_ == 1)
+ qDebug() << "exit: dinput refcount now" << refcnt_;
+
+ if (refcnt_ == 0)
{
qDebug() << "exit: deleting di handle";
free_di();
@@ -66,3 +81,18 @@ dinput_handle::di_t::~di_t()
init_lock.clear();
}
+
+dinput_handle::di_t::di_t(LPDIRECTINPUT8& handle) : handle(&handle)
+{
+ ref_di();
+}
+
+dinput_handle::di_t::di_t() : handle(nullptr) {}
+
+dinput_handle::di_t::~di_t()
+{
+ if (handle)
+ unref_di();
+}
+
+#endif
diff --git a/opentrack-dinput/dinput.hpp b/opentrack-dinput/dinput.hpp
index 37a6d723..53f1c4af 100644
--- a/opentrack-dinput/dinput.hpp
+++ b/opentrack-dinput/dinput.hpp
@@ -35,16 +35,19 @@ public:
{
friend class dinput_handle;
- LPDIRECTINPUT8& handle;
+ LPDIRECTINPUT8* handle;
di_t(LPDIRECTINPUT8& handle);
void free_di();
+ void unref_di();
+ void ref_di();
public:
- LPDIRECTINPUT8 operator->() { return handle; }
- operator LPDIRECTINPUT8() { return handle; }
- LPDIRECTINPUT8 di() { return handle; }
-
+ LPDIRECTINPUT8 operator->() { return *handle; }
+ operator LPDIRECTINPUT8() { return *handle; }
+ LPDIRECTINPUT8 di() { return *handle; }
+ di_t& operator=(const di_t& new_di);
+ di_t();
~di_t();
};