xref: /aosp_15_r20/external/openscreen/platform/api/task_runner.h (revision 3f982cf4871df8771c9d4abe6e9a6f8d829b2736)
1*3f982cf4SFabien Sanglard // Copyright 2019 The Chromium Authors. All rights reserved.
2*3f982cf4SFabien Sanglard // Use of this source code is governed by a BSD-style license that can be
3*3f982cf4SFabien Sanglard // found in the LICENSE file.
4*3f982cf4SFabien Sanglard 
5*3f982cf4SFabien Sanglard #ifndef PLATFORM_API_TASK_RUNNER_H_
6*3f982cf4SFabien Sanglard #define PLATFORM_API_TASK_RUNNER_H_
7*3f982cf4SFabien Sanglard 
8*3f982cf4SFabien Sanglard #include <future>
9*3f982cf4SFabien Sanglard #include <utility>
10*3f982cf4SFabien Sanglard 
11*3f982cf4SFabien Sanglard #include "platform/api/time.h"
12*3f982cf4SFabien Sanglard 
13*3f982cf4SFabien Sanglard namespace openscreen {
14*3f982cf4SFabien Sanglard 
15*3f982cf4SFabien Sanglard // A thread-safe API surface that allows for posting tasks. The underlying
16*3f982cf4SFabien Sanglard // implementation may be single or multi-threaded, and all complication should
17*3f982cf4SFabien Sanglard // be handled by the implementation class. The implementation must guarantee:
18*3f982cf4SFabien Sanglard // (1) Tasks shall not overlap in time/CPU.
19*3f982cf4SFabien Sanglard // (2) Tasks shall run sequentially, e.g. posting task A then B implies
20*3f982cf4SFabien Sanglard //     that A shall run before B.
21*3f982cf4SFabien Sanglard // (3) If task A is posted before task B, then any mutation in A happens-before
22*3f982cf4SFabien Sanglard //     B runs (even if A and B run on different threads).
23*3f982cf4SFabien Sanglard class TaskRunner {
24*3f982cf4SFabien Sanglard  public:
25*3f982cf4SFabien Sanglard // Seem to get an error using clang when compiling with -fno-exceptions:
26*3f982cf4SFabien Sanglard //   error: implicit instantiation of undefined template 'std::__1::packaged_task<void () noexcept>'
27*3f982cf4SFabien Sanglard #if __has_feature(cxx_exceptions)
28*3f982cf4SFabien Sanglard   using Task = std::packaged_task<void() noexcept>;
29*3f982cf4SFabien Sanglard #else
30*3f982cf4SFabien Sanglard   using Task = std::packaged_task<void()>;
31*3f982cf4SFabien Sanglard #endif
32*3f982cf4SFabien Sanglard 
33*3f982cf4SFabien Sanglard   virtual ~TaskRunner() = default;
34*3f982cf4SFabien Sanglard 
35*3f982cf4SFabien Sanglard   // Takes any callable target (function, lambda-expression, std::bind result,
36*3f982cf4SFabien Sanglard   // etc.) that should be run at the first convenient time.
37*3f982cf4SFabien Sanglard   template <typename Functor>
PostTask(Functor f)38*3f982cf4SFabien Sanglard   inline void PostTask(Functor f) {
39*3f982cf4SFabien Sanglard     PostPackagedTask(Task(std::move(f)));
40*3f982cf4SFabien Sanglard   }
41*3f982cf4SFabien Sanglard 
42*3f982cf4SFabien Sanglard   // Takes any callable target (function, lambda-expression, std::bind result,
43*3f982cf4SFabien Sanglard   // etc.) that should be run no sooner than |delay| time from now. Note that
44*3f982cf4SFabien Sanglard   // the Task might run after an additional delay, especially under heavier
45*3f982cf4SFabien Sanglard   // system load. There is no deadline concept.
46*3f982cf4SFabien Sanglard   template <typename Functor>
PostTaskWithDelay(Functor f,Clock::duration delay)47*3f982cf4SFabien Sanglard   inline void PostTaskWithDelay(Functor f, Clock::duration delay) {
48*3f982cf4SFabien Sanglard     PostPackagedTaskWithDelay(Task(std::move(f)), delay);
49*3f982cf4SFabien Sanglard   }
50*3f982cf4SFabien Sanglard 
51*3f982cf4SFabien Sanglard   // Implementations should provide the behavior explained in the comments above
52*3f982cf4SFabien Sanglard   // for PostTask[WithDelay](). Client code may also call these directly when
53*3f982cf4SFabien Sanglard   // passing an existing Task object.
54*3f982cf4SFabien Sanglard   virtual void PostPackagedTask(Task task) = 0;
55*3f982cf4SFabien Sanglard   virtual void PostPackagedTaskWithDelay(Task task, Clock::duration delay) = 0;
56*3f982cf4SFabien Sanglard 
57*3f982cf4SFabien Sanglard   // Return true if the calling thread is the thread that task runner is using
58*3f982cf4SFabien Sanglard   // to run tasks, false otherwise.
59*3f982cf4SFabien Sanglard   virtual bool IsRunningOnTaskRunner() = 0;
60*3f982cf4SFabien Sanglard };
61*3f982cf4SFabien Sanglard 
62*3f982cf4SFabien Sanglard }  // namespace openscreen
63*3f982cf4SFabien Sanglard 
64*3f982cf4SFabien Sanglard #endif  // PLATFORM_API_TASK_RUNNER_H_
65