Index: ios/web/app/web_main_loop.mm |
diff --git a/ios/web/app/web_main_loop.mm b/ios/web/app/web_main_loop.mm |
new file mode 100644 |
index 0000000000000000000000000000000000000000..b3300d4cf9320ad3ac79340e8f0c7f412fbd8c51 |
--- /dev/null |
+++ b/ios/web/app/web_main_loop.mm |
@@ -0,0 +1,278 @@ |
+// Copyright 2014 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "ios/web/app/web_main_loop.h" |
+ |
+#include "base/bind.h" |
+#include "base/command_line.h" |
+#include "base/logging.h" |
+#include "base/message_loop/message_loop.h" |
+#include "base/metrics/histogram.h" |
+#include "base/path_service.h" |
+#include "base/power_monitor/power_monitor.h" |
+#include "base/power_monitor/power_monitor_device_source.h" |
+#include "base/process/process_metrics.h" |
+#include "base/system_monitor/system_monitor.h" |
+#include "base/threading/thread_restrictions.h" |
+#include "crypto/nss_util.h" |
+#include "ios/web/net/cookie_notification_bridge.h" |
+#include "ios/web/public/app/web_main_parts.h" |
+#include "ios/web/public/web_client.h" |
+#include "ios/web/web_thread_impl.h" |
+#include "net/base/network_change_notifier.h" |
+ |
+namespace web { |
+ |
+// The currently-running WebMainLoop. There can be one or zero. |
+// TODO(rohitrao): Desktop uses this to implement |
+// ImmediateShutdownAndExitProcess. If we don't need that functionality, we can |
+// remove this. |
+WebMainLoop* g_current_web_main_loop = nullptr; |
+ |
+WebMainLoop::WebMainLoop() : result_code_(0), created_threads_(false) { |
+ DCHECK(!g_current_web_main_loop); |
+ g_current_web_main_loop = this; |
+} |
+ |
+WebMainLoop::~WebMainLoop() { |
+ DCHECK_EQ(this, g_current_web_main_loop); |
+ g_current_web_main_loop = nullptr; |
+} |
+ |
+void WebMainLoop::Init() { |
+ parts_.reset(web::GetWebClient()->CreateWebMainParts()); |
+} |
+ |
+void WebMainLoop::EarlyInitialization() { |
+ if (parts_) { |
+ parts_->PreEarlyInitialization(); |
+ } |
+ |
+#if !defined(USE_OPENSSL) |
+ // We want to be sure to init NSPR on the main thread. |
+ crypto::EnsureNSPRInit(); |
+#endif // !defined(USE_OPENSSL) |
+ |
+ if (parts_) { |
+ parts_->PostEarlyInitialization(); |
+ } |
+} |
+ |
+void WebMainLoop::MainMessageLoopStart() { |
+ if (parts_) { |
+ parts_->PreMainMessageLoopStart(); |
+ } |
+ |
+ // Create a MessageLoop if one does not already exist for the current thread. |
+ if (!base::MessageLoop::current()) { |
+ main_message_loop_.reset(new base::MessageLoopForUI); |
+ } |
+ // Note: In Chrome, Attach() is called in |
+ // ChromeBrowserMainPartsIOS::PreMainMessageLoopStart(). |
+ base::MessageLoopForUI::current()->Attach(); |
+ |
+ InitializeMainThread(); |
+ |
+#if 0 |
+ // TODO(droger): SystemMonitor is not working properly on iOS. |
+ // See http://crbug.com/228014. |
+ system_monitor_.reset(new base::SystemMonitor); |
+#endif |
+ // TODO(rohitrao): Do we need PowerMonitor on iOS, or can we get rid of it? |
+ scoped_ptr<base::PowerMonitorSource> power_monitor_source( |
+ new base::PowerMonitorDeviceSource()); |
+ power_monitor_.reset(new base::PowerMonitor(power_monitor_source.Pass())); |
+ network_change_notifier_.reset(net::NetworkChangeNotifier::Create()); |
+ |
+ if (parts_) { |
+ parts_->PostMainMessageLoopStart(); |
+ } |
+} |
+ |
+void WebMainLoop::CreateStartupTasks() { |
+ int result = 0; |
+ result = PreCreateThreads(); |
+ if (result > 0) |
+ return; |
+ |
+ result = CreateThreads(); |
+ if (result > 0) |
+ return; |
+ |
+ result = WebThreadsStarted(); |
+ if (result > 0) |
+ return; |
+ |
+ result = PreMainMessageLoopRun(); |
+ if (result > 0) |
+ return; |
+} |
+ |
+int WebMainLoop::PreCreateThreads() { |
+ if (parts_) { |
+ result_code_ = parts_->PreCreateThreads(); |
+ } |
+ |
+ return result_code_; |
+} |
+ |
+int WebMainLoop::CreateThreads() { |
+ base::Thread::Options default_options; |
+ base::Thread::Options io_message_loop_options; |
+ io_message_loop_options.message_loop_type = base::MessageLoop::TYPE_IO; |
+ base::Thread::Options ui_message_loop_options; |
+ ui_message_loop_options.message_loop_type = base::MessageLoop::TYPE_UI; |
+ |
+ // Start threads in the order they occur in the WebThread::ID |
+ // enumeration, except for WebThread::UI which is the main |
+ // thread. |
+ // |
+ // Must be size_t so we can increment it. |
+ for (size_t thread_id = WebThread::UI + 1; thread_id < WebThread::ID_COUNT; |
+ ++thread_id) { |
+ scoped_ptr<WebThreadImpl>* thread_to_start = nullptr; |
+ base::Thread::Options* options = &default_options; |
+ |
+ switch (thread_id) { |
+ // TODO(rohitrao): We probably do not need all of these threads. Remove |
+ // the ones that serve no purpose. http://crbug.com/365909 |
+ case WebThread::DB: |
+ thread_to_start = &db_thread_; |
+ break; |
+ case WebThread::FILE_USER_BLOCKING: |
+ thread_to_start = &file_user_blocking_thread_; |
+ break; |
+ case WebThread::FILE: |
+ thread_to_start = &file_thread_; |
+ options = &io_message_loop_options; |
+ break; |
+ case WebThread::CACHE: |
+ thread_to_start = &cache_thread_; |
+ options = &io_message_loop_options; |
+ break; |
+ case WebThread::IO: |
+ thread_to_start = &io_thread_; |
+ options = &io_message_loop_options; |
+ break; |
+ case WebThread::UI: |
+ case WebThread::ID_COUNT: |
+ default: |
+ NOTREACHED(); |
+ break; |
+ } |
+ |
+ WebThread::ID id = static_cast<WebThread::ID>(thread_id); |
+ |
+ if (thread_to_start) { |
+ (*thread_to_start).reset(new WebThreadImpl(id)); |
+ (*thread_to_start)->StartWithOptions(*options); |
+ } else { |
+ NOTREACHED(); |
+ } |
+ } |
+ created_threads_ = true; |
+ return result_code_; |
+} |
+ |
+int WebMainLoop::PreMainMessageLoopRun() { |
+ if (parts_) { |
+ parts_->PreMainMessageLoopRun(); |
+ } |
+ |
+ // If the UI thread blocks, the whole UI is unresponsive. |
+ // Do not allow disk IO from the UI thread. |
+ base::ThreadRestrictions::SetIOAllowed(false); |
+ base::ThreadRestrictions::DisallowWaiting(); |
+ return result_code_; |
+} |
+ |
+void WebMainLoop::ShutdownThreadsAndCleanUp() { |
+ if (!created_threads_) { |
+ // Called early, nothing to do |
+ return; |
+ } |
+ |
+ // Teardown may start in PostMainMessageLoopRun, and during teardown we |
+ // need to be able to perform IO. |
+ base::ThreadRestrictions::SetIOAllowed(true); |
+ WebThread::PostTask( |
+ WebThread::IO, FROM_HERE, |
+ base::Bind(base::IgnoreResult(&base::ThreadRestrictions::SetIOAllowed), |
+ true)); |
+ |
+ if (parts_) { |
+ parts_->PostMainMessageLoopRun(); |
+ } |
+ |
+ // Must be size_t so we can subtract from it. |
+ for (size_t thread_id = WebThread::ID_COUNT - 1; |
+ thread_id >= (WebThread::UI + 1); --thread_id) { |
+ // Find the thread object we want to stop. Looping over all valid |
+ // WebThread IDs and DCHECKing on a missing case in the switch |
+ // statement helps avoid a mismatch between this code and the |
+ // WebThread::ID enumeration. |
+ // |
+ // The destruction order is the reverse order of occurrence in the |
+ // WebThread::ID list. The rationale for the order is as |
+ // follows (need to be filled in a bit): |
+ // |
+ // |
+ // - The IO thread is the only user of the CACHE thread. |
+ // |
+ // - (Not sure why DB stops last.) |
+ switch (thread_id) { |
+ case WebThread::DB: |
+ db_thread_.reset(); |
+ break; |
+ case WebThread::FILE_USER_BLOCKING: |
+ file_user_blocking_thread_.reset(); |
+ break; |
+ case WebThread::FILE: |
+ file_thread_.reset(); |
+ break; |
+ case WebThread::CACHE: |
+ cache_thread_.reset(); |
+ break; |
+ case WebThread::IO: |
+ io_thread_.reset(); |
+ break; |
+ case WebThread::UI: |
+ case WebThread::ID_COUNT: |
+ default: |
+ NOTREACHED(); |
+ break; |
+ } |
+ } |
+ |
+ // Close the blocking I/O pool after the other threads. Other threads such |
+ // as the I/O thread may need to schedule work like closing files or flushing |
+ // data during shutdown, so the blocking pool needs to be available. There |
+ // may also be slow operations pending that will block shutdown, so closing |
+ // it here (which will block until required operations are complete) gives |
+ // more head start for those operations to finish. |
+ WebThreadImpl::ShutdownThreadPool(); |
+ |
+ if (parts_) { |
+ parts_->PostDestroyThreads(); |
+ } |
+} |
+ |
+void WebMainLoop::InitializeMainThread() { |
+ const char* kThreadName = "CrWebMain"; |
+ base::PlatformThread::SetName(kThreadName); |
+ if (main_message_loop_) { |
+ main_message_loop_->set_thread_name(kThreadName); |
+ } |
+ |
+ // Register the main thread by instantiating it, but don't call any methods. |
+ main_thread_.reset( |
+ new WebThreadImpl(WebThread::UI, base::MessageLoop::current())); |
+} |
+ |
+int WebMainLoop::WebThreadsStarted() { |
+ cookie_notification_bridge_.reset(new CookieNotificationBridge); |
+ return result_code_; |
+} |
+ |
+} // namespace web |