blob: 1b9ed4f0ddb708cc55cde9d4af5472d06d337360 [file] [log] [blame]
[email protected]4306df72012-04-20 18:58:571// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]54fd1d32009-09-01 00:12:582// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "chrome/browser/memory_details.h"
[email protected]b90d7e802011-01-09 16:32:206
[email protected]54fd1d32009-09-01 00:12:587#include <psapi.h>
[email protected]d09a4ce1c2013-07-24 17:37:028#include <TlHelp32.h>
[email protected]54fd1d32009-09-01 00:12:589
[email protected]24d69692011-10-21 18:26:5110#include "base/bind.h"
[email protected]54fd1d32009-09-01 00:12:5811#include "base/file_version_info.h"
[email protected]57999812013-02-24 05:40:5212#include "base/files/file_path.h"
[email protected]f9b294362013-06-10 20:22:3113#include "base/strings/string_util.h"
[email protected]112158af2013-06-07 23:46:1814#include "base/strings/utf_string_conversions.h"
[email protected]b90d7e802011-01-09 16:32:2015#include "base/win/scoped_handle.h"
[email protected]1e67c2b2011-03-04 01:17:3716#include "base/win/windows_version.h"
[email protected]1eeb5e02010-07-20 23:02:1117#include "chrome/common/chrome_version_info.h"
[email protected]54fd1d32009-09-01 00:12:5818#include "chrome/common/url_constants.h"
[email protected]af39f002014-08-22 10:18:1819#include "chrome/grit/chromium_strings.h"
[email protected]c38831a12011-10-28 12:44:4920#include "content/public/browser/browser_thread.h"
[email protected]bd5d6cf2011-12-01 00:39:1221#include "content/public/common/process_type.h"
[email protected]c051a1b2011-01-21 23:30:1722#include "ui/base/l10n/l10n_util.h"
[email protected]54fd1d32009-09-01 00:12:5823
[email protected]631bb742011-11-02 11:29:3924using content::BrowserThread;
25
[email protected]54fd1d32009-09-01 00:12:5826// Known browsers which we collect details for.
27enum {
28 CHROME_BROWSER = 0,
[email protected]aef8d5ae2010-03-17 22:40:5229 CHROME_NACL_PROCESS,
[email protected]54fd1d32009-09-01 00:12:5830 IE_BROWSER,
31 FIREFOX_BROWSER,
32 OPERA_BROWSER,
33 SAFARI_BROWSER,
34 IE_64BIT_BROWSER,
35 KONQUEROR_BROWSER,
36 MAX_BROWSERS
37} BrowserProcess;
38
[email protected]4306df72012-04-20 18:58:5739MemoryDetails::MemoryDetails()
[email protected]8a5b2da2014-07-07 10:56:5140 : user_metrics_mode_(UPDATE_USER_METRICS),
41 memory_growth_tracker_(NULL) {
[email protected]299e3d002014-06-03 08:46:4942 static const base::string16 google_browser_name =
43 l10n_util::GetStringUTF16(IDS_PRODUCT_NAME);
[email protected]93aa89c72010-10-20 21:32:0444 struct {
45 const wchar_t* name;
46 const wchar_t* process_name;
47 } process_template[MAX_BROWSERS] = {
[email protected]54fd1d32009-09-01 00:12:5848 { google_browser_name.c_str(), L"chrome.exe", },
[email protected]aef8d5ae2010-03-17 22:40:5249 { google_browser_name.c_str(), L"nacl64.exe", },
[email protected]54fd1d32009-09-01 00:12:5850 { L"IE", L"iexplore.exe", },
51 { L"Firefox", L"firefox.exe", },
52 { L"Opera", L"opera.exe", },
53 { L"Safari", L"safari.exe", },
54 { L"IE (64bit)", L"iexplore.exe", },
55 { L"Konqueror", L"konqueror.exe", },
56 };
57
[email protected]93aa89c72010-10-20 21:32:0458 for (int index = 0; index < MAX_BROWSERS; ++index) {
[email protected]54fd1d32009-09-01 00:12:5859 ProcessData process;
[email protected]93aa89c72010-10-20 21:32:0460 process.name = process_template[index].name;
61 process.process_name = process_template[index].process_name;
[email protected]54fd1d32009-09-01 00:12:5862 process_data_.push_back(process);
63 }
64}
65
66ProcessData* MemoryDetails::ChromeBrowser() {
67 return &process_data_[CHROME_BROWSER];
68}
69
70void MemoryDetails::CollectProcessData(
[email protected]4df3ac62011-03-11 04:38:5271 const std::vector<ProcessMemoryInformation>& child_info) {
[email protected]f8b3ef82010-10-11 02:45:5272 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
[email protected]54fd1d32009-09-01 00:12:5873
74 // Clear old data.
[email protected]93aa89c72010-10-20 21:32:0475 for (unsigned int index = 0; index < process_data_.size(); index++)
[email protected]54fd1d32009-09-01 00:12:5876 process_data_[index].processes.clear();
77
[email protected]f481221192011-04-07 22:15:3478 base::win::OSInfo::WindowsArchitecture windows_architecture =
79 base::win::OSInfo::GetInstance()->architecture();
[email protected]54fd1d32009-09-01 00:12:5880
[email protected]b90d7e802011-01-09 16:32:2081 base::win::ScopedHandle snapshot(
82 ::CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0));
[email protected]54fd1d32009-09-01 00:12:5883 PROCESSENTRY32 process_entry = {sizeof(PROCESSENTRY32)};
84 if (!snapshot.Get()) {
85 LOG(ERROR) << "CreateToolhelp32Snaphot failed: " << GetLastError();
86 return;
87 }
rvargasd7743ba2014-09-25 01:35:2888 if (!::Process32First(snapshot.Get(), &process_entry)) {
[email protected]54fd1d32009-09-01 00:12:5889 LOG(ERROR) << "Process32First failed: " << GetLastError();
90 return;
91 }
92 do {
[email protected]a4dc33f2009-10-20 15:09:5593 base::ProcessId pid = process_entry.th32ProcessID;
[email protected]1e67c2b2011-03-04 01:17:3794 base::win::ScopedHandle process_handle(::OpenProcess(
[email protected]54fd1d32009-09-01 00:12:5895 PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, pid));
rvargasd7743ba2014-09-25 01:35:2896 if (!process_handle.IsValid())
[email protected]54fd1d32009-09-01 00:12:5897 continue;
[email protected]af67f202011-03-04 19:12:2998 bool is_64bit_process =
[email protected]f481221192011-04-07 22:15:3499 ((windows_architecture == base::win::OSInfo::X64_ARCHITECTURE) ||
100 (windows_architecture == base::win::OSInfo::IA64_ARCHITECTURE)) &&
rvargasd7743ba2014-09-25 01:35:28101 (base::win::OSInfo::GetWOW64StatusForProcess(process_handle.Get()) ==
[email protected]f481221192011-04-07 22:15:34102 base::win::OSInfo::WOW64_DISABLED);
[email protected]93aa89c72010-10-20 21:32:04103 for (unsigned int index2 = 0; index2 < process_data_.size(); index2++) {
[email protected]54fd1d32009-09-01 00:12:58104 if (_wcsicmp(process_data_[index2].process_name.c_str(),
[email protected]b6128aa2010-04-29 17:44:42105 process_entry.szExeFile) != 0)
[email protected]54fd1d32009-09-01 00:12:58106 continue;
107 if (index2 == IE_BROWSER && is_64bit_process)
108 continue; // Should use IE_64BIT_BROWSER
109 // Get Memory Information.
110 ProcessMemoryInformation info;
111 info.pid = pid;
112 if (info.pid == GetCurrentProcessId())
[email protected]f3b357692013-03-22 05:16:13113 info.process_type = content::PROCESS_TYPE_BROWSER;
[email protected]54fd1d32009-09-01 00:12:58114 else
[email protected]f3b357692013-03-22 05:16:13115 info.process_type = content::PROCESS_TYPE_UNKNOWN;
[email protected]54fd1d32009-09-01 00:12:58116
117 scoped_ptr<base::ProcessMetrics> metrics;
rvargasd7743ba2014-09-25 01:35:28118 metrics.reset(base::ProcessMetrics::CreateProcessMetrics(
119 process_handle.Get()));
[email protected]54fd1d32009-09-01 00:12:58120 metrics->GetCommittedKBytes(&info.committed);
121 metrics->GetWorkingSetKBytes(&info.working_set);
122
123 // Get Version Information.
124 TCHAR name[MAX_PATH];
[email protected]aef8d5ae2010-03-17 22:40:52125 if (index2 == CHROME_BROWSER || index2 == CHROME_NACL_PROCESS) {
[email protected]0211f57e2010-08-27 20:28:42126 chrome::VersionInfo version_info;
[email protected]30c91802010-12-18 00:40:17127 if (version_info.is_valid())
[email protected]6778fed2013-12-24 20:09:37128 info.version = base::ASCIIToWide(version_info.Version());
[email protected]54fd1d32009-09-01 00:12:58129 // Check if this is one of the child processes whose data we collected
130 // on the IO thread, and if so copy over that data.
131 for (size_t child = 0; child < child_info.size(); child++) {
132 if (child_info[child].pid != info.pid)
133 continue;
134 info.titles = child_info[child].titles;
[email protected]f3b357692013-03-22 05:16:13135 info.process_type = child_info[child].process_type;
[email protected]54fd1d32009-09-01 00:12:58136 break;
137 }
rvargasd7743ba2014-09-25 01:35:28138 } else if (GetModuleFileNameEx(process_handle.Get(), NULL, name,
[email protected]1e67c2b2011-03-04 01:17:37139 MAX_PATH - 1)) {
[email protected]54fd1d32009-09-01 00:12:58140 std::wstring str_name(name);
141 scoped_ptr<FileVersionInfo> version_info(
[email protected]650b2d52013-02-10 03:41:45142 FileVersionInfo::CreateFileVersionInfo(base::FilePath(str_name)));
[email protected]54fd1d32009-09-01 00:12:58143 if (version_info != NULL) {
144 info.version = version_info->product_version();
145 info.product_name = version_info->product_name();
146 }
147 }
148
149 // Add the process info to our list.
[email protected]aef8d5ae2010-03-17 22:40:52150 if (index2 == CHROME_NACL_PROCESS) {
151 // Add NaCl processes to Chrome's list
152 process_data_[CHROME_BROWSER].processes.push_back(info);
153 } else {
154 process_data_[index2].processes.push_back(info);
155 }
[email protected]54fd1d32009-09-01 00:12:58156 break;
157 }
rvargasd7743ba2014-09-25 01:35:28158 } while (::Process32Next(snapshot.Get(), &process_entry));
[email protected]54fd1d32009-09-01 00:12:58159
160 // Finally return to the browser thread.
[email protected]f8b3ef82010-10-11 02:45:52161 BrowserThread::PostTask(
162 BrowserThread::UI, FROM_HERE,
[email protected]24d69692011-10-21 18:26:51163 base::Bind(&MemoryDetails::CollectChildInfoOnUIThread, this));
[email protected]54fd1d32009-09-01 00:12:58164}