| // 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 "extensions/shell/browser/shell_browser_main_parts.h" |
| |
| #include <string> |
| |
| #include "apps/browser_context_keyed_service_factories.h" |
| #include "base/command_line.h" |
| #include "base/task/post_task.h" |
| #include "build/build_config.h" |
| #include "components/keyed_service/content/browser_context_dependency_manager.h" |
| #include "components/nacl/common/buildflags.h" |
| #include "components/prefs/pref_service.h" |
| #include "components/sessions/core/session_id_generator.h" |
| #include "components/storage_monitor/storage_monitor.h" |
| #include "components/update_client/update_query_params.h" |
| #include "content/public/browser/browser_task_traits.h" |
| #include "content/public/browser/child_process_security_policy.h" |
| #include "content/public/browser/context_factory.h" |
| #include "content/public/browser/devtools_agent_host.h" |
| #include "content/public/common/content_switches.h" |
| #include "content/public/common/result_codes.h" |
| #include "content/public/common/service_manager_connection.h" |
| #include "content/shell/browser/shell_devtools_manager_delegate.h" |
| #include "extensions/browser/browser_context_keyed_service_factories.h" |
| #include "extensions/browser/extension_system.h" |
| #include "extensions/browser/updater/update_service.h" |
| #include "extensions/common/constants.h" |
| #include "extensions/shell/browser/shell_browser_context.h" |
| #include "extensions/shell/browser/shell_browser_context_keyed_service_factories.h" |
| #include "extensions/shell/browser/shell_browser_main_delegate.h" |
| #include "extensions/shell/browser/shell_desktop_controller_aura.h" |
| #include "extensions/shell/browser/shell_device_client.h" |
| #include "extensions/shell/browser/shell_extension_system.h" |
| #include "extensions/shell/browser/shell_extension_system_factory.h" |
| #include "extensions/shell/browser/shell_extensions_browser_client.h" |
| #include "extensions/shell/browser/shell_oauth2_token_service.h" |
| #include "extensions/shell/browser/shell_prefs.h" |
| #include "extensions/shell/browser/shell_update_query_params_delegate.h" |
| #include "extensions/shell/common/shell_extensions_client.h" |
| #include "extensions/shell/common/switches.h" |
| #include "ui/base/ime/input_method_initializer.h" |
| #include "ui/base/resource/resource_bundle.h" |
| |
| #if defined(USE_AURA) |
| #include "ui/aura/env.h" |
| #endif |
| |
| #if defined(OS_CHROMEOS) |
| #include "chromeos/audio/audio_devices_pref_handler_impl.h" |
| #include "chromeos/audio/cras_audio_handler.h" |
| #include "chromeos/disks/disk_mount_manager.h" |
| #include "chromeos/network/network_handler.h" |
| #include "extensions/shell/browser/shell_audio_controller_chromeos.h" |
| #include "extensions/shell/browser/shell_network_controller_chromeos.h" |
| #endif |
| |
| #if defined(OS_LINUX) |
| #include "device/bluetooth/bluetooth_adapter_factory.h" |
| #include "device/bluetooth/dbus/bluez_dbus_manager.h" |
| #endif |
| |
| #if defined(OS_CHROMEOS) |
| #include "chromeos/dbus/dbus_thread_manager.h" |
| #elif defined(OS_LINUX) |
| #include "device/bluetooth/dbus/bluez_dbus_thread_manager.h" |
| #endif |
| |
| #if BUILDFLAG(ENABLE_NACL) |
| #include "components/nacl/browser/nacl_browser.h" |
| #include "components/nacl/browser/nacl_process_host.h" |
| #include "content/public/browser/browser_thread.h" |
| #include "extensions/shell/browser/shell_nacl_browser_delegate.h" |
| #endif |
| |
| #if defined(USE_AURA) && defined(USE_X11) |
| #include "ui/events/devices/x11/touch_factory_x11.h" // nogncheck |
| #endif |
| |
| using base::CommandLine; |
| using content::BrowserContext; |
| |
| #if BUILDFLAG(ENABLE_NACL) |
| using content::BrowserThread; |
| #endif |
| |
| namespace extensions { |
| |
| namespace { |
| |
| // Intentionally dereferences a null pointer to test the crash reporter. |
| void CrashForTest() { |
| int* bad_pointer = nullptr; |
| *bad_pointer = 0; |
| } |
| |
| } // namespace |
| |
| ShellBrowserMainParts::ShellBrowserMainParts( |
| const content::MainFunctionParams& parameters, |
| ShellBrowserMainDelegate* browser_main_delegate) |
| : extension_system_(nullptr), |
| parameters_(parameters), |
| run_message_loop_(true), |
| browser_main_delegate_(browser_main_delegate) { |
| } |
| |
| ShellBrowserMainParts::~ShellBrowserMainParts() { |
| } |
| |
| void ShellBrowserMainParts::PreMainMessageLoopStart() { |
| #if defined(USE_AURA) && defined(USE_X11) |
| ui::TouchFactory::SetTouchDeviceListFromCommandLine(); |
| #endif |
| } |
| |
| void ShellBrowserMainParts::PostMainMessageLoopStart() { |
| #if defined(OS_CHROMEOS) |
| // Perform initialization of D-Bus objects here rather than in the below |
| // helper classes so those classes' tests can initialize stub versions of the |
| // D-Bus objects. |
| chromeos::DBusThreadManager::Initialize(); |
| chromeos::disks::DiskMountManager::Initialize(); |
| |
| bluez::BluezDBusManager::Initialize(); |
| |
| chromeos::NetworkHandler::Initialize(); |
| network_controller_.reset(new ShellNetworkController( |
| base::CommandLine::ForCurrentProcess()->GetSwitchValueNative( |
| switches::kAppShellPreferredNetwork))); |
| |
| if (base::CommandLine::ForCurrentProcess()->HasSwitch( |
| switches::kAppShellAllowRoaming)) { |
| network_controller_->SetCellularAllowRoaming(true); |
| } |
| #elif defined(OS_LINUX) |
| // app_shell doesn't need GTK, so the fake input method context can work. |
| // See crbug.com/381852 and revision fb69f142. |
| // TODO(michaelpg): Verify this works for target environments. |
| ui::InitializeInputMethodForTesting(); |
| |
| bluez::BluezDBusThreadManager::Initialize(); |
| bluez::BluezDBusManager::Initialize(); |
| #else |
| ui::InitializeInputMethodForTesting(); |
| #endif |
| } |
| |
| int ShellBrowserMainParts::PreEarlyInitialization() { |
| return service_manager::RESULT_CODE_NORMAL_EXIT; |
| } |
| |
| int ShellBrowserMainParts::PreCreateThreads() { |
| // TODO(jamescook): Initialize chromeos::CrosSettings here? |
| |
| content::ChildProcessSecurityPolicy::GetInstance()->RegisterWebSafeScheme( |
| kExtensionScheme); |
| |
| // Return no error. |
| return 0; |
| } |
| |
| void ShellBrowserMainParts::PreMainMessageLoopRun() { |
| extensions_client_ = std::make_unique<ShellExtensionsClient>(); |
| ExtensionsClient::Set(extensions_client_.get()); |
| |
| // BrowserContextKeyedAPIServiceFactories require an ExtensionsBrowserClient. |
| extensions_browser_client_ = std::make_unique<ShellExtensionsBrowserClient>(); |
| ExtensionsBrowserClient::Set(extensions_browser_client_.get()); |
| |
| apps::EnsureBrowserContextKeyedServiceFactoriesBuilt(); |
| EnsureBrowserContextKeyedServiceFactoriesBuilt(); |
| shell::EnsureBrowserContextKeyedServiceFactoriesBuilt(); |
| |
| // Initialize our "profile" equivalent. |
| browser_context_ = std::make_unique<ShellBrowserContext>(this); |
| |
| // app_shell only supports a single user, so all preferences live in the user |
| // data directory, including the device-wide local state. |
| local_state_ = shell_prefs::CreateLocalState(browser_context_->GetPath()); |
| sessions::SessionIdGenerator::GetInstance()->Init(local_state_.get()); |
| user_pref_service_ = |
| shell_prefs::CreateUserPrefService(browser_context_.get()); |
| extensions_browser_client_->InitWithBrowserContext(browser_context_.get(), |
| user_pref_service_.get()); |
| |
| #if defined(OS_CHROMEOS) |
| chromeos::CrasAudioHandler::Initialize( |
| new chromeos::AudioDevicesPrefHandlerImpl(local_state_.get())); |
| audio_controller_.reset(new ShellAudioController()); |
| #endif |
| |
| // Create BrowserContextKeyedServices now that we have an |
| // ExtensionsBrowserClient that BrowserContextKeyedAPIServices can query. |
| BrowserContextDependencyManager::GetInstance()->CreateBrowserContextServices( |
| browser_context_.get()); |
| |
| #if defined(USE_AURA) |
| aura::Env::GetInstance()->set_context_factory(content::GetContextFactory()); |
| aura::Env::GetInstance()->set_context_factory_private( |
| content::GetContextFactoryPrivate()); |
| #endif |
| |
| storage_monitor::StorageMonitor::Create( |
| content::ServiceManagerConnection::GetForProcess() |
| ->GetConnector() |
| ->Clone()); |
| |
| desktop_controller_.reset( |
| browser_main_delegate_->CreateDesktopController(browser_context_.get())); |
| |
| // TODO(jamescook): Initialize user_manager::UserManager. |
| |
| device_client_.reset(new ShellDeviceClient); |
| |
| update_query_params_delegate_.reset(new ShellUpdateQueryParamsDelegate); |
| update_client::UpdateQueryParams::SetDelegate( |
| update_query_params_delegate_.get()); |
| |
| InitExtensionSystem(); |
| |
| // Initialize OAuth2 support from command line. |
| base::CommandLine* cmd = base::CommandLine::ForCurrentProcess(); |
| oauth2_token_service_.reset(new ShellOAuth2TokenService( |
| cmd->GetSwitchValueASCII(switches::kAppShellUser), |
| cmd->GetSwitchValueASCII(switches::kAppShellRefreshToken))); |
| |
| #if BUILDFLAG(ENABLE_NACL) |
| nacl::NaClBrowser::SetDelegate( |
| std::make_unique<ShellNaClBrowserDelegate>(browser_context_.get())); |
| // Track the task so it can be canceled if app_shell shuts down very quickly, |
| // such as in browser tests. |
| task_tracker_.PostTask( |
| base::CreateSingleThreadTaskRunnerWithTraits({BrowserThread::IO}).get(), |
| FROM_HERE, base::Bind(nacl::NaClProcessHost::EarlyStartup)); |
| #endif |
| |
| content::ShellDevToolsManagerDelegate::StartHttpHandler( |
| browser_context_.get()); |
| |
| if (cmd->HasSwitch(::switches::kBrowserCrashTest)) |
| CrashForTest(); |
| |
| if (parameters_.ui_task) { |
| // For running browser tests. |
| parameters_.ui_task->Run(); |
| delete parameters_.ui_task; |
| run_message_loop_ = false; |
| } else { |
| browser_main_delegate_->Start(browser_context_.get()); |
| } |
| } |
| |
| bool ShellBrowserMainParts::MainMessageLoopRun(int* result_code) { |
| if (!run_message_loop_) |
| return true; |
| desktop_controller_->Run(); |
| *result_code = service_manager::RESULT_CODE_NORMAL_EXIT; |
| return true; |
| } |
| |
| void ShellBrowserMainParts::PostMainMessageLoopRun() { |
| // Close apps before shutting down browser context and extensions system. |
| desktop_controller_->CloseAppWindows(); |
| |
| // NOTE: Please destroy objects in the reverse order of their creation. |
| browser_main_delegate_->Shutdown(); |
| content::ShellDevToolsManagerDelegate::StopHttpHandler(); |
| |
| #if BUILDFLAG(ENABLE_NACL) |
| task_tracker_.TryCancelAll(); |
| #endif |
| |
| oauth2_token_service_.reset(); |
| BrowserContextDependencyManager::GetInstance()->DestroyBrowserContextServices( |
| browser_context_.get()); |
| extension_system_ = NULL; |
| |
| desktop_controller_.reset(); |
| |
| storage_monitor::StorageMonitor::Destroy(); |
| |
| #if defined(OS_CHROMEOS) |
| audio_controller_.reset(); |
| chromeos::CrasAudioHandler::Shutdown(); |
| #endif |
| |
| sessions::SessionIdGenerator::GetInstance()->Shutdown(); |
| |
| user_pref_service_->CommitPendingWrite(); |
| user_pref_service_.reset(); |
| local_state_->CommitPendingWrite(); |
| local_state_.reset(); |
| |
| browser_context_.reset(); |
| } |
| |
| void ShellBrowserMainParts::PostDestroyThreads() { |
| extensions_browser_client_.reset(); |
| ExtensionsBrowserClient::Set(nullptr); |
| |
| #if defined(OS_CHROMEOS) |
| network_controller_.reset(); |
| chromeos::NetworkHandler::Shutdown(); |
| chromeos::disks::DiskMountManager::Shutdown(); |
| device::BluetoothAdapterFactory::Shutdown(); |
| bluez::BluezDBusManager::Shutdown(); |
| chromeos::DBusThreadManager::Shutdown(); |
| #elif defined(OS_LINUX) |
| device::BluetoothAdapterFactory::Shutdown(); |
| bluez::BluezDBusManager::Shutdown(); |
| bluez::BluezDBusThreadManager::Shutdown(); |
| #endif |
| } |
| |
| void ShellBrowserMainParts::InitExtensionSystem() { |
| DCHECK(browser_context_); |
| extension_system_ = static_cast<ShellExtensionSystem*>( |
| ExtensionSystem::Get(browser_context_.get())); |
| extension_system_->InitForRegularProfile(true /* extensions_enabled */); |
| } |
| |
| } // namespace extensions |