blob: 2aa4def95706a24df2075205c032021543709ab0 [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"
[email protected]8e937c1e2012-06-28 22:57:309
[email protected]b9f12d8f2014-04-16 05:29:4910#if defined(OS_WIN)
[email protected]440e3572014-01-17 00:10:2911#include "base/message_loop/message_pump_dispatcher.h"
12#endif
13
[email protected]8e937c1e2012-06-28 22:57:3014namespace base {
15
16RunLoop::RunLoop()
17 : loop_(MessageLoop::current()),
[email protected]8e937c1e2012-06-28 22:57:3018 previous_run_loop_(NULL),
19 run_depth_(0),
20 run_called_(false),
21 quit_called_(false),
22 running_(false),
[email protected]dcf10632013-10-08 19:23:3323 quit_when_idle_received_(false),
24 weak_factory_(this) {
[email protected]9a2e75d2014-04-05 15:24:0325#if defined(OS_WIN)
[email protected]8e937c1e2012-06-28 22:57:3026 dispatcher_ = NULL;
27#endif
28}
29
[email protected]9a2e75d2014-04-05 15:24:0330#if defined(OS_WIN)
[email protected]440e3572014-01-17 00:10:2931RunLoop::RunLoop(MessagePumpDispatcher* dispatcher)
[email protected]8e937c1e2012-06-28 22:57:3032 : loop_(MessageLoop::current()),
[email protected]8e937c1e2012-06-28 22:57:3033 previous_run_loop_(NULL),
34 dispatcher_(dispatcher),
35 run_depth_(0),
36 run_called_(false),
37 quit_called_(false),
38 running_(false),
[email protected]dcf10632013-10-08 19:23:3339 quit_when_idle_received_(false),
40 weak_factory_(this) {
[email protected]8e937c1e2012-06-28 22:57:3041}
42#endif
43
44RunLoop::~RunLoop() {
45}
46
47void RunLoop::Run() {
48 if (!BeforeRun())
49 return;
vadimt12f0f7d2014-09-15 19:19:3850
51 // Use task stopwatch to exclude the loop run time from the current task, if
52 // any.
53 tracked_objects::TaskStopwatch stopwatch;
vadimt20175532014-10-28 20:14:2054 stopwatch.Start();
[email protected]8e937c1e2012-06-28 22:57:3055 loop_->RunHandler();
vadimt12f0f7d2014-09-15 19:19:3856 stopwatch.Stop();
57
[email protected]8e937c1e2012-06-28 22:57:3058 AfterRun();
59}
60
61void RunLoop::RunUntilIdle() {
62 quit_when_idle_received_ = true;
63 Run();
64}
65
66void RunLoop::Quit() {
67 quit_called_ = true;
68 if (running_ && loop_->run_loop_ == this) {
69 // This is the inner-most RunLoop, so quit now.
70 loop_->QuitNow();
71 }
72}
73
74base::Closure RunLoop::QuitClosure() {
75 return base::Bind(&RunLoop::Quit, weak_factory_.GetWeakPtr());
76}
77
78bool RunLoop::BeforeRun() {
79 DCHECK(!run_called_);
80 run_called_ = true;
81
82 // Allow Quit to be called before Run.
83 if (quit_called_)
84 return false;
85
86 // Push RunLoop stack:
87 previous_run_loop_ = loop_->run_loop_;
88 run_depth_ = previous_run_loop_? previous_run_loop_->run_depth_ + 1 : 1;
89 loop_->run_loop_ = this;
90
91 running_ = true;
92 return true;
93}
94
95void RunLoop::AfterRun() {
96 running_ = false;
97
98 // Pop RunLoop stack:
99 loop_->run_loop_ = previous_run_loop_;
100
101 // Execute deferred QuitNow, if any:
102 if (previous_run_loop_ && previous_run_loop_->quit_called_)
103 loop_->QuitNow();
104}
105
106} // namespace base