xref: /aosp_15_r20/external/cronet/base/run_loop_unittest.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2016 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "base/run_loop.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <functional>
8*6777b538SAndroid Build Coastguard Worker #include <memory>
9*6777b538SAndroid Build Coastguard Worker #include <utility>
10*6777b538SAndroid Build Coastguard Worker 
11*6777b538SAndroid Build Coastguard Worker #include "base/containers/queue.h"
12*6777b538SAndroid Build Coastguard Worker #include "base/functional/bind.h"
13*6777b538SAndroid Build Coastguard Worker #include "base/functional/callback_helpers.h"
14*6777b538SAndroid Build Coastguard Worker #include "base/location.h"
15*6777b538SAndroid Build Coastguard Worker #include "base/memory/ptr_util.h"
16*6777b538SAndroid Build Coastguard Worker #include "base/memory/ref_counted.h"
17*6777b538SAndroid Build Coastguard Worker #include "base/synchronization/lock.h"
18*6777b538SAndroid Build Coastguard Worker #include "base/synchronization/waitable_event.h"
19*6777b538SAndroid Build Coastguard Worker #include "base/task/sequenced_task_runner.h"
20*6777b538SAndroid Build Coastguard Worker #include "base/task/single_thread_task_runner.h"
21*6777b538SAndroid Build Coastguard Worker #include "base/test/bind.h"
22*6777b538SAndroid Build Coastguard Worker #include "base/test/gtest_util.h"
23*6777b538SAndroid Build Coastguard Worker #include "base/test/scoped_run_loop_timeout.h"
24*6777b538SAndroid Build Coastguard Worker #include "base/test/task_environment.h"
25*6777b538SAndroid Build Coastguard Worker #include "base/test/test_timeouts.h"
26*6777b538SAndroid Build Coastguard Worker #include "base/threading/platform_thread.h"
27*6777b538SAndroid Build Coastguard Worker #include "base/threading/thread.h"
28*6777b538SAndroid Build Coastguard Worker #include "base/threading/thread_checker_impl.h"
29*6777b538SAndroid Build Coastguard Worker #include "build/build_config.h"
30*6777b538SAndroid Build Coastguard Worker #include "testing/gmock/include/gmock/gmock.h"
31*6777b538SAndroid Build Coastguard Worker #include "testing/gtest/include/gtest/gtest.h"
32*6777b538SAndroid Build Coastguard Worker 
33*6777b538SAndroid Build Coastguard Worker namespace base {
34*6777b538SAndroid Build Coastguard Worker 
35*6777b538SAndroid Build Coastguard Worker namespace {
36*6777b538SAndroid Build Coastguard Worker 
QuitWhenIdleTask(RunLoop * run_loop,int * counter)37*6777b538SAndroid Build Coastguard Worker void QuitWhenIdleTask(RunLoop* run_loop, int* counter) {
38*6777b538SAndroid Build Coastguard Worker   run_loop->QuitWhenIdle();
39*6777b538SAndroid Build Coastguard Worker   ++(*counter);
40*6777b538SAndroid Build Coastguard Worker }
41*6777b538SAndroid Build Coastguard Worker 
RunNestedLoopTask(int * counter)42*6777b538SAndroid Build Coastguard Worker void RunNestedLoopTask(int* counter) {
43*6777b538SAndroid Build Coastguard Worker   RunLoop nested_run_loop(RunLoop::Type::kNestableTasksAllowed);
44*6777b538SAndroid Build Coastguard Worker 
45*6777b538SAndroid Build Coastguard Worker   // This task should quit |nested_run_loop| but not the main RunLoop.
46*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
47*6777b538SAndroid Build Coastguard Worker       FROM_HERE, BindOnce(&QuitWhenIdleTask, Unretained(&nested_run_loop),
48*6777b538SAndroid Build Coastguard Worker                           Unretained(counter)));
49*6777b538SAndroid Build Coastguard Worker 
50*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask(
51*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedNotRunClosure(FROM_HERE), Days(1));
52*6777b538SAndroid Build Coastguard Worker 
53*6777b538SAndroid Build Coastguard Worker   nested_run_loop.Run();
54*6777b538SAndroid Build Coastguard Worker 
55*6777b538SAndroid Build Coastguard Worker   ++(*counter);
56*6777b538SAndroid Build Coastguard Worker }
57*6777b538SAndroid Build Coastguard Worker 
58*6777b538SAndroid Build Coastguard Worker // A simple SingleThreadTaskRunner that just queues undelayed tasks (and ignores
59*6777b538SAndroid Build Coastguard Worker // delayed tasks). Tasks can then be processed one by one by ProcessTask() which
60*6777b538SAndroid Build Coastguard Worker // will return true if it processed a task and false otherwise.
61*6777b538SAndroid Build Coastguard Worker class SimpleSingleThreadTaskRunner : public SingleThreadTaskRunner {
62*6777b538SAndroid Build Coastguard Worker  public:
63*6777b538SAndroid Build Coastguard Worker   SimpleSingleThreadTaskRunner() = default;
64*6777b538SAndroid Build Coastguard Worker   SimpleSingleThreadTaskRunner(const SimpleSingleThreadTaskRunner&) = delete;
65*6777b538SAndroid Build Coastguard Worker   SimpleSingleThreadTaskRunner& operator=(const SimpleSingleThreadTaskRunner&) =
66*6777b538SAndroid Build Coastguard Worker       delete;
67*6777b538SAndroid Build Coastguard Worker 
PostDelayedTask(const Location & from_here,OnceClosure task,base::TimeDelta delay)68*6777b538SAndroid Build Coastguard Worker   bool PostDelayedTask(const Location& from_here,
69*6777b538SAndroid Build Coastguard Worker                        OnceClosure task,
70*6777b538SAndroid Build Coastguard Worker                        base::TimeDelta delay) override {
71*6777b538SAndroid Build Coastguard Worker     if (delay.is_positive())
72*6777b538SAndroid Build Coastguard Worker       return false;
73*6777b538SAndroid Build Coastguard Worker     AutoLock auto_lock(tasks_lock_);
74*6777b538SAndroid Build Coastguard Worker     pending_tasks_.push(std::move(task));
75*6777b538SAndroid Build Coastguard Worker     return true;
76*6777b538SAndroid Build Coastguard Worker   }
77*6777b538SAndroid Build Coastguard Worker 
PostNonNestableDelayedTask(const Location & from_here,OnceClosure task,base::TimeDelta delay)78*6777b538SAndroid Build Coastguard Worker   bool PostNonNestableDelayedTask(const Location& from_here,
79*6777b538SAndroid Build Coastguard Worker                                   OnceClosure task,
80*6777b538SAndroid Build Coastguard Worker                                   base::TimeDelta delay) override {
81*6777b538SAndroid Build Coastguard Worker     return PostDelayedTask(from_here, std::move(task), delay);
82*6777b538SAndroid Build Coastguard Worker   }
83*6777b538SAndroid Build Coastguard Worker 
RunsTasksInCurrentSequence() const84*6777b538SAndroid Build Coastguard Worker   bool RunsTasksInCurrentSequence() const override {
85*6777b538SAndroid Build Coastguard Worker     return origin_thread_checker_.CalledOnValidThread();
86*6777b538SAndroid Build Coastguard Worker   }
87*6777b538SAndroid Build Coastguard Worker 
ProcessSingleTask()88*6777b538SAndroid Build Coastguard Worker   bool ProcessSingleTask() {
89*6777b538SAndroid Build Coastguard Worker     OnceClosure task;
90*6777b538SAndroid Build Coastguard Worker     {
91*6777b538SAndroid Build Coastguard Worker       AutoLock auto_lock(tasks_lock_);
92*6777b538SAndroid Build Coastguard Worker       if (pending_tasks_.empty())
93*6777b538SAndroid Build Coastguard Worker         return false;
94*6777b538SAndroid Build Coastguard Worker       task = std::move(pending_tasks_.front());
95*6777b538SAndroid Build Coastguard Worker       pending_tasks_.pop();
96*6777b538SAndroid Build Coastguard Worker     }
97*6777b538SAndroid Build Coastguard Worker     // It's important to Run() after pop() and outside the lock as |task| may
98*6777b538SAndroid Build Coastguard Worker     // run a nested loop which will re-enter ProcessSingleTask().
99*6777b538SAndroid Build Coastguard Worker     std::move(task).Run();
100*6777b538SAndroid Build Coastguard Worker     return true;
101*6777b538SAndroid Build Coastguard Worker   }
102*6777b538SAndroid Build Coastguard Worker 
103*6777b538SAndroid Build Coastguard Worker  private:
104*6777b538SAndroid Build Coastguard Worker   ~SimpleSingleThreadTaskRunner() override = default;
105*6777b538SAndroid Build Coastguard Worker 
106*6777b538SAndroid Build Coastguard Worker   Lock tasks_lock_;
107*6777b538SAndroid Build Coastguard Worker   base::queue<OnceClosure> pending_tasks_;
108*6777b538SAndroid Build Coastguard Worker 
109*6777b538SAndroid Build Coastguard Worker   // RunLoop relies on RunsTasksInCurrentSequence() signal. Use a
110*6777b538SAndroid Build Coastguard Worker   // ThreadCheckerImpl to be able to reliably provide that signal even in
111*6777b538SAndroid Build Coastguard Worker   // non-dcheck builds.
112*6777b538SAndroid Build Coastguard Worker   ThreadCheckerImpl origin_thread_checker_;
113*6777b538SAndroid Build Coastguard Worker };
114*6777b538SAndroid Build Coastguard Worker 
115*6777b538SAndroid Build Coastguard Worker // The basis of all TestDelegates, allows safely injecting a OnceClosure to be
116*6777b538SAndroid Build Coastguard Worker // run in the next idle phase of this delegate's Run() implementation. This can
117*6777b538SAndroid Build Coastguard Worker // be used to have code run on a thread that is otherwise livelocked in an idle
118*6777b538SAndroid Build Coastguard Worker // phase (sometimes a simple PostTask() won't do it -- e.g. when processing
119*6777b538SAndroid Build Coastguard Worker // application tasks is disallowed).
120*6777b538SAndroid Build Coastguard Worker class InjectableTestDelegate : public RunLoop::Delegate {
121*6777b538SAndroid Build Coastguard Worker  public:
InjectClosureOnDelegate(OnceClosure closure)122*6777b538SAndroid Build Coastguard Worker   void InjectClosureOnDelegate(OnceClosure closure) {
123*6777b538SAndroid Build Coastguard Worker     AutoLock auto_lock(closure_lock_);
124*6777b538SAndroid Build Coastguard Worker     closure_ = std::move(closure);
125*6777b538SAndroid Build Coastguard Worker   }
126*6777b538SAndroid Build Coastguard Worker 
RunInjectedClosure()127*6777b538SAndroid Build Coastguard Worker   bool RunInjectedClosure() {
128*6777b538SAndroid Build Coastguard Worker     AutoLock auto_lock(closure_lock_);
129*6777b538SAndroid Build Coastguard Worker     if (closure_.is_null())
130*6777b538SAndroid Build Coastguard Worker       return false;
131*6777b538SAndroid Build Coastguard Worker     std::move(closure_).Run();
132*6777b538SAndroid Build Coastguard Worker     return true;
133*6777b538SAndroid Build Coastguard Worker   }
134*6777b538SAndroid Build Coastguard Worker 
135*6777b538SAndroid Build Coastguard Worker  private:
136*6777b538SAndroid Build Coastguard Worker   Lock closure_lock_;
137*6777b538SAndroid Build Coastguard Worker   OnceClosure closure_;
138*6777b538SAndroid Build Coastguard Worker };
139*6777b538SAndroid Build Coastguard Worker 
140*6777b538SAndroid Build Coastguard Worker // A simple test RunLoop::Delegate to exercise Runloop logic independent of any
141*6777b538SAndroid Build Coastguard Worker // other base constructs. BindToCurrentThread() must be called before this
142*6777b538SAndroid Build Coastguard Worker // TestBoundDelegate is operational.
143*6777b538SAndroid Build Coastguard Worker class TestBoundDelegate final : public InjectableTestDelegate {
144*6777b538SAndroid Build Coastguard Worker  public:
145*6777b538SAndroid Build Coastguard Worker   TestBoundDelegate() = default;
146*6777b538SAndroid Build Coastguard Worker 
147*6777b538SAndroid Build Coastguard Worker   // Makes this TestBoundDelegate become the RunLoop::Delegate and
148*6777b538SAndroid Build Coastguard Worker   // SingleThreadTaskRunner::CurrentDefaultHandle for this thread.
BindToCurrentThread()149*6777b538SAndroid Build Coastguard Worker   void BindToCurrentThread() {
150*6777b538SAndroid Build Coastguard Worker     thread_task_runner_handle_ =
151*6777b538SAndroid Build Coastguard Worker         std::make_unique<SingleThreadTaskRunner::CurrentDefaultHandle>(
152*6777b538SAndroid Build Coastguard Worker             simple_task_runner_);
153*6777b538SAndroid Build Coastguard Worker     RunLoop::RegisterDelegateForCurrentThread(this);
154*6777b538SAndroid Build Coastguard Worker   }
155*6777b538SAndroid Build Coastguard Worker 
156*6777b538SAndroid Build Coastguard Worker  private:
Run(bool application_tasks_allowed,TimeDelta timeout)157*6777b538SAndroid Build Coastguard Worker   void Run(bool application_tasks_allowed, TimeDelta timeout) override {
158*6777b538SAndroid Build Coastguard Worker     if (nested_run_allowing_tasks_incoming_) {
159*6777b538SAndroid Build Coastguard Worker       EXPECT_TRUE(RunLoop::IsNestedOnCurrentThread());
160*6777b538SAndroid Build Coastguard Worker       EXPECT_TRUE(application_tasks_allowed);
161*6777b538SAndroid Build Coastguard Worker     } else if (RunLoop::IsNestedOnCurrentThread()) {
162*6777b538SAndroid Build Coastguard Worker       EXPECT_FALSE(application_tasks_allowed);
163*6777b538SAndroid Build Coastguard Worker     }
164*6777b538SAndroid Build Coastguard Worker     nested_run_allowing_tasks_incoming_ = false;
165*6777b538SAndroid Build Coastguard Worker 
166*6777b538SAndroid Build Coastguard Worker     while (!should_quit_) {
167*6777b538SAndroid Build Coastguard Worker       if (application_tasks_allowed && simple_task_runner_->ProcessSingleTask())
168*6777b538SAndroid Build Coastguard Worker         continue;
169*6777b538SAndroid Build Coastguard Worker 
170*6777b538SAndroid Build Coastguard Worker       if (ShouldQuitWhenIdle())
171*6777b538SAndroid Build Coastguard Worker         break;
172*6777b538SAndroid Build Coastguard Worker 
173*6777b538SAndroid Build Coastguard Worker       if (RunInjectedClosure())
174*6777b538SAndroid Build Coastguard Worker         continue;
175*6777b538SAndroid Build Coastguard Worker 
176*6777b538SAndroid Build Coastguard Worker       PlatformThread::YieldCurrentThread();
177*6777b538SAndroid Build Coastguard Worker     }
178*6777b538SAndroid Build Coastguard Worker     should_quit_ = false;
179*6777b538SAndroid Build Coastguard Worker   }
180*6777b538SAndroid Build Coastguard Worker 
Quit()181*6777b538SAndroid Build Coastguard Worker   void Quit() override { should_quit_ = true; }
182*6777b538SAndroid Build Coastguard Worker 
EnsureWorkScheduled()183*6777b538SAndroid Build Coastguard Worker   void EnsureWorkScheduled() override {
184*6777b538SAndroid Build Coastguard Worker     nested_run_allowing_tasks_incoming_ = true;
185*6777b538SAndroid Build Coastguard Worker   }
186*6777b538SAndroid Build Coastguard Worker 
187*6777b538SAndroid Build Coastguard Worker   // True if the next invocation of Run() is expected to be from a
188*6777b538SAndroid Build Coastguard Worker   // kNestableTasksAllowed RunLoop.
189*6777b538SAndroid Build Coastguard Worker   bool nested_run_allowing_tasks_incoming_ = false;
190*6777b538SAndroid Build Coastguard Worker 
191*6777b538SAndroid Build Coastguard Worker   scoped_refptr<SimpleSingleThreadTaskRunner> simple_task_runner_ =
192*6777b538SAndroid Build Coastguard Worker       MakeRefCounted<SimpleSingleThreadTaskRunner>();
193*6777b538SAndroid Build Coastguard Worker 
194*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<SingleThreadTaskRunner::CurrentDefaultHandle>
195*6777b538SAndroid Build Coastguard Worker       thread_task_runner_handle_;
196*6777b538SAndroid Build Coastguard Worker 
197*6777b538SAndroid Build Coastguard Worker   bool should_quit_ = false;
198*6777b538SAndroid Build Coastguard Worker };
199*6777b538SAndroid Build Coastguard Worker 
200*6777b538SAndroid Build Coastguard Worker enum class RunLoopTestType {
201*6777b538SAndroid Build Coastguard Worker   // Runs all RunLoopTests under a TaskEnvironment to make sure real world
202*6777b538SAndroid Build Coastguard Worker   // scenarios work.
203*6777b538SAndroid Build Coastguard Worker   kRealEnvironment,
204*6777b538SAndroid Build Coastguard Worker 
205*6777b538SAndroid Build Coastguard Worker   // Runs all RunLoopTests under a test RunLoop::Delegate to make sure the
206*6777b538SAndroid Build Coastguard Worker   // delegate interface fully works standalone.
207*6777b538SAndroid Build Coastguard Worker   kTestDelegate,
208*6777b538SAndroid Build Coastguard Worker };
209*6777b538SAndroid Build Coastguard Worker 
210*6777b538SAndroid Build Coastguard Worker // The task environment for the RunLoopTest of a given type. A separate class
211*6777b538SAndroid Build Coastguard Worker // so it can be instantiated on the stack in the RunLoopTest fixture.
212*6777b538SAndroid Build Coastguard Worker class RunLoopTestEnvironment {
213*6777b538SAndroid Build Coastguard Worker  public:
RunLoopTestEnvironment(RunLoopTestType type)214*6777b538SAndroid Build Coastguard Worker   explicit RunLoopTestEnvironment(RunLoopTestType type) {
215*6777b538SAndroid Build Coastguard Worker     switch (type) {
216*6777b538SAndroid Build Coastguard Worker       case RunLoopTestType::kRealEnvironment: {
217*6777b538SAndroid Build Coastguard Worker         task_environment_ = std::make_unique<test::TaskEnvironment>();
218*6777b538SAndroid Build Coastguard Worker         break;
219*6777b538SAndroid Build Coastguard Worker       }
220*6777b538SAndroid Build Coastguard Worker       case RunLoopTestType::kTestDelegate: {
221*6777b538SAndroid Build Coastguard Worker         auto test_delegate = std::make_unique<TestBoundDelegate>();
222*6777b538SAndroid Build Coastguard Worker         test_delegate->BindToCurrentThread();
223*6777b538SAndroid Build Coastguard Worker         test_delegate_ = std::move(test_delegate);
224*6777b538SAndroid Build Coastguard Worker         break;
225*6777b538SAndroid Build Coastguard Worker       }
226*6777b538SAndroid Build Coastguard Worker     }
227*6777b538SAndroid Build Coastguard Worker   }
228*6777b538SAndroid Build Coastguard Worker 
229*6777b538SAndroid Build Coastguard Worker  private:
230*6777b538SAndroid Build Coastguard Worker   // Instantiates one or the other based on the RunLoopTestType.
231*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<test::TaskEnvironment> task_environment_;
232*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<InjectableTestDelegate> test_delegate_;
233*6777b538SAndroid Build Coastguard Worker };
234*6777b538SAndroid Build Coastguard Worker 
235*6777b538SAndroid Build Coastguard Worker class RunLoopTest : public testing::TestWithParam<RunLoopTestType> {
236*6777b538SAndroid Build Coastguard Worker  public:
237*6777b538SAndroid Build Coastguard Worker   RunLoopTest(const RunLoopTest&) = delete;
238*6777b538SAndroid Build Coastguard Worker   RunLoopTest& operator=(const RunLoopTest&) = delete;
239*6777b538SAndroid Build Coastguard Worker 
240*6777b538SAndroid Build Coastguard Worker  protected:
RunLoopTest()241*6777b538SAndroid Build Coastguard Worker   RunLoopTest() : test_environment_(GetParam()) {}
242*6777b538SAndroid Build Coastguard Worker 
243*6777b538SAndroid Build Coastguard Worker   RunLoopTestEnvironment test_environment_;
244*6777b538SAndroid Build Coastguard Worker   RunLoop run_loop_;
245*6777b538SAndroid Build Coastguard Worker };
246*6777b538SAndroid Build Coastguard Worker 
247*6777b538SAndroid Build Coastguard Worker }  // namespace
248*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,QuitWhenIdle)249*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitWhenIdle) {
250*6777b538SAndroid Build Coastguard Worker   int counter = 0;
251*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
252*6777b538SAndroid Build Coastguard Worker       FROM_HERE, BindOnce(&QuitWhenIdleTask, Unretained(&run_loop_),
253*6777b538SAndroid Build Coastguard Worker                           Unretained(&counter)));
254*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
255*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
256*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask(
257*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedNotRunClosure(FROM_HERE), Days(1));
258*6777b538SAndroid Build Coastguard Worker 
259*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
260*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(1, counter);
261*6777b538SAndroid Build Coastguard Worker }
262*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,QuitWhenIdleNestedLoop)263*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitWhenIdleNestedLoop) {
264*6777b538SAndroid Build Coastguard Worker   int counter = 0;
265*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
266*6777b538SAndroid Build Coastguard Worker       FROM_HERE, BindOnce(&RunNestedLoopTask, Unretained(&counter)));
267*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
268*6777b538SAndroid Build Coastguard Worker       FROM_HERE, BindOnce(&QuitWhenIdleTask, Unretained(&run_loop_),
269*6777b538SAndroid Build Coastguard Worker                           Unretained(&counter)));
270*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
271*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
272*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask(
273*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedNotRunClosure(FROM_HERE), Days(1));
274*6777b538SAndroid Build Coastguard Worker 
275*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
276*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(3, counter);
277*6777b538SAndroid Build Coastguard Worker }
278*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,QuitWhenIdleClosure)279*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitWhenIdleClosure) {
280*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
281*6777b538SAndroid Build Coastguard Worker       FROM_HERE, run_loop_.QuitWhenIdleClosure());
282*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
283*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
284*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask(
285*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedNotRunClosure(FROM_HERE), Days(1));
286*6777b538SAndroid Build Coastguard Worker 
287*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
288*6777b538SAndroid Build Coastguard Worker }
289*6777b538SAndroid Build Coastguard Worker 
290*6777b538SAndroid Build Coastguard Worker // Verify that the QuitWhenIdleClosure() can run after the RunLoop has been
291*6777b538SAndroid Build Coastguard Worker // deleted. It should have no effect.
TEST_P(RunLoopTest,QuitWhenIdleClosureAfterRunLoopScope)292*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitWhenIdleClosureAfterRunLoopScope) {
293*6777b538SAndroid Build Coastguard Worker   RepeatingClosure quit_when_idle_closure;
294*6777b538SAndroid Build Coastguard Worker   {
295*6777b538SAndroid Build Coastguard Worker     RunLoop run_loop;
296*6777b538SAndroid Build Coastguard Worker     quit_when_idle_closure = run_loop.QuitWhenIdleClosure();
297*6777b538SAndroid Build Coastguard Worker     run_loop.RunUntilIdle();
298*6777b538SAndroid Build Coastguard Worker   }
299*6777b538SAndroid Build Coastguard Worker   quit_when_idle_closure.Run();
300*6777b538SAndroid Build Coastguard Worker }
301*6777b538SAndroid Build Coastguard Worker 
302*6777b538SAndroid Build Coastguard Worker // Verify that Quit can be executed from another sequence.
TEST_P(RunLoopTest,QuitFromOtherSequence)303*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitFromOtherSequence) {
304*6777b538SAndroid Build Coastguard Worker   Thread other_thread("test");
305*6777b538SAndroid Build Coastguard Worker   other_thread.Start();
306*6777b538SAndroid Build Coastguard Worker   scoped_refptr<SequencedTaskRunner> other_sequence =
307*6777b538SAndroid Build Coastguard Worker       other_thread.task_runner();
308*6777b538SAndroid Build Coastguard Worker 
309*6777b538SAndroid Build Coastguard Worker   // Always expected to run before asynchronous Quit() kicks in.
310*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
311*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
312*6777b538SAndroid Build Coastguard Worker 
313*6777b538SAndroid Build Coastguard Worker   WaitableEvent loop_was_quit(WaitableEvent::ResetPolicy::MANUAL,
314*6777b538SAndroid Build Coastguard Worker                               WaitableEvent::InitialState::NOT_SIGNALED);
315*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(
316*6777b538SAndroid Build Coastguard Worker       FROM_HERE, base::BindOnce([](RunLoop* run_loop) { run_loop->Quit(); },
317*6777b538SAndroid Build Coastguard Worker                                 Unretained(&run_loop_)));
318*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(
319*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
320*6777b538SAndroid Build Coastguard Worker       base::BindOnce(&WaitableEvent::Signal, base::Unretained(&loop_was_quit)));
321*6777b538SAndroid Build Coastguard Worker 
322*6777b538SAndroid Build Coastguard Worker   // Anything that's posted after the Quit closure was posted back to this
323*6777b538SAndroid Build Coastguard Worker   // sequence shouldn't get a chance to run.
324*6777b538SAndroid Build Coastguard Worker   loop_was_quit.Wait();
325*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
326*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedNotRunClosure(FROM_HERE));
327*6777b538SAndroid Build Coastguard Worker 
328*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
329*6777b538SAndroid Build Coastguard Worker }
330*6777b538SAndroid Build Coastguard Worker 
331*6777b538SAndroid Build Coastguard Worker // Verify that QuitClosure can be executed from another sequence.
TEST_P(RunLoopTest,QuitFromOtherSequenceWithClosure)332*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitFromOtherSequenceWithClosure) {
333*6777b538SAndroid Build Coastguard Worker   Thread other_thread("test");
334*6777b538SAndroid Build Coastguard Worker   other_thread.Start();
335*6777b538SAndroid Build Coastguard Worker   scoped_refptr<SequencedTaskRunner> other_sequence =
336*6777b538SAndroid Build Coastguard Worker       other_thread.task_runner();
337*6777b538SAndroid Build Coastguard Worker 
338*6777b538SAndroid Build Coastguard Worker   // Always expected to run before asynchronous Quit() kicks in.
339*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
340*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
341*6777b538SAndroid Build Coastguard Worker 
342*6777b538SAndroid Build Coastguard Worker   WaitableEvent loop_was_quit(WaitableEvent::ResetPolicy::MANUAL,
343*6777b538SAndroid Build Coastguard Worker                               WaitableEvent::InitialState::NOT_SIGNALED);
344*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(FROM_HERE, run_loop_.QuitClosure());
345*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(
346*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
347*6777b538SAndroid Build Coastguard Worker       base::BindOnce(&WaitableEvent::Signal, base::Unretained(&loop_was_quit)));
348*6777b538SAndroid Build Coastguard Worker 
349*6777b538SAndroid Build Coastguard Worker   // Anything that's posted after the Quit closure was posted back to this
350*6777b538SAndroid Build Coastguard Worker   // sequence shouldn't get a chance to run.
351*6777b538SAndroid Build Coastguard Worker   loop_was_quit.Wait();
352*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
353*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedNotRunClosure(FROM_HERE));
354*6777b538SAndroid Build Coastguard Worker 
355*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
356*6777b538SAndroid Build Coastguard Worker }
357*6777b538SAndroid Build Coastguard Worker 
358*6777b538SAndroid Build Coastguard Worker // Verify that Quit can be executed from another sequence even when the
359*6777b538SAndroid Build Coastguard Worker // Quit is racing with Run() -- i.e. forgo the WaitableEvent used above.
TEST_P(RunLoopTest,QuitFromOtherSequenceRacy)360*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitFromOtherSequenceRacy) {
361*6777b538SAndroid Build Coastguard Worker   Thread other_thread("test");
362*6777b538SAndroid Build Coastguard Worker   other_thread.Start();
363*6777b538SAndroid Build Coastguard Worker   scoped_refptr<SequencedTaskRunner> other_sequence =
364*6777b538SAndroid Build Coastguard Worker       other_thread.task_runner();
365*6777b538SAndroid Build Coastguard Worker 
366*6777b538SAndroid Build Coastguard Worker   // Always expected to run before asynchronous Quit() kicks in.
367*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
368*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
369*6777b538SAndroid Build Coastguard Worker 
370*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(FROM_HERE, run_loop_.QuitClosure());
371*6777b538SAndroid Build Coastguard Worker 
372*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
373*6777b538SAndroid Build Coastguard Worker }
374*6777b538SAndroid Build Coastguard Worker 
375*6777b538SAndroid Build Coastguard Worker // Verify that QuitClosure can be executed from another sequence even when the
376*6777b538SAndroid Build Coastguard Worker // Quit is racing with Run() -- i.e. forgo the WaitableEvent used above.
TEST_P(RunLoopTest,QuitFromOtherSequenceRacyWithClosure)377*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitFromOtherSequenceRacyWithClosure) {
378*6777b538SAndroid Build Coastguard Worker   Thread other_thread("test");
379*6777b538SAndroid Build Coastguard Worker   other_thread.Start();
380*6777b538SAndroid Build Coastguard Worker   scoped_refptr<SequencedTaskRunner> other_sequence =
381*6777b538SAndroid Build Coastguard Worker       other_thread.task_runner();
382*6777b538SAndroid Build Coastguard Worker 
383*6777b538SAndroid Build Coastguard Worker   // Always expected to run before asynchronous Quit() kicks in.
384*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
385*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
386*6777b538SAndroid Build Coastguard Worker 
387*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(FROM_HERE, run_loop_.QuitClosure());
388*6777b538SAndroid Build Coastguard Worker 
389*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
390*6777b538SAndroid Build Coastguard Worker }
391*6777b538SAndroid Build Coastguard Worker 
392*6777b538SAndroid Build Coastguard Worker // Verify that QuitWhenIdle can be executed from another sequence.
TEST_P(RunLoopTest,QuitWhenIdleFromOtherSequence)393*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitWhenIdleFromOtherSequence) {
394*6777b538SAndroid Build Coastguard Worker   Thread other_thread("test");
395*6777b538SAndroid Build Coastguard Worker   other_thread.Start();
396*6777b538SAndroid Build Coastguard Worker   scoped_refptr<SequencedTaskRunner> other_sequence =
397*6777b538SAndroid Build Coastguard Worker       other_thread.task_runner();
398*6777b538SAndroid Build Coastguard Worker 
399*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
400*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
401*6777b538SAndroid Build Coastguard Worker 
402*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(
403*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
404*6777b538SAndroid Build Coastguard Worker       base::BindOnce([](RunLoop* run_loop) { run_loop->QuitWhenIdle(); },
405*6777b538SAndroid Build Coastguard Worker                      Unretained(&run_loop_)));
406*6777b538SAndroid Build Coastguard Worker 
407*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
408*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
409*6777b538SAndroid Build Coastguard Worker 
410*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
411*6777b538SAndroid Build Coastguard Worker 
412*6777b538SAndroid Build Coastguard Worker   // Regardless of the outcome of the race this thread shouldn't have been idle
413*6777b538SAndroid Build Coastguard Worker   // until both tasks posted to this sequence have run.
414*6777b538SAndroid Build Coastguard Worker }
415*6777b538SAndroid Build Coastguard Worker 
416*6777b538SAndroid Build Coastguard Worker // Verify that QuitWhenIdleClosure can be executed from another sequence.
TEST_P(RunLoopTest,QuitWhenIdleFromOtherSequenceWithClosure)417*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, QuitWhenIdleFromOtherSequenceWithClosure) {
418*6777b538SAndroid Build Coastguard Worker   Thread other_thread("test");
419*6777b538SAndroid Build Coastguard Worker   other_thread.Start();
420*6777b538SAndroid Build Coastguard Worker   scoped_refptr<SequencedTaskRunner> other_sequence =
421*6777b538SAndroid Build Coastguard Worker       other_thread.task_runner();
422*6777b538SAndroid Build Coastguard Worker 
423*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
424*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
425*6777b538SAndroid Build Coastguard Worker 
426*6777b538SAndroid Build Coastguard Worker   other_sequence->PostTask(FROM_HERE, run_loop_.QuitWhenIdleClosure());
427*6777b538SAndroid Build Coastguard Worker 
428*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
429*6777b538SAndroid Build Coastguard Worker       FROM_HERE, MakeExpectedRunClosure(FROM_HERE));
430*6777b538SAndroid Build Coastguard Worker 
431*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
432*6777b538SAndroid Build Coastguard Worker 
433*6777b538SAndroid Build Coastguard Worker   // Regardless of the outcome of the race this thread shouldn't have been idle
434*6777b538SAndroid Build Coastguard Worker   // until the both tasks posted to this sequence have run.
435*6777b538SAndroid Build Coastguard Worker }
436*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,IsRunningOnCurrentThread)437*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, IsRunningOnCurrentThread) {
438*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(RunLoop::IsRunningOnCurrentThread());
439*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
440*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
441*6777b538SAndroid Build Coastguard Worker       BindOnce([]() { EXPECT_TRUE(RunLoop::IsRunningOnCurrentThread()); }));
442*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
443*6777b538SAndroid Build Coastguard Worker       FROM_HERE, run_loop_.QuitClosure());
444*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
445*6777b538SAndroid Build Coastguard Worker }
446*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,IsNestedOnCurrentThread)447*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, IsNestedOnCurrentThread) {
448*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(RunLoop::IsNestedOnCurrentThread());
449*6777b538SAndroid Build Coastguard Worker 
450*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
451*6777b538SAndroid Build Coastguard Worker       FROM_HERE, BindOnce([]() {
452*6777b538SAndroid Build Coastguard Worker         EXPECT_FALSE(RunLoop::IsNestedOnCurrentThread());
453*6777b538SAndroid Build Coastguard Worker 
454*6777b538SAndroid Build Coastguard Worker         RunLoop nested_run_loop(RunLoop::Type::kNestableTasksAllowed);
455*6777b538SAndroid Build Coastguard Worker 
456*6777b538SAndroid Build Coastguard Worker         SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
457*6777b538SAndroid Build Coastguard Worker             FROM_HERE, BindOnce([]() {
458*6777b538SAndroid Build Coastguard Worker               EXPECT_TRUE(RunLoop::IsNestedOnCurrentThread());
459*6777b538SAndroid Build Coastguard Worker             }));
460*6777b538SAndroid Build Coastguard Worker         SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
461*6777b538SAndroid Build Coastguard Worker             FROM_HERE, nested_run_loop.QuitClosure());
462*6777b538SAndroid Build Coastguard Worker 
463*6777b538SAndroid Build Coastguard Worker         EXPECT_FALSE(RunLoop::IsNestedOnCurrentThread());
464*6777b538SAndroid Build Coastguard Worker         nested_run_loop.Run();
465*6777b538SAndroid Build Coastguard Worker         EXPECT_FALSE(RunLoop::IsNestedOnCurrentThread());
466*6777b538SAndroid Build Coastguard Worker       }));
467*6777b538SAndroid Build Coastguard Worker 
468*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
469*6777b538SAndroid Build Coastguard Worker       FROM_HERE, run_loop_.QuitClosure());
470*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
471*6777b538SAndroid Build Coastguard Worker }
472*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,CannotRunMoreThanOnce)473*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, CannotRunMoreThanOnce) {
474*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
475*6777b538SAndroid Build Coastguard Worker       FROM_HERE, run_loop_.QuitClosure());
476*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
477*6777b538SAndroid Build Coastguard Worker   EXPECT_DCHECK_DEATH({ run_loop_.Run(); });
478*6777b538SAndroid Build Coastguard Worker }
479*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,CanRunUntilIdleMoreThanOnce)480*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, CanRunUntilIdleMoreThanOnce) {
481*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(FROM_HERE, DoNothing());
482*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
483*6777b538SAndroid Build Coastguard Worker 
484*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(FROM_HERE, DoNothing());
485*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
486*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
487*6777b538SAndroid Build Coastguard Worker }
488*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,CanRunUntilIdleThenRunIfNotQuit)489*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, CanRunUntilIdleThenRunIfNotQuit) {
490*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(FROM_HERE, DoNothing());
491*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
492*6777b538SAndroid Build Coastguard Worker 
493*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
494*6777b538SAndroid Build Coastguard Worker       FROM_HERE, run_loop_.QuitClosure());
495*6777b538SAndroid Build Coastguard Worker   run_loop_.Run();
496*6777b538SAndroid Build Coastguard Worker }
497*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,CannotRunUntilIdleThenRunIfQuit)498*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, CannotRunUntilIdleThenRunIfQuit) {
499*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
500*6777b538SAndroid Build Coastguard Worker       FROM_HERE, run_loop_.QuitClosure());
501*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
502*6777b538SAndroid Build Coastguard Worker 
503*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(FROM_HERE, DoNothing());
504*6777b538SAndroid Build Coastguard Worker   EXPECT_DCHECK_DEATH({ run_loop_.Run(); });
505*6777b538SAndroid Build Coastguard Worker }
506*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,CannotRunAgainIfQuitWhenIdle)507*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, CannotRunAgainIfQuitWhenIdle) {
508*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
509*6777b538SAndroid Build Coastguard Worker       FROM_HERE, run_loop_.QuitWhenIdleClosure());
510*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
511*6777b538SAndroid Build Coastguard Worker 
512*6777b538SAndroid Build Coastguard Worker   EXPECT_DCHECK_DEATH({ run_loop_.RunUntilIdle(); });
513*6777b538SAndroid Build Coastguard Worker }
514*6777b538SAndroid Build Coastguard Worker 
515*6777b538SAndroid Build Coastguard Worker namespace {
516*6777b538SAndroid Build Coastguard Worker 
517*6777b538SAndroid Build Coastguard Worker class MockNestingObserver : public RunLoop::NestingObserver {
518*6777b538SAndroid Build Coastguard Worker  public:
519*6777b538SAndroid Build Coastguard Worker   MockNestingObserver() = default;
520*6777b538SAndroid Build Coastguard Worker   MockNestingObserver(const MockNestingObserver&) = delete;
521*6777b538SAndroid Build Coastguard Worker   MockNestingObserver& operator=(const MockNestingObserver&) = delete;
522*6777b538SAndroid Build Coastguard Worker 
523*6777b538SAndroid Build Coastguard Worker   // RunLoop::NestingObserver:
524*6777b538SAndroid Build Coastguard Worker   MOCK_METHOD0(OnBeginNestedRunLoop, void());
525*6777b538SAndroid Build Coastguard Worker   MOCK_METHOD0(OnExitNestedRunLoop, void());
526*6777b538SAndroid Build Coastguard Worker };
527*6777b538SAndroid Build Coastguard Worker 
528*6777b538SAndroid Build Coastguard Worker class MockTask {
529*6777b538SAndroid Build Coastguard Worker  public:
530*6777b538SAndroid Build Coastguard Worker   MockTask() = default;
531*6777b538SAndroid Build Coastguard Worker   MockTask(const MockTask&) = delete;
532*6777b538SAndroid Build Coastguard Worker   MockTask& operator=(const MockTask&) = delete;
533*6777b538SAndroid Build Coastguard Worker 
534*6777b538SAndroid Build Coastguard Worker   MOCK_METHOD0(Task, void());
535*6777b538SAndroid Build Coastguard Worker };
536*6777b538SAndroid Build Coastguard Worker 
537*6777b538SAndroid Build Coastguard Worker }  // namespace
538*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,NestingObservers)539*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, NestingObservers) {
540*6777b538SAndroid Build Coastguard Worker   testing::StrictMock<MockNestingObserver> nesting_observer;
541*6777b538SAndroid Build Coastguard Worker   testing::StrictMock<MockTask> mock_task_a;
542*6777b538SAndroid Build Coastguard Worker   testing::StrictMock<MockTask> mock_task_b;
543*6777b538SAndroid Build Coastguard Worker 
544*6777b538SAndroid Build Coastguard Worker   RunLoop::AddNestingObserverOnCurrentThread(&nesting_observer);
545*6777b538SAndroid Build Coastguard Worker 
546*6777b538SAndroid Build Coastguard Worker   const RepeatingClosure run_nested_loop = BindRepeating([]() {
547*6777b538SAndroid Build Coastguard Worker     RunLoop nested_run_loop(RunLoop::Type::kNestableTasksAllowed);
548*6777b538SAndroid Build Coastguard Worker     SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
549*6777b538SAndroid Build Coastguard Worker         FROM_HERE, nested_run_loop.QuitClosure());
550*6777b538SAndroid Build Coastguard Worker     nested_run_loop.Run();
551*6777b538SAndroid Build Coastguard Worker   });
552*6777b538SAndroid Build Coastguard Worker 
553*6777b538SAndroid Build Coastguard Worker   // Generate a stack of nested RunLoops. OnBeginNestedRunLoop() is expected
554*6777b538SAndroid Build Coastguard Worker   // when beginning each nesting depth and OnExitNestedRunLoop() is expected
555*6777b538SAndroid Build Coastguard Worker   // when exiting each nesting depth. Each one of these tasks is ahead of the
556*6777b538SAndroid Build Coastguard Worker   // QuitClosures as those are only posted at the end of the queue when
557*6777b538SAndroid Build Coastguard Worker   // |run_nested_loop| is executed.
558*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(FROM_HERE,
559*6777b538SAndroid Build Coastguard Worker                                                         run_nested_loop);
560*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
561*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
562*6777b538SAndroid Build Coastguard Worker       base::BindOnce(&MockTask::Task, base::Unretained(&mock_task_a)));
563*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(FROM_HERE,
564*6777b538SAndroid Build Coastguard Worker                                                         run_nested_loop);
565*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
566*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
567*6777b538SAndroid Build Coastguard Worker       base::BindOnce(&MockTask::Task, base::Unretained(&mock_task_b)));
568*6777b538SAndroid Build Coastguard Worker 
569*6777b538SAndroid Build Coastguard Worker   {
570*6777b538SAndroid Build Coastguard Worker     testing::InSequence in_sequence;
571*6777b538SAndroid Build Coastguard Worker     EXPECT_CALL(nesting_observer, OnBeginNestedRunLoop());
572*6777b538SAndroid Build Coastguard Worker     EXPECT_CALL(mock_task_a, Task());
573*6777b538SAndroid Build Coastguard Worker     EXPECT_CALL(nesting_observer, OnBeginNestedRunLoop());
574*6777b538SAndroid Build Coastguard Worker     EXPECT_CALL(mock_task_b, Task());
575*6777b538SAndroid Build Coastguard Worker     EXPECT_CALL(nesting_observer, OnExitNestedRunLoop()).Times(2);
576*6777b538SAndroid Build Coastguard Worker   }
577*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
578*6777b538SAndroid Build Coastguard Worker 
579*6777b538SAndroid Build Coastguard Worker   RunLoop::RemoveNestingObserverOnCurrentThread(&nesting_observer);
580*6777b538SAndroid Build Coastguard Worker }
581*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,DisallowRunning)582*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, DisallowRunning) {
583*6777b538SAndroid Build Coastguard Worker   ScopedDisallowRunningRunLoop disallow_running;
584*6777b538SAndroid Build Coastguard Worker   EXPECT_DCHECK_DEATH({ run_loop_.RunUntilIdle(); });
585*6777b538SAndroid Build Coastguard Worker }
586*6777b538SAndroid Build Coastguard Worker 
TEST_P(RunLoopTest,ExpiredDisallowRunning)587*6777b538SAndroid Build Coastguard Worker TEST_P(RunLoopTest, ExpiredDisallowRunning) {
588*6777b538SAndroid Build Coastguard Worker   { ScopedDisallowRunningRunLoop disallow_running; }
589*6777b538SAndroid Build Coastguard Worker   // Running should be fine after |disallow_running| goes out of scope.
590*6777b538SAndroid Build Coastguard Worker   run_loop_.RunUntilIdle();
591*6777b538SAndroid Build Coastguard Worker }
592*6777b538SAndroid Build Coastguard Worker 
593*6777b538SAndroid Build Coastguard Worker INSTANTIATE_TEST_SUITE_P(Real,
594*6777b538SAndroid Build Coastguard Worker                          RunLoopTest,
595*6777b538SAndroid Build Coastguard Worker                          testing::Values(RunLoopTestType::kRealEnvironment));
596*6777b538SAndroid Build Coastguard Worker INSTANTIATE_TEST_SUITE_P(Mock,
597*6777b538SAndroid Build Coastguard Worker                          RunLoopTest,
598*6777b538SAndroid Build Coastguard Worker                          testing::Values(RunLoopTestType::kTestDelegate));
599*6777b538SAndroid Build Coastguard Worker 
TEST(RunLoopDeathTest,MustRegisterBeforeInstantiating)600*6777b538SAndroid Build Coastguard Worker TEST(RunLoopDeathTest, MustRegisterBeforeInstantiating) {
601*6777b538SAndroid Build Coastguard Worker   TestBoundDelegate unbound_test_delegate_;
602*6777b538SAndroid Build Coastguard Worker   // RunLoop::RunLoop() should CHECK fetching the
603*6777b538SAndroid Build Coastguard Worker   // SingleThreadTaskRunner::CurrentDefaultHandle.
604*6777b538SAndroid Build Coastguard Worker   EXPECT_DEATH_IF_SUPPORTED({ RunLoop(); }, "");
605*6777b538SAndroid Build Coastguard Worker }
606*6777b538SAndroid Build Coastguard Worker 
TEST(RunLoopDelegateTest,NestableTasksDontRunInDefaultNestedLoops)607*6777b538SAndroid Build Coastguard Worker TEST(RunLoopDelegateTest, NestableTasksDontRunInDefaultNestedLoops) {
608*6777b538SAndroid Build Coastguard Worker   TestBoundDelegate test_delegate;
609*6777b538SAndroid Build Coastguard Worker   test_delegate.BindToCurrentThread();
610*6777b538SAndroid Build Coastguard Worker 
611*6777b538SAndroid Build Coastguard Worker   base::Thread other_thread("test");
612*6777b538SAndroid Build Coastguard Worker   other_thread.Start();
613*6777b538SAndroid Build Coastguard Worker 
614*6777b538SAndroid Build Coastguard Worker   RunLoop main_loop;
615*6777b538SAndroid Build Coastguard Worker   // A nested run loop which isn't kNestableTasksAllowed.
616*6777b538SAndroid Build Coastguard Worker   RunLoop nested_run_loop(RunLoop::Type::kDefault);
617*6777b538SAndroid Build Coastguard Worker 
618*6777b538SAndroid Build Coastguard Worker   bool nested_run_loop_ended = false;
619*6777b538SAndroid Build Coastguard Worker 
620*6777b538SAndroid Build Coastguard Worker   // The first task on the main loop will result in a nested run loop. Since
621*6777b538SAndroid Build Coastguard Worker   // it's not kNestableTasksAllowed, no further task should be processed until
622*6777b538SAndroid Build Coastguard Worker   // it's quit.
623*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
624*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
625*6777b538SAndroid Build Coastguard Worker       BindOnce([](RunLoop* nested_run_loop) { nested_run_loop->Run(); },
626*6777b538SAndroid Build Coastguard Worker                Unretained(&nested_run_loop)));
627*6777b538SAndroid Build Coastguard Worker 
628*6777b538SAndroid Build Coastguard Worker   // Post a task that will fail if it runs inside the nested run loop.
629*6777b538SAndroid Build Coastguard Worker   SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
630*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
631*6777b538SAndroid Build Coastguard Worker       BindOnce(
632*6777b538SAndroid Build Coastguard Worker           [](const bool& nested_run_loop_ended,
633*6777b538SAndroid Build Coastguard Worker              OnceClosure continuation_callback) {
634*6777b538SAndroid Build Coastguard Worker             EXPECT_TRUE(nested_run_loop_ended);
635*6777b538SAndroid Build Coastguard Worker             EXPECT_FALSE(RunLoop::IsNestedOnCurrentThread());
636*6777b538SAndroid Build Coastguard Worker             std::move(continuation_callback).Run();
637*6777b538SAndroid Build Coastguard Worker           },
638*6777b538SAndroid Build Coastguard Worker           std::cref(nested_run_loop_ended), main_loop.QuitClosure()));
639*6777b538SAndroid Build Coastguard Worker 
640*6777b538SAndroid Build Coastguard Worker   // Post a task flipping the boolean bit for extra verification right before
641*6777b538SAndroid Build Coastguard Worker   // quitting |nested_run_loop|.
642*6777b538SAndroid Build Coastguard Worker   other_thread.task_runner()->PostDelayedTask(
643*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
644*6777b538SAndroid Build Coastguard Worker       BindOnce(
645*6777b538SAndroid Build Coastguard Worker           [](bool* nested_run_loop_ended) {
646*6777b538SAndroid Build Coastguard Worker             EXPECT_FALSE(*nested_run_loop_ended);
647*6777b538SAndroid Build Coastguard Worker             *nested_run_loop_ended = true;
648*6777b538SAndroid Build Coastguard Worker           },
649*6777b538SAndroid Build Coastguard Worker           Unretained(&nested_run_loop_ended)),
650*6777b538SAndroid Build Coastguard Worker       TestTimeouts::tiny_timeout());
651*6777b538SAndroid Build Coastguard Worker   // Post an async delayed task to exit the run loop when idle. This confirms
652*6777b538SAndroid Build Coastguard Worker   // that (1) the test task only ran in the main loop after the nested loop
653*6777b538SAndroid Build Coastguard Worker   // exited and (2) the nested run loop actually considers itself idle while
654*6777b538SAndroid Build Coastguard Worker   // spinning. Note: The quit closure needs to be injected directly on the
655*6777b538SAndroid Build Coastguard Worker   // delegate as invoking QuitWhenIdle() off-thread results in a thread bounce
656*6777b538SAndroid Build Coastguard Worker   // which will not processed because of the very logic under test (nestable
657*6777b538SAndroid Build Coastguard Worker   // tasks don't run in |nested_run_loop|).
658*6777b538SAndroid Build Coastguard Worker   other_thread.task_runner()->PostDelayedTask(
659*6777b538SAndroid Build Coastguard Worker       FROM_HERE,
660*6777b538SAndroid Build Coastguard Worker       BindOnce(
661*6777b538SAndroid Build Coastguard Worker           [](TestBoundDelegate* test_delegate, OnceClosure injected_closure) {
662*6777b538SAndroid Build Coastguard Worker             test_delegate->InjectClosureOnDelegate(std::move(injected_closure));
663*6777b538SAndroid Build Coastguard Worker           },
664*6777b538SAndroid Build Coastguard Worker           Unretained(&test_delegate), nested_run_loop.QuitWhenIdleClosure()),
665*6777b538SAndroid Build Coastguard Worker       TestTimeouts::tiny_timeout());
666*6777b538SAndroid Build Coastguard Worker 
667*6777b538SAndroid Build Coastguard Worker   main_loop.Run();
668*6777b538SAndroid Build Coastguard Worker }
669*6777b538SAndroid Build Coastguard Worker 
670*6777b538SAndroid Build Coastguard Worker }  // namespace base
671