blob: a2322f849582c3b22cc78b9c9980b3fd8a1a92b8 [file] [log] [blame]
[email protected]8e937c1e2012-06-28 22:57:301// Copyright (c) 2012 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "base/run_loop.h"
6
7#include "base/bind.h"
vadimt12f0f7d2014-09-15 19:19:388#include "base/tracked_objects.h"
avi9b6f42932015-12-26 22:15:149#include "build/build_config.h"
[email protected]8e937c1e2012-06-28 22:57:3010
11namespace base {
12
13RunLoop::RunLoop()
14 : loop_(MessageLoop::current()),
[email protected]8e937c1e2012-06-28 22:57:3015 previous_run_loop_(NULL),
16 run_depth_(0),
17 run_called_(false),
18 quit_called_(false),
19 running_(false),
[email protected]dcf10632013-10-08 19:23:3320 quit_when_idle_received_(false),
21 weak_factory_(this) {
[email protected]8e937c1e2012-06-28 22:57:3022}
23
[email protected]8e937c1e2012-06-28 22:57:3024RunLoop::~RunLoop() {
25}
26
27void RunLoop::Run() {
28 if (!BeforeRun())
29 return;
vadimt12f0f7d2014-09-15 19:19:3830
31 // Use task stopwatch to exclude the loop run time from the current task, if
32 // any.
33 tracked_objects::TaskStopwatch stopwatch;
vadimt20175532014-10-28 20:14:2034 stopwatch.Start();
[email protected]8e937c1e2012-06-28 22:57:3035 loop_->RunHandler();
vadimt12f0f7d2014-09-15 19:19:3836 stopwatch.Stop();
37
[email protected]8e937c1e2012-06-28 22:57:3038 AfterRun();
39}
40
41void RunLoop::RunUntilIdle() {
42 quit_when_idle_received_ = true;
43 Run();
44}
45
46void RunLoop::Quit() {
47 quit_called_ = true;
48 if (running_ && loop_->run_loop_ == this) {
49 // This is the inner-most RunLoop, so quit now.
50 loop_->QuitNow();
51 }
52}
53
fdoraya4f28ec2016-06-10 00:08:5854void RunLoop::QuitWhenIdle() {
55 quit_when_idle_received_ = true;
56}
57
[email protected]8e937c1e2012-06-28 22:57:3058base::Closure RunLoop::QuitClosure() {
59 return base::Bind(&RunLoop::Quit, weak_factory_.GetWeakPtr());
60}
61
fdoraya3658602016-06-10 18:23:1562base::Closure RunLoop::QuitWhenIdleClosure() {
63 return base::Bind(&RunLoop::QuitWhenIdle, weak_factory_.GetWeakPtr());
64}
65
[email protected]8e937c1e2012-06-28 22:57:3066bool RunLoop::BeforeRun() {
67 DCHECK(!run_called_);
68 run_called_ = true;
69
70 // Allow Quit to be called before Run.
71 if (quit_called_)
72 return false;
73
74 // Push RunLoop stack:
75 previous_run_loop_ = loop_->run_loop_;
76 run_depth_ = previous_run_loop_? previous_run_loop_->run_depth_ + 1 : 1;
77 loop_->run_loop_ = this;
78
jamescookaacdfd02016-04-28 00:50:0379 if (run_depth_ > 1)
80 loop_->NotifyBeginNestedLoop();
81
[email protected]8e937c1e2012-06-28 22:57:3082 running_ = true;
83 return true;
84}
85
86void RunLoop::AfterRun() {
87 running_ = false;
88
89 // Pop RunLoop stack:
90 loop_->run_loop_ = previous_run_loop_;
91
92 // Execute deferred QuitNow, if any:
93 if (previous_run_loop_ && previous_run_loop_->quit_called_)
94 loop_->QuitNow();
95}
96
97} // namespace base