CodePorting.Translator Cs2Cpp
CodePorting.Translator.Cs2Cpp.Framework
async.h
1
3#pragma once
4
5#include <functional>
6#include <system/object_ext.h>
7#include <system/action.h>
8#include <system/threading/tasks/result_task.h>
9#include <system/threading/tasks/task_status.h>
10#include <system/details/dereference.h>
11
12namespace System { namespace Details {
13
17struct ASPOSECPP_SHARED_CLASS AsyncContextBase
18{
20 AsyncContextBase() : stage(0), m_executing(false) {}
22 AsyncContextBase(const AsyncContextBase& other) : AsyncContextBase() {}
23
25 ASPOSECPP_SHARED_API void Continue();
33 template<typename T>
34 bool Await(const T& awaitee, int32_t step)
35 {
36 return AwaitImpl(awaitee, step, [](const auto& awaiter) { awaiter.GetResult(); });
37 }
47 template <typename T, typename V>
48 bool Await(const T& awaitee, V& value, int32_t step)
49 {
50 return AwaitImpl(awaitee, step, [&value](const auto& awaiter) { value = ExplicitCast<V>(awaiter.GetResult()); });
51 }
52
54 ASPOSECPP_SHARED_API void operator()();
55
57 int32_t stage;
58
59protected:
61 using Continuer = std::function<void()>;
62
64 virtual Threading::Tasks::Task* get_Task() const = 0;
66 virtual void Execute() = 0;
75 template <typename AwaiteeT, typename ContinuerT>
76 bool AwaitImpl(const AwaiteeT& awaitee, int32_t stage, const ContinuerT& continuer)
77 {
78 // Getting awaiter instance
79 auto awaiter = Dereference(awaitee).GetAwaiter();
80
81 // Trying to avoid sleeping if awaiter completed already
82 if (awaiter.get_IsCompleted())
83 {
84 continuer(awaiter);
85 return false;
86 }
87
88 // Going to suspend
89 Suspend([awaiter, continuer]() { continuer(awaiter); }, stage);
90
91 // Getting resume context (we need to get it here because resuming should be executed in other context)
92 auto continuationContext = GetResumeScheduler(awaiter.continueOnCapturedContext);
93
94 // Setting up resume action
95 awaiter.OnCompleted(Action<>([this, continuationContext] { Resume(continuationContext); }));
96
97 return true;
98 }
100 ASPOSECPP_SHARED_API void Suspend(Continuer&& continuer, int32_t stage);
102 ASPOSECPP_SHARED_API SharedPtr<Threading::Tasks::TaskScheduler> GetResumeScheduler(bool continueOnCapturedContext) const;
104 ASPOSECPP_SHARED_API void Resume(const SharedPtr<Threading::Tasks::TaskScheduler>& context);
105
107 Continuer m_continuer;
109 bool m_executing;
111 SharedPtr<Threading::Tasks::TaskScheduler> m_activator;
113 std::mutex m_state_mutex;
114};
115
118using AsyncFunction = std::function<void(struct AsyncContext&)>;
119
122struct AsyncContext : AsyncContextBase
123{
126 AsyncContext(Threading::Tasks::Task* task, const AsyncFunction& fnc) : m_task(task), m_function(fnc)
127 {}
128
130 ASPOSECPP_SHARED_API void Return();
131
132protected:
134 Threading::Tasks::Task* get_Task() const override {return m_task;}
136 void Execute() override {m_function(*this);}
137
139 Threading::Tasks::Task* m_task;
141 AsyncFunction m_function;
142};
143
144template<typename T>
145struct ResultAsyncContext;
146
151template <typename T>
152using ResultAsyncFunction = std::function<void(ResultAsyncContext<T>&)>;
153
157template<typename T>
158struct ResultAsyncContext : AsyncContextBase
159{
162 ResultAsyncContext(Threading::Tasks::ResultTask<T>* task, const ResultAsyncFunction<T>& fnc) : m_result_task(task), m_function(fnc)
163 {}
164
169 void Return(const T& result)
170 {
171 m_result_task->set_Result(result);
172 m_result_task->set_Status(Threading::Tasks::TaskStatus::RanToCompletion);
173 }
174
175protected:
177 Threading::Tasks::Task* get_Task() const override {return m_result_task;}
179 void Execute() override {m_function(*this);}
180
182 Threading::Tasks::ResultTask<T>* m_result_task;
184 ResultAsyncFunction<T> m_function;
185};
186
187} // namespace Details
188
189ASPOSECPP_SHARED_API TaskPtr MakeAsync(const Details::AsyncFunction& fnc);
190
191template<typename T>
192RTaskPtr<T> MakeAsync(const Details::ResultAsyncFunction<T>& fnc)
193{
194 auto result = System::MakeObject<Threading::Tasks::ResultTask<T>>();
195 result->set_Function(Details::ResultAsyncContext<T>(result.GetPointer(), fnc));
196 result->RunSynchronously();
197 return result;
198}
199
200} // namespace System
void set_Function(const FunctionT &fnc)
Sets the internal function to execute.
Definition: task.h:131
@ Suspend
Suspension of a logical operation.
@ Resume
Resumption of a logical operation.
@ Continue
The 'Continue' status code that equals to HTTP status 100.
Definition: db_command.h:9
TaskPtr MakeAsync(const Details::AsyncFunction &fnc)