Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(142)

Side by Side Diff: base/process/process_metrics_mac.cc

Issue 278923002: Use the new ScopedMachVM class and the MACH_LOG family of logging macros (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Rebase onto r269793 Created 6 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « base/process/memory_mac.mm ('k') | base/threading/platform_thread_mac.mm » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "base/process/process_metrics.h" 5 #include "base/process/process_metrics.h"
6 6
7 #include <mach/mach.h> 7 #include <mach/mach.h>
8 #include <mach/mach_vm.h> 8 #include <mach/mach_vm.h>
9 #include <mach/shared_region.h> 9 #include <mach/shared_region.h>
10 #include <sys/sysctl.h> 10 #include <sys/sysctl.h>
11 11
12 #include "base/containers/hash_tables.h" 12 #include "base/containers/hash_tables.h"
13 #include "base/logging.h" 13 #include "base/logging.h"
14 #include "base/mac/mach_logging.h"
14 #include "base/mac/scoped_mach_port.h" 15 #include "base/mac/scoped_mach_port.h"
15 #include "base/sys_info.h" 16 #include "base/sys_info.h"
16 17
17 #if !defined(TASK_POWER_INFO) 18 #if !defined(TASK_POWER_INFO)
18 // Doesn't exist in the 10.6 or 10.7 SDKs. 19 // Doesn't exist in the 10.6 or 10.7 SDKs.
19 #define TASK_POWER_INFO 21 20 #define TASK_POWER_INFO 21
20 struct task_power_info { 21 struct task_power_info {
21 uint64_t total_user; 22 uint64_t total_user;
22 uint64_t total_system; 23 uint64_t total_system;
23 uint64_t task_interrupt_wakeups; 24 uint64_t task_interrupt_wakeups;
(...skipping 85 matching lines...) Expand 10 before | Expand all | Expand 10 after
109 110
110 size_t ProcessMetrics::GetPeakWorkingSetSize() const { 111 size_t ProcessMetrics::GetPeakWorkingSetSize() const {
111 return 0; 112 return 0;
112 } 113 }
113 114
114 // This is a rough approximation of the algorithm that libtop uses. 115 // This is a rough approximation of the algorithm that libtop uses.
115 // private_bytes is the size of private resident memory. 116 // private_bytes is the size of private resident memory.
116 // shared_bytes is the size of shared resident memory. 117 // shared_bytes is the size of shared resident memory.
117 bool ProcessMetrics::GetMemoryBytes(size_t* private_bytes, 118 bool ProcessMetrics::GetMemoryBytes(size_t* private_bytes,
118 size_t* shared_bytes) { 119 size_t* shared_bytes) {
119 kern_return_t kr;
120 size_t private_pages_count = 0; 120 size_t private_pages_count = 0;
121 size_t shared_pages_count = 0; 121 size_t shared_pages_count = 0;
122 122
123 if (!private_bytes && !shared_bytes) 123 if (!private_bytes && !shared_bytes)
124 return true; 124 return true;
125 125
126 mach_port_t task = TaskForPid(process_); 126 mach_port_t task = TaskForPid(process_);
127 if (task == MACH_PORT_NULL) { 127 if (task == MACH_PORT_NULL) {
128 DLOG(ERROR) << "Invalid process"; 128 DLOG(ERROR) << "Invalid process";
129 return false; 129 return false;
(...skipping 16 matching lines...) Expand all
146 // Private memory is much simpler. We simply count the pages that are marked 146 // Private memory is much simpler. We simply count the pages that are marked
147 // as private or copy on write (COW). 147 // as private or copy on write (COW).
148 // 148 //
149 // See libtop_update_vm_regions in 149 // See libtop_update_vm_regions in
150 // http://www.opensource.apple.com/source/top/top-67/libtop.c 150 // http://www.opensource.apple.com/source/top/top-67/libtop.c
151 mach_vm_size_t size = 0; 151 mach_vm_size_t size = 0;
152 for (mach_vm_address_t address = MACH_VM_MIN_ADDRESS;; address += size) { 152 for (mach_vm_address_t address = MACH_VM_MIN_ADDRESS;; address += size) {
153 vm_region_top_info_data_t info; 153 vm_region_top_info_data_t info;
154 mach_msg_type_number_t info_count = VM_REGION_TOP_INFO_COUNT; 154 mach_msg_type_number_t info_count = VM_REGION_TOP_INFO_COUNT;
155 mach_port_t object_name; 155 mach_port_t object_name;
156 kr = mach_vm_region(task, 156 kern_return_t kr = mach_vm_region(task,
157 &address, 157 &address,
158 &size, 158 &size,
159 VM_REGION_TOP_INFO, 159 VM_REGION_TOP_INFO,
160 (vm_region_info_t)&info, 160 reinterpret_cast<vm_region_info_t>(&info),
161 &info_count, 161 &info_count,
162 &object_name); 162 &object_name);
163 if (kr == KERN_INVALID_ADDRESS) { 163 if (kr == KERN_INVALID_ADDRESS) {
164 // We're at the end of the address space. 164 // We're at the end of the address space.
165 break; 165 break;
166 } else if (kr != KERN_SUCCESS) { 166 } else if (kr != KERN_SUCCESS) {
167 DLOG(ERROR) << "Calling mach_vm_region failed with error: " 167 MACH_DLOG(ERROR, kr) << "mach_vm_region";
168 << mach_error_string(kr);
169 return false; 168 return false;
170 } 169 }
171 170
171 // The kernel always returns a null object for VM_REGION_TOP_INFO, but
172 // balance it with a deallocate in case this ever changes. See 10.9.2
173 // xnu-2422.90.20/osfmk/vm/vm_map.c vm_map_region.
174 mach_port_deallocate(mach_task_self(), object_name);
175
172 if (IsAddressInSharedRegion(address, cpu_type) && 176 if (IsAddressInSharedRegion(address, cpu_type) &&
173 info.share_mode != SM_PRIVATE) 177 info.share_mode != SM_PRIVATE)
174 continue; 178 continue;
175 179
176 if (info.share_mode == SM_COW && info.ref_count == 1) 180 if (info.share_mode == SM_COW && info.ref_count == 1)
177 info.share_mode = SM_PRIVATE; 181 info.share_mode = SM_PRIVATE;
178 182
179 switch (info.share_mode) { 183 switch (info.share_mode) {
180 case SM_PRIVATE: 184 case SM_PRIVATE:
181 private_pages_count += info.private_pages_resident; 185 private_pages_count += info.private_pages_resident;
182 private_pages_count += info.shared_pages_resident; 186 private_pages_count += info.shared_pages_resident;
183 break; 187 break;
184 case SM_COW: 188 case SM_COW:
185 private_pages_count += info.private_pages_resident; 189 private_pages_count += info.private_pages_resident;
186 // Fall through 190 // Fall through
187 case SM_SHARED: 191 case SM_SHARED:
188 if (seen_objects.count(info.obj_id) == 0) { 192 if (seen_objects.count(info.obj_id) == 0) {
189 // Only count the first reference to this region. 193 // Only count the first reference to this region.
190 seen_objects.insert(info.obj_id); 194 seen_objects.insert(info.obj_id);
191 shared_pages_count += info.shared_pages_resident; 195 shared_pages_count += info.shared_pages_resident;
192 } 196 }
193 break; 197 break;
194 default: 198 default:
195 break; 199 break;
196 } 200 }
197 } 201 }
198 202
199 vm_size_t page_size;
200 kr = host_page_size(task, &page_size);
201 if (kr != KERN_SUCCESS) {
202 DLOG(ERROR) << "Failed to fetch host page size, error: "
203 << mach_error_string(kr);
204 return false;
205 }
206
207 if (private_bytes) 203 if (private_bytes)
208 *private_bytes = private_pages_count * page_size; 204 *private_bytes = private_pages_count * PAGE_SIZE;
209 if (shared_bytes) 205 if (shared_bytes)
210 *shared_bytes = shared_pages_count * page_size; 206 *shared_bytes = shared_pages_count * PAGE_SIZE;
211 207
212 return true; 208 return true;
213 } 209 }
214 210
215 void ProcessMetrics::GetCommittedKBytes(CommittedKBytes* usage) const { 211 void ProcessMetrics::GetCommittedKBytes(CommittedKBytes* usage) const {
216 } 212 }
217 213
218 bool ProcessMetrics::GetWorkingSetKBytes(WorkingSetKBytes* ws_usage) const { 214 bool ProcessMetrics::GetWorkingSetKBytes(WorkingSetKBytes* ws_usage) const {
219 size_t priv = GetWorkingSetSize(); 215 size_t priv = GetWorkingSetSize();
220 if (!priv) 216 if (!priv)
221 return false; 217 return false;
222 ws_usage->priv = priv / 1024; 218 ws_usage->priv = priv / 1024;
223 ws_usage->shareable = 0; 219 ws_usage->shareable = 0;
224 ws_usage->shared = 0; 220 ws_usage->shared = 0;
225 return true; 221 return true;
226 } 222 }
227 223
228 #define TIME_VALUE_TO_TIMEVAL(a, r) do { \ 224 #define TIME_VALUE_TO_TIMEVAL(a, r) do { \
229 (r)->tv_sec = (a)->seconds; \ 225 (r)->tv_sec = (a)->seconds; \
230 (r)->tv_usec = (a)->microseconds; \ 226 (r)->tv_usec = (a)->microseconds; \
231 } while (0) 227 } while (0)
232 228
233 double ProcessMetrics::GetCPUUsage() { 229 double ProcessMetrics::GetCPUUsage() {
234 mach_port_t task = TaskForPid(process_); 230 mach_port_t task = TaskForPid(process_);
235 if (task == MACH_PORT_NULL) 231 if (task == MACH_PORT_NULL)
236 return 0; 232 return 0;
237 233
238 kern_return_t kr;
239
240 // Libtop explicitly loops over the threads (libtop_pinfo_update_cpu_usage() 234 // Libtop explicitly loops over the threads (libtop_pinfo_update_cpu_usage()
241 // in libtop.c), but this is more concise and gives the same results: 235 // in libtop.c), but this is more concise and gives the same results:
242 task_thread_times_info thread_info_data; 236 task_thread_times_info thread_info_data;
243 mach_msg_type_number_t thread_info_count = TASK_THREAD_TIMES_INFO_COUNT; 237 mach_msg_type_number_t thread_info_count = TASK_THREAD_TIMES_INFO_COUNT;
244 kr = task_info(task, 238 kern_return_t kr = task_info(task,
245 TASK_THREAD_TIMES_INFO, 239 TASK_THREAD_TIMES_INFO,
246 reinterpret_cast<task_info_t>(&thread_info_data), 240 reinterpret_cast<task_info_t>(&thread_info_data),
247 &thread_info_count); 241 &thread_info_count);
248 if (kr != KERN_SUCCESS) { 242 if (kr != KERN_SUCCESS) {
249 // Most likely cause: |task| is a zombie. 243 // Most likely cause: |task| is a zombie.
250 return 0; 244 return 0;
251 } 245 }
252 246
253 task_basic_info_64 task_info_data; 247 task_basic_info_64 task_info_data;
254 if (!GetTaskInfo(task, &task_info_data)) 248 if (!GetTaskInfo(task, &task_info_data))
255 return 0; 249 return 0;
256 250
257 /* Set total_time. */ 251 /* Set total_time. */
(...skipping 29 matching lines...) Expand all
287 last_system_time_ = task_time; 281 last_system_time_ = task_time;
288 282
289 return static_cast<double>(system_time_delta * 100.0) / time_delta; 283 return static_cast<double>(system_time_delta * 100.0) / time_delta;
290 } 284 }
291 285
292 int ProcessMetrics::GetIdleWakeupsPerSecond() { 286 int ProcessMetrics::GetIdleWakeupsPerSecond() {
293 mach_port_t task = TaskForPid(process_); 287 mach_port_t task = TaskForPid(process_);
294 if (task == MACH_PORT_NULL) 288 if (task == MACH_PORT_NULL)
295 return 0; 289 return 0;
296 290
297 kern_return_t kr;
298
299 task_power_info power_info_data; 291 task_power_info power_info_data;
300 mach_msg_type_number_t power_info_count = TASK_POWER_INFO_COUNT; 292 mach_msg_type_number_t power_info_count = TASK_POWER_INFO_COUNT;
301 kr = task_info(task, 293 kern_return_t kr = task_info(task,
302 TASK_POWER_INFO, 294 TASK_POWER_INFO,
303 reinterpret_cast<task_info_t>(&power_info_data), 295 reinterpret_cast<task_info_t>(&power_info_data),
304 &power_info_count); 296 &power_info_count);
305 if (kr != KERN_SUCCESS) { 297 if (kr != KERN_SUCCESS) {
306 // Most likely cause: |task| is a zombie, or this is on a pre-10.8.4 system 298 // Most likely cause: |task| is a zombie, or this is on a pre-10.8.4 system
307 // where TASK_POWER_INFO isn't supported yet. 299 // where TASK_POWER_INFO isn't supported yet.
308 return 0; 300 return 0;
309 } 301 }
310 uint64_t absolute_idle_wakeups = power_info_data.task_platform_idle_wakeups; 302 uint64_t absolute_idle_wakeups = power_info_data.task_platform_idle_wakeups;
311 303
312 TimeTicks time = TimeTicks::Now(); 304 TimeTicks time = TimeTicks::Now();
313 305
314 if (last_absolute_idle_wakeups_ == 0) { 306 if (last_absolute_idle_wakeups_ == 0) {
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after
355 } 347 }
356 348
357 // Bytes committed by the system. 349 // Bytes committed by the system.
358 size_t GetSystemCommitCharge() { 350 size_t GetSystemCommitCharge() {
359 base::mac::ScopedMachPort host(mach_host_self()); 351 base::mac::ScopedMachPort host(mach_host_self());
360 mach_msg_type_number_t count = HOST_VM_INFO_COUNT; 352 mach_msg_type_number_t count = HOST_VM_INFO_COUNT;
361 vm_statistics_data_t data; 353 vm_statistics_data_t data;
362 kern_return_t kr = host_statistics(host, HOST_VM_INFO, 354 kern_return_t kr = host_statistics(host, HOST_VM_INFO,
363 reinterpret_cast<host_info_t>(&data), 355 reinterpret_cast<host_info_t>(&data),
364 &count); 356 &count);
365 if (kr) { 357 if (kr != KERN_SUCCESS) {
366 DLOG(WARNING) << "Failed to fetch host statistics."; 358 MACH_DLOG(WARNING, kr) << "host_statistics";
367 return 0; 359 return 0;
368 } 360 }
369 361
370 vm_size_t page_size; 362 return (data.active_count * PAGE_SIZE) / 1024;
371 kr = host_page_size(host, &page_size);
372 if (kr) {
373 DLOG(ERROR) << "Failed to fetch host page size.";
374 return 0;
375 }
376
377 return (data.active_count * page_size) / 1024;
378 } 363 }
379 364
380 } // namespace base 365 } // namespace base
OLDNEW
« no previous file with comments | « base/process/memory_mac.mm ('k') | base/threading/platform_thread_mac.mm » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698