/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtConcurrent module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QTCONCURRENT_STOREDFUNCTIONCALL_H #define QTCONCURRENT_STOREDFUNCTIONCALL_H #include #ifndef QT_NO_CONCURRENT #include #include #include QT_BEGIN_NAMESPACE #ifndef Q_QDOC namespace QtConcurrent { template struct NonMemberFunctionResolver; template struct NonMemberFunctionResolver { using Type = std::tuple, QPromise &, std::decay_t...>; static_assert(std::is_invocable_v, QPromise &, std::decay_t...>, "It's not possible to invoke the function with passed arguments."); static_assert(std::is_void_v, QPromise &, std::decay_t...>>, "The function must return void type."); static constexpr void invoke(std::decay_t function, QPromise &promise, std::decay_t... args) { std::invoke(function, promise, args...); } static Type initData(Function &&f, QPromise &promise, Args &&...args) { return Type { std::forward(f), std::ref(promise), std::forward(args)... }; } }; template struct MemberFunctionResolver; template struct MemberFunctionResolver { using Type = std::tuple, std::decay_t, QPromise &, std::decay_t...>; static_assert(std::is_invocable_v, std::decay_t, QPromise &, std::decay_t...>, "It's not possible to invoke the function with passed arguments."); static_assert(std::is_void_v, std::decay_t, QPromise &, std::decay_t...>>, "The function must return void type."); static constexpr void invoke(std::decay_t function, std::decay_t object, QPromise &promise, std::decay_t... args) { std::invoke(function, object, promise, args...); } static Type initData(Function &&f, QPromise &promise, Arg &&fa, Args &&...args) { return Type { std::forward(f), std::forward(fa), std::ref(promise), std::forward(args)... }; } }; template struct FunctionResolverHelper; template struct FunctionResolverHelper : public NonMemberFunctionResolver { }; template struct FunctionResolverHelper : public MemberFunctionResolver { }; template struct FunctionResolver : public FunctionResolverHelper>::type, Function, PromiseType, Args...> { }; template struct InvokeResult { static_assert(std::is_invocable_v, std::decay_t...>, "It's not possible to invoke the function with passed arguments."); using Type = std::invoke_result_t, std::decay_t...>; }; template using InvokeResultType = typename InvokeResult::Type; template using DecayedTuple = std::tuple...>; template struct StoredFunctionCall : public RunFunctionTask> { StoredFunctionCall(DecayedTuple &&_data) : data(std::move(_data)) {} protected: void runFunctor() override { constexpr auto invoke = [] (std::decay_t function, std::decay_t... args) -> auto { return std::invoke(function, args...); }; if constexpr (std::is_void_v>) std::apply(invoke, std::move(data)); else this->result = std::apply(invoke, std::move(data)); } private: DecayedTuple data; }; template struct StoredFunctionCallWithPromise : public RunFunctionTaskBase { using Resolver = FunctionResolver; using DataType = typename Resolver::Type; StoredFunctionCallWithPromise(Function &&f, Args &&...args) : prom(this->promise), data(std::move(Resolver::initData(std::forward(f), std::ref(prom), std::forward(args)...))) {} StoredFunctionCallWithPromise(DecayedTuple &&_data) : StoredFunctionCallWithPromise(std::move(_data), std::index_sequence_for, std::decay_t...>()) {} protected: void runFunctor() override { std::apply(Resolver::invoke, std::move(data)); } private: // helper to pack back the tuple into parameter pack template StoredFunctionCallWithPromise(DecayedTuple &&_data, std::index_sequence) : StoredFunctionCallWithPromise(std::move(std::get(_data))...) {} QPromise prom; DataType data; }; template struct NonPromiseTaskResolver; template struct NonPromiseTaskResolver { using TaskWithArgs = DecayedTuple; static auto run(TaskWithArgs &&args, const TaskStartParameters &startParameters) { return (new StoredFunctionCall(std::move(args))) ->start(startParameters); } }; template struct PromiseTaskResolver; template struct PromiseTaskResolver { static_assert(QtPrivate::ArgResolver::IsPromise::value, "The first argument of passed callable object isn't a QPromise & type. " "Did you intend to pass a callable which takes a QPromise & type as a first argument? " "Otherwise it's not possible to invoke the function with passed arguments."); using TaskWithArgs = DecayedTuple; static auto run(TaskWithArgs &&args, const TaskStartParameters &startParameters) { using PromiseType = typename QtPrivate::ArgResolver::PromiseType; return (new StoredFunctionCallWithPromise(std::move(args))) ->start(startParameters); } }; template struct TaskResolverHelper; template struct TaskResolverHelper : public NonPromiseTaskResolver { }; template struct TaskResolverHelper : public PromiseTaskResolver { }; template struct TaskResolver : public TaskResolverHelper, std::decay_t...>::type, Function, Args...> { }; } //namespace QtConcurrent #endif // Q_QDOC QT_END_NAMESPACE #endif // QT_NO_CONCURRENT #endif