blob: e114db6886027c337118b82a7f6eb17c8a758296 [file] [log] [blame]
Jordan Baylesb0c191e2019-03-26 15:49:57 -07001// Copyright 2019 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
btolschc92ba2f2019-04-10 11:46:01 -07005#ifndef PLATFORM_API_TASK_RUNNER_H_
6#define PLATFORM_API_TASK_RUNNER_H_
Jordan Baylesb0c191e2019-03-26 15:49:57 -07007
Yuri Wiitala75ea15d2019-12-03 16:01:48 -08008#include <future> // NOLINT
9#include <utility>
Jordan Baylesb0c191e2019-03-26 15:49:57 -070010
11#include "platform/api/time.h"
12
13namespace openscreen {
Jordan Baylesb0c191e2019-03-26 15:49:57 -070014
15// A thread-safe API surface that allows for posting tasks. The underlying
16// implementation may be single or multi-threaded, and all complication should
mark a. foltz39dda172020-02-05 16:44:03 -080017// be handled by the implementation class. The implementation must guarantee:
Jordan Baylesb0c191e2019-03-26 15:49:57 -070018// (1) Tasks shall not overlap in time/CPU.
19// (2) Tasks shall run sequentially, e.g. posting task A then B implies
20// that A shall run before B.
mark a. foltz39dda172020-02-05 16:44:03 -080021// (3) If task A is posted before task B, then any mutation in A happens-before
22// B runs (even if A and B run on different threads).
Jordan Baylesb0c191e2019-03-26 15:49:57 -070023class TaskRunner {
24 public:
Yuri Wiitalab929b832019-06-05 17:13:15 -070025 using Task = std::packaged_task<void() noexcept>;
Jordan Baylesb0c191e2019-03-26 15:49:57 -070026
27 virtual ~TaskRunner() = default;
28
Yuri Wiitalab929b832019-06-05 17:13:15 -070029 // Takes any callable target (function, lambda-expression, std::bind result,
30 // etc.) that should be run at the first convenient time.
31 template <typename Functor>
32 inline void PostTask(Functor f) {
33 PostPackagedTask(Task(std::move(f)));
34 }
Jordan Baylesb0c191e2019-03-26 15:49:57 -070035
Yuri Wiitalab929b832019-06-05 17:13:15 -070036 // Takes any callable target (function, lambda-expression, std::bind result,
37 // etc.) that should be run no sooner than |delay| time from now. Note that
38 // the Task might run after an additional delay, especially under heavier
39 // system load. There is no deadline concept.
40 template <typename Functor>
41 inline void PostTaskWithDelay(Functor f, Clock::duration delay) {
42 PostPackagedTaskWithDelay(Task(std::move(f)), delay);
43 }
44
45 // Implementations should provide the behavior explained in the comments above
46 // for PostTask[WithDelay](). Client code may also call these directly when
47 // passing an existing Task object.
48 virtual void PostPackagedTask(Task task) = 0;
49 virtual void PostPackagedTaskWithDelay(Task task, Clock::duration delay) = 0;
Max Yakimakhabf567dc2019-09-20 13:37:04 -070050
51 // Return true if the calling thread is the thread that task runner is using
52 // to run tasks, false otherwise.
mark a. foltzc4a3e082020-02-07 15:19:31 -080053 virtual bool IsRunningOnTaskRunner() = 0;
Jordan Baylesb0c191e2019-03-26 15:49:57 -070054};
Yuri Wiitalab929b832019-06-05 17:13:15 -070055
Jordan Baylesb0c191e2019-03-26 15:49:57 -070056} // namespace openscreen
57
btolschc92ba2f2019-04-10 11:46:01 -070058#endif // PLATFORM_API_TASK_RUNNER_H_