|  | // Copyright 2013 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "base/process/process_metrics.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  | #include <stdint.h> | 
|  |  | 
|  | #include <memory> | 
|  | #include <sstream> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/command_line.h" | 
|  | #include "base/files/file.h" | 
|  | #include "base/files/file_util.h" | 
|  | #include "base/files/scoped_temp_dir.h" | 
|  | #include "base/memory/shared_memory_mapping.h" | 
|  | #include "base/memory/writable_shared_memory_region.h" | 
|  | #include "base/ranges/algorithm.h" | 
|  | #include "base/strings/string_number_conversions.h" | 
|  | #include "base/strings/string_util.h" | 
|  | #include "base/strings/stringprintf.h" | 
|  | #include "base/system/sys_info.h" | 
|  | #include "base/test/multiprocess_test.h" | 
|  | #include "base/threading/thread.h" | 
|  | #include "build/build_config.h" | 
|  | #include "build/chromeos_buildflags.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  | #include "testing/multiprocess_func_list.h" | 
|  |  | 
|  | #if BUILDFLAG(IS_APPLE) | 
|  | #include <sys/mman.h> | 
|  | #endif | 
|  |  | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) | 
|  | #include "base/process/internal_linux.h" | 
|  | #endif | 
|  |  | 
|  | namespace base { | 
|  | namespace debug { | 
|  |  | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||      \ | 
|  | BUILDFLAG(IS_CHROMEOS_LACROS) || BUILDFLAG(IS_WIN) || \ | 
|  | BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_FUCHSIA) | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | void BusyWork(std::vector<std::string>* vec) { | 
|  | int64_t test_value = 0; | 
|  | for (int i = 0; i < 100000; ++i) { | 
|  | ++test_value; | 
|  | vec->push_back(NumberToString(test_value)); | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || | 
|  | // BUILDFLAG(IS_CHROMEOS_LACROS) || BUILDFLAG(IS_WIN) || | 
|  | // BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_FUCHSIA) | 
|  |  | 
|  | // Tests for SystemMetrics. | 
|  | // Exists as a class so it can be a friend of SystemMetrics. | 
|  | class SystemMetricsTest : public testing::Test { | 
|  | public: | 
|  | SystemMetricsTest() = default; | 
|  |  | 
|  | SystemMetricsTest(const SystemMetricsTest&) = delete; | 
|  | SystemMetricsTest& operator=(const SystemMetricsTest&) = delete; | 
|  | }; | 
|  |  | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) | 
|  | TEST_F(SystemMetricsTest, IsValidDiskName) { | 
|  | const char invalid_input1[] = ""; | 
|  | const char invalid_input2[] = "s"; | 
|  | const char invalid_input3[] = "sdz+"; | 
|  | const char invalid_input4[] = "hda0"; | 
|  | const char invalid_input5[] = "mmcbl"; | 
|  | const char invalid_input6[] = "mmcblka"; | 
|  | const char invalid_input7[] = "mmcblkb"; | 
|  | const char invalid_input8[] = "mmmblk0"; | 
|  |  | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input1)); | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input2)); | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input3)); | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input4)); | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input5)); | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input6)); | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input7)); | 
|  | EXPECT_FALSE(IsValidDiskName(invalid_input8)); | 
|  |  | 
|  | const char valid_input1[] = "sda"; | 
|  | const char valid_input2[] = "sdaaaa"; | 
|  | const char valid_input3[] = "hdz"; | 
|  | const char valid_input4[] = "mmcblk0"; | 
|  | const char valid_input5[] = "mmcblk999"; | 
|  |  | 
|  | EXPECT_TRUE(IsValidDiskName(valid_input1)); | 
|  | EXPECT_TRUE(IsValidDiskName(valid_input2)); | 
|  | EXPECT_TRUE(IsValidDiskName(valid_input3)); | 
|  | EXPECT_TRUE(IsValidDiskName(valid_input4)); | 
|  | EXPECT_TRUE(IsValidDiskName(valid_input5)); | 
|  | } | 
|  |  | 
|  | TEST_F(SystemMetricsTest, ParseMeminfo) { | 
|  | SystemMemoryInfoKB meminfo; | 
|  | const char invalid_input1[] = "abc"; | 
|  | const char invalid_input2[] = "MemTotal:"; | 
|  | // Partial file with no MemTotal | 
|  | const char invalid_input3[] = | 
|  | "MemFree:         3913968 kB\n" | 
|  | "Buffers:         2348340 kB\n" | 
|  | "Cached:         49071596 kB\n" | 
|  | "SwapCached:           12 kB\n" | 
|  | "Active:         36393900 kB\n" | 
|  | "Inactive:       21221496 kB\n" | 
|  | "Active(anon):    5674352 kB\n" | 
|  | "Inactive(anon):   633992 kB\n"; | 
|  | EXPECT_FALSE(ParseProcMeminfo(invalid_input1, &meminfo)); | 
|  | EXPECT_FALSE(ParseProcMeminfo(invalid_input2, &meminfo)); | 
|  | EXPECT_FALSE(ParseProcMeminfo(invalid_input3, &meminfo)); | 
|  |  | 
|  | const char valid_input1[] = | 
|  | "MemTotal:        3981504 kB\n" | 
|  | "MemFree:          140764 kB\n" | 
|  | "MemAvailable:     535413 kB\n" | 
|  | "Buffers:          116480 kB\n" | 
|  | "Cached:           406160 kB\n" | 
|  | "SwapCached:        21304 kB\n" | 
|  | "Active:          3152040 kB\n" | 
|  | "Inactive:         472856 kB\n" | 
|  | "Active(anon):    2972352 kB\n" | 
|  | "Inactive(anon):   270108 kB\n" | 
|  | "Active(file):     179688 kB\n" | 
|  | "Inactive(file):   202748 kB\n" | 
|  | "Unevictable:           0 kB\n" | 
|  | "Mlocked:               0 kB\n" | 
|  | "SwapTotal:       5832280 kB\n" | 
|  | "SwapFree:        3672368 kB\n" | 
|  | "Dirty:               184 kB\n" | 
|  | "Writeback:             0 kB\n" | 
|  | "AnonPages:       3101224 kB\n" | 
|  | "Mapped:           142296 kB\n" | 
|  | "Shmem:            140204 kB\n" | 
|  | "Slab:              54212 kB\n" | 
|  | "SReclaimable:      30936 kB\n" | 
|  | "SUnreclaim:        23276 kB\n" | 
|  | "KernelStack:        2464 kB\n" | 
|  | "PageTables:        24812 kB\n" | 
|  | "NFS_Unstable:          0 kB\n" | 
|  | "Bounce:                0 kB\n" | 
|  | "WritebackTmp:          0 kB\n" | 
|  | "CommitLimit:     7823032 kB\n" | 
|  | "Committed_AS:    7973536 kB\n" | 
|  | "VmallocTotal:   34359738367 kB\n" | 
|  | "VmallocUsed:      375940 kB\n" | 
|  | "VmallocChunk:   34359361127 kB\n" | 
|  | "DirectMap4k:       72448 kB\n" | 
|  | "DirectMap2M:     4061184 kB\n"; | 
|  | // output from a much older kernel where the Active and Inactive aren't | 
|  | // broken down into anon and file and Huge Pages are enabled | 
|  | const char valid_input2[] = | 
|  | "MemTotal:       255908 kB\n" | 
|  | "MemFree:         69936 kB\n" | 
|  | "Buffers:         15812 kB\n" | 
|  | "Cached:         115124 kB\n" | 
|  | "SwapCached:          0 kB\n" | 
|  | "Active:          92700 kB\n" | 
|  | "Inactive:        63792 kB\n" | 
|  | "HighTotal:           0 kB\n" | 
|  | "HighFree:            0 kB\n" | 
|  | "LowTotal:       255908 kB\n" | 
|  | "LowFree:         69936 kB\n" | 
|  | "SwapTotal:      524280 kB\n" | 
|  | "SwapFree:       524200 kB\n" | 
|  | "Dirty:               4 kB\n" | 
|  | "Writeback:           0 kB\n" | 
|  | "Mapped:          42236 kB\n" | 
|  | "Slab:            25912 kB\n" | 
|  | "Committed_AS:   118680 kB\n" | 
|  | "PageTables:       1236 kB\n" | 
|  | "VmallocTotal:  3874808 kB\n" | 
|  | "VmallocUsed:      1416 kB\n" | 
|  | "VmallocChunk:  3872908 kB\n" | 
|  | "HugePages_Total:     0\n" | 
|  | "HugePages_Free:      0\n" | 
|  | "Hugepagesize:     4096 kB\n"; | 
|  |  | 
|  | EXPECT_TRUE(ParseProcMeminfo(valid_input1, &meminfo)); | 
|  | EXPECT_EQ(meminfo.total, 3981504); | 
|  | EXPECT_EQ(meminfo.free, 140764); | 
|  | EXPECT_EQ(meminfo.available, 535413); | 
|  | EXPECT_EQ(meminfo.buffers, 116480); | 
|  | EXPECT_EQ(meminfo.cached, 406160); | 
|  | EXPECT_EQ(meminfo.active_anon, 2972352); | 
|  | EXPECT_EQ(meminfo.active_file, 179688); | 
|  | EXPECT_EQ(meminfo.inactive_anon, 270108); | 
|  | EXPECT_EQ(meminfo.inactive_file, 202748); | 
|  | EXPECT_EQ(meminfo.swap_total, 5832280); | 
|  | EXPECT_EQ(meminfo.swap_free, 3672368); | 
|  | EXPECT_EQ(meminfo.dirty, 184); | 
|  | EXPECT_EQ(meminfo.reclaimable, 30936); | 
|  | #if BUILDFLAG(IS_CHROMEOS) | 
|  | EXPECT_EQ(meminfo.shmem, 140204); | 
|  | EXPECT_EQ(meminfo.slab, 54212); | 
|  | #endif | 
|  | EXPECT_EQ(355725u, | 
|  | base::SysInfo::AmountOfAvailablePhysicalMemory(meminfo) / 1024); | 
|  | // Simulate as if there is no MemAvailable. | 
|  | meminfo.available = 0; | 
|  | EXPECT_EQ(374448u, | 
|  | base::SysInfo::AmountOfAvailablePhysicalMemory(meminfo) / 1024); | 
|  | meminfo = {}; | 
|  | EXPECT_TRUE(ParseProcMeminfo(valid_input2, &meminfo)); | 
|  | EXPECT_EQ(meminfo.total, 255908); | 
|  | EXPECT_EQ(meminfo.free, 69936); | 
|  | EXPECT_EQ(meminfo.available, 0); | 
|  | EXPECT_EQ(meminfo.buffers, 15812); | 
|  | EXPECT_EQ(meminfo.cached, 115124); | 
|  | EXPECT_EQ(meminfo.swap_total, 524280); | 
|  | EXPECT_EQ(meminfo.swap_free, 524200); | 
|  | EXPECT_EQ(meminfo.dirty, 4); | 
|  | EXPECT_EQ(69936u, | 
|  | base::SysInfo::AmountOfAvailablePhysicalMemory(meminfo) / 1024); | 
|  | } | 
|  |  | 
|  | TEST_F(SystemMetricsTest, ParseVmstat) { | 
|  | VmStatInfo vmstat; | 
|  | // Part of vmstat from a 4.19 kernel. | 
|  | const char valid_input1[] = | 
|  | "pgpgin 2358216\n" | 
|  | "pgpgout 296072\n" | 
|  | "pswpin 345219\n" | 
|  | "pswpout 2605828\n" | 
|  | "pgalloc_dma32 8380235\n" | 
|  | "pgalloc_normal 3384525\n" | 
|  | "pgalloc_movable 0\n" | 
|  | "allocstall_dma32 0\n" | 
|  | "allocstall_normal 2028\n" | 
|  | "allocstall_movable 32559\n" | 
|  | "pgskip_dma32 0\n" | 
|  | "pgskip_normal 0\n" | 
|  | "pgskip_movable 0\n" | 
|  | "pgfree 11802722\n" | 
|  | "pgactivate 894917\n" | 
|  | "pgdeactivate 3255711\n" | 
|  | "pglazyfree 48\n" | 
|  | "pgfault 10043657\n" | 
|  | "pgmajfault 358901\n" | 
|  | "pgmajfault_s 2100\n" | 
|  | "pgmajfault_a 343211\n" | 
|  | "pgmajfault_f 13590\n" | 
|  | "pglazyfreed 0\n" | 
|  | "pgrefill 3429488\n" | 
|  | "pgsteal_kswapd 1466893\n" | 
|  | "pgsteal_direct 1771759\n" | 
|  | "pgscan_kswapd 1907332\n" | 
|  | "pgscan_direct 2118930\n" | 
|  | "pgscan_direct_throttle 154\n" | 
|  | "pginodesteal 3176\n" | 
|  | "slabs_scanned 293804\n" | 
|  | "kswapd_inodesteal 16753\n" | 
|  | "kswapd_low_wmark_hit_quickly 10\n" | 
|  | "kswapd_high_wmark_hit_quickly 423\n" | 
|  | "pageoutrun 441\n" | 
|  | "pgrotated 1636\n" | 
|  | "drop_pagecache 0\n" | 
|  | "drop_slab 0\n" | 
|  | "oom_kill 18\n"; | 
|  | const char valid_input2[] = | 
|  | "pgpgin 2606135\n" | 
|  | "pgpgout 1359128\n" | 
|  | "pswpin 899959\n" | 
|  | "pswpout 19761244\n" | 
|  | "pgalloc_dma 31\n" | 
|  | "pgalloc_dma32 18139339\n" | 
|  | "pgalloc_normal 44085950\n" | 
|  | "pgalloc_movable 0\n" | 
|  | "allocstall_dma 0\n" | 
|  | "allocstall_dma32 0\n" | 
|  | "allocstall_normal 18881\n" | 
|  | "allocstall_movable 169527\n" | 
|  | "pgskip_dma 0\n" | 
|  | "pgskip_dma32 0\n" | 
|  | "pgskip_normal 0\n" | 
|  | "pgskip_movable 0\n" | 
|  | "pgfree 63060999\n" | 
|  | "pgactivate 1703494\n" | 
|  | "pgdeactivate 20537803\n" | 
|  | "pglazyfree 163\n" | 
|  | "pgfault 45201169\n" | 
|  | "pgmajfault 609626\n" | 
|  | "pgmajfault_s 7488\n" | 
|  | "pgmajfault_a 591793\n" | 
|  | "pgmajfault_f 10345\n" | 
|  | "pglazyfreed 0\n" | 
|  | "pgrefill 20673453\n" | 
|  | "pgsteal_kswapd 11802772\n" | 
|  | "pgsteal_direct 8618160\n" | 
|  | "pgscan_kswapd 12640517\n" | 
|  | "pgscan_direct 9092230\n" | 
|  | "pgscan_direct_throttle 638\n" | 
|  | "pginodesteal 1716\n" | 
|  | "slabs_scanned 2594642\n" | 
|  | "kswapd_inodesteal 67358\n" | 
|  | "kswapd_low_wmark_hit_quickly 52\n" | 
|  | "kswapd_high_wmark_hit_quickly 11\n" | 
|  | "pageoutrun 83\n" | 
|  | "pgrotated 977\n" | 
|  | "drop_pagecache 1\n" | 
|  | "drop_slab 1\n" | 
|  | "oom_kill 1\n" | 
|  | "pgmigrate_success 3202\n" | 
|  | "pgmigrate_fail 795\n"; | 
|  | const char valid_input3[] = | 
|  | "pswpin 12\n" | 
|  | "pswpout 901\n" | 
|  | "pgmajfault 18881\n"; | 
|  | EXPECT_TRUE(ParseProcVmstat(valid_input1, &vmstat)); | 
|  | EXPECT_EQ(345219LU, vmstat.pswpin); | 
|  | EXPECT_EQ(2605828LU, vmstat.pswpout); | 
|  | EXPECT_EQ(358901LU, vmstat.pgmajfault); | 
|  | EXPECT_EQ(18LU, vmstat.oom_kill); | 
|  | EXPECT_TRUE(ParseProcVmstat(valid_input2, &vmstat)); | 
|  | EXPECT_EQ(899959LU, vmstat.pswpin); | 
|  | EXPECT_EQ(19761244LU, vmstat.pswpout); | 
|  | EXPECT_EQ(609626LU, vmstat.pgmajfault); | 
|  | EXPECT_EQ(1LU, vmstat.oom_kill); | 
|  | EXPECT_TRUE(ParseProcVmstat(valid_input3, &vmstat)); | 
|  | EXPECT_EQ(12LU, vmstat.pswpin); | 
|  | EXPECT_EQ(901LU, vmstat.pswpout); | 
|  | EXPECT_EQ(18881LU, vmstat.pgmajfault); | 
|  | EXPECT_EQ(0LU, vmstat.oom_kill); | 
|  |  | 
|  | const char missing_pgmajfault_input[] = | 
|  | "pswpin 12\n" | 
|  | "pswpout 901\n"; | 
|  | EXPECT_FALSE(ParseProcVmstat(missing_pgmajfault_input, &vmstat)); | 
|  | const char empty_input[] = ""; | 
|  | EXPECT_FALSE(ParseProcVmstat(empty_input, &vmstat)); | 
|  | } | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || | 
|  | // BUILDFLAG(IS_ANDROID) | 
|  |  | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||      \ | 
|  | BUILDFLAG(IS_CHROMEOS_LACROS) || BUILDFLAG(IS_WIN) || \ | 
|  | BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_FUCHSIA) | 
|  |  | 
|  | // Test that ProcessMetrics::GetPlatformIndependentCPUUsage() doesn't return | 
|  | // negative values when the number of threads running on the process decreases | 
|  | // between two successive calls to it. | 
|  | TEST_F(SystemMetricsTest, TestNoNegativeCpuUsage) { | 
|  | ProcessHandle handle = GetCurrentProcessHandle(); | 
|  | std::unique_ptr<ProcessMetrics> metrics( | 
|  | ProcessMetrics::CreateProcessMetrics(handle)); | 
|  |  | 
|  | EXPECT_GE(metrics->GetPlatformIndependentCPUUsage(), 0.0); | 
|  | Thread thread1("thread1"); | 
|  | Thread thread2("thread2"); | 
|  | Thread thread3("thread3"); | 
|  |  | 
|  | thread1.StartAndWaitForTesting(); | 
|  | thread2.StartAndWaitForTesting(); | 
|  | thread3.StartAndWaitForTesting(); | 
|  |  | 
|  | ASSERT_TRUE(thread1.IsRunning()); | 
|  | ASSERT_TRUE(thread2.IsRunning()); | 
|  | ASSERT_TRUE(thread3.IsRunning()); | 
|  |  | 
|  | std::vector<std::string> vec1; | 
|  | std::vector<std::string> vec2; | 
|  | std::vector<std::string> vec3; | 
|  |  | 
|  | thread1.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec1)); | 
|  | thread2.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec2)); | 
|  | thread3.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec3)); | 
|  |  | 
|  | TimeDelta prev_cpu_usage = metrics->GetCumulativeCPUUsage(); | 
|  | EXPECT_GE(prev_cpu_usage, TimeDelta()); | 
|  | EXPECT_GE(metrics->GetPlatformIndependentCPUUsage(), 0.0); | 
|  |  | 
|  | thread1.Stop(); | 
|  | TimeDelta current_cpu_usage = metrics->GetCumulativeCPUUsage(); | 
|  | EXPECT_GE(current_cpu_usage, prev_cpu_usage); | 
|  | prev_cpu_usage = current_cpu_usage; | 
|  | EXPECT_GE(metrics->GetPlatformIndependentCPUUsage(), 0.0); | 
|  |  | 
|  | thread2.Stop(); | 
|  | current_cpu_usage = metrics->GetCumulativeCPUUsage(); | 
|  | EXPECT_GE(current_cpu_usage, prev_cpu_usage); | 
|  | prev_cpu_usage = current_cpu_usage; | 
|  | EXPECT_GE(metrics->GetPlatformIndependentCPUUsage(), 0.0); | 
|  |  | 
|  | thread3.Stop(); | 
|  | current_cpu_usage = metrics->GetCumulativeCPUUsage(); | 
|  | EXPECT_GE(current_cpu_usage, prev_cpu_usage); | 
|  | EXPECT_GE(metrics->GetPlatformIndependentCPUUsage(), 0.0); | 
|  | } | 
|  |  | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || | 
|  | // BUILDFLAG(IS_CHROMEOS_LACROS) || BUILDFLAG(IS_WIN) || | 
|  | // BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_FUCHSIA) | 
|  |  | 
|  | #if BUILDFLAG(IS_CHROMEOS) | 
|  | TEST_F(SystemMetricsTest, ParseZramMmStat) { | 
|  | SwapInfo swapinfo; | 
|  |  | 
|  | const char invalid_input1[] = "aaa"; | 
|  | const char invalid_input2[] = "1 2 3 4 5 6"; | 
|  | const char invalid_input3[] = "a 2 3 4 5 6 7"; | 
|  | EXPECT_FALSE(ParseZramMmStat(invalid_input1, &swapinfo)); | 
|  | EXPECT_FALSE(ParseZramMmStat(invalid_input2, &swapinfo)); | 
|  | EXPECT_FALSE(ParseZramMmStat(invalid_input3, &swapinfo)); | 
|  |  | 
|  | const char valid_input1[] = | 
|  | "17715200 5008166 566062  0 1225715712  127 183842"; | 
|  | EXPECT_TRUE(ParseZramMmStat(valid_input1, &swapinfo)); | 
|  | EXPECT_EQ(17715200ULL, swapinfo.orig_data_size); | 
|  | EXPECT_EQ(5008166ULL, swapinfo.compr_data_size); | 
|  | EXPECT_EQ(566062ULL, swapinfo.mem_used_total); | 
|  | } | 
|  |  | 
|  | TEST_F(SystemMetricsTest, ParseZramStat) { | 
|  | SwapInfo swapinfo; | 
|  |  | 
|  | const char invalid_input1[] = "aaa"; | 
|  | const char invalid_input2[] = "1 2 3 4 5 6 7 8 9 10"; | 
|  | const char invalid_input3[] = "a 2 3 4 5 6 7 8 9 10 11"; | 
|  | EXPECT_FALSE(ParseZramStat(invalid_input1, &swapinfo)); | 
|  | EXPECT_FALSE(ParseZramStat(invalid_input2, &swapinfo)); | 
|  | EXPECT_FALSE(ParseZramStat(invalid_input3, &swapinfo)); | 
|  |  | 
|  | const char valid_input1[] = | 
|  | "299    0    2392    0    1    0    8    0    0    0    0"; | 
|  | EXPECT_TRUE(ParseZramStat(valid_input1, &swapinfo)); | 
|  | EXPECT_EQ(299ULL, swapinfo.num_reads); | 
|  | EXPECT_EQ(1ULL, swapinfo.num_writes); | 
|  | } | 
|  | #endif  // BUILDFLAG(IS_CHROMEOS) | 
|  |  | 
|  | #if BUILDFLAG(IS_WIN) || BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_LINUX) || \ | 
|  | BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) | 
|  | TEST(SystemMetrics2Test, GetSystemMemoryInfo) { | 
|  | SystemMemoryInfoKB info; | 
|  | EXPECT_TRUE(GetSystemMemoryInfo(&info)); | 
|  |  | 
|  | // Ensure each field received a value. | 
|  | EXPECT_GT(info.total, 0); | 
|  | #if BUILDFLAG(IS_WIN) | 
|  | EXPECT_GT(info.avail_phys, 0); | 
|  | #else | 
|  | EXPECT_GT(info.free, 0); | 
|  | #endif | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) | 
|  | EXPECT_GT(info.buffers, 0); | 
|  | EXPECT_GT(info.cached, 0); | 
|  | EXPECT_GT(info.active_anon + info.inactive_anon, 0); | 
|  | EXPECT_GT(info.active_file + info.inactive_file, 0); | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || | 
|  | // BUILDFLAG(IS_ANDROID) | 
|  |  | 
|  | // All the values should be less than the total amount of memory. | 
|  | #if !BUILDFLAG(IS_WIN) && !BUILDFLAG(IS_IOS) | 
|  | // TODO(crbug.com/711450): re-enable the following assertion on iOS. | 
|  | EXPECT_LT(info.free, info.total); | 
|  | #endif | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) | 
|  | EXPECT_LT(info.buffers, info.total); | 
|  | EXPECT_LT(info.cached, info.total); | 
|  | EXPECT_LT(info.active_anon, info.total); | 
|  | EXPECT_LT(info.inactive_anon, info.total); | 
|  | EXPECT_LT(info.active_file, info.total); | 
|  | EXPECT_LT(info.inactive_file, info.total); | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || | 
|  | // BUILDFLAG(IS_ANDROID) | 
|  |  | 
|  | #if BUILDFLAG(IS_APPLE) | 
|  | EXPECT_GT(info.file_backed, 0); | 
|  | #endif | 
|  |  | 
|  | #if BUILDFLAG(IS_CHROMEOS) | 
|  | // Chrome OS exposes shmem. | 
|  | EXPECT_GT(info.shmem, 0); | 
|  | EXPECT_LT(info.shmem, info.total); | 
|  | #endif | 
|  | } | 
|  | #endif  // BUILDFLAG(IS_WIN) || BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_LINUX) || | 
|  | // BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) | 
|  |  | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) | 
|  | TEST(ProcessMetricsTest, ParseProcStatCPU) { | 
|  | // /proc/self/stat for a process running "top". | 
|  | const char kTopStat[] = "960 (top) S 16230 960 16230 34818 960 " | 
|  | "4202496 471 0 0 0 " | 
|  | "12 16 0 0 "  // <- These are the goods. | 
|  | "20 0 1 0 121946157 15077376 314 18446744073709551615 4194304 " | 
|  | "4246868 140733983044336 18446744073709551615 140244213071219 " | 
|  | "0 0 0 138047495 0 0 0 17 1 0 0 0 0 0"; | 
|  | EXPECT_EQ(12 + 16, ParseProcStatCPU(kTopStat)); | 
|  |  | 
|  | // cat /proc/self/stat on a random other machine I have. | 
|  | const char kSelfStat[] = "5364 (cat) R 5354 5364 5354 34819 5364 " | 
|  | "0 142 0 0 0 " | 
|  | "0 0 0 0 "  // <- No CPU, apparently. | 
|  | "16 0 1 0 1676099790 2957312 114 4294967295 134512640 134528148 " | 
|  | "3221224832 3221224344 3086339742 0 0 0 0 0 0 0 17 0 0 0"; | 
|  |  | 
|  | EXPECT_EQ(0, ParseProcStatCPU(kSelfStat)); | 
|  |  | 
|  | // Some weird long-running process with a weird name that I created for the | 
|  | // purposes of this test. | 
|  | const char kWeirdNameStat[] = "26115 (Hello) You ()))  ) R 24614 26115 24614" | 
|  | " 34839 26115 4218880 227 0 0 0 " | 
|  | "5186 11 0 0 " | 
|  | "20 0 1 0 36933953 4296704 90 18446744073709551615 4194304 4196116 " | 
|  | "140735857761568 140735857761160 4195644 0 0 0 0 0 0 0 17 14 0 0 0 0 0 " | 
|  | "6295056 6295616 16519168 140735857770710 140735857770737 " | 
|  | "140735857770737 140735857774557 0"; | 
|  | EXPECT_EQ(5186 + 11, ParseProcStatCPU(kWeirdNameStat)); | 
|  | } | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || | 
|  | // BUILDFLAG(IS_ANDROID) | 
|  |  | 
|  | // Disable on Android because base_unittests runs inside a Dalvik VM that | 
|  | // starts and stop threads (crbug.com/175563). | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) | 
|  | // http://crbug.com/396455 | 
|  | TEST(ProcessMetricsTest, DISABLED_GetNumberOfThreads) { | 
|  | const ProcessHandle current = GetCurrentProcessHandle(); | 
|  | const int64_t initial_threads = GetNumberOfThreads(current); | 
|  | ASSERT_GT(initial_threads, 0); | 
|  | const int kNumAdditionalThreads = 10; | 
|  | { | 
|  | std::unique_ptr<Thread> my_threads[kNumAdditionalThreads]; | 
|  | for (int i = 0; i < kNumAdditionalThreads; ++i) { | 
|  | my_threads[i] = std::make_unique<Thread>("GetNumberOfThreadsTest"); | 
|  | my_threads[i]->Start(); | 
|  | ASSERT_EQ(GetNumberOfThreads(current), initial_threads + 1 + i); | 
|  | } | 
|  | } | 
|  | // The Thread destructor will stop them. | 
|  | ASSERT_EQ(initial_threads, GetNumberOfThreads(current)); | 
|  | } | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) | 
|  |  | 
|  | #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_MAC) | 
|  | namespace { | 
|  |  | 
|  | // Keep these in sync so the GetChildOpenFdCount test can refer to correct test | 
|  | // main. | 
|  | #define ChildMain ChildFdCount | 
|  | #define ChildMainString "ChildFdCount" | 
|  |  | 
|  | // Command line flag name and file name used for synchronization. | 
|  | const char kTempDirFlag[] = "temp-dir"; | 
|  |  | 
|  | const char kSignalReady[] = "ready"; | 
|  | const char kSignalReadyAck[] = "ready-ack"; | 
|  | const char kSignalOpened[] = "opened"; | 
|  | const char kSignalOpenedAck[] = "opened-ack"; | 
|  | const char kSignalClosed[] = "closed"; | 
|  |  | 
|  | const int kChildNumFilesToOpen = 100; | 
|  |  | 
|  | bool SignalEvent(const FilePath& signal_dir, const char* signal_file) { | 
|  | File file(signal_dir.AppendASCII(signal_file), | 
|  | File::FLAG_CREATE | File::FLAG_WRITE); | 
|  | return file.IsValid(); | 
|  | } | 
|  |  | 
|  | // Check whether an event was signaled. | 
|  | bool CheckEvent(const FilePath& signal_dir, const char* signal_file) { | 
|  | File file(signal_dir.AppendASCII(signal_file), | 
|  | File::FLAG_OPEN | File::FLAG_READ); | 
|  | return file.IsValid(); | 
|  | } | 
|  |  | 
|  | // Busy-wait for an event to be signaled. | 
|  | void WaitForEvent(const FilePath& signal_dir, const char* signal_file) { | 
|  | while (!CheckEvent(signal_dir, signal_file)) | 
|  | PlatformThread::Sleep(Milliseconds(10)); | 
|  | } | 
|  |  | 
|  | // Subprocess to test the number of open file descriptors. | 
|  | MULTIPROCESS_TEST_MAIN(ChildMain) { | 
|  | CommandLine* command_line = CommandLine::ForCurrentProcess(); | 
|  | const FilePath temp_path = command_line->GetSwitchValuePath(kTempDirFlag); | 
|  | CHECK(DirectoryExists(temp_path)); | 
|  |  | 
|  | CHECK(SignalEvent(temp_path, kSignalReady)); | 
|  | WaitForEvent(temp_path, kSignalReadyAck); | 
|  |  | 
|  | std::vector<File> files; | 
|  | for (int i = 0; i < kChildNumFilesToOpen; ++i) { | 
|  | files.emplace_back(temp_path.AppendASCII(StringPrintf("file.%d", i)), | 
|  | File::FLAG_CREATE | File::FLAG_WRITE); | 
|  | } | 
|  |  | 
|  | CHECK(SignalEvent(temp_path, kSignalOpened)); | 
|  | WaitForEvent(temp_path, kSignalOpenedAck); | 
|  |  | 
|  | files.clear(); | 
|  |  | 
|  | CHECK(SignalEvent(temp_path, kSignalClosed)); | 
|  |  | 
|  | // Wait to be terminated. | 
|  | while (true) | 
|  | PlatformThread::Sleep(Seconds(1)); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | TEST(ProcessMetricsTest, GetChildOpenFdCount) { | 
|  | ScopedTempDir temp_dir; | 
|  | ASSERT_TRUE(temp_dir.CreateUniqueTempDir()); | 
|  | const FilePath temp_path = temp_dir.GetPath(); | 
|  | CommandLine child_command_line(GetMultiProcessTestChildBaseCommandLine()); | 
|  | child_command_line.AppendSwitchPath(kTempDirFlag, temp_path); | 
|  | Process child = SpawnMultiProcessTestChild( | 
|  | ChildMainString, child_command_line, LaunchOptions()); | 
|  | ASSERT_TRUE(child.IsValid()); | 
|  |  | 
|  | WaitForEvent(temp_path, kSignalReady); | 
|  |  | 
|  | std::unique_ptr<ProcessMetrics> metrics = | 
|  | #if BUILDFLAG(IS_APPLE) | 
|  | ProcessMetrics::CreateProcessMetrics(child.Handle(), nullptr); | 
|  | #else | 
|  | ProcessMetrics::CreateProcessMetrics(child.Handle()); | 
|  | #endif  // BUILDFLAG(IS_APPLE) | 
|  |  | 
|  | const int fd_count = metrics->GetOpenFdCount(); | 
|  | EXPECT_GE(fd_count, 0); | 
|  |  | 
|  | ASSERT_TRUE(SignalEvent(temp_path, kSignalReadyAck)); | 
|  | WaitForEvent(temp_path, kSignalOpened); | 
|  |  | 
|  | EXPECT_EQ(fd_count + kChildNumFilesToOpen, metrics->GetOpenFdCount()); | 
|  | ASSERT_TRUE(SignalEvent(temp_path, kSignalOpenedAck)); | 
|  |  | 
|  | WaitForEvent(temp_path, kSignalClosed); | 
|  |  | 
|  | EXPECT_EQ(fd_count, metrics->GetOpenFdCount()); | 
|  |  | 
|  | ASSERT_TRUE(child.Terminate(0, true)); | 
|  | } | 
|  |  | 
|  | TEST(ProcessMetricsTest, GetOpenFdCount) { | 
|  | base::ProcessHandle process = base::GetCurrentProcessHandle(); | 
|  | std::unique_ptr<base::ProcessMetrics> metrics = | 
|  | #if BUILDFLAG(IS_APPLE) | 
|  | ProcessMetrics::CreateProcessMetrics(process, nullptr); | 
|  | #else | 
|  | ProcessMetrics::CreateProcessMetrics(process); | 
|  | #endif  // BUILDFLAG(IS_APPLE) | 
|  |  | 
|  | ScopedTempDir temp_dir; | 
|  | ASSERT_TRUE(temp_dir.CreateUniqueTempDir()); | 
|  |  | 
|  | int fd_count = metrics->GetOpenFdCount(); | 
|  | EXPECT_GT(fd_count, 0); | 
|  | File file(temp_dir.GetPath().AppendASCII("file"), | 
|  | File::FLAG_CREATE | File::FLAG_WRITE); | 
|  | int new_fd_count = metrics->GetOpenFdCount(); | 
|  | EXPECT_GT(new_fd_count, 0); | 
|  | EXPECT_EQ(new_fd_count, fd_count + 1); | 
|  | } | 
|  |  | 
|  | #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_MAC) | 
|  |  | 
|  | #if BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) | 
|  |  | 
|  | TEST(ProcessMetricsTestLinux, GetPageFaultCounts) { | 
|  | std::unique_ptr<base::ProcessMetrics> process_metrics( | 
|  | base::ProcessMetrics::CreateProcessMetrics( | 
|  | base::GetCurrentProcessHandle())); | 
|  |  | 
|  | PageFaultCounts counts; | 
|  | ASSERT_TRUE(process_metrics->GetPageFaultCounts(&counts)); | 
|  | ASSERT_GT(counts.minor, 0); | 
|  | ASSERT_GE(counts.major, 0); | 
|  |  | 
|  | // Allocate and touch memory. Touching it is required to make sure that the | 
|  | // page fault count goes up, as memory is typically mapped lazily. | 
|  | { | 
|  | const size_t kMappedSize = 4 << 20;  // 4 MiB. | 
|  |  | 
|  | WritableSharedMemoryRegion region = | 
|  | WritableSharedMemoryRegion::Create(kMappedSize); | 
|  | ASSERT_TRUE(region.IsValid()); | 
|  |  | 
|  | WritableSharedMemoryMapping mapping = region.Map(); | 
|  | ASSERT_TRUE(mapping.IsValid()); | 
|  |  | 
|  | memset(mapping.memory(), 42, kMappedSize); | 
|  | } | 
|  |  | 
|  | PageFaultCounts counts_after; | 
|  | ASSERT_TRUE(process_metrics->GetPageFaultCounts(&counts_after)); | 
|  | ASSERT_GT(counts_after.minor, counts.minor); | 
|  | ASSERT_GE(counts_after.major, counts.major); | 
|  | } | 
|  |  | 
|  | TEST(ProcessMetricsTestLinux, GetCumulativeCPUUsagePerThread) { | 
|  | ProcessHandle handle = GetCurrentProcessHandle(); | 
|  | std::unique_ptr<ProcessMetrics> metrics( | 
|  | ProcessMetrics::CreateProcessMetrics(handle)); | 
|  |  | 
|  | Thread thread1("thread1"); | 
|  | thread1.StartAndWaitForTesting(); | 
|  | ASSERT_TRUE(thread1.IsRunning()); | 
|  |  | 
|  | std::vector<std::string> vec1; | 
|  | thread1.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec1)); | 
|  |  | 
|  | ProcessMetrics::CPUUsagePerThread prev_thread_times; | 
|  | EXPECT_TRUE(metrics->GetCumulativeCPUUsagePerThread(prev_thread_times)); | 
|  |  | 
|  | // Should have at least the test runner thread and the thread spawned above. | 
|  | EXPECT_GE(prev_thread_times.size(), 2u); | 
|  | EXPECT_TRUE(ranges::any_of( | 
|  | prev_thread_times, | 
|  | [&thread1](const std::pair<PlatformThreadId, base::TimeDelta>& entry) { | 
|  | return entry.first == thread1.GetThreadId(); | 
|  | })); | 
|  | EXPECT_TRUE(ranges::any_of( | 
|  | prev_thread_times, | 
|  | [](const std::pair<PlatformThreadId, base::TimeDelta>& entry) { | 
|  | return entry.first == base::PlatformThread::CurrentId(); | 
|  | })); | 
|  |  | 
|  | for (const auto& entry : prev_thread_times) { | 
|  | EXPECT_GE(entry.second, base::TimeDelta()); | 
|  | } | 
|  |  | 
|  | thread1.Stop(); | 
|  |  | 
|  | ProcessMetrics::CPUUsagePerThread current_thread_times; | 
|  | EXPECT_TRUE(metrics->GetCumulativeCPUUsagePerThread(current_thread_times)); | 
|  |  | 
|  | // The stopped thread may still be reported until the kernel cleans it up. | 
|  | EXPECT_GE(prev_thread_times.size(), 1u); | 
|  | EXPECT_TRUE(ranges::any_of( | 
|  | current_thread_times, | 
|  | [](const std::pair<PlatformThreadId, base::TimeDelta>& entry) { | 
|  | return entry.first == base::PlatformThread::CurrentId(); | 
|  | })); | 
|  |  | 
|  | // Reported times should not decrease. | 
|  | for (const auto& entry : current_thread_times) { | 
|  | auto prev_it = ranges::find_if( | 
|  | prev_thread_times, | 
|  | [&entry]( | 
|  | const std::pair<PlatformThreadId, base::TimeDelta>& prev_entry) { | 
|  | return entry.first == prev_entry.first; | 
|  | }); | 
|  |  | 
|  | if (prev_it != prev_thread_times.end()) | 
|  | EXPECT_GE(entry.second, prev_it->second); | 
|  | } | 
|  | } | 
|  | #endif  // BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_LINUX) || | 
|  | // BUILDFLAG(IS_CHROMEOS) | 
|  |  | 
|  | }  // namespace debug | 
|  | }  // namespace base |