// 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);
}