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