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