// 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/memory_details.h" #include <set> #include <string> #include "base/basictypes.h" #include "base/bind.h" #include "base/file_version_info.h" #include "base/files/file_path.h" #include "base/mac/mac_util.h" #include "base/process/process_iterator.h" #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "base/threading/thread.h" #include "chrome/browser/process_info_snapshot.h" #include "chrome/common/chrome_constants.h" #include "chrome/common/chrome_version_info.h" #include "chrome/common/url_constants.h" #include "content/public/browser/browser_thread.h" #include "content/public/common/process_type.h" #include "grit/chromium_strings.h" #include "ui/base/l10n/l10n_util.h" using content::BrowserThread; // TODO(viettrungluu): Many of the TODOs below are subsumed by a general need to // refactor the about:memory code (not just on Mac, but probably on other // platforms as well). I've filed crbug.com/25456. // Known browsers which we collect details for. |CHROME_BROWSER| *must* be the // first browser listed. The order here must match those in |process_template| // (in |MemoryDetails::MemoryDetails()| below). // TODO(viettrungluu): In the big refactoring (see above), get rid of this order // dependence. enum BrowserType { // TODO(viettrungluu): possibly add more? CHROME_BROWSER = 0, SAFARI_BROWSER, FIREFOX_BROWSER, CAMINO_BROWSER, OPERA_BROWSER, OMNIWEB_BROWSER, MAX_BROWSERS } BrowserProcess; MemoryDetails::MemoryDetails() : user_metrics_mode_(UPDATE_USER_METRICS) { const std::string google_browser_name = l10n_util::GetStringUTF8(IDS_PRODUCT_NAME); // (Human and process) names of browsers; should match the ordering for // |BrowserProcess| (i.e., |BrowserType|). // TODO(viettrungluu): The current setup means that we can't detect both // Chrome and Chromium at the same time! // TODO(viettrungluu): Get localized browser names for other browsers // (crbug.com/25779). struct { const char* name; const char* process_name; } process_template[MAX_BROWSERS] = { { google_browser_name.c_str(), chrome::kBrowserProcessExecutableName, }, { "Safari", "Safari", }, { "Firefox", "firefox-bin", }, { "Camino", "Camino", }, { "Opera", "Opera", }, { "OmniWeb", "OmniWeb", }, }; for (size_t index = 0; index < MAX_BROWSERS; ++index) { ProcessData process; process.name = UTF8ToUTF16(process_template[index].name); process.process_name = UTF8ToUTF16(process_template[index].process_name); process_data_.push_back(process); } } ProcessData* MemoryDetails::ChromeBrowser() { return &process_data_[CHROME_BROWSER]; } void MemoryDetails::CollectProcessData( const std::vector<ProcessMemoryInformation>& child_info) { // This must be run on the file thread to avoid jank (|ProcessInfoSnapshot| // runs /bin/ps, which isn't instantaneous). DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); // Clear old data. for (size_t index = 0; index < MAX_BROWSERS; index++) process_data_[index].processes.clear(); // First, we use |NamedProcessIterator| to get the PIDs of the processes we're // interested in; we save our results to avoid extra calls to // |NamedProcessIterator| (for performance reasons) and to avoid additional // inconsistencies caused by racing. Then we run |/bin/ps| *once* to get // information on those PIDs. Then we used our saved information to iterate // over browsers, then over PIDs. // Get PIDs of main browser processes. std::vector<base::ProcessId> pids_by_browser[MAX_BROWSERS]; std::vector<base::ProcessId> all_pids; for (size_t index = CHROME_BROWSER; index < MAX_BROWSERS; index++) { base::NamedProcessIterator process_it( UTF16ToUTF8(process_data_[index].process_name), NULL); while (const base::ProcessEntry* entry = process_it.NextProcessEntry()) { pids_by_browser[index].push_back(entry->pid()); all_pids.push_back(entry->pid()); } } // The helper might show up as these different flavors depending on the // executable flags required. std::vector<std::string> helper_names; helper_names.push_back(chrome::kHelperProcessExecutableName); for (const char* const* suffix = chrome::kHelperFlavorSuffixes; *suffix; ++suffix) { std::string helper_name = chrome::kHelperProcessExecutableName; helper_name.append(1, ' '); helper_name.append(*suffix); helper_names.push_back(helper_name); } // Get PIDs of helpers. std::vector<base::ProcessId> helper_pids; for (size_t i = 0; i < helper_names.size(); ++i) { std::string helper_name = helper_names[i]; base::NamedProcessIterator helper_it(helper_name, NULL); while (const base::ProcessEntry* entry = helper_it.NextProcessEntry()) { helper_pids.push_back(entry->pid()); all_pids.push_back(entry->pid()); } } // Capture information about the processes we're interested in. ProcessInfoSnapshot process_info; process_info.Sample(all_pids); // Handle the other processes first. for (size_t index = CHROME_BROWSER + 1; index < MAX_BROWSERS; index++) { for (std::vector<base::ProcessId>::const_iterator it = pids_by_browser[index].begin(); it != pids_by_browser[index].end(); ++it) { ProcessMemoryInformation info; info.pid = *it; info.process_type = content::PROCESS_TYPE_UNKNOWN; // Try to get version information. To do this, we need first to get the // executable's name (we can only believe |proc_info.command| if it looks // like an absolute path). Then we need strip the executable's name back // to the bundle's name. And only then can we try to get the version. scoped_ptr<FileVersionInfo> version_info; ProcessInfoSnapshot::ProcInfoEntry proc_info; if (process_info.GetProcInfo(info.pid, &proc_info)) { if (proc_info.command.length() > 1 && proc_info.command[0] == '/') { base::FilePath bundle_name = base::mac::GetAppBundlePath(base::FilePath(proc_info.command)); if (!bundle_name.empty()) { version_info.reset(FileVersionInfo::CreateFileVersionInfo( bundle_name)); } } } if (version_info.get()) { info.product_name = version_info->product_name(); info.version = version_info->product_version(); } else { info.product_name = process_data_[index].name; info.version = base::string16(); } // Memory info. process_info.GetCommittedKBytesOfPID(info.pid, &info.committed); process_info.GetWorkingSetKBytesOfPID(info.pid, &info.working_set); // Add the process info to our list. process_data_[index].processes.push_back(info); } } // Collect data about Chrome/Chromium. for (std::vector<base::ProcessId>::const_iterator it = pids_by_browser[CHROME_BROWSER].begin(); it != pids_by_browser[CHROME_BROWSER].end(); ++it) { CollectProcessDataChrome(child_info, *it, process_info); } // And collect data about the helpers. for (std::vector<base::ProcessId>::const_iterator it = helper_pids.begin(); it != helper_pids.end(); ++it) { CollectProcessDataChrome(child_info, *it, process_info); } // Finally return to the browser thread. BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&MemoryDetails::CollectChildInfoOnUIThread, this)); } void MemoryDetails::CollectProcessDataChrome( const std::vector<ProcessMemoryInformation>& child_info, base::ProcessId pid, const ProcessInfoSnapshot& process_info) { ProcessMemoryInformation info; info.pid = pid; if (info.pid == base::GetCurrentProcId()) info.process_type = content::PROCESS_TYPE_BROWSER; else info.process_type = content::PROCESS_TYPE_UNKNOWN; chrome::VersionInfo version_info; if (version_info.is_valid()) { info.product_name = ASCIIToUTF16(version_info.Name()); info.version = ASCIIToUTF16(version_info.Version()); } else { info.product_name = process_data_[CHROME_BROWSER].name; info.version = base::string16(); } // Check if this is one of the child processes whose data we collected // on the IO thread, and if so copy over that data. for (size_t child = 0; child < child_info.size(); child++) { if (child_info[child].pid == info.pid) { info.titles = child_info[child].titles; info.process_type = child_info[child].process_type; break; } } // Memory info. process_info.GetCommittedKBytesOfPID(info.pid, &info.committed); process_info.GetWorkingSetKBytesOfPID(info.pid, &info.working_set); // Add the process info to our list. process_data_[CHROME_BROWSER].processes.push_back(info); }