// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #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 RunFunctionTaskBase> { 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 { auto result = std::apply(invoke, std::move(data)); using T = InvokeResultType; if constexpr (std::is_move_constructible_v) this->promise.reportAndMoveResult(std::move(result)); else if constexpr (std::is_copy_constructible_v) this->promise.reportResult(result); } } 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