2026-05-17 16:52:04 -04:00
|
|
|
#ifndef PROMISES_H
|
|
|
|
|
#define PROMISES_H
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <coroutine>
|
|
|
|
|
#include <exception>
|
|
|
|
|
#include <functional>
|
|
|
|
|
#include <iostream>
|
|
|
|
|
#include <thread>
|
|
|
|
|
#include <type_traits>
|
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
|
|
#include <boost/asio/io_service.hpp>
|
|
|
|
|
#include <boost/asio/post.hpp>
|
|
|
|
|
|
|
|
|
|
#include <spinscale/componentThread.h>
|
|
|
|
|
#include <spinscale/co/coQutex.h>
|
|
|
|
|
#include <spinscale/spinLock.h>
|
|
|
|
|
|
|
|
|
|
namespace sscl::co {
|
|
|
|
|
|
|
|
|
|
template <typename PromiseType, typename T>
|
|
|
|
|
class PostingInvoker;
|
|
|
|
|
|
|
|
|
|
template <typename T, bool IsVoid = std::is_void_v<T>>
|
|
|
|
|
struct ReturnValueStorage;
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
struct ReturnValueStorage<T, false>
|
|
|
|
|
{
|
|
|
|
|
T myReturnValue{};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
struct ReturnValueStorage<T, true>
|
|
|
|
|
{
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
struct ReturnValues
|
|
|
|
|
: public ReturnValueStorage<T>
|
|
|
|
|
{
|
|
|
|
|
ReturnValues() noexcept
|
|
|
|
|
: myExceptionPtr(myMemberExceptionPtr)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
explicit ReturnValues(std::exception_ptr &callerExceptionPtr) noexcept
|
|
|
|
|
: myExceptionPtr(callerExceptionPtr)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
~ReturnValues() noexcept
|
|
|
|
|
{
|
|
|
|
|
#ifdef CONFIG_LIBSSCL_DEBUG_CO
|
|
|
|
|
std::cout << __func__ << ": " << std::this_thread::get_id() << " Destructing.\n";
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** EXPLANATION:
|
|
|
|
|
* The exception_ptr ref here can either point to the exception_ptr
|
|
|
|
|
* a non-viral coroutine supplied to us as its storage space for
|
|
|
|
|
* where we should store any exception that is thrown;
|
|
|
|
|
*
|
|
|
|
|
* Or it could point to the member exception_ptr in this very class,
|
|
|
|
|
* which is used for viral coroutines that can bubble their exception
|
|
|
|
|
* up and automatically via the language runtime.
|
|
|
|
|
*/
|
|
|
|
|
std::exception_ptr &myExceptionPtr;
|
|
|
|
|
std::exception_ptr myMemberExceptionPtr = nullptr;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** `return_value` / `return_void` only. ThreadTag is not a template parameter here:
|
|
|
|
|
* for tagged promises, PromiseType is `TaggedPostingPromise<T, ThreadTag>`.
|
|
|
|
|
*/
|
|
|
|
|
template <typename PromiseType, typename T, bool IsVoid = std::is_void_v<T>>
|
|
|
|
|
struct PostingPromiseReturnOps;
|
|
|
|
|
|
|
|
|
|
template <typename PromiseType, typename T>
|
|
|
|
|
struct PostingPromiseReturnOps<PromiseType, T, false>
|
|
|
|
|
{
|
|
|
|
|
void return_value(T returnValue) noexcept
|
|
|
|
|
{
|
|
|
|
|
static_cast<PromiseType *>(this)->returnValues.myReturnValue = std::move(returnValue);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename PromiseType, typename T>
|
|
|
|
|
struct PostingPromiseReturnOps<PromiseType, T, true>
|
|
|
|
|
{
|
|
|
|
|
void return_void() noexcept
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
struct PostingPromise
|
|
|
|
|
: public PromiseChainLink
|
|
|
|
|
{
|
|
|
|
|
struct PostBackStatus
|
|
|
|
|
{
|
|
|
|
|
struct CalleeFlowExecutor;
|
|
|
|
|
struct CallerFlowExecutor;
|
|
|
|
|
friend struct CalleeFlowExecutor;
|
|
|
|
|
friend struct CallerFlowExecutor;
|
|
|
|
|
|
|
|
|
|
explicit PostBackStatus(PostingPromise &calleePromiseIn) noexcept
|
|
|
|
|
: calleePromise(calleePromiseIn)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
void reset() noexcept
|
|
|
|
|
{
|
|
|
|
|
sscl::SpinLock::Guard guard(lock);
|
|
|
|
|
callerHasSetCallerSchedHandle = false;
|
|
|
|
|
calleeIsReadyToPostBack = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct FlowExecutor
|
|
|
|
|
{
|
|
|
|
|
explicit FlowExecutor(PostBackStatus &parentIn) noexcept
|
|
|
|
|
: parent(parentIn)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
PostBackStatus &parent;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct CalleeFlowExecutor
|
|
|
|
|
: public FlowExecutor
|
|
|
|
|
{
|
|
|
|
|
explicit CalleeFlowExecutor(PostBackStatus &parentIn) noexcept
|
|
|
|
|
: FlowExecutor(parentIn)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
void operator()() noexcept
|
|
|
|
|
{
|
|
|
|
|
sscl::SpinLock::Guard guard(this->parent.lock);
|
|
|
|
|
this->parent.calleeIsReadyToPostBack = true;
|
|
|
|
|
if (this->parent.callerHasSetCallerSchedHandle)
|
|
|
|
|
{
|
|
|
|
|
boost::asio::post(
|
|
|
|
|
this->parent.calleePromise.callerIoContext,
|
|
|
|
|
this->parent.calleePromise.callerSchedHandle);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct CallerFlowExecutor
|
|
|
|
|
: public FlowExecutor
|
|
|
|
|
{
|
|
|
|
|
explicit CallerFlowExecutor(PostBackStatus &parentIn) noexcept
|
|
|
|
|
: FlowExecutor(parentIn)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
bool operator()() noexcept
|
|
|
|
|
{
|
|
|
|
|
sscl::SpinLock::Guard guard(this->parent.lock);
|
|
|
|
|
this->parent.callerHasSetCallerSchedHandle = true;
|
|
|
|
|
if (this->parent.calleeIsReadyToPostBack) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CalleeFlowExecutor getCalleeFlowExecutor() noexcept
|
|
|
|
|
{
|
|
|
|
|
return CalleeFlowExecutor(*this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CallerFlowExecutor getCallerFlowExecutor() noexcept
|
|
|
|
|
{
|
|
|
|
|
return CallerFlowExecutor(*this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PostingPromise &calleePromise;
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
sscl::SpinLock lock;
|
|
|
|
|
bool callerHasSetCallerSchedHandle = false;
|
|
|
|
|
bool calleeIsReadyToPostBack = false;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** Post-to must run from this awaiter's await_suspend, not synchronously inside
|
|
|
|
|
* promise.initial_suspend() before it returns: the implementation's hidden coroutine
|
|
|
|
|
* state (async segment / suspend index used on the next resume()) is only updated
|
|
|
|
|
* after initial_suspend has finished returning its awaiter. Posting the handle too
|
|
|
|
|
* early lets the callee resume before that update and re-enter initial_suspend from
|
|
|
|
|
* the start, duplicating the post. See docs/prompts/post-to-and-back-in-invokables.md.
|
|
|
|
|
*/
|
|
|
|
|
struct InitialSuspendPostingInvoker
|
|
|
|
|
: public std::suspend_always
|
|
|
|
|
{
|
|
|
|
|
InitialSuspendPostingInvoker(
|
2026-05-17 17:12:11 -04:00
|
|
|
boost::asio::io_service &targetIoServiceIn,
|
2026-05-17 16:52:04 -04:00
|
|
|
std::coroutine_handle<> targetSchedHandleIn) noexcept
|
2026-05-17 17:12:11 -04:00
|
|
|
: targetIoService(targetIoServiceIn),
|
2026-05-17 16:52:04 -04:00
|
|
|
targetSchedHandle(targetSchedHandleIn)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
bool await_suspend(std::coroutine_handle<> const) noexcept
|
|
|
|
|
{
|
2026-05-17 17:12:11 -04:00
|
|
|
boost::asio::post(targetIoService, targetSchedHandle);
|
2026-05-17 16:52:04 -04:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-17 17:12:11 -04:00
|
|
|
boost::asio::io_service &targetIoService;
|
2026-05-17 16:52:04 -04:00
|
|
|
std::coroutine_handle<> targetSchedHandle;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** Post-back (non-viral completion post; viral CalleeFlowExecutor) must run from this
|
|
|
|
|
* awaiter's await_suspend, not synchronously inside promise.final_suspend() before it
|
|
|
|
|
* returns: the hidden coroutine segment index in the coroutine state is only advanced
|
|
|
|
|
* after final_suspend exits. Doing that work inside final_suspend's body risks the same
|
|
|
|
|
* kind of ordering bug as initial_suspend—resume observing the wrong segment. See
|
|
|
|
|
* docs/prompts/post-to-and-back-in-invokables.md.
|
|
|
|
|
*/
|
|
|
|
|
struct FinalSuspendPostingInvoker
|
|
|
|
|
: public std::suspend_always
|
|
|
|
|
{
|
|
|
|
|
explicit FinalSuspendPostingInvoker(PostingPromise &calleePromiseIn) noexcept
|
|
|
|
|
: calleePromise(calleePromiseIn)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
bool await_suspend(std::coroutine_handle<> const) noexcept
|
|
|
|
|
{
|
|
|
|
|
if (calleePromise.callerLambda)
|
|
|
|
|
{
|
|
|
|
|
#ifdef CONFIG_LIBSSCL_DEBUG_CO
|
|
|
|
|
std::cout << "final_suspend" << ": " << std::this_thread::get_id()
|
|
|
|
|
<< " Non-viral: posting callerLambda completion to callerIoContext.\n";
|
|
|
|
|
#endif
|
|
|
|
|
boost::asio::post(
|
|
|
|
|
calleePromise.callerIoContext,
|
|
|
|
|
[&calleeRef = calleePromise]()
|
|
|
|
|
{
|
|
|
|
|
if (calleeRef.returnValues.myExceptionPtr) {
|
|
|
|
|
std::rethrow_exception(calleeRef.returnValues.myExceptionPtr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
calleeRef.callerLambda();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
#ifdef CONFIG_LIBSSCL_DEBUG_CO
|
|
|
|
|
std::cout << "final_suspend" << ": " << std::this_thread::get_id()
|
|
|
|
|
<< " Viral: running CalleeFlowExecutor.\n";
|
|
|
|
|
#endif
|
|
|
|
|
calleePromise.postBackStatus.getCalleeFlowExecutor()();
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PostingPromise &calleePromise;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
PostingPromise() noexcept
|
|
|
|
|
: returnValues(), postBackStatus(*this)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
template <typename... TailArgs>
|
|
|
|
|
PostingPromise(
|
|
|
|
|
std::exception_ptr &_callerExceptionPtr,
|
|
|
|
|
std::function<void()> _callerLambda,
|
|
|
|
|
TailArgs &&...) noexcept
|
|
|
|
|
: returnValues(_callerExceptionPtr),
|
|
|
|
|
callerLambda(std::move(_callerLambda)),
|
|
|
|
|
postBackStatus(*this)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
~PostingPromise() noexcept
|
|
|
|
|
{
|
|
|
|
|
#ifdef CONFIG_LIBSSCL_DEBUG_CO
|
|
|
|
|
std::cout << __func__ << ": " << std::this_thread::get_id() << " Destructing.\n";
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void unhandled_exception() noexcept
|
|
|
|
|
{
|
|
|
|
|
returnValues.myExceptionPtr = std::current_exception();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void removeAcquiredLock(CoQutex &coQutex) noexcept override
|
|
|
|
|
{
|
|
|
|
|
eraseFirstMatchingAcquiredLock(coQutex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const PromiseChainLink *callerPromiseChainLink() const noexcept override
|
|
|
|
|
{ return callerChainLink; }
|
|
|
|
|
|
|
|
|
|
PromiseChainLink *callerPromiseChainLink() noexcept override
|
|
|
|
|
{ return callerChainLink; }
|
|
|
|
|
|
|
|
|
|
/** Non-viral: post completion lambda to callerIoContext from this thread.
|
|
|
|
|
* Viral: run CalleeFlowExecutor (handshake flags); caller may post caller resume
|
|
|
|
|
* later via CallerFlowExecutor. See docs/caller-posts-to-own-io-context.md.
|
|
|
|
|
* Work runs in FinalSuspendPostingInvoker::await_suspend after the suspend point
|
|
|
|
|
* advances (see docs/prompts/post-to-and-back-in-invokables.md).
|
|
|
|
|
*/
|
|
|
|
|
auto final_suspend() noexcept
|
|
|
|
|
{
|
|
|
|
|
return FinalSuspendPostingInvoker(*this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ReturnValues<T> returnValues;
|
|
|
|
|
std::function<void()> callerLambda;
|
|
|
|
|
boost::asio::io_service &callerIoContext =
|
|
|
|
|
sscl::ComponentThread::getSelf()->getIoService();
|
|
|
|
|
std::coroutine_handle<> selfSchedHandle;
|
|
|
|
|
std::coroutine_handle<void> callerSchedHandle;
|
|
|
|
|
PromiseChainLink *callerChainLink = nullptr;
|
|
|
|
|
PostBackStatus postBackStatus;
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
void setSelfSchedHandle(std::coroutine_handle<> schedHandle) noexcept
|
|
|
|
|
{
|
|
|
|
|
selfSchedHandle = schedHandle;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void setCallerPromiseChainLink(PromiseChainLink *chainLink) noexcept
|
|
|
|
|
{
|
|
|
|
|
callerChainLink = chainLink;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename, typename>
|
|
|
|
|
friend class PostingInvoker;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename ThreadTag>
|
|
|
|
|
struct TaggedPostingPromise
|
|
|
|
|
: public PostingPromise<T>,
|
|
|
|
|
public PostingPromiseReturnOps<TaggedPostingPromise<T, ThreadTag>, T>
|
|
|
|
|
{
|
|
|
|
|
TaggedPostingPromise() noexcept
|
|
|
|
|
: PostingPromise<T>()
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
template <typename... TailArgs>
|
|
|
|
|
TaggedPostingPromise(
|
|
|
|
|
std::exception_ptr &_exceptionPtr,
|
|
|
|
|
std::function<void()> _callerLambda,
|
|
|
|
|
TailArgs &&... tailArgs) noexcept
|
|
|
|
|
: PostingPromise<T>(
|
|
|
|
|
_exceptionPtr,
|
|
|
|
|
std::move(_callerLambda),
|
|
|
|
|
std::forward<TailArgs>(tailArgs)...)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
auto initial_suspend() noexcept
|
|
|
|
|
{
|
|
|
|
|
#ifdef CONFIG_LIBSSCL_DEBUG_CO
|
|
|
|
|
std::cout << __func__ << ": " << std::this_thread::get_id() << " About to post selfSchedHandle to " << typeid(ThreadTag).name() << ".\n";
|
|
|
|
|
std::cout << __func__ << ": " << std::this_thread::get_id() << " Returning InitialSuspendPostingInvoker.\n";
|
|
|
|
|
#endif
|
|
|
|
|
return typename PostingPromise<T>::InitialSuspendPostingInvoker(
|
2026-05-17 17:12:11 -04:00
|
|
|
ThreadTag::io_service(),
|
2026-05-17 16:52:04 -04:00
|
|
|
this->selfSchedHandle);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace sscl::co
|
|
|
|
|
|
|
|
|
|
#endif // PROMISES_H
|