Gaia-ECS v0.9.3
A simple and powerful entity component system
Loading...
Searching...
No Matches
sched.h
1#pragma once
2
3#include "gaia/config/config.h"
4
5#include <cstdint>
6
7#include "gaia/cnt/darray.h"
8#include "gaia/mem/smallblock_allocator.h"
9#include "gaia/mt/jobcommon.h"
10#include "gaia/mt/jobhandle.h"
11#include "gaia/mt/threadpool.h"
12
13namespace gaia {
14 namespace ecs {
15 enum class QueryExecType : uint32_t;
16
19 struct SchedToken {
21 uintptr_t value[2]{};
22 };
23
25 enum class SchedFlags : uint8_t {
27 Default = 0,
29 Background = 0x01
30 };
31
36 GAIA_NODISCARD inline bool sched_flags_has(SchedFlags flags, SchedFlags flag) {
37 return ((uint8_t)flags & (uint8_t)flag) != 0U;
38 }
39
43 void* pCtx = nullptr;
45 void (*invoke)(void* pCtx) = nullptr;
47 QueryExecType execType{};
49 SchedFlags flags = SchedFlags::Default;
50 };
51
53 struct SchedParDesc {
55 void* pCtx = nullptr;
57 void (*invoke)(void* pCtx, uint32_t idxStart, uint32_t idxEnd) = nullptr;
59 uint32_t itemCount = 0;
61 uint32_t groupSize = 0;
63 QueryExecType execType{};
65 SchedFlags flags = SchedFlags::Default;
66 };
67
71 struct Sched {
73 void* pCtx = nullptr;
78 SchedToken (*sched)(void* pCtx, const SchedTaskDesc* pDesc) = nullptr;
83 SchedToken (*sched_par)(void* pCtx, const SchedParDesc* pDesc) = nullptr;
88 SchedToken (*add)(void* pCtx, const SchedTaskDesc* pDesc) = nullptr;
93 SchedToken (*add_par)(void* pCtx, const SchedParDesc* pDesc) = nullptr;
97 void (*submit)(void* pCtx, SchedToken token) = nullptr;
102 void (*dep)(void* pCtx, SchedToken tokenFirst, SchedToken tokenSecond) = nullptr;
106 void (*wait)(void* pCtx, SchedToken token) = nullptr;
110 void (*del)(void* pCtx, SchedToken token) = nullptr;
111 };
112
118 class SchedJob {
119 Sched m_sched;
120 SchedToken m_token{};
121 void* m_pCleanupCtx = nullptr;
122 void (*m_cleanup)(void* pCtx) = nullptr;
123 bool m_valid = false;
124 bool m_submitted = false;
125 bool m_waited = false;
126
127 void cleanup() {
128 if (m_cleanup != nullptr) {
129 m_cleanup(m_pCleanupCtx);
130 m_cleanup = nullptr;
131 m_pCleanupCtx = nullptr;
132 }
133 }
134
135 GAIA_NODISCARD static bool same_sched(const Sched& a, const Sched& b) {
136 return a.pCtx == b.pCtx && a.sched == b.sched && a.sched_par == b.sched_par && a.add == b.add &&
137 a.add_par == b.add_par && a.submit == b.submit && a.dep == b.dep && a.wait == b.wait && a.del == b.del;
138 }
139
140 public:
142 SchedJob() = default;
149 SchedJob(Sched sched, SchedToken token, bool submitted, void* pCleanupCtx, void (*cleanup)(void* pCtx)):
150 m_sched(sched), m_token(token), m_pCleanupCtx(pCleanupCtx), m_cleanup(cleanup), m_valid(true),
151 m_submitted(submitted) {}
152
155 del();
156 }
157
158 SchedJob(const SchedJob&) = delete;
159 SchedJob& operator=(const SchedJob&) = delete;
160
163 SchedJob(SchedJob&& other) noexcept:
164 m_sched(other.m_sched), m_token(other.m_token), m_pCleanupCtx(other.m_pCleanupCtx),
165 m_cleanup(other.m_cleanup), m_valid(other.m_valid), m_submitted(other.m_submitted), m_waited(other.m_waited) {
166 other.m_valid = false;
167 other.m_cleanup = nullptr;
168 other.m_pCleanupCtx = nullptr;
169 }
170
174 SchedJob& operator=(SchedJob&& other) noexcept {
175 if (this == &other)
176 return *this;
177 del();
178 m_sched = other.m_sched;
179 m_token = other.m_token;
180 m_pCleanupCtx = other.m_pCleanupCtx;
181 m_cleanup = other.m_cleanup;
182 m_valid = other.m_valid;
183 m_submitted = other.m_submitted;
184 m_waited = other.m_waited;
185 other.m_valid = false;
186 other.m_cleanup = nullptr;
187 other.m_pCleanupCtx = nullptr;
188 return *this;
189 }
190
193 GAIA_NODISCARD bool valid() const {
194 return m_valid;
195 }
196
199 GAIA_NODISCARD SchedToken token() const {
200 return m_token;
201 }
202
204 void submit();
215 void dep(const SchedJob& jobFirst);
218 void wait();
220 void del();
221 };
222
223 namespace detail {
224 inline mt::JobPriority exec_prio(QueryExecType execType) {
225 // QueryExecType::ParallelEff is encoded as value 3. Keep the scheduler bridge independent
226 // from the enum definition point so this header stays C-like and forward-declarable.
227 return (uint32_t)execType == 3U ? mt::JobPriority::Low : mt::JobPriority::High;
228 }
229
230 inline bool sched_flags_background(SchedFlags flags) {
231 return sched_flags_has(flags, SchedFlags::Background);
232 }
233
234 inline mt::JobCreationFlags job_creation_flags(SchedFlags flags) {
235 uint8_t jobFlags = (uint8_t)mt::JobCreationFlags::ManualDelete;
236 if (sched_flags_background(flags))
237 jobFlags |= (uint8_t)mt::JobCreationFlags::Background;
238 return (mt::JobCreationFlags)jobFlags;
239 }
240
241 enum class SchedTokenKind : uint32_t { None, Single, Parallel };
242
245 SchedTokenKind kind = SchedTokenKind::None;
246 bool submitted = false;
247
248 GAIA_USE_SMALLBLOCK(SchedTokenDefData)
249 };
250
251 inline SchedToken make_sched_token(SchedTokenDefData* pData) {
252 SchedToken token{};
253 token.value[0] = (uintptr_t)pData;
254 return token;
255 }
256
257 inline SchedTokenDefData* sched_token_data(SchedToken token) {
258 return reinterpret_cast<SchedTokenDefData*>(token.value[0]);
259 }
260
261 inline SchedToken add_one_def([[maybe_unused]] void* pCtx, const SchedTaskDesc* pDesc) {
262 GAIA_ASSERT(pDesc != nullptr);
263 GAIA_ASSERT(pDesc->invoke != nullptr);
264 if (pDesc == nullptr || pDesc->invoke == nullptr)
265 return {};
266
267 auto* pData = new SchedTokenDefData();
268 pData->kind = SchedTokenKind::Single;
269 pData->handles.resize(1);
270
271 mt::Job job;
272 job.priority = exec_prio(pDesc->execType);
273 job.flags = job_creation_flags(pDesc->flags);
274 job.func = [pCtx = pDesc->pCtx, invoke = pDesc->invoke]() {
275 invoke(pCtx);
276 };
277
278 pData->handles[0] = mt::ThreadPool::get().add(GAIA_MOV(job));
279 return make_sched_token(pData);
280 }
281
282 inline SchedToken sched_one_def([[maybe_unused]] void* pCtx, const SchedTaskDesc* pDesc) {
283 auto token = add_one_def(pCtx, pDesc);
284 auto* pData = sched_token_data(token);
285 if (pData != nullptr) {
286 mt::ThreadPool::get().submit(pData->handles[0]);
287 pData->submitted = true;
288 }
289 return token;
290 }
291
292 inline SchedToken add_par_def([[maybe_unused]] void* pCtx, const SchedParDesc* pDesc) {
293 GAIA_ASSERT(pDesc != nullptr);
294 GAIA_ASSERT(pDesc->invoke != nullptr);
295 if (pDesc == nullptr || pDesc->invoke == nullptr || pDesc->itemCount == 0)
296 return {};
297
298 auto& tp = mt::ThreadPool::get();
299 const auto prio = exec_prio(pDesc->execType);
300 uint32_t groupSize = pDesc->groupSize;
301 if (groupSize == 0) {
302 const bool background = sched_flags_background(pDesc->flags);
303 const auto workers =
304 background ? core::get_max(1U, tp.background_workers()) : core::get_max(1U, tp.workers() + 1U);
305 groupSize = (pDesc->itemCount + workers - 1) / workers;
306 constexpr uint32_t maxUnitsOfWorkPerGroup = 8;
307 groupSize = groupSize / maxUnitsOfWorkPerGroup;
308 if (groupSize == 0)
309 groupSize = 1;
310 }
311
312 const auto jobs = (pDesc->itemCount + groupSize - 1) / groupSize;
313 auto* pData = new SchedTokenDefData();
314 pData->kind = SchedTokenKind::Parallel;
315 pData->handles.resize(jobs + 1);
316
317 for (uint32_t jobIndex = 0; jobIndex < jobs; ++jobIndex) {
318 const uint32_t idxStart = jobIndex * groupSize;
319 const uint32_t idxEnd = core::get_min(idxStart + groupSize, pDesc->itemCount);
320
321 mt::Job job;
322 job.priority = prio;
323 job.flags = job_creation_flags(pDesc->flags);
324 job.func = [desc = *pDesc, idxStart, idxEnd]() {
325 desc.invoke(desc.pCtx, idxStart, idxEnd);
326 };
327 pData->handles[jobIndex] = tp.add(GAIA_MOV(job));
328 }
329 {
330 mt::Job syncJob;
331 syncJob.priority = prio;
332 syncJob.flags = job_creation_flags(pDesc->flags);
333 pData->handles[jobs] = tp.add(GAIA_MOV(syncJob));
334 }
335 tp.dep(std::span(pData->handles.data(), jobs), pData->handles[jobs]);
336 return make_sched_token(pData);
337 }
338
339 inline SchedToken sched_par_def([[maybe_unused]] void* pCtx, const SchedParDesc* pDesc) {
340 auto token = add_par_def(pCtx, pDesc);
341 auto* pData = sched_token_data(token);
342 if (pData != nullptr) {
343 mt::ThreadPool::get().submit(std::span(pData->handles.data(), pData->handles.size()));
344 pData->submitted = true;
345 }
346 return token;
347 }
348
349 inline void sched_submit_def([[maybe_unused]] void* pCtx, SchedToken token) {
350 auto* pData = sched_token_data(token);
351 if (pData == nullptr || pData->submitted)
352 return;
353 auto& tp = mt::ThreadPool::get();
354 tp.submit(std::span(pData->handles.data(), pData->handles.size()));
355 pData->submitted = true;
356 }
357
358 inline void sched_dep_def([[maybe_unused]] void* pCtx, SchedToken tokenFirst, SchedToken tokenSecond) {
359 auto* pFirst = sched_token_data(tokenFirst);
360 auto* pSecond = sched_token_data(tokenSecond);
361 if (pFirst == nullptr || pSecond == nullptr || pFirst->handles.empty() || pSecond->handles.empty())
362 return;
363 auto& tp = mt::ThreadPool::get();
364 const auto firstDone = pFirst->handles.back();
365 if (pSecond->kind == SchedTokenKind::Parallel && pSecond->handles.size() > 1) {
366 const auto childCount = (uint32_t)pSecond->handles.size() - 1;
367 for (uint32_t i = 0; i < childCount; ++i)
368 tp.dep(firstDone, pSecond->handles[i]);
369 return;
370 }
371 tp.dep(firstDone, pSecond->handles.back());
372 }
373
374 inline void sched_wait_def([[maybe_unused]] void* pCtx, SchedToken token) {
375 auto* pData = sched_token_data(token);
376 if (pData == nullptr || pData->handles.empty())
377 return;
378 mt::ThreadPool::get().wait(pData->handles.back());
379 }
380
381 inline void sched_del_def([[maybe_unused]] void* pCtx, [[maybe_unused]] SchedToken token) {
382 auto* pData = sched_token_data(token);
383 if (pData == nullptr)
384 return;
385 auto& tp = mt::ThreadPool::get();
386 for (auto handle: pData->handles) {
387 if (handle != mt::JobNull)
388 tp.del(handle);
389 }
390 delete pData;
391 }
392 } // namespace detail
393
396 GAIA_NODISCARD inline const Sched& sched_def() {
397 static const Sched sched = [] {
398 Sched b{};
399 b.sched = &detail::sched_one_def;
400 b.sched_par = &detail::sched_par_def;
401 b.add = &detail::add_one_def;
402 b.add_par = &detail::add_par_def;
403 b.submit = &detail::sched_submit_def;
404 b.dep = &detail::sched_dep_def;
405 b.wait = &detail::sched_wait_def;
406 b.del = &detail::sched_del_def;
407 return b;
408 }();
409 return sched;
410 }
411
415 GAIA_NODISCARD inline const Sched& sched_resolve(const Sched& sched) {
416 if (sched.sched == nullptr && sched.sched_par == nullptr && sched.add == nullptr && sched.add_par == nullptr &&
417 sched.submit == nullptr && sched.dep == nullptr && sched.wait == nullptr && sched.del == nullptr)
418 return sched_def();
419 return sched;
420 }
421
426 GAIA_NODISCARD inline SchedToken sched_one(const Sched& sched, const SchedTaskDesc& desc) {
427 const auto& resolved = sched_resolve(sched);
428 if (resolved.sched != nullptr)
429 return resolved.sched(resolved.pCtx, &desc);
430 GAIA_ASSERT(resolved.add != nullptr);
431 GAIA_ASSERT(resolved.submit != nullptr);
432 const auto token = resolved.add != nullptr ? resolved.add(resolved.pCtx, &desc) : SchedToken{};
433 if (resolved.submit != nullptr)
434 resolved.submit(resolved.pCtx, token);
435 return token;
436 }
437
445 GAIA_NODISCARD inline SchedJob
446 sched_add(const Sched& sched, const SchedTaskDesc& desc, void* pCleanupCtx, void (*cleanup)(void* pCtx)) {
447 const auto& resolved = sched_resolve(sched);
448 if (resolved.add != nullptr) {
449 if (resolved.submit == nullptr || resolved.wait == nullptr || resolved.del == nullptr) {
450 GAIA_ASSERT(false);
451 if (cleanup != nullptr)
452 cleanup(pCleanupCtx);
453 return {};
454 }
455 return SchedJob(resolved, resolved.add(resolved.pCtx, &desc), false, pCleanupCtx, cleanup);
456 }
457
458 if (resolved.sched == nullptr || resolved.wait == nullptr || resolved.del == nullptr) {
459 GAIA_ASSERT(false);
460 if (cleanup != nullptr)
461 cleanup(pCleanupCtx);
462 return {};
463 }
464 return SchedJob(resolved, resolved.sched(resolved.pCtx, &desc), true, pCleanupCtx, cleanup);
465 }
466
471 GAIA_NODISCARD inline SchedToken sched_par(const Sched& sched, const SchedParDesc& desc) {
472 const auto& resolved = sched_resolve(sched);
473 if (resolved.sched_par != nullptr)
474 return resolved.sched_par(resolved.pCtx, &desc);
475 GAIA_ASSERT(resolved.add_par != nullptr);
476 GAIA_ASSERT(resolved.submit != nullptr);
477 const auto token = resolved.add_par != nullptr ? resolved.add_par(resolved.pCtx, &desc) : SchedToken{};
478 if (resolved.submit != nullptr)
479 resolved.submit(resolved.pCtx, token);
480 return token;
481 }
482
490 GAIA_NODISCARD inline SchedJob
491 sched_add_par(const Sched& sched, const SchedParDesc& desc, void* pCleanupCtx, void (*cleanup)(void* pCtx)) {
492 const auto& resolved = sched_resolve(sched);
493 if (resolved.add_par != nullptr) {
494 if (resolved.submit == nullptr || resolved.wait == nullptr || resolved.del == nullptr) {
495 GAIA_ASSERT(false);
496 if (cleanup != nullptr)
497 cleanup(pCleanupCtx);
498 return {};
499 }
500 return SchedJob(resolved, resolved.add_par(resolved.pCtx, &desc), false, pCleanupCtx, cleanup);
501 }
502
503 if (resolved.sched_par == nullptr || resolved.wait == nullptr || resolved.del == nullptr) {
504 GAIA_ASSERT(false);
505 if (cleanup != nullptr)
506 cleanup(pCleanupCtx);
507 return {};
508 }
509 return SchedJob(resolved, resolved.sched_par(resolved.pCtx, &desc), true, pCleanupCtx, cleanup);
510 }
511
515 inline void sched_submit(const Sched& sched, SchedToken token) {
516 const auto& resolved = sched_resolve(sched);
517 if (resolved.submit != nullptr)
518 resolved.submit(resolved.pCtx, token);
519 }
520
525 inline void sched_dep(const Sched& sched, SchedToken tokenFirst, SchedToken tokenSecond) {
526 const auto& resolved = sched_resolve(sched);
527 if (resolved.dep != nullptr)
528 resolved.dep(resolved.pCtx, tokenFirst, tokenSecond);
529 }
530
534 inline void sched_wait(const Sched& sched, SchedToken token) {
535 const auto& resolved = sched_resolve(sched);
536 if (resolved.wait != nullptr)
537 resolved.wait(resolved.pCtx, token);
538 }
539
543 inline void sched_del(const Sched& sched, SchedToken token) {
544 const auto& resolved = sched_resolve(sched);
545 if (resolved.del != nullptr)
546 resolved.del(resolved.pCtx, token);
547 }
548
549 inline void SchedJob::submit() {
550 if (!m_valid || m_submitted)
551 return;
552 GAIA_ASSERT(m_sched.submit != nullptr);
553 if (m_sched.submit == nullptr)
554 return;
555 sched_submit(m_sched, m_token);
556 m_submitted = true;
557 }
558
559 inline void SchedJob::dep(const SchedJob& jobFirst) {
560 if (!m_valid || !jobFirst.m_valid)
561 return;
562 GAIA_ASSERT(same_sched(m_sched, jobFirst.m_sched));
563 GAIA_ASSERT(!m_submitted && !jobFirst.m_submitted);
564 if (!same_sched(m_sched, jobFirst.m_sched) || m_submitted || jobFirst.m_submitted)
565 return;
566 sched_dep(m_sched, jobFirst.m_token, m_token);
567 }
568
569 inline void SchedJob::wait() {
570 if (!m_valid || m_waited)
571 return;
572 GAIA_ASSERT(m_submitted);
573 if (!m_submitted)
574 return;
575 sched_wait(m_sched, m_token);
576 m_waited = true;
577 cleanup();
578 }
579
580 inline void SchedJob::del() {
581 if (!m_valid)
582 return;
583 if (m_submitted && !m_waited)
584 wait();
585 sched_del(m_sched, m_token);
586 cleanup();
587 m_valid = false;
588 m_submitted = false;
589 m_waited = false;
590 }
591 } // namespace ecs
592} // namespace gaia
Array with variable size of elements of type.
Definition darray_impl.h:25
Definition span_impl.h:99
Move-only wrapper for scheduler-owned ECS work.
Definition sched.h:118
GAIA_NODISCARD bool valid() const
Returns true if this wrapper owns scheduler work.
Definition sched.h:193
void submit()
Submits the added work if it has not been submitted yet.
Definition sched.h:549
SchedJob & operator=(SchedJob &&other) noexcept
Move-assigns a scheduler job wrapper.
Definition sched.h:174
GAIA_NODISCARD SchedToken token() const
Returns the opaque scheduler token.
Definition sched.h:199
SchedJob()=default
Creates an empty wrapper.
void del()
Deletes scheduler resources and runs wrapper cleanup.
Definition sched.h:580
SchedJob(Sched sched, SchedToken token, bool submitted, void *pCleanupCtx, void(*cleanup)(void *pCtx))
Creates a wrapper around token from sched.
Definition sched.h:149
void wait()
Waits for submitted work to complete.
Definition sched.h:569
~SchedJob()
Waits for and deletes a still-owned token.
Definition sched.h:154
void dep(const SchedJob &jobFirst)
Adds a dependency edge so this job runs after jobFirst.
Definition sched.h:559
SchedJob(SchedJob &&other) noexcept
Move-constructs a scheduler job wrapper.
Definition sched.h:163
Checks if endianess was detected correctly at compile-time.
Definition bitset.h:9
Description of a parallel-for submission to a scheduler.
Definition sched.h:53
void * pCtx
Opaque callback context forwarded to invoke().
Definition sched.h:55
void(* invoke)(void *pCtx, uint32_t idxStart, uint32_t idxEnd)
Parallel-for entry point receiving a half-open item range [idxStart, idxEnd).
Definition sched.h:57
uint32_t groupSize
Preferred group size. A value of 0 lets the scheduler choose.
Definition sched.h:61
SchedFlags flags
Scheduler flags describing non-default execution requirements.
Definition sched.h:65
QueryExecType execType
Execution hint selected by the scheduler caller.
Definition sched.h:63
uint32_t itemCount
Total number of items to process.
Definition sched.h:59
Description of a single task submitted to a scheduler.
Definition sched.h:41
SchedFlags flags
Scheduler flags describing non-default execution requirements.
Definition sched.h:49
void(* invoke)(void *pCtx)
Task entry point.
Definition sched.h:45
QueryExecType execType
Execution hint selected by the scheduler caller.
Definition sched.h:47
void * pCtx
Opaque callback context forwarded to invoke().
Definition sched.h:43
Opaque synchronization token returned by a scheduler. The scheduler owns the meaning of the payload.
Definition sched.h:19
uintptr_t value[2]
Scheduler-defined token payload.
Definition sched.h:21
Scheduler descriptor used by ECS runtime code. All callbacks may be null when the descriptor is only ...
Definition sched.h:71
SchedToken(* add_par)(void *pCtx, const SchedParDesc *pDesc)
Adds a parallel-for workload without submitting it for execution.
Definition sched.h:93
void * pCtx
Opaque scheduler-owned context passed back to every callback.
Definition sched.h:73
void(* dep)(void *pCtx, SchedToken tokenFirst, SchedToken tokenSecond)
Adds a dependency edge so tokenSecond runs after tokenFirst.
Definition sched.h:102
void(* wait)(void *pCtx, SchedToken token)
Waits until the scheduled work referenced by token finishes.
Definition sched.h:106
void(* del)(void *pCtx, SchedToken token)
Deletes any scheduler-owned resources associated with token.
Definition sched.h:110
SchedToken(* sched)(void *pCtx, const SchedTaskDesc *pDesc)
Schedules one task for execution.
Definition sched.h:78
void(* submit)(void *pCtx, SchedToken token)
Submits a previously added token.
Definition sched.h:97
SchedToken(* add)(void *pCtx, const SchedTaskDesc *pDesc)
Adds one task without submitting it for execution.
Definition sched.h:88
SchedToken(* sched_par)(void *pCtx, const SchedParDesc *pDesc)
Schedules a parallel-for workload for execution.
Definition sched.h:83