diff options
-rw-r--r-- | compat/run-in-thread.hpp | 54 |
1 files changed, 24 insertions, 30 deletions
diff --git a/compat/run-in-thread.hpp b/compat/run-in-thread.hpp index b8ffc179..afb41c98 100644 --- a/compat/run-in-thread.hpp +++ b/compat/run-in-thread.hpp @@ -19,25 +19,18 @@ namespace qt_impl_detail { -template<typename t> +template<typename u> struct run_in_thread_traits { - using type = t; - using ret_type = t; - static inline void assign(t& lvalue, const t& rvalue) { lvalue = rvalue; } - static inline t pass(const t& val) { return val; } - template<typename F> static inline t call(F&& fun) { return std::move(fun()); } -}; + using ret_type = std::remove_reference_t<u>; -template<typename u> -struct run_in_thread_traits<u&&> -{ - using t = typename std::remove_reference<u>::type; - using type = t; - using ret_type = u; - static inline void assign(t& lvalue, t&& rvalue) { lvalue = rvalue; } - static inline t&& pass(t&& val) { return val; } - template<typename F> static inline t&& call(F&& fun) { return std::move(fun()); } + template<typename t> + static inline void assign(u& lvalue, t&& rvalue) { std::forward<u>(lvalue) = std::forward<t>(rvalue); } + + template<typename t> + static inline auto pass(t&& val) -> decltype(auto) { return std::forward<t>(val); } + + template<typename F> static inline auto call(F&& fun) -> decltype(auto) { return std::forward<F>(fun)(); } }; template<> @@ -45,9 +38,9 @@ struct run_in_thread_traits<void> { using type = unsigned char; using ret_type = void; - static inline void assign(unsigned char&, unsigned char&&) {} - static inline void pass(type&&) {} - template<typename F> static type call(F&& fun) { fun(); return type(0); } + static inline void assign(unsigned char&, unsigned char) {} + static inline void pass(type) {} + template<typename F> static type call(F&& fun) { std::forward<F>(fun)(); return type(0); } }; } @@ -67,9 +60,9 @@ run_in_thread_sync(QObject* obj, F&& fun) { std::mutex mtx; std::condition_variable cvar; - bool flag; + bool flag = false; - semaphore() : flag(false) {} + semaphore() = default; void wait() { @@ -86,18 +79,19 @@ run_in_thread_sync(QObject* obj, F&& fun) } }; + if (obj->thread() == QThread::currentThread()) + return traits::pass(traits::call(fun)); + semaphore sem; { QObject src; - QObject::connect(&src, - &QObject::destroyed, - obj, - [&] { + src.moveToThread(QThread::currentThread()); + QObject::connect(&src, &QObject::destroyed, obj, [&] { traits::assign(ret, traits::call(fun)); sem.notify(); }, - Qt::AutoConnection); + Qt::QueuedConnection); } sem.wait(); @@ -107,9 +101,9 @@ run_in_thread_sync(QObject* obj, F&& fun) template<typename F> void run_in_thread_async(QObject* obj, F&& fun) { + if (obj->thread() == QThread::currentThread()) + return (void)fun(); + QObject src; - QThread* t = obj->thread(); - if (!t) abort(); - src.moveToThread(t); - QObject::connect(&src, &QObject::destroyed, obj, std::move(fun), Qt::AutoConnection); + QObject::connect(&src, &QObject::destroyed, obj, std::forward<F>(fun), Qt::QueuedConnection); } |