Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(139)

Side by Side Diff: ios/web/app/web_main_loop.mm

Issue 1149323004: Upstream the iOS web_shell and supporting code (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@web-thread-impl
Patch Set: Address review comments Created 5 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « ios/web/app/web_main_loop.h ('k') | ios/web/app/web_main_runner.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // Copyright 2014 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 "ios/web/app/web_main_loop.h"
6
7 #include "base/bind.h"
8 #include "base/command_line.h"
9 #include "base/logging.h"
10 #include "base/message_loop/message_loop.h"
11 #include "base/metrics/histogram.h"
12 #include "base/path_service.h"
13 #include "base/power_monitor/power_monitor.h"
14 #include "base/power_monitor/power_monitor_device_source.h"
15 #include "base/process/process_metrics.h"
16 #include "base/system_monitor/system_monitor.h"
17 #include "base/threading/thread_restrictions.h"
18 #include "crypto/nss_util.h"
19 #include "ios/web/net/cookie_notification_bridge.h"
20 #include "ios/web/public/app/web_main_parts.h"
21 #include "ios/web/public/web_client.h"
22 #include "ios/web/web_thread_impl.h"
23 #include "net/base/network_change_notifier.h"
24
25 namespace web {
26
27 // The currently-running WebMainLoop. There can be one or zero.
28 // TODO(rohitrao): Desktop uses this to implement
29 // ImmediateShutdownAndExitProcess. If we don't need that functionality, we can
30 // remove this.
31 WebMainLoop* g_current_web_main_loop = nullptr;
32
33 WebMainLoop::WebMainLoop() : result_code_(0), created_threads_(false) {
34 DCHECK(!g_current_web_main_loop);
35 g_current_web_main_loop = this;
36 }
37
38 WebMainLoop::~WebMainLoop() {
39 DCHECK_EQ(this, g_current_web_main_loop);
40 g_current_web_main_loop = nullptr;
41 }
42
43 void WebMainLoop::Init() {
44 parts_.reset(web::GetWebClient()->CreateWebMainParts());
45 }
46
47 void WebMainLoop::EarlyInitialization() {
48 if (parts_) {
49 parts_->PreEarlyInitialization();
50 }
51
52 #if !defined(USE_OPENSSL)
53 // We want to be sure to init NSPR on the main thread.
54 crypto::EnsureNSPRInit();
55 #endif // !defined(USE_OPENSSL)
56
57 if (parts_) {
58 parts_->PostEarlyInitialization();
59 }
60 }
61
62 void WebMainLoop::MainMessageLoopStart() {
63 if (parts_) {
64 parts_->PreMainMessageLoopStart();
65 }
66
67 // Create a MessageLoop if one does not already exist for the current thread.
68 if (!base::MessageLoop::current()) {
69 main_message_loop_.reset(new base::MessageLoopForUI);
70 }
71 // Note: In Chrome, Attach() is called in
72 // ChromeBrowserMainPartsIOS::PreMainMessageLoopStart().
73 base::MessageLoopForUI::current()->Attach();
74
75 InitializeMainThread();
76
77 #if 0
78 // TODO(droger): SystemMonitor is not working properly on iOS.
79 // See http://crbug.com/228014.
80 system_monitor_.reset(new base::SystemMonitor);
81 #endif
82 // TODO(rohitrao): Do we need PowerMonitor on iOS, or can we get rid of it?
83 scoped_ptr<base::PowerMonitorSource> power_monitor_source(
84 new base::PowerMonitorDeviceSource());
85 power_monitor_.reset(new base::PowerMonitor(power_monitor_source.Pass()));
86 network_change_notifier_.reset(net::NetworkChangeNotifier::Create());
87
88 if (parts_) {
89 parts_->PostMainMessageLoopStart();
90 }
91 }
92
93 void WebMainLoop::CreateStartupTasks() {
94 int result = 0;
95 result = PreCreateThreads();
96 if (result > 0)
97 return;
98
99 result = CreateThreads();
100 if (result > 0)
101 return;
102
103 result = WebThreadsStarted();
104 if (result > 0)
105 return;
106
107 result = PreMainMessageLoopRun();
108 if (result > 0)
109 return;
110 }
111
112 int WebMainLoop::PreCreateThreads() {
113 if (parts_) {
114 result_code_ = parts_->PreCreateThreads();
115 }
116
117 return result_code_;
118 }
119
120 int WebMainLoop::CreateThreads() {
121 base::Thread::Options default_options;
122 base::Thread::Options io_message_loop_options;
123 io_message_loop_options.message_loop_type = base::MessageLoop::TYPE_IO;
124 base::Thread::Options ui_message_loop_options;
125 ui_message_loop_options.message_loop_type = base::MessageLoop::TYPE_UI;
126
127 // Start threads in the order they occur in the WebThread::ID
128 // enumeration, except for WebThread::UI which is the main
129 // thread.
130 //
131 // Must be size_t so we can increment it.
132 for (size_t thread_id = WebThread::UI + 1; thread_id < WebThread::ID_COUNT;
133 ++thread_id) {
134 scoped_ptr<WebThreadImpl>* thread_to_start = nullptr;
135 base::Thread::Options* options = &default_options;
136
137 switch (thread_id) {
138 // TODO(rohitrao): We probably do not need all of these threads. Remove
139 // the ones that serve no purpose. http://crbug.com/365909
140 case WebThread::DB:
141 thread_to_start = &db_thread_;
142 break;
143 case WebThread::FILE_USER_BLOCKING:
144 thread_to_start = &file_user_blocking_thread_;
145 break;
146 case WebThread::FILE:
147 thread_to_start = &file_thread_;
148 options = &io_message_loop_options;
149 break;
150 case WebThread::CACHE:
151 thread_to_start = &cache_thread_;
152 options = &io_message_loop_options;
153 break;
154 case WebThread::IO:
155 thread_to_start = &io_thread_;
156 options = &io_message_loop_options;
157 break;
158 case WebThread::UI:
159 case WebThread::ID_COUNT:
160 default:
161 NOTREACHED();
162 break;
163 }
164
165 WebThread::ID id = static_cast<WebThread::ID>(thread_id);
166
167 if (thread_to_start) {
168 (*thread_to_start).reset(new WebThreadImpl(id));
169 (*thread_to_start)->StartWithOptions(*options);
170 } else {
171 NOTREACHED();
172 }
173 }
174 created_threads_ = true;
175 return result_code_;
176 }
177
178 int WebMainLoop::PreMainMessageLoopRun() {
179 if (parts_) {
180 parts_->PreMainMessageLoopRun();
181 }
182
183 // If the UI thread blocks, the whole UI is unresponsive.
184 // Do not allow disk IO from the UI thread.
185 base::ThreadRestrictions::SetIOAllowed(false);
186 base::ThreadRestrictions::DisallowWaiting();
187 return result_code_;
188 }
189
190 void WebMainLoop::ShutdownThreadsAndCleanUp() {
191 if (!created_threads_) {
192 // Called early, nothing to do
193 return;
194 }
195
196 // Teardown may start in PostMainMessageLoopRun, and during teardown we
197 // need to be able to perform IO.
198 base::ThreadRestrictions::SetIOAllowed(true);
199 WebThread::PostTask(
200 WebThread::IO, FROM_HERE,
201 base::Bind(base::IgnoreResult(&base::ThreadRestrictions::SetIOAllowed),
202 true));
203
204 if (parts_) {
205 parts_->PostMainMessageLoopRun();
206 }
207
208 // Must be size_t so we can subtract from it.
209 for (size_t thread_id = WebThread::ID_COUNT - 1;
210 thread_id >= (WebThread::UI + 1); --thread_id) {
211 // Find the thread object we want to stop. Looping over all valid
212 // WebThread IDs and DCHECKing on a missing case in the switch
213 // statement helps avoid a mismatch between this code and the
214 // WebThread::ID enumeration.
215 //
216 // The destruction order is the reverse order of occurrence in the
217 // WebThread::ID list. The rationale for the order is as
218 // follows (need to be filled in a bit):
219 //
220 //
221 // - The IO thread is the only user of the CACHE thread.
222 //
223 // - (Not sure why DB stops last.)
224 switch (thread_id) {
225 case WebThread::DB:
226 db_thread_.reset();
227 break;
228 case WebThread::FILE_USER_BLOCKING:
229 file_user_blocking_thread_.reset();
230 break;
231 case WebThread::FILE:
232 file_thread_.reset();
233 break;
234 case WebThread::CACHE:
235 cache_thread_.reset();
236 break;
237 case WebThread::IO:
238 io_thread_.reset();
239 break;
240 case WebThread::UI:
241 case WebThread::ID_COUNT:
242 default:
243 NOTREACHED();
244 break;
245 }
246 }
247
248 // Close the blocking I/O pool after the other threads. Other threads such
249 // as the I/O thread may need to schedule work like closing files or flushing
250 // data during shutdown, so the blocking pool needs to be available. There
251 // may also be slow operations pending that will block shutdown, so closing
252 // it here (which will block until required operations are complete) gives
253 // more head start for those operations to finish.
254 WebThreadImpl::ShutdownThreadPool();
255
256 if (parts_) {
257 parts_->PostDestroyThreads();
258 }
259 }
260
261 void WebMainLoop::InitializeMainThread() {
262 const char* kThreadName = "CrWebMain";
263 base::PlatformThread::SetName(kThreadName);
264 if (main_message_loop_) {
265 main_message_loop_->set_thread_name(kThreadName);
266 }
267
268 // Register the main thread by instantiating it, but don't call any methods.
269 main_thread_.reset(
270 new WebThreadImpl(WebThread::UI, base::MessageLoop::current()));
271 }
272
273 int WebMainLoop::WebThreadsStarted() {
274 cookie_notification_bridge_.reset(new CookieNotificationBridge);
275 return result_code_;
276 }
277
278 } // namespace web
OLDNEW
« no previous file with comments | « ios/web/app/web_main_loop.h ('k') | ios/web/app/web_main_runner.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698