// 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 <sys/types.h> #include <unistd.h> #include <map> #include <set> #include "base/bind.h" #include "base/file_util.h" #include "base/process/process_iterator.h" #include "base/process/process_metrics.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "chrome/common/chrome_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 base::ProcessEntry; using content::BrowserThread; // Known browsers which we collect details for. enum BrowserType { CHROME = 0, FIREFOX, ICEWEASEL, OPERA, KONQUEROR, EPIPHANY, MIDORI, MAX_BROWSERS } BrowserProcess; // The pretty printed names of those browsers. Matches up with enum // BrowserType. static const char kBrowserPrettyNames[][10] = { "Chrome", "Firefox", "Iceweasel", "Opera", "Konqueror", "Epiphany", "Midori", }; // A mapping from process name to the type of browser. static const struct { const char process_name[16]; BrowserType browser; } kBrowserBinaryNames[] = { { "firefox", FIREFOX }, { "firefox-3.5", FIREFOX }, { "firefox-3.0", FIREFOX }, { "firefox-bin", FIREFOX }, { "iceweasel", ICEWEASEL }, { "opera", OPERA }, { "konqueror", KONQUEROR }, { "epiphany-browse", EPIPHANY }, { "epiphany", EPIPHANY }, { "midori", MIDORI }, { "", MAX_BROWSERS }, }; MemoryDetails::MemoryDetails() : user_metrics_mode_(UPDATE_USER_METRICS) { } ProcessData* MemoryDetails::ChromeBrowser() { return &process_data_[0]; } struct Process { pid_t pid; pid_t parent; std::string name; }; typedef std::map<pid_t, Process> ProcessMap; // Get information on all the processes running on the system. static ProcessMap GetProcesses() { ProcessMap map; base::ProcessIterator process_iter(NULL); while (const ProcessEntry* process_entry = process_iter.NextProcessEntry()) { Process process; process.pid = process_entry->pid(); process.parent = process_entry->parent_pid(); process.name = process_entry->exe_file(); map.insert(std::make_pair(process.pid, process)); } return map; } // Given a process name, return the type of the browser which created that // process, or |MAX_BROWSERS| if we don't know about it. static BrowserType GetBrowserType(const std::string& process_name) { for (unsigned i = 0; kBrowserBinaryNames[i].process_name[0]; ++i) { if (strcmp(process_name.c_str(), kBrowserBinaryNames[i].process_name) == 0) return kBrowserBinaryNames[i].browser; } return MAX_BROWSERS; } // For each of a list of pids, collect memory information about that process. static ProcessData GetProcessDataMemoryInformation( const std::vector<pid_t>& pids) { ProcessData process_data; for (std::vector<pid_t>::const_iterator iter = pids.begin(); iter != pids.end(); ++iter) { ProcessMemoryInformation pmi; pmi.pid = *iter; pmi.num_processes = 1; if (pmi.pid == base::GetCurrentProcId()) pmi.process_type = content::PROCESS_TYPE_BROWSER; else pmi.process_type = content::PROCESS_TYPE_UNKNOWN; base::ProcessMetrics* metrics = base::ProcessMetrics::CreateProcessMetrics(*iter); metrics->GetWorkingSetKBytes(&pmi.working_set); delete metrics; process_data.processes.push_back(pmi); } return process_data; } // Find all children of the given process with pid |root|. static std::vector<pid_t> GetAllChildren(const ProcessMap& processes, const pid_t root) { std::vector<pid_t> children; children.push_back(root); std::set<pid_t> wavefront, next_wavefront; wavefront.insert(root); while (wavefront.size()) { for (ProcessMap::const_iterator iter = processes.begin(); iter != processes.end(); ++iter) { const Process& process = iter->second; if (wavefront.count(process.parent)) { children.push_back(process.pid); next_wavefront.insert(process.pid); } } wavefront.clear(); wavefront.swap(next_wavefront); } return children; } void MemoryDetails::CollectProcessData( const std::vector<ProcessMemoryInformation>& child_info) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); ProcessMap process_map = GetProcesses(); std::set<pid_t> browsers_found; // For each process on the system, if it appears to be a browser process and // it's parent isn't a browser process, then record it in |browsers_found|. for (ProcessMap::const_iterator iter = process_map.begin(); iter != process_map.end(); ++iter) { const Process& current_process = iter->second; const BrowserType type = GetBrowserType(current_process.name); if (type == MAX_BROWSERS) continue; ProcessMap::const_iterator parent_iter = process_map.find(current_process.parent); if (parent_iter == process_map.end()) { browsers_found.insert(current_process.pid); continue; } if (GetBrowserType(parent_iter->second.name) != type) { // We found a process whose type is diffent from its parent's type. // That means it is the root process of the browser. browsers_found.insert(current_process.pid); break; } } ProcessData current_browser = GetProcessDataMemoryInformation(GetAllChildren(process_map, getpid())); current_browser.name = l10n_util::GetStringUTF16(IDS_SHORT_PRODUCT_NAME); current_browser.process_name = base::ASCIIToUTF16("chrome"); for (std::vector<ProcessMemoryInformation>::iterator i = current_browser.processes.begin(); i != current_browser.processes.end(); ++i) { // 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 != i->pid) continue; i->titles = child_info[child].titles; i->process_type = child_info[child].process_type; break; } } process_data_.push_back(current_browser); // For each browser process, collect a list of its children and get the // memory usage of each. for (std::set<pid_t>::const_iterator iter = browsers_found.begin(); iter != browsers_found.end(); ++iter) { std::vector<pid_t> browser_processes = GetAllChildren(process_map, *iter); ProcessData browser = GetProcessDataMemoryInformation(browser_processes); ProcessMap::const_iterator process_iter = process_map.find(*iter); if (process_iter == process_map.end()) continue; BrowserType type = GetBrowserType(process_iter->second.name); if (type != MAX_BROWSERS) browser.name = base::ASCIIToUTF16(kBrowserPrettyNames[type]); process_data_.push_back(browser); } #if defined(OS_CHROMEOS) base::GetSwapInfo(&swap_info_); #endif // Finally return to the browser thread. BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&MemoryDetails::CollectChildInfoOnUIThread, this)); }