// Copyright (c) 2009 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_list.h" #include "base/histogram.h" #include "base/logging.h" #include "base/message_loop.h" #include "build/build_config.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/browser_shutdown.h" #include "chrome/browser/browser_window.h" #include "chrome/browser/profile_manager.h" #include "chrome/browser/renderer_host/render_process_host.h" #include "chrome/browser/tab_contents/navigation_controller.h" #include "chrome/common/notification_registrar.h" #include "chrome/common/notification_service.h" #include "chrome/common/result_codes.h" #if defined(OS_MACOSX) #include "chrome/browser/chrome_browser_application_mac.h" #endif namespace { // This object is instantiated when the first Browser object is added to the // list and delete when the last one is removed. It watches for loads and // creates histograms of some global object counts. class BrowserActivityObserver : public NotificationObserver { public: BrowserActivityObserver() { registrar_.Add(this, NotificationType::NAV_ENTRY_COMMITTED, NotificationService::AllSources()); } ~BrowserActivityObserver() {} private: // NotificationObserver implementation. virtual void Observe(NotificationType type, const NotificationSource& source, const NotificationDetails& details) { DCHECK(type == NotificationType::NAV_ENTRY_COMMITTED); const NavigationController::LoadCommittedDetails& load = *Details(details).ptr(); if (!load.is_main_frame || load.is_auto || load.is_in_page) return; // Don't log for subframes or other trivial types. LogRenderProcessHostCount(); LogBrowserTabCount(); } // Counts the number of active RenderProcessHosts and logs them. void LogRenderProcessHostCount() const { int hosts_count = 0; for (RenderProcessHost::iterator i(RenderProcessHost::AllHostsIterator()); !i.IsAtEnd(); i.Advance()) ++hosts_count; UMA_HISTOGRAM_CUSTOM_COUNTS("MPArch.RPHCountPerLoad", hosts_count, 1, 50, 50); } // Counts the number of tabs in each browser window and logs them. This is // different than the number of TabContents objects since TabContents objects // can be used for popups and in dialog boxes. We're just counting toplevel // tabs here. void LogBrowserTabCount() const { int tab_count = 0; for (BrowserList::const_iterator browser_iterator = BrowserList::begin(); browser_iterator != BrowserList::end(); browser_iterator++) tab_count += (*browser_iterator)->tab_count(); UMA_HISTOGRAM_CUSTOM_COUNTS("Tabs.TabCountPerLoad", tab_count, 1, 200, 50); } NotificationRegistrar registrar_; DISALLOW_COPY_AND_ASSIGN(BrowserActivityObserver); }; BrowserActivityObserver* activity_observer = NULL; // Returns true if the specified |browser| has a matching profile and type to // those specified. |type| can also be TYPE_ANY, which means only |profile| // must be matched. If |match_incognito| is true, we check both the incognito // and regular versions of the profile. bool BrowserMatchesProfileAndType(Browser* browser, Profile* profile, Browser::Type type, bool match_incognito) { bool profile_match = match_incognito ? browser->profile()->GetOriginalProfile() == profile->GetOriginalProfile() : browser->profile() == profile; return (type == Browser::TYPE_ANY || browser->type() == type) && profile_match; } // Finds a registered Browser object matching |profile| and |type|. This // walks the list of Browsers that have ever been activated from most recently // activated to least. If a Browser has never been activated, such as in a test // scenario, this function will _not_ find it. Fall back to // FindBrowserMatching() in that case. Browser* FindInLastActiveMatching(Profile* profile, Browser::Type type, bool match_incognito) { for (BrowserList::list_type::const_reverse_iterator i = BrowserList::begin_last_active(); i != BrowserList::end_last_active(); ++i) { if (BrowserMatchesProfileAndType(*i, profile, type, match_incognito)) return *i; } return NULL; } // Finds a registered Browser object matching |profile| and |type| even if that // Browser has never been activated. This is a forward walk, and intended as a // last ditch fallback mostly to handle tests run on machines where no window is // ever activated. The user experience if this function is relied on is not good // since matching browsers will be returned in registration (creation) order. Browser* FindBrowserMatching(Profile* profile, Browser::Type type, bool match_incognito) { for (BrowserList::const_iterator i = BrowserList::begin(); i != BrowserList::end(); ++i) { if (BrowserMatchesProfileAndType(*i, profile, type, match_incognito)) return *i; } return NULL; } } // namespace BrowserList::list_type BrowserList::browsers_; std::vector BrowserList::observers_; // static void BrowserList::AddBrowser(Browser* browser) { DCHECK(browser); browsers_.push_back(browser); g_browser_process->AddRefModule(); if (!activity_observer) activity_observer = new BrowserActivityObserver; NotificationService::current()->Notify( NotificationType::BROWSER_OPENED, Source(browser), NotificationService::NoDetails()); // Send out notifications after add has occurred. Do some basic checking to // try to catch evil observers that change the list from under us. size_t original_count = observers_.size(); for (size_t i = 0; i < observers_.size(); ++i) observers_[i]->OnBrowserAdded(browser); DCHECK_EQ(original_count, observers_.size()) << "observer list modified during notification"; } // static void BrowserList::RemoveBrowser(Browser* browser) { RemoveBrowserFrom(browser, &last_active_browsers_); // Closing all windows does not indicate quitting the application on the Mac, // however, many UI tests rely on this behavior so leave it be for now and // simply ignore the behavior on the Mac outside of unit tests. // TODO(andybons): Fix the UI tests to Do The Right Thing. bool close_app_non_mac = (browsers_.size() == 1); NotificationService::current()->Notify( NotificationType::BROWSER_CLOSED, Source(browser), Details(&close_app_non_mac)); // Send out notifications before anything changes. Do some basic checking to // try to catch evil observers that change the list from under us. size_t original_count = observers_.size(); for (size_t i = 0; i < observers_.size(); ++i) observers_[i]->OnBrowserRemoving(browser); DCHECK_EQ(original_count, observers_.size()) << "observer list modified during notification"; RemoveBrowserFrom(browser, &browsers_); // If the last Browser object was destroyed, make sure we try to close any // remaining dependent windows too. if (browsers_.empty()) { AllBrowsersClosed(); delete activity_observer; activity_observer = NULL; } g_browser_process->ReleaseModule(); } // static void BrowserList::AddObserver(BrowserList::Observer* observer) { DCHECK(std::find(observers_.begin(), observers_.end(), observer) == observers_.end()) << "Adding an observer twice"; observers_.push_back(observer); } // static void BrowserList::RemoveObserver(BrowserList::Observer* observer) { std::vector::iterator i = std::find(observers_.begin(), observers_.end(), observer); if (i == observers_.end()) { NOTREACHED() << "Removing an observer that isn't registered."; return; } observers_.erase(i); } // static void BrowserList::CloseAllBrowsers(bool use_post) { // Before we close the browsers shutdown all session services. That way an // exit can restore all browsers open before exiting. ProfileManager::ShutdownSessionServices(); for (BrowserList::const_iterator i = BrowserList::begin(); i != BrowserList::end();) { if (use_post) { (*i)->window()->Close(); ++i; } else { // This path is hit during logoff/power-down. In this case we won't get // a final message and so we force the browser to be deleted. Browser* browser = *i; browser->window()->Close(); // Close doesn't immediately destroy the browser // (Browser::TabStripEmpty() uses invoke later) but when we're ending the // session we need to make sure the browser is destroyed now. So, invoke // DestroyBrowser to make sure the browser is deleted and cleanup can // happen. browser->window()->DestroyBrowser(); i = BrowserList::begin(); if (i != BrowserList::end() && browser == *i) { // Destroying the browser should have removed it from the browser list. // We should never get here. NOTREACHED(); return; } } } } // static void BrowserList::CloseAllBrowsersAndExit() { NotificationService::current()->Notify( NotificationType::APP_EXITING, NotificationService::AllSources(), NotificationService::NoDetails()); #if !defined(OS_MACOSX) // On most platforms, closing all windows causes the application to exit. CloseAllBrowsers(true); #else // On the Mac, the application continues to run once all windows are closed. // Terminate will result in a CloseAllBrowsers(true) call, and once (and if) // that is done, will cause the application to exit cleanly. chrome_browser_application_mac::Terminate(); #endif } // static void BrowserList::WindowsSessionEnding() { // EndSession is invoked once per frame. Only do something the first time. static bool already_ended = false; if (already_ended) return; already_ended = true; browser_shutdown::OnShutdownStarting(browser_shutdown::END_SESSION); NotificationService::current()->Notify( NotificationType::APP_EXITING, NotificationService::AllSources(), NotificationService::NoDetails()); // Write important data first. g_browser_process->EndSession(); // Close all the browsers. BrowserList::CloseAllBrowsers(false); // Send out notification. This is used during testing so that the test harness // can properly shutdown before we exit. NotificationService::current()->Notify( NotificationType::SESSION_END, NotificationService::AllSources(), NotificationService::NoDetails()); // And shutdown. browser_shutdown::Shutdown(); #if defined(OS_WIN) // At this point the message loop is still running yet we've shut everything // down. If any messages are processed we'll likely crash. Exit now. ExitProcess(ResultCodes::NORMAL_EXIT); #else NOTIMPLEMENTED(); #endif } // static bool BrowserList::HasBrowserWithProfile(Profile* profile) { for (BrowserList::const_iterator i = BrowserList::begin(); i != BrowserList::end(); ++i) { if (BrowserMatchesProfileAndType(*i, profile, Browser::TYPE_ANY, false)) return true; } return false; } // static bool BrowserList::IsInPersistentMode() { // TODO(atwilson): check the boolean state variable that you will set for // persisent instances. return false; } // static BrowserList::list_type BrowserList::last_active_browsers_; // static void BrowserList::SetLastActive(Browser* browser) { RemoveBrowserFrom(browser, &last_active_browsers_); last_active_browsers_.push_back(browser); for (std::vector::const_iterator i = observers_.begin(); i != observers_.end(); ++i) { (*i)->OnBrowserSetLastActive(browser); } } // static Browser* BrowserList::GetLastActive() { if (!last_active_browsers_.empty()) return *(last_active_browsers_.rbegin()); return NULL; } // static Browser* BrowserList::GetLastActiveWithProfile(Profile* p) { // We are only interested in last active browsers, so we don't fall back to all // browsers like FindBrowserWith* do. return FindInLastActiveMatching(p, Browser::TYPE_ANY, false); } // static Browser* BrowserList::FindBrowserWithType(Profile* p, Browser::Type t, bool match_incognito) { Browser* browser = FindInLastActiveMatching(p, t, match_incognito); // Fall back to a forward scan of all Browsers if no active one was found. return browser ? browser : FindBrowserMatching(p, t, match_incognito); } // static Browser* BrowserList::FindBrowserWithProfile(Profile* p) { return FindBrowserWithType(p, Browser::TYPE_ANY, false); } // static Browser* BrowserList::FindBrowserWithID(SessionID::id_type desired_id) { for (BrowserList::const_iterator i = BrowserList::begin(); i != BrowserList::end(); ++i) { if ((*i)->session_id().id() == desired_id) return *i; } return NULL; } // static size_t BrowserList::GetBrowserCountForType(Profile* p, Browser::Type type) { size_t result = 0; for (BrowserList::const_iterator i = BrowserList::begin(); i != BrowserList::end(); ++i) { if (BrowserMatchesProfileAndType(*i, p, type, false)) ++result; } return result; } // static size_t BrowserList::GetBrowserCount(Profile* p) { size_t result = 0; for (BrowserList::const_iterator i = BrowserList::begin(); i != BrowserList::end(); ++i) { if (BrowserMatchesProfileAndType(*i, p, Browser::TYPE_ANY, false)) result++; } return result; } // static bool BrowserList::IsOffTheRecordSessionActive() { for (BrowserList::const_iterator i = BrowserList::begin(); i != BrowserList::end(); ++i) { if ((*i)->profile()->IsOffTheRecord()) return true; } return false; } // static void BrowserList::RemoveBrowserFrom(Browser* browser, list_type* browser_list) { const iterator remove_browser = std::find(browser_list->begin(), browser_list->end(), browser); if (remove_browser != browser_list->end()) browser_list->erase(remove_browser); } TabContentsIterator::TabContentsIterator() : browser_iterator_(BrowserList::begin()), web_view_index_(-1), cur_(NULL) { Advance(); } void TabContentsIterator::Advance() { // Unless we're at the beginning (index = -1) or end (iterator = end()), // then the current TabContents should be valid. DCHECK(web_view_index_ || browser_iterator_ == BrowserList::end() || cur_) << "Trying to advance past the end"; // Update cur_ to the next TabContents in the list. while (browser_iterator_ != BrowserList::end()) { web_view_index_++; while (web_view_index_ >= (*browser_iterator_)->tab_count()) { // advance browsers ++browser_iterator_; web_view_index_ = 0; if (browser_iterator_ == BrowserList::end()) { cur_ = NULL; return; } } TabContents* next_tab = (*browser_iterator_)->GetTabContentsAt(web_view_index_); if (next_tab) { cur_ = next_tab; return; } } }