// Copyright (c) 2012 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 "chrome/browser/browser_shutdown.h" #include #include #include "base/bind.h" #include "base/command_line.h" #include "base/file_util.h" #include "base/files/file_path.h" #include "base/metrics/histogram.h" #include "base/path_service.h" #include "base/prefs/pref_registry_simple.h" #include "base/prefs/pref_service.h" #include "base/strings/string_number_conversions.h" #include "base/strings/stringprintf.h" #include "base/threading/thread.h" #include "base/time/time.h" #include "chrome/browser/about_flags.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/first_run/upgrade_util.h" #include "chrome/browser/jankometer.h" #include "chrome/browser/lifetime/application_lifetime.h" #include "chrome/browser/profiles/profile_manager.h" #include "chrome/browser/service_process/service_process_control.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/crash_keys.h" #include "chrome/common/pref_names.h" #include "chrome/common/switch_utils.h" #include "components/metrics/metrics_service.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/render_process_host.h" #if defined(OS_WIN) #include "chrome/browser/browser_util_win.h" #include "chrome/browser/first_run/upgrade_util_win.h" #endif #if defined(ENABLE_RLZ) #include "chrome/browser/rlz/rlz.h" #endif #if defined(OS_CHROMEOS) #include "chrome/browser/chromeos/boot_times_loader.h" #endif using base::Time; using base::TimeDelta; using content::BrowserThread; namespace browser_shutdown { namespace { // Whether the browser is trying to quit (e.g., Quit chosen from menu). bool g_trying_to_quit = false; #if defined(OS_WIN) upgrade_util::RelaunchMode g_relaunch_mode = upgrade_util::RELAUNCH_MODE_DEFAULT; #endif Time* shutdown_started_ = NULL; ShutdownType shutdown_type_ = NOT_VALID; int shutdown_num_processes_; int shutdown_num_processes_slow_; const char kShutdownMsFile[] = "chrome_shutdown_ms.txt"; const char* ToShutdownTypeString(ShutdownType type) { switch (type) { case NOT_VALID: NOTREACHED(); return ""; case WINDOW_CLOSE: return "close"; case BROWSER_EXIT: return "exit"; case END_SESSION: return "end"; } return ""; } } // namespace void RegisterPrefs(PrefRegistrySimple* registry) { registry->RegisterIntegerPref(prefs::kShutdownType, NOT_VALID); registry->RegisterIntegerPref(prefs::kShutdownNumProcesses, 0); registry->RegisterIntegerPref(prefs::kShutdownNumProcessesSlow, 0); } ShutdownType GetShutdownType() { return shutdown_type_; } void OnShutdownStarting(ShutdownType type) { if (shutdown_type_ != NOT_VALID) return; base::debug::SetCrashKeyValue(crash_keys::kShutdownType, ToShutdownTypeString(type)); #if !defined(OS_CHROMEOS) // Start the shutdown tracing. Note that On ChromeOS we have started this // already. chrome::StartShutdownTracing(); #endif shutdown_type_ = type; // For now, we're only counting the number of renderer processes // since we can't safely count the number of plugin processes from this // thread, and we'd really like to avoid anything which might add further // delays to shutdown time. DCHECK(!shutdown_started_); shutdown_started_ = new Time(Time::Now()); // Call FastShutdown on all of the RenderProcessHosts. This will be // a no-op in some cases, so we still need to go through the normal // shutdown path for the ones that didn't exit here. shutdown_num_processes_ = 0; shutdown_num_processes_slow_ = 0; for (content::RenderProcessHost::iterator i( content::RenderProcessHost::AllHostsIterator()); !i.IsAtEnd(); i.Advance()) { ++shutdown_num_processes_; if (!i.GetCurrentValue()->FastShutdownIfPossible()) ++shutdown_num_processes_slow_; } } base::FilePath GetShutdownMsPath() { base::FilePath shutdown_ms_file; PathService::Get(chrome::DIR_USER_DATA, &shutdown_ms_file); return shutdown_ms_file.AppendASCII(kShutdownMsFile); } bool ShutdownPreThreadsStop() { #if defined(OS_CHROMEOS) chromeos::BootTimesLoader::Get()->AddLogoutTimeMarker( "BrowserShutdownStarted", false); #endif // Shutdown the IPC channel to the service processes. ServiceProcessControl::GetInstance()->Disconnect(); // WARNING: During logoff/shutdown (WM_ENDSESSION) we may not have enough // time to get here. If you have something that *must* happen on end session, // consider putting it in BrowserProcessImpl::EndSession. PrefService* prefs = g_browser_process->local_state(); MetricsService* metrics = g_browser_process->metrics_service(); if (metrics) metrics->RecordCompletedSessionEnd(); if (shutdown_type_ > NOT_VALID && shutdown_num_processes_ > 0) { // Record the shutdown info so that we can put it into a histogram at next // startup. prefs->SetInteger(prefs::kShutdownType, shutdown_type_); prefs->SetInteger(prefs::kShutdownNumProcesses, shutdown_num_processes_); prefs->SetInteger(prefs::kShutdownNumProcessesSlow, shutdown_num_processes_slow_); } // Check local state for the restart flag so we can restart the session below. bool restart_last_session = false; if (prefs->HasPrefPath(prefs::kRestartLastSessionOnShutdown)) { restart_last_session = prefs->GetBoolean(prefs::kRestartLastSessionOnShutdown); prefs->ClearPref(prefs::kRestartLastSessionOnShutdown); #if defined(OS_WIN) if (restart_last_session) { if (prefs->HasPrefPath(prefs::kRelaunchMode)) { g_relaunch_mode = upgrade_util::RelaunchModeStringToEnum( prefs->GetString(prefs::kRelaunchMode)); prefs->ClearPref(prefs::kRelaunchMode); } } #endif } prefs->CommitPendingWrite(); #if defined(ENABLE_RLZ) // Cleanup any statics created by RLZ. Must be done before NotificationService // is destroyed. RLZTracker::CleanupRlz(); #endif return restart_last_session; } void ShutdownPostThreadsStop(bool restart_last_session) { // The jank'o'meter requires that the browser process has been destroyed // before calling UninstallJankometer(). delete g_browser_process; g_browser_process = NULL; // crbug.com/95079 - This needs to happen after the browser process object // goes away. ProfileManager::NukeDeletedProfilesFromDisk(); #if defined(OS_CHROMEOS) chromeos::BootTimesLoader::Get()->AddLogoutTimeMarker("BrowserDeleted", true); #endif // Uninstall Jank-O-Meter here after the IO thread is no longer running. UninstallJankometer(); #if defined(OS_WIN) if (!browser_util::IsBrowserAlreadyRunning() && shutdown_type_ != browser_shutdown::END_SESSION) { upgrade_util::SwapNewChromeExeIfPresent(); } #endif if (restart_last_session) { #if !defined(OS_CHROMEOS) // Make sure to relaunch the browser with the original command line plus // the Restore Last Session flag. Note that Chrome can be launched (ie. // through ShellExecute on Windows) with a switch argument terminator at // the end (double dash, as described in b/1366444) plus a URL, // which prevents us from appending to the command line directly (issue // 46182). We therefore use GetSwitches to copy the command line (it stops // at the switch argument terminator). CommandLine old_cl(*CommandLine::ForCurrentProcess()); scoped_ptr new_cl(new CommandLine(old_cl.GetProgram())); std::map switches = old_cl.GetSwitches(); // Remove the switches that shouldn't persist across restart. about_flags::RemoveFlagsSwitches(&switches); switches::RemoveSwitchesForAutostart(&switches); // Append the old switches to the new command line. for (std::map::const_iterator i = switches.begin(); i != switches.end(); ++i) { CommandLine::StringType switch_value = i->second; if (!switch_value.empty()) new_cl->AppendSwitchNative(i->first, i->second); else new_cl->AppendSwitch(i->first); } #if defined(OS_WIN) upgrade_util::RelaunchChromeWithMode(*new_cl.get(), g_relaunch_mode); #else upgrade_util::RelaunchChromeBrowser(*new_cl.get()); #endif // defined(OS_WIN) #else NOTIMPLEMENTED(); #endif // !defined(OS_CHROMEOS) } if (shutdown_type_ > NOT_VALID && shutdown_num_processes_ > 0) { // Measure total shutdown time as late in the process as possible // and then write it to a file to be read at startup. // We can't use prefs since all services are shutdown at this point. TimeDelta shutdown_delta = Time::Now() - *shutdown_started_; std::string shutdown_ms = base::Int64ToString(shutdown_delta.InMilliseconds()); int len = static_cast(shutdown_ms.length()) + 1; base::FilePath shutdown_ms_file = GetShutdownMsPath(); base::WriteFile(shutdown_ms_file, shutdown_ms.c_str(), len); } #if defined(OS_CHROMEOS) chrome::NotifyAndTerminate(false); #endif } void ReadLastShutdownFile(ShutdownType type, int num_procs, int num_procs_slow) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); base::FilePath shutdown_ms_file = GetShutdownMsPath(); std::string shutdown_ms_str; int64 shutdown_ms = 0; if (base::ReadFileToString(shutdown_ms_file, &shutdown_ms_str)) base::StringToInt64(shutdown_ms_str, &shutdown_ms); base::DeleteFile(shutdown_ms_file, false); if (type == NOT_VALID || shutdown_ms == 0 || num_procs == 0) return; const char* time_fmt = "Shutdown.%s.time"; const char* time_per_fmt = "Shutdown.%s.time_per_process"; std::string time; std::string time_per; if (type == WINDOW_CLOSE) { time = base::StringPrintf(time_fmt, "window_close"); time_per = base::StringPrintf(time_per_fmt, "window_close"); } else if (type == BROWSER_EXIT) { time = base::StringPrintf(time_fmt, "browser_exit"); time_per = base::StringPrintf(time_per_fmt, "browser_exit"); } else if (type == END_SESSION) { time = base::StringPrintf(time_fmt, "end_session"); time_per = base::StringPrintf(time_per_fmt, "end_session"); } else { NOTREACHED(); } if (time.empty()) return; // TODO(erikkay): change these to UMA histograms after a bit more testing. UMA_HISTOGRAM_TIMES(time.c_str(), TimeDelta::FromMilliseconds(shutdown_ms)); UMA_HISTOGRAM_TIMES(time_per.c_str(), TimeDelta::FromMilliseconds(shutdown_ms / num_procs)); UMA_HISTOGRAM_COUNTS_100("Shutdown.renderers.total", num_procs); UMA_HISTOGRAM_COUNTS_100("Shutdown.renderers.slow", num_procs_slow); } void ReadLastShutdownInfo() { PrefService* prefs = g_browser_process->local_state(); ShutdownType type = static_cast(prefs->GetInteger(prefs::kShutdownType)); int num_procs = prefs->GetInteger(prefs::kShutdownNumProcesses); int num_procs_slow = prefs->GetInteger(prefs::kShutdownNumProcessesSlow); // clear the prefs immediately so we don't pick them up on a future run prefs->SetInteger(prefs::kShutdownType, NOT_VALID); prefs->SetInteger(prefs::kShutdownNumProcesses, 0); prefs->SetInteger(prefs::kShutdownNumProcessesSlow, 0); // Read and delete the file on the file thread. BrowserThread::PostTask( BrowserThread::FILE, FROM_HERE, base::Bind(&ReadLastShutdownFile, type, num_procs, num_procs_slow)); } void SetTryingToQuit(bool quitting) { g_trying_to_quit = quitting; } bool IsTryingToQuit() { return g_trying_to_quit; } } // namespace browser_shutdown