xref: /aosp_15_r20/external/cronet/base/process/process_metrics_unittest.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2013 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "base/process/process_metrics.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <stddef.h>
8*6777b538SAndroid Build Coastguard Worker #include <stdint.h>
9*6777b538SAndroid Build Coastguard Worker 
10*6777b538SAndroid Build Coastguard Worker #include <memory>
11*6777b538SAndroid Build Coastguard Worker #include <sstream>
12*6777b538SAndroid Build Coastguard Worker #include <string>
13*6777b538SAndroid Build Coastguard Worker #include <utility>
14*6777b538SAndroid Build Coastguard Worker #include <vector>
15*6777b538SAndroid Build Coastguard Worker 
16*6777b538SAndroid Build Coastguard Worker #include "base/command_line.h"
17*6777b538SAndroid Build Coastguard Worker #include "base/files/file.h"
18*6777b538SAndroid Build Coastguard Worker #include "base/files/file_path.h"
19*6777b538SAndroid Build Coastguard Worker #include "base/files/file_util.h"
20*6777b538SAndroid Build Coastguard Worker #include "base/files/scoped_temp_dir.h"
21*6777b538SAndroid Build Coastguard Worker #include "base/functional/bind.h"
22*6777b538SAndroid Build Coastguard Worker #include "base/memory/shared_memory_mapping.h"
23*6777b538SAndroid Build Coastguard Worker #include "base/memory/writable_shared_memory_region.h"
24*6777b538SAndroid Build Coastguard Worker #include "base/process/launch.h"
25*6777b538SAndroid Build Coastguard Worker #include "base/process/process.h"
26*6777b538SAndroid Build Coastguard Worker #include "base/process/process_handle.h"
27*6777b538SAndroid Build Coastguard Worker #include "base/ranges/algorithm.h"
28*6777b538SAndroid Build Coastguard Worker #include "base/strings/string_number_conversions.h"
29*6777b538SAndroid Build Coastguard Worker #include "base/strings/string_util.h"
30*6777b538SAndroid Build Coastguard Worker #include "base/strings/stringprintf.h"
31*6777b538SAndroid Build Coastguard Worker #include "base/system/sys_info.h"
32*6777b538SAndroid Build Coastguard Worker #include "base/test/gmock_expected_support.h"
33*6777b538SAndroid Build Coastguard Worker #include "base/test/gtest_util.h"
34*6777b538SAndroid Build Coastguard Worker #include "base/test/multiprocess_test.h"
35*6777b538SAndroid Build Coastguard Worker #include "base/test/test_timeouts.h"
36*6777b538SAndroid Build Coastguard Worker #include "base/threading/thread.h"
37*6777b538SAndroid Build Coastguard Worker #include "base/types/expected.h"
38*6777b538SAndroid Build Coastguard Worker #include "build/blink_buildflags.h"
39*6777b538SAndroid Build Coastguard Worker #include "build/build_config.h"
40*6777b538SAndroid Build Coastguard Worker #include "build/chromeos_buildflags.h"
41*6777b538SAndroid Build Coastguard Worker #include "testing/gmock/include/gmock/gmock.h"
42*6777b538SAndroid Build Coastguard Worker #include "testing/gtest/include/gtest/gtest.h"
43*6777b538SAndroid Build Coastguard Worker #include "testing/multiprocess_func_list.h"
44*6777b538SAndroid Build Coastguard Worker 
45*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_APPLE)
46*6777b538SAndroid Build Coastguard Worker #include <sys/mman.h>
47*6777b538SAndroid Build Coastguard Worker #endif
48*6777b538SAndroid Build Coastguard Worker 
49*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_MAC)
50*6777b538SAndroid Build Coastguard Worker #include <mach/mach.h>
51*6777b538SAndroid Build Coastguard Worker 
52*6777b538SAndroid Build Coastguard Worker #include "base/apple/mach_logging.h"
53*6777b538SAndroid Build Coastguard Worker #include "base/apple/scoped_mach_port.h"
54*6777b538SAndroid Build Coastguard Worker #include "base/mac/mach_port_rendezvous.h"
55*6777b538SAndroid Build Coastguard Worker #include "base/process/port_provider_mac.h"
56*6777b538SAndroid Build Coastguard Worker #endif
57*6777b538SAndroid Build Coastguard Worker 
58*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||      \
59*6777b538SAndroid Build Coastguard Worker     BUILDFLAG(IS_CHROMEOS_LACROS) || BUILDFLAG(IS_WIN) || \
60*6777b538SAndroid Build Coastguard Worker     BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_FUCHSIA) || BUILDFLAG(IS_APPLE)
61*6777b538SAndroid Build Coastguard Worker #define ENABLE_CPU_TESTS 1
62*6777b538SAndroid Build Coastguard Worker #else
63*6777b538SAndroid Build Coastguard Worker #define ENABLE_CPU_TESTS 0
64*6777b538SAndroid Build Coastguard Worker #endif
65*6777b538SAndroid Build Coastguard Worker 
66*6777b538SAndroid Build Coastguard Worker namespace base::debug {
67*6777b538SAndroid Build Coastguard Worker 
68*6777b538SAndroid Build Coastguard Worker namespace {
69*6777b538SAndroid Build Coastguard Worker 
70*6777b538SAndroid Build Coastguard Worker using base::test::ErrorIs;
71*6777b538SAndroid Build Coastguard Worker using base::test::ValueIs;
72*6777b538SAndroid Build Coastguard Worker using ::testing::_;
73*6777b538SAndroid Build Coastguard Worker using ::testing::AssertionFailure;
74*6777b538SAndroid Build Coastguard Worker using ::testing::AssertionResult;
75*6777b538SAndroid Build Coastguard Worker using ::testing::AssertionSuccess;
76*6777b538SAndroid Build Coastguard Worker using ::testing::Ge;
77*6777b538SAndroid Build Coastguard Worker 
78*6777b538SAndroid Build Coastguard Worker #if ENABLE_CPU_TESTS
79*6777b538SAndroid Build Coastguard Worker 
BusyWork(std::vector<std::string> * vec)80*6777b538SAndroid Build Coastguard Worker void BusyWork(std::vector<std::string>* vec) {
81*6777b538SAndroid Build Coastguard Worker   int64_t test_value = 0;
82*6777b538SAndroid Build Coastguard Worker   for (int i = 0; i < 100000; ++i) {
83*6777b538SAndroid Build Coastguard Worker     ++test_value;
84*6777b538SAndroid Build Coastguard Worker     vec->push_back(NumberToString(test_value));
85*6777b538SAndroid Build Coastguard Worker   }
86*6777b538SAndroid Build Coastguard Worker }
87*6777b538SAndroid Build Coastguard Worker 
88*6777b538SAndroid Build Coastguard Worker // Tests that GetCumulativeCPUUsage() returns a valid result that's equal to or
89*6777b538SAndroid Build Coastguard Worker // greater than `prev_cpu_usage`, and returns the result. If
90*6777b538SAndroid Build Coastguard Worker // GetCumulativeCPUUsage() returns an error, records a failed expectation and
91*6777b538SAndroid Build Coastguard Worker // returns an empty TimeDelta so that each test doesn't need to check for
92*6777b538SAndroid Build Coastguard Worker // nullopt repeatedly.
TestCumulativeCPU(ProcessMetrics * metrics,TimeDelta prev_cpu_usage)93*6777b538SAndroid Build Coastguard Worker TimeDelta TestCumulativeCPU(ProcessMetrics* metrics, TimeDelta prev_cpu_usage) {
94*6777b538SAndroid Build Coastguard Worker   const base::expected<TimeDelta, ProcessCPUUsageError> current_cpu_usage =
95*6777b538SAndroid Build Coastguard Worker       metrics->GetCumulativeCPUUsage();
96*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(current_cpu_usage, ValueIs(Ge(prev_cpu_usage)));
97*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(metrics->GetPlatformIndependentCPUUsage(), ValueIs(Ge(0.0)));
98*6777b538SAndroid Build Coastguard Worker   return current_cpu_usage.value_or(TimeDelta());
99*6777b538SAndroid Build Coastguard Worker }
100*6777b538SAndroid Build Coastguard Worker 
101*6777b538SAndroid Build Coastguard Worker #endif  // ENABLE_CPU_TESTS
102*6777b538SAndroid Build Coastguard Worker 
103*6777b538SAndroid Build Coastguard Worker // Helper to deal with Mac process launching complexity. On other platforms this
104*6777b538SAndroid Build Coastguard Worker // is just a thin wrapper around SpawnMultiProcessTestChild.
105*6777b538SAndroid Build Coastguard Worker class TestChildLauncher {
106*6777b538SAndroid Build Coastguard Worker  public:
107*6777b538SAndroid Build Coastguard Worker   TestChildLauncher() = default;
108*6777b538SAndroid Build Coastguard Worker   ~TestChildLauncher() = default;
109*6777b538SAndroid Build Coastguard Worker 
110*6777b538SAndroid Build Coastguard Worker   TestChildLauncher(const TestChildLauncher&) = delete;
111*6777b538SAndroid Build Coastguard Worker   TestChildLauncher& operator=(const TestChildLauncher&) = delete;
112*6777b538SAndroid Build Coastguard Worker 
113*6777b538SAndroid Build Coastguard Worker   // Returns a reference to the command line for the child process. This can be
114*6777b538SAndroid Build Coastguard Worker   // used to add extra parameters before calling SpawnChildProcess().
command_line()115*6777b538SAndroid Build Coastguard Worker   CommandLine& command_line() { return command_line_; }
116*6777b538SAndroid Build Coastguard Worker 
117*6777b538SAndroid Build Coastguard Worker   // Returns a reference to the child process object, which will be invalid
118*6777b538SAndroid Build Coastguard Worker   // until SpawnChildProcess() is called.
child_process()119*6777b538SAndroid Build Coastguard Worker   Process& child_process() { return child_process_; }
120*6777b538SAndroid Build Coastguard Worker 
121*6777b538SAndroid Build Coastguard Worker   // Spawns a multiprocess test child to execute the function `procname`.
122*6777b538SAndroid Build Coastguard Worker   AssertionResult SpawnChildProcess(const std::string& procname);
123*6777b538SAndroid Build Coastguard Worker 
124*6777b538SAndroid Build Coastguard Worker   // Returns a ProcessMetrics object for the child process created by
125*6777b538SAndroid Build Coastguard Worker   // SpawnChildProcess().
126*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> CreateChildProcessMetrics();
127*6777b538SAndroid Build Coastguard Worker 
128*6777b538SAndroid Build Coastguard Worker   // Terminates the child process created by SpawnChildProcess(). Returns true
129*6777b538SAndroid Build Coastguard Worker   // if the process successfully terminates within the allowed time.
130*6777b538SAndroid Build Coastguard Worker   bool TerminateChildProcess();
131*6777b538SAndroid Build Coastguard Worker 
132*6777b538SAndroid Build Coastguard Worker   // Called from the child process to send data back to the parent if needed.
133*6777b538SAndroid Build Coastguard Worker   static void NotifyParent();
134*6777b538SAndroid Build Coastguard Worker 
135*6777b538SAndroid Build Coastguard Worker  private:
136*6777b538SAndroid Build Coastguard Worker   CommandLine command_line_ = GetMultiProcessTestChildBaseCommandLine();
137*6777b538SAndroid Build Coastguard Worker   Process child_process_;
138*6777b538SAndroid Build Coastguard Worker 
139*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_MAC)
140*6777b538SAndroid Build Coastguard Worker   class TestChildPortProvider;
141*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<TestChildPortProvider> port_provider_;
142*6777b538SAndroid Build Coastguard Worker #endif
143*6777b538SAndroid Build Coastguard Worker };
144*6777b538SAndroid Build Coastguard Worker 
145*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_MAC)
146*6777b538SAndroid Build Coastguard Worker 
147*6777b538SAndroid Build Coastguard Worker // Adapted from base/mac/mach_port_rendezvous_unittest.cc and
148*6777b538SAndroid Build Coastguard Worker // https://mw.foldr.org/posts/computers/macosx/task-info-fun-with-mach/
149*6777b538SAndroid Build Coastguard Worker 
150*6777b538SAndroid Build Coastguard Worker constexpr MachPortsForRendezvous::key_type kTestChildRendezvousKey = 'test';
151*6777b538SAndroid Build Coastguard Worker 
152*6777b538SAndroid Build Coastguard Worker // A PortProvider that tracks child processes spawned by TestChildLauncher.
153*6777b538SAndroid Build Coastguard Worker class TestChildLauncher::TestChildPortProvider final : public PortProvider {
154*6777b538SAndroid Build Coastguard Worker  public:
TestChildPortProvider(ProcessHandle handle,apple::ScopedMachSendRight port)155*6777b538SAndroid Build Coastguard Worker   TestChildPortProvider(ProcessHandle handle, apple::ScopedMachSendRight port)
156*6777b538SAndroid Build Coastguard Worker       : handle_(handle), port_(std::move(port)) {}
157*6777b538SAndroid Build Coastguard Worker 
158*6777b538SAndroid Build Coastguard Worker   ~TestChildPortProvider() final = default;
159*6777b538SAndroid Build Coastguard Worker 
160*6777b538SAndroid Build Coastguard Worker   TestChildPortProvider(const TestChildPortProvider&) = delete;
161*6777b538SAndroid Build Coastguard Worker   TestChildPortProvider& operator=(const TestChildPortProvider&) = delete;
162*6777b538SAndroid Build Coastguard Worker 
TaskForHandle(ProcessHandle process_handle) const163*6777b538SAndroid Build Coastguard Worker   mach_port_t TaskForHandle(ProcessHandle process_handle) const final {
164*6777b538SAndroid Build Coastguard Worker     return process_handle == handle_ ? port_.get() : MACH_PORT_NULL;
165*6777b538SAndroid Build Coastguard Worker   }
166*6777b538SAndroid Build Coastguard Worker 
167*6777b538SAndroid Build Coastguard Worker  private:
168*6777b538SAndroid Build Coastguard Worker   ProcessHandle handle_;
169*6777b538SAndroid Build Coastguard Worker   apple::ScopedMachSendRight port_;
170*6777b538SAndroid Build Coastguard Worker };
171*6777b538SAndroid Build Coastguard Worker 
SpawnChildProcess(const std::string & procname)172*6777b538SAndroid Build Coastguard Worker AssertionResult TestChildLauncher::SpawnChildProcess(
173*6777b538SAndroid Build Coastguard Worker     const std::string& procname) {
174*6777b538SAndroid Build Coastguard Worker   // Allocate a port for the parent to receive details from the child process.
175*6777b538SAndroid Build Coastguard Worker   apple::ScopedMachReceiveRight receive_port;
176*6777b538SAndroid Build Coastguard Worker   if (!apple::CreateMachPort(&receive_port, nullptr)) {
177*6777b538SAndroid Build Coastguard Worker     return AssertionFailure() << "Failed to allocate receive port";
178*6777b538SAndroid Build Coastguard Worker   }
179*6777b538SAndroid Build Coastguard Worker 
180*6777b538SAndroid Build Coastguard Worker   // Pass the sending end of the port to the child.
181*6777b538SAndroid Build Coastguard Worker   LaunchOptions options = LaunchOptionsForTest();
182*6777b538SAndroid Build Coastguard Worker   options.mach_ports_for_rendezvous.emplace(
183*6777b538SAndroid Build Coastguard Worker       kTestChildRendezvousKey,
184*6777b538SAndroid Build Coastguard Worker       MachRendezvousPort(receive_port.get(), MACH_MSG_TYPE_MAKE_SEND));
185*6777b538SAndroid Build Coastguard Worker   child_process_ =
186*6777b538SAndroid Build Coastguard Worker       SpawnMultiProcessTestChild(procname, command_line_, std::move(options));
187*6777b538SAndroid Build Coastguard Worker   if (!child_process_.IsValid()) {
188*6777b538SAndroid Build Coastguard Worker     return AssertionFailure() << "Failed to launch child process.";
189*6777b538SAndroid Build Coastguard Worker   }
190*6777b538SAndroid Build Coastguard Worker 
191*6777b538SAndroid Build Coastguard Worker   // Wait for the child to send back its mach_task_self().
192*6777b538SAndroid Build Coastguard Worker   struct : mach_msg_base_t {
193*6777b538SAndroid Build Coastguard Worker     mach_msg_port_descriptor_t task_port;
194*6777b538SAndroid Build Coastguard Worker     mach_msg_trailer_t trailer;
195*6777b538SAndroid Build Coastguard Worker   } msg{};
196*6777b538SAndroid Build Coastguard Worker   kern_return_t kr =
197*6777b538SAndroid Build Coastguard Worker       mach_msg(&msg.header, MACH_RCV_MSG | MACH_RCV_TIMEOUT, 0, sizeof(msg),
198*6777b538SAndroid Build Coastguard Worker                receive_port.get(),
199*6777b538SAndroid Build Coastguard Worker                TestTimeouts::action_timeout().InMilliseconds(), MACH_PORT_NULL);
200*6777b538SAndroid Build Coastguard Worker   if (kr != KERN_SUCCESS) {
201*6777b538SAndroid Build Coastguard Worker     return AssertionFailure()
202*6777b538SAndroid Build Coastguard Worker            << "Failed to read mach_task_self from child process: "
203*6777b538SAndroid Build Coastguard Worker            << mach_error_string(kr);
204*6777b538SAndroid Build Coastguard Worker   }
205*6777b538SAndroid Build Coastguard Worker   port_provider_ = std::make_unique<TestChildPortProvider>(
206*6777b538SAndroid Build Coastguard Worker       child_process_.Handle(), apple::ScopedMachSendRight(msg.task_port.name));
207*6777b538SAndroid Build Coastguard Worker   return AssertionSuccess();
208*6777b538SAndroid Build Coastguard Worker }
209*6777b538SAndroid Build Coastguard Worker 
CreateChildProcessMetrics()210*6777b538SAndroid Build Coastguard Worker std::unique_ptr<ProcessMetrics> TestChildLauncher::CreateChildProcessMetrics() {
211*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_MAC)
212*6777b538SAndroid Build Coastguard Worker   return ProcessMetrics::CreateProcessMetrics(child_process_.Handle(),
213*6777b538SAndroid Build Coastguard Worker                                               port_provider_.get());
214*6777b538SAndroid Build Coastguard Worker #else
215*6777b538SAndroid Build Coastguard Worker   return ProcessMetrics::CreateProcessMetrics(child_process_.Handle());
216*6777b538SAndroid Build Coastguard Worker #endif
217*6777b538SAndroid Build Coastguard Worker }
218*6777b538SAndroid Build Coastguard Worker 
TerminateChildProcess()219*6777b538SAndroid Build Coastguard Worker bool TestChildLauncher::TerminateChildProcess() {
220*6777b538SAndroid Build Coastguard Worker   return TerminateMultiProcessTestChild(child_process_, /*exit_code=*/0,
221*6777b538SAndroid Build Coastguard Worker                                         /*wait=*/true);
222*6777b538SAndroid Build Coastguard Worker }
223*6777b538SAndroid Build Coastguard Worker 
224*6777b538SAndroid Build Coastguard Worker // static
NotifyParent()225*6777b538SAndroid Build Coastguard Worker void TestChildLauncher::NotifyParent() {
226*6777b538SAndroid Build Coastguard Worker   auto* client = MachPortRendezvousClient::GetInstance();
227*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(client);
228*6777b538SAndroid Build Coastguard Worker   apple::ScopedMachSendRight send_port =
229*6777b538SAndroid Build Coastguard Worker       client->TakeSendRight(kTestChildRendezvousKey);
230*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(send_port.is_valid());
231*6777b538SAndroid Build Coastguard Worker 
232*6777b538SAndroid Build Coastguard Worker   // Send mach_task_self to the parent process so that it can use the port to
233*6777b538SAndroid Build Coastguard Worker   // create ProcessMetrics.
234*6777b538SAndroid Build Coastguard Worker   struct : mach_msg_base_t {
235*6777b538SAndroid Build Coastguard Worker     mach_msg_port_descriptor_t task_port;
236*6777b538SAndroid Build Coastguard Worker   } msg{};
237*6777b538SAndroid Build Coastguard Worker   msg.header.msgh_bits =
238*6777b538SAndroid Build Coastguard Worker       MACH_MSGH_BITS_REMOTE(MACH_MSG_TYPE_COPY_SEND) | MACH_MSGH_BITS_COMPLEX;
239*6777b538SAndroid Build Coastguard Worker   msg.header.msgh_remote_port = send_port.get();
240*6777b538SAndroid Build Coastguard Worker   msg.header.msgh_size = sizeof(msg);
241*6777b538SAndroid Build Coastguard Worker   msg.body.msgh_descriptor_count = 1;
242*6777b538SAndroid Build Coastguard Worker   msg.task_port.name = mach_task_self();
243*6777b538SAndroid Build Coastguard Worker   msg.task_port.disposition = MACH_MSG_TYPE_COPY_SEND;
244*6777b538SAndroid Build Coastguard Worker   msg.task_port.type = MACH_MSG_PORT_DESCRIPTOR;
245*6777b538SAndroid Build Coastguard Worker   kern_return_t kr =
246*6777b538SAndroid Build Coastguard Worker       mach_msg(&msg.header, MACH_SEND_MSG, msg.header.msgh_size, 0,
247*6777b538SAndroid Build Coastguard Worker                MACH_PORT_NULL, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
248*6777b538SAndroid Build Coastguard Worker   MACH_CHECK(kr == KERN_SUCCESS, kr);
249*6777b538SAndroid Build Coastguard Worker }
250*6777b538SAndroid Build Coastguard Worker 
251*6777b538SAndroid Build Coastguard Worker #else
252*6777b538SAndroid Build Coastguard Worker 
SpawnChildProcess(const std::string & procname)253*6777b538SAndroid Build Coastguard Worker AssertionResult TestChildLauncher::SpawnChildProcess(
254*6777b538SAndroid Build Coastguard Worker     const std::string& procname) {
255*6777b538SAndroid Build Coastguard Worker   child_process_ = SpawnMultiProcessTestChild(procname, command_line_,
256*6777b538SAndroid Build Coastguard Worker                                               LaunchOptionsForTest());
257*6777b538SAndroid Build Coastguard Worker   return child_process_.IsValid()
258*6777b538SAndroid Build Coastguard Worker              ? AssertionSuccess()
259*6777b538SAndroid Build Coastguard Worker              : AssertionFailure() << "Failed to launch child process.";
260*6777b538SAndroid Build Coastguard Worker }
261*6777b538SAndroid Build Coastguard Worker 
CreateChildProcessMetrics()262*6777b538SAndroid Build Coastguard Worker std::unique_ptr<ProcessMetrics> TestChildLauncher::CreateChildProcessMetrics() {
263*6777b538SAndroid Build Coastguard Worker   return ProcessMetrics::CreateProcessMetrics(child_process_.Handle());
264*6777b538SAndroid Build Coastguard Worker }
265*6777b538SAndroid Build Coastguard Worker 
TerminateChildProcess()266*6777b538SAndroid Build Coastguard Worker bool TestChildLauncher::TerminateChildProcess() {
267*6777b538SAndroid Build Coastguard Worker   [[maybe_unused]] const ProcessHandle child_handle = child_process_.Handle();
268*6777b538SAndroid Build Coastguard Worker   if (!TerminateMultiProcessTestChild(child_process_, /*exit_code=*/0,
269*6777b538SAndroid Build Coastguard Worker                                       /*wait=*/true)) {
270*6777b538SAndroid Build Coastguard Worker     return false;
271*6777b538SAndroid Build Coastguard Worker   }
272*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
273*6777b538SAndroid Build Coastguard Worker   // After the process exits, ProcessMetrics races to read /proc/<pid>/stat
274*6777b538SAndroid Build Coastguard Worker   // before it's deleted. Wait until it's definitely gone.
275*6777b538SAndroid Build Coastguard Worker   const auto stat_path = FilePath(FILE_PATH_LITERAL("/proc"))
276*6777b538SAndroid Build Coastguard Worker                              .AppendASCII(NumberToString(child_handle))
277*6777b538SAndroid Build Coastguard Worker                              .Append(FILE_PATH_LITERAL("stat"));
278*6777b538SAndroid Build Coastguard Worker 
279*6777b538SAndroid Build Coastguard Worker   while (PathExists(stat_path)) {
280*6777b538SAndroid Build Coastguard Worker     PlatformThread::Sleep(TestTimeouts::tiny_timeout());
281*6777b538SAndroid Build Coastguard Worker   }
282*6777b538SAndroid Build Coastguard Worker #endif
283*6777b538SAndroid Build Coastguard Worker   return true;
284*6777b538SAndroid Build Coastguard Worker }
285*6777b538SAndroid Build Coastguard Worker 
286*6777b538SAndroid Build Coastguard Worker // static
NotifyParent()287*6777b538SAndroid Build Coastguard Worker void TestChildLauncher::NotifyParent() {
288*6777b538SAndroid Build Coastguard Worker   // Do nothing.
289*6777b538SAndroid Build Coastguard Worker }
290*6777b538SAndroid Build Coastguard Worker 
291*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_MAC)
292*6777b538SAndroid Build Coastguard Worker 
293*6777b538SAndroid Build Coastguard Worker }  // namespace
294*6777b538SAndroid Build Coastguard Worker 
295*6777b538SAndroid Build Coastguard Worker // Tests for SystemMetrics.
296*6777b538SAndroid Build Coastguard Worker // Exists as a class so it can be a friend of SystemMetrics.
297*6777b538SAndroid Build Coastguard Worker class SystemMetricsTest : public testing::Test {
298*6777b538SAndroid Build Coastguard Worker  public:
299*6777b538SAndroid Build Coastguard Worker   SystemMetricsTest() = default;
300*6777b538SAndroid Build Coastguard Worker 
301*6777b538SAndroid Build Coastguard Worker   SystemMetricsTest(const SystemMetricsTest&) = delete;
302*6777b538SAndroid Build Coastguard Worker   SystemMetricsTest& operator=(const SystemMetricsTest&) = delete;
303*6777b538SAndroid Build Coastguard Worker };
304*6777b538SAndroid Build Coastguard Worker 
305*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
TEST_F(SystemMetricsTest,IsValidDiskName)306*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, IsValidDiskName) {
307*6777b538SAndroid Build Coastguard Worker   const char invalid_input1[] = "";
308*6777b538SAndroid Build Coastguard Worker   const char invalid_input2[] = "s";
309*6777b538SAndroid Build Coastguard Worker   const char invalid_input3[] = "sdz+";
310*6777b538SAndroid Build Coastguard Worker   const char invalid_input4[] = "hda0";
311*6777b538SAndroid Build Coastguard Worker   const char invalid_input5[] = "mmcbl";
312*6777b538SAndroid Build Coastguard Worker   const char invalid_input6[] = "mmcblka";
313*6777b538SAndroid Build Coastguard Worker   const char invalid_input7[] = "mmcblkb";
314*6777b538SAndroid Build Coastguard Worker   const char invalid_input8[] = "mmmblk0";
315*6777b538SAndroid Build Coastguard Worker 
316*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input1));
317*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input2));
318*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input3));
319*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input4));
320*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input5));
321*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input6));
322*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input7));
323*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(IsValidDiskName(invalid_input8));
324*6777b538SAndroid Build Coastguard Worker 
325*6777b538SAndroid Build Coastguard Worker   const char valid_input1[] = "sda";
326*6777b538SAndroid Build Coastguard Worker   const char valid_input2[] = "sdaaaa";
327*6777b538SAndroid Build Coastguard Worker   const char valid_input3[] = "hdz";
328*6777b538SAndroid Build Coastguard Worker   const char valid_input4[] = "mmcblk0";
329*6777b538SAndroid Build Coastguard Worker   const char valid_input5[] = "mmcblk999";
330*6777b538SAndroid Build Coastguard Worker 
331*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(IsValidDiskName(valid_input1));
332*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(IsValidDiskName(valid_input2));
333*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(IsValidDiskName(valid_input3));
334*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(IsValidDiskName(valid_input4));
335*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(IsValidDiskName(valid_input5));
336*6777b538SAndroid Build Coastguard Worker }
337*6777b538SAndroid Build Coastguard Worker 
TEST_F(SystemMetricsTest,ParseMeminfo)338*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, ParseMeminfo) {
339*6777b538SAndroid Build Coastguard Worker   SystemMemoryInfoKB meminfo;
340*6777b538SAndroid Build Coastguard Worker   const char invalid_input1[] = "abc";
341*6777b538SAndroid Build Coastguard Worker   const char invalid_input2[] = "MemTotal:";
342*6777b538SAndroid Build Coastguard Worker   // Partial file with no MemTotal
343*6777b538SAndroid Build Coastguard Worker   const char invalid_input3[] =
344*6777b538SAndroid Build Coastguard Worker       "MemFree:         3913968 kB\n"
345*6777b538SAndroid Build Coastguard Worker       "Buffers:         2348340 kB\n"
346*6777b538SAndroid Build Coastguard Worker       "Cached:         49071596 kB\n"
347*6777b538SAndroid Build Coastguard Worker       "SwapCached:           12 kB\n"
348*6777b538SAndroid Build Coastguard Worker       "Active:         36393900 kB\n"
349*6777b538SAndroid Build Coastguard Worker       "Inactive:       21221496 kB\n"
350*6777b538SAndroid Build Coastguard Worker       "Active(anon):    5674352 kB\n"
351*6777b538SAndroid Build Coastguard Worker       "Inactive(anon):   633992 kB\n";
352*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseProcMeminfo(invalid_input1, &meminfo));
353*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseProcMeminfo(invalid_input2, &meminfo));
354*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseProcMeminfo(invalid_input3, &meminfo));
355*6777b538SAndroid Build Coastguard Worker 
356*6777b538SAndroid Build Coastguard Worker   const char valid_input1[] =
357*6777b538SAndroid Build Coastguard Worker       "MemTotal:        3981504 kB\n"
358*6777b538SAndroid Build Coastguard Worker       "MemFree:          140764 kB\n"
359*6777b538SAndroid Build Coastguard Worker       "MemAvailable:     535413 kB\n"
360*6777b538SAndroid Build Coastguard Worker       "Buffers:          116480 kB\n"
361*6777b538SAndroid Build Coastguard Worker       "Cached:           406160 kB\n"
362*6777b538SAndroid Build Coastguard Worker       "SwapCached:        21304 kB\n"
363*6777b538SAndroid Build Coastguard Worker       "Active:          3152040 kB\n"
364*6777b538SAndroid Build Coastguard Worker       "Inactive:         472856 kB\n"
365*6777b538SAndroid Build Coastguard Worker       "Active(anon):    2972352 kB\n"
366*6777b538SAndroid Build Coastguard Worker       "Inactive(anon):   270108 kB\n"
367*6777b538SAndroid Build Coastguard Worker       "Active(file):     179688 kB\n"
368*6777b538SAndroid Build Coastguard Worker       "Inactive(file):   202748 kB\n"
369*6777b538SAndroid Build Coastguard Worker       "Unevictable:           0 kB\n"
370*6777b538SAndroid Build Coastguard Worker       "Mlocked:               0 kB\n"
371*6777b538SAndroid Build Coastguard Worker       "SwapTotal:       5832280 kB\n"
372*6777b538SAndroid Build Coastguard Worker       "SwapFree:        3672368 kB\n"
373*6777b538SAndroid Build Coastguard Worker       "Dirty:               184 kB\n"
374*6777b538SAndroid Build Coastguard Worker       "Writeback:             0 kB\n"
375*6777b538SAndroid Build Coastguard Worker       "AnonPages:       3101224 kB\n"
376*6777b538SAndroid Build Coastguard Worker       "Mapped:           142296 kB\n"
377*6777b538SAndroid Build Coastguard Worker       "Shmem:            140204 kB\n"
378*6777b538SAndroid Build Coastguard Worker       "Slab:              54212 kB\n"
379*6777b538SAndroid Build Coastguard Worker       "SReclaimable:      30936 kB\n"
380*6777b538SAndroid Build Coastguard Worker       "SUnreclaim:        23276 kB\n"
381*6777b538SAndroid Build Coastguard Worker       "KernelStack:        2464 kB\n"
382*6777b538SAndroid Build Coastguard Worker       "PageTables:        24812 kB\n"
383*6777b538SAndroid Build Coastguard Worker       "NFS_Unstable:          0 kB\n"
384*6777b538SAndroid Build Coastguard Worker       "Bounce:                0 kB\n"
385*6777b538SAndroid Build Coastguard Worker       "WritebackTmp:          0 kB\n"
386*6777b538SAndroid Build Coastguard Worker       "CommitLimit:     7823032 kB\n"
387*6777b538SAndroid Build Coastguard Worker       "Committed_AS:    7973536 kB\n"
388*6777b538SAndroid Build Coastguard Worker       "VmallocTotal:   34359738367 kB\n"
389*6777b538SAndroid Build Coastguard Worker       "VmallocUsed:      375940 kB\n"
390*6777b538SAndroid Build Coastguard Worker       "VmallocChunk:   34359361127 kB\n"
391*6777b538SAndroid Build Coastguard Worker       "DirectMap4k:       72448 kB\n"
392*6777b538SAndroid Build Coastguard Worker       "DirectMap2M:     4061184 kB\n";
393*6777b538SAndroid Build Coastguard Worker   // output from a much older kernel where the Active and Inactive aren't
394*6777b538SAndroid Build Coastguard Worker   // broken down into anon and file and Huge Pages are enabled
395*6777b538SAndroid Build Coastguard Worker   const char valid_input2[] =
396*6777b538SAndroid Build Coastguard Worker       "MemTotal:       255908 kB\n"
397*6777b538SAndroid Build Coastguard Worker       "MemFree:         69936 kB\n"
398*6777b538SAndroid Build Coastguard Worker       "Buffers:         15812 kB\n"
399*6777b538SAndroid Build Coastguard Worker       "Cached:         115124 kB\n"
400*6777b538SAndroid Build Coastguard Worker       "SwapCached:          0 kB\n"
401*6777b538SAndroid Build Coastguard Worker       "Active:          92700 kB\n"
402*6777b538SAndroid Build Coastguard Worker       "Inactive:        63792 kB\n"
403*6777b538SAndroid Build Coastguard Worker       "HighTotal:           0 kB\n"
404*6777b538SAndroid Build Coastguard Worker       "HighFree:            0 kB\n"
405*6777b538SAndroid Build Coastguard Worker       "LowTotal:       255908 kB\n"
406*6777b538SAndroid Build Coastguard Worker       "LowFree:         69936 kB\n"
407*6777b538SAndroid Build Coastguard Worker       "SwapTotal:      524280 kB\n"
408*6777b538SAndroid Build Coastguard Worker       "SwapFree:       524200 kB\n"
409*6777b538SAndroid Build Coastguard Worker       "Dirty:               4 kB\n"
410*6777b538SAndroid Build Coastguard Worker       "Writeback:           0 kB\n"
411*6777b538SAndroid Build Coastguard Worker       "Mapped:          42236 kB\n"
412*6777b538SAndroid Build Coastguard Worker       "Slab:            25912 kB\n"
413*6777b538SAndroid Build Coastguard Worker       "Committed_AS:   118680 kB\n"
414*6777b538SAndroid Build Coastguard Worker       "PageTables:       1236 kB\n"
415*6777b538SAndroid Build Coastguard Worker       "VmallocTotal:  3874808 kB\n"
416*6777b538SAndroid Build Coastguard Worker       "VmallocUsed:      1416 kB\n"
417*6777b538SAndroid Build Coastguard Worker       "VmallocChunk:  3872908 kB\n"
418*6777b538SAndroid Build Coastguard Worker       "HugePages_Total:     0\n"
419*6777b538SAndroid Build Coastguard Worker       "HugePages_Free:      0\n"
420*6777b538SAndroid Build Coastguard Worker       "Hugepagesize:     4096 kB\n";
421*6777b538SAndroid Build Coastguard Worker 
422*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseProcMeminfo(valid_input1, &meminfo));
423*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.total, 3981504);
424*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.free, 140764);
425*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.available, 535413);
426*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.buffers, 116480);
427*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.cached, 406160);
428*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.active_anon, 2972352);
429*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.active_file, 179688);
430*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.inactive_anon, 270108);
431*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.inactive_file, 202748);
432*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.swap_total, 5832280);
433*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.swap_free, 3672368);
434*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.dirty, 184);
435*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.reclaimable, 30936);
436*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_CHROMEOS)
437*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.shmem, 140204);
438*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.slab, 54212);
439*6777b538SAndroid Build Coastguard Worker #endif
440*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(355725u,
441*6777b538SAndroid Build Coastguard Worker             base::SysInfo::AmountOfAvailablePhysicalMemory(meminfo) / 1024);
442*6777b538SAndroid Build Coastguard Worker   // Simulate as if there is no MemAvailable.
443*6777b538SAndroid Build Coastguard Worker   meminfo.available = 0;
444*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(374448u,
445*6777b538SAndroid Build Coastguard Worker             base::SysInfo::AmountOfAvailablePhysicalMemory(meminfo) / 1024);
446*6777b538SAndroid Build Coastguard Worker   meminfo = {};
447*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseProcMeminfo(valid_input2, &meminfo));
448*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.total, 255908);
449*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.free, 69936);
450*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.available, 0);
451*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.buffers, 15812);
452*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.cached, 115124);
453*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.swap_total, 524280);
454*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.swap_free, 524200);
455*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.dirty, 4);
456*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(69936u,
457*6777b538SAndroid Build Coastguard Worker             base::SysInfo::AmountOfAvailablePhysicalMemory(meminfo) / 1024);
458*6777b538SAndroid Build Coastguard Worker 
459*6777b538SAndroid Build Coastguard Worker   // output from a system with a large page cache, to catch arithmetic errors
460*6777b538SAndroid Build Coastguard Worker   // that incorrectly assume free + buffers + cached <= total. (Copied from
461*6777b538SAndroid Build Coastguard Worker   // ash/components/arc/test/data/mem_profile/16G.)
462*6777b538SAndroid Build Coastguard Worker   const char large_cache_input[] =
463*6777b538SAndroid Build Coastguard Worker       "MemTotal:       18025572 kB\n"
464*6777b538SAndroid Build Coastguard Worker       "MemFree:        13150176 kB\n"
465*6777b538SAndroid Build Coastguard Worker       "MemAvailable:   15447672 kB\n"
466*6777b538SAndroid Build Coastguard Worker       "Buffers:         1524852 kB\n"
467*6777b538SAndroid Build Coastguard Worker       "Cached:         12645260 kB\n"
468*6777b538SAndroid Build Coastguard Worker       "SwapCached:            0 kB\n"
469*6777b538SAndroid Build Coastguard Worker       "Active:          2572904 kB\n"
470*6777b538SAndroid Build Coastguard Worker       "Inactive:        1064976 kB\n"
471*6777b538SAndroid Build Coastguard Worker       "Active(anon):    1047836 kB\n"
472*6777b538SAndroid Build Coastguard Worker       "Inactive(anon):    11736 kB\n"
473*6777b538SAndroid Build Coastguard Worker       "Active(file):    1525068 kB\n"
474*6777b538SAndroid Build Coastguard Worker       "Inactive(file):  1053240 kB\n"
475*6777b538SAndroid Build Coastguard Worker       "Unevictable:      611904 kB\n"
476*6777b538SAndroid Build Coastguard Worker       "Mlocked:           32884 kB\n"
477*6777b538SAndroid Build Coastguard Worker       "SwapTotal:      11756208 kB\n"
478*6777b538SAndroid Build Coastguard Worker       "SwapFree:       11756208 kB\n"
479*6777b538SAndroid Build Coastguard Worker       "Dirty:              4152 kB\n"
480*6777b538SAndroid Build Coastguard Worker       "Writeback:             0 kB\n"
481*6777b538SAndroid Build Coastguard Worker       "AnonPages:       1079660 kB\n"
482*6777b538SAndroid Build Coastguard Worker       "Mapped:           782152 kB\n"
483*6777b538SAndroid Build Coastguard Worker       "Shmem:            591820 kB\n"
484*6777b538SAndroid Build Coastguard Worker       "Slab:             366104 kB\n"
485*6777b538SAndroid Build Coastguard Worker       "SReclaimable:     254356 kB\n"
486*6777b538SAndroid Build Coastguard Worker       "SUnreclaim:       111748 kB\n"
487*6777b538SAndroid Build Coastguard Worker       "KernelStack:       22652 kB\n"
488*6777b538SAndroid Build Coastguard Worker       "PageTables:        41540 kB\n"
489*6777b538SAndroid Build Coastguard Worker       "NFS_Unstable:          0 kB\n"
490*6777b538SAndroid Build Coastguard Worker       "Bounce:                0 kB\n"
491*6777b538SAndroid Build Coastguard Worker       "WritebackTmp:          0 kB\n"
492*6777b538SAndroid Build Coastguard Worker       "CommitLimit:    15768992 kB\n"
493*6777b538SAndroid Build Coastguard Worker       "Committed_AS:   36120244 kB\n"
494*6777b538SAndroid Build Coastguard Worker       "VmallocTotal:   34359738367 kB\n"
495*6777b538SAndroid Build Coastguard Worker       "VmallocUsed:           0 kB\n"
496*6777b538SAndroid Build Coastguard Worker       "VmallocChunk:          0 kB\n"
497*6777b538SAndroid Build Coastguard Worker       "Percpu:             3328 kB\n"
498*6777b538SAndroid Build Coastguard Worker       "AnonHugePages:     32768 kB\n"
499*6777b538SAndroid Build Coastguard Worker       "ShmemHugePages:        0 kB\n"
500*6777b538SAndroid Build Coastguard Worker       "ShmemPmdMapped:        0 kB\n"
501*6777b538SAndroid Build Coastguard Worker       "DirectMap4k:      293036 kB\n"
502*6777b538SAndroid Build Coastguard Worker       "DirectMap2M:     6918144 kB\n"
503*6777b538SAndroid Build Coastguard Worker       "DirectMap1G:     2097152 kB\n";
504*6777b538SAndroid Build Coastguard Worker 
505*6777b538SAndroid Build Coastguard Worker   meminfo = {};
506*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseProcMeminfo(large_cache_input, &meminfo));
507*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.total, 18025572);
508*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.free, 13150176);
509*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.buffers, 1524852);
510*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(meminfo.cached, 12645260);
511*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(GetSystemCommitChargeFromMeminfo(meminfo), 0u);
512*6777b538SAndroid Build Coastguard Worker }
513*6777b538SAndroid Build Coastguard Worker 
TEST_F(SystemMetricsTest,ParseVmstat)514*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, ParseVmstat) {
515*6777b538SAndroid Build Coastguard Worker   VmStatInfo vmstat;
516*6777b538SAndroid Build Coastguard Worker   // Part of vmstat from a 4.19 kernel.
517*6777b538SAndroid Build Coastguard Worker   const char valid_input1[] =
518*6777b538SAndroid Build Coastguard Worker       "pgpgin 2358216\n"
519*6777b538SAndroid Build Coastguard Worker       "pgpgout 296072\n"
520*6777b538SAndroid Build Coastguard Worker       "pswpin 345219\n"
521*6777b538SAndroid Build Coastguard Worker       "pswpout 2605828\n"
522*6777b538SAndroid Build Coastguard Worker       "pgalloc_dma32 8380235\n"
523*6777b538SAndroid Build Coastguard Worker       "pgalloc_normal 3384525\n"
524*6777b538SAndroid Build Coastguard Worker       "pgalloc_movable 0\n"
525*6777b538SAndroid Build Coastguard Worker       "allocstall_dma32 0\n"
526*6777b538SAndroid Build Coastguard Worker       "allocstall_normal 2028\n"
527*6777b538SAndroid Build Coastguard Worker       "allocstall_movable 32559\n"
528*6777b538SAndroid Build Coastguard Worker       "pgskip_dma32 0\n"
529*6777b538SAndroid Build Coastguard Worker       "pgskip_normal 0\n"
530*6777b538SAndroid Build Coastguard Worker       "pgskip_movable 0\n"
531*6777b538SAndroid Build Coastguard Worker       "pgfree 11802722\n"
532*6777b538SAndroid Build Coastguard Worker       "pgactivate 894917\n"
533*6777b538SAndroid Build Coastguard Worker       "pgdeactivate 3255711\n"
534*6777b538SAndroid Build Coastguard Worker       "pglazyfree 48\n"
535*6777b538SAndroid Build Coastguard Worker       "pgfault 10043657\n"
536*6777b538SAndroid Build Coastguard Worker       "pgmajfault 358901\n"
537*6777b538SAndroid Build Coastguard Worker       "pgmajfault_s 2100\n"
538*6777b538SAndroid Build Coastguard Worker       "pgmajfault_a 343211\n"
539*6777b538SAndroid Build Coastguard Worker       "pgmajfault_f 13590\n"
540*6777b538SAndroid Build Coastguard Worker       "pglazyfreed 0\n"
541*6777b538SAndroid Build Coastguard Worker       "pgrefill 3429488\n"
542*6777b538SAndroid Build Coastguard Worker       "pgsteal_kswapd 1466893\n"
543*6777b538SAndroid Build Coastguard Worker       "pgsteal_direct 1771759\n"
544*6777b538SAndroid Build Coastguard Worker       "pgscan_kswapd 1907332\n"
545*6777b538SAndroid Build Coastguard Worker       "pgscan_direct 2118930\n"
546*6777b538SAndroid Build Coastguard Worker       "pgscan_direct_throttle 154\n"
547*6777b538SAndroid Build Coastguard Worker       "pginodesteal 3176\n"
548*6777b538SAndroid Build Coastguard Worker       "slabs_scanned 293804\n"
549*6777b538SAndroid Build Coastguard Worker       "kswapd_inodesteal 16753\n"
550*6777b538SAndroid Build Coastguard Worker       "kswapd_low_wmark_hit_quickly 10\n"
551*6777b538SAndroid Build Coastguard Worker       "kswapd_high_wmark_hit_quickly 423\n"
552*6777b538SAndroid Build Coastguard Worker       "pageoutrun 441\n"
553*6777b538SAndroid Build Coastguard Worker       "pgrotated 1636\n"
554*6777b538SAndroid Build Coastguard Worker       "drop_pagecache 0\n"
555*6777b538SAndroid Build Coastguard Worker       "drop_slab 0\n"
556*6777b538SAndroid Build Coastguard Worker       "oom_kill 18\n";
557*6777b538SAndroid Build Coastguard Worker   const char valid_input2[] =
558*6777b538SAndroid Build Coastguard Worker       "pgpgin 2606135\n"
559*6777b538SAndroid Build Coastguard Worker       "pgpgout 1359128\n"
560*6777b538SAndroid Build Coastguard Worker       "pswpin 899959\n"
561*6777b538SAndroid Build Coastguard Worker       "pswpout 19761244\n"
562*6777b538SAndroid Build Coastguard Worker       "pgalloc_dma 31\n"
563*6777b538SAndroid Build Coastguard Worker       "pgalloc_dma32 18139339\n"
564*6777b538SAndroid Build Coastguard Worker       "pgalloc_normal 44085950\n"
565*6777b538SAndroid Build Coastguard Worker       "pgalloc_movable 0\n"
566*6777b538SAndroid Build Coastguard Worker       "allocstall_dma 0\n"
567*6777b538SAndroid Build Coastguard Worker       "allocstall_dma32 0\n"
568*6777b538SAndroid Build Coastguard Worker       "allocstall_normal 18881\n"
569*6777b538SAndroid Build Coastguard Worker       "allocstall_movable 169527\n"
570*6777b538SAndroid Build Coastguard Worker       "pgskip_dma 0\n"
571*6777b538SAndroid Build Coastguard Worker       "pgskip_dma32 0\n"
572*6777b538SAndroid Build Coastguard Worker       "pgskip_normal 0\n"
573*6777b538SAndroid Build Coastguard Worker       "pgskip_movable 0\n"
574*6777b538SAndroid Build Coastguard Worker       "pgfree 63060999\n"
575*6777b538SAndroid Build Coastguard Worker       "pgactivate 1703494\n"
576*6777b538SAndroid Build Coastguard Worker       "pgdeactivate 20537803\n"
577*6777b538SAndroid Build Coastguard Worker       "pglazyfree 163\n"
578*6777b538SAndroid Build Coastguard Worker       "pgfault 45201169\n"
579*6777b538SAndroid Build Coastguard Worker       "pgmajfault 609626\n"
580*6777b538SAndroid Build Coastguard Worker       "pgmajfault_s 7488\n"
581*6777b538SAndroid Build Coastguard Worker       "pgmajfault_a 591793\n"
582*6777b538SAndroid Build Coastguard Worker       "pgmajfault_f 10345\n"
583*6777b538SAndroid Build Coastguard Worker       "pglazyfreed 0\n"
584*6777b538SAndroid Build Coastguard Worker       "pgrefill 20673453\n"
585*6777b538SAndroid Build Coastguard Worker       "pgsteal_kswapd 11802772\n"
586*6777b538SAndroid Build Coastguard Worker       "pgsteal_direct 8618160\n"
587*6777b538SAndroid Build Coastguard Worker       "pgscan_kswapd 12640517\n"
588*6777b538SAndroid Build Coastguard Worker       "pgscan_direct 9092230\n"
589*6777b538SAndroid Build Coastguard Worker       "pgscan_direct_throttle 638\n"
590*6777b538SAndroid Build Coastguard Worker       "pginodesteal 1716\n"
591*6777b538SAndroid Build Coastguard Worker       "slabs_scanned 2594642\n"
592*6777b538SAndroid Build Coastguard Worker       "kswapd_inodesteal 67358\n"
593*6777b538SAndroid Build Coastguard Worker       "kswapd_low_wmark_hit_quickly 52\n"
594*6777b538SAndroid Build Coastguard Worker       "kswapd_high_wmark_hit_quickly 11\n"
595*6777b538SAndroid Build Coastguard Worker       "pageoutrun 83\n"
596*6777b538SAndroid Build Coastguard Worker       "pgrotated 977\n"
597*6777b538SAndroid Build Coastguard Worker       "drop_pagecache 1\n"
598*6777b538SAndroid Build Coastguard Worker       "drop_slab 1\n"
599*6777b538SAndroid Build Coastguard Worker       "oom_kill 1\n"
600*6777b538SAndroid Build Coastguard Worker       "pgmigrate_success 3202\n"
601*6777b538SAndroid Build Coastguard Worker       "pgmigrate_fail 795\n";
602*6777b538SAndroid Build Coastguard Worker   const char valid_input3[] =
603*6777b538SAndroid Build Coastguard Worker       "pswpin 12\n"
604*6777b538SAndroid Build Coastguard Worker       "pswpout 901\n"
605*6777b538SAndroid Build Coastguard Worker       "pgmajfault 18881\n";
606*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseProcVmstat(valid_input1, &vmstat));
607*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(345219LU, vmstat.pswpin);
608*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(2605828LU, vmstat.pswpout);
609*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(358901LU, vmstat.pgmajfault);
610*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(18LU, vmstat.oom_kill);
611*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseProcVmstat(valid_input2, &vmstat));
612*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(899959LU, vmstat.pswpin);
613*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(19761244LU, vmstat.pswpout);
614*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(609626LU, vmstat.pgmajfault);
615*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(1LU, vmstat.oom_kill);
616*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseProcVmstat(valid_input3, &vmstat));
617*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(12LU, vmstat.pswpin);
618*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(901LU, vmstat.pswpout);
619*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(18881LU, vmstat.pgmajfault);
620*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(0LU, vmstat.oom_kill);
621*6777b538SAndroid Build Coastguard Worker 
622*6777b538SAndroid Build Coastguard Worker   const char missing_pgmajfault_input[] =
623*6777b538SAndroid Build Coastguard Worker       "pswpin 12\n"
624*6777b538SAndroid Build Coastguard Worker       "pswpout 901\n";
625*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseProcVmstat(missing_pgmajfault_input, &vmstat));
626*6777b538SAndroid Build Coastguard Worker   const char empty_input[] = "";
627*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseProcVmstat(empty_input, &vmstat));
628*6777b538SAndroid Build Coastguard Worker }
629*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
630*6777b538SAndroid Build Coastguard Worker         // BUILDFLAG(IS_ANDROID)
631*6777b538SAndroid Build Coastguard Worker 
632*6777b538SAndroid Build Coastguard Worker #if ENABLE_CPU_TESTS
633*6777b538SAndroid Build Coastguard Worker // Test that ProcessMetrics::GetPlatformIndependentCPUUsage() doesn't return
634*6777b538SAndroid Build Coastguard Worker // negative values when the number of threads running on the process decreases
635*6777b538SAndroid Build Coastguard Worker // between two successive calls to it.
TEST_F(SystemMetricsTest,TestNoNegativeCpuUsage)636*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, TestNoNegativeCpuUsage) {
637*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> metrics =
638*6777b538SAndroid Build Coastguard Worker       ProcessMetrics::CreateCurrentProcessMetrics();
639*6777b538SAndroid Build Coastguard Worker 
640*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(metrics->GetPlatformIndependentCPUUsage(), ValueIs(Ge(0.0)));
641*6777b538SAndroid Build Coastguard Worker 
642*6777b538SAndroid Build Coastguard Worker   Thread thread1("thread1");
643*6777b538SAndroid Build Coastguard Worker   Thread thread2("thread2");
644*6777b538SAndroid Build Coastguard Worker   Thread thread3("thread3");
645*6777b538SAndroid Build Coastguard Worker 
646*6777b538SAndroid Build Coastguard Worker   thread1.StartAndWaitForTesting();
647*6777b538SAndroid Build Coastguard Worker   thread2.StartAndWaitForTesting();
648*6777b538SAndroid Build Coastguard Worker   thread3.StartAndWaitForTesting();
649*6777b538SAndroid Build Coastguard Worker 
650*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(thread1.IsRunning());
651*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(thread2.IsRunning());
652*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(thread3.IsRunning());
653*6777b538SAndroid Build Coastguard Worker 
654*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> vec1;
655*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> vec2;
656*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> vec3;
657*6777b538SAndroid Build Coastguard Worker 
658*6777b538SAndroid Build Coastguard Worker   thread1.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec1));
659*6777b538SAndroid Build Coastguard Worker   thread2.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec2));
660*6777b538SAndroid Build Coastguard Worker   thread3.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec3));
661*6777b538SAndroid Build Coastguard Worker 
662*6777b538SAndroid Build Coastguard Worker   TimeDelta prev_cpu_usage = TestCumulativeCPU(metrics.get(), TimeDelta());
663*6777b538SAndroid Build Coastguard Worker 
664*6777b538SAndroid Build Coastguard Worker   thread1.Stop();
665*6777b538SAndroid Build Coastguard Worker   prev_cpu_usage = TestCumulativeCPU(metrics.get(), prev_cpu_usage);
666*6777b538SAndroid Build Coastguard Worker 
667*6777b538SAndroid Build Coastguard Worker   thread2.Stop();
668*6777b538SAndroid Build Coastguard Worker   prev_cpu_usage = TestCumulativeCPU(metrics.get(), prev_cpu_usage);
669*6777b538SAndroid Build Coastguard Worker 
670*6777b538SAndroid Build Coastguard Worker   thread3.Stop();
671*6777b538SAndroid Build Coastguard Worker   prev_cpu_usage = TestCumulativeCPU(metrics.get(), prev_cpu_usage);
672*6777b538SAndroid Build Coastguard Worker }
673*6777b538SAndroid Build Coastguard Worker 
674*6777b538SAndroid Build Coastguard Worker #if !BUILDFLAG(IS_APPLE)
675*6777b538SAndroid Build Coastguard Worker 
676*6777b538SAndroid Build Coastguard Worker // Subprocess to test the child CPU usage.
MULTIPROCESS_TEST_MAIN(CPUUsageChildMain)677*6777b538SAndroid Build Coastguard Worker MULTIPROCESS_TEST_MAIN(CPUUsageChildMain) {
678*6777b538SAndroid Build Coastguard Worker   TestChildLauncher::NotifyParent();
679*6777b538SAndroid Build Coastguard Worker   // Busy wait until terminated.
680*6777b538SAndroid Build Coastguard Worker   while (true) {
681*6777b538SAndroid Build Coastguard Worker     std::vector<std::string> vec;
682*6777b538SAndroid Build Coastguard Worker     BusyWork(&vec);
683*6777b538SAndroid Build Coastguard Worker   }
684*6777b538SAndroid Build Coastguard Worker }
685*6777b538SAndroid Build Coastguard Worker 
TEST_F(SystemMetricsTest,MeasureChildCpuUsage)686*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, MeasureChildCpuUsage) {
687*6777b538SAndroid Build Coastguard Worker   TestChildLauncher child_launcher;
688*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(child_launcher.SpawnChildProcess("CPUUsageChildMain"));
689*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> metrics =
690*6777b538SAndroid Build Coastguard Worker       child_launcher.CreateChildProcessMetrics();
691*6777b538SAndroid Build Coastguard Worker 
692*6777b538SAndroid Build Coastguard Worker   const TimeDelta cpu_usage1 = TestCumulativeCPU(metrics.get(), TimeDelta());
693*6777b538SAndroid Build Coastguard Worker   PlatformThread::Sleep(TestTimeouts::tiny_timeout());
694*6777b538SAndroid Build Coastguard Worker 
695*6777b538SAndroid Build Coastguard Worker   const TimeDelta cpu_usage2 = TestCumulativeCPU(metrics.get(), cpu_usage1);
696*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(cpu_usage2.is_positive());
697*6777b538SAndroid Build Coastguard Worker 
698*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(child_launcher.TerminateChildProcess());
699*6777b538SAndroid Build Coastguard Worker 
700*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN) || BUILDFLAG(IS_FUCHSIA)
701*6777b538SAndroid Build Coastguard Worker   // Windows and Fuchsia return final measurements of a process after it exits.
702*6777b538SAndroid Build Coastguard Worker   TestCumulativeCPU(metrics.get(), cpu_usage2);
703*6777b538SAndroid Build Coastguard Worker #else
704*6777b538SAndroid Build Coastguard Worker   // All other platforms return an error.
705*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(metrics->GetCumulativeCPUUsage(), ErrorIs(_));
706*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(metrics->GetPlatformIndependentCPUUsage(), ErrorIs(_));
707*6777b538SAndroid Build Coastguard Worker #endif
708*6777b538SAndroid Build Coastguard Worker }
709*6777b538SAndroid Build Coastguard Worker 
710*6777b538SAndroid Build Coastguard Worker #endif  // !BUILDFLAG(IS_APPLE)
711*6777b538SAndroid Build Coastguard Worker 
TEST_F(SystemMetricsTest,InvalidProcessCpuUsage)712*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, InvalidProcessCpuUsage) {
713*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_MAC)
714*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> metrics =
715*6777b538SAndroid Build Coastguard Worker       ProcessMetrics::CreateProcessMetrics(kNullProcessHandle, nullptr);
716*6777b538SAndroid Build Coastguard Worker #else
717*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> metrics =
718*6777b538SAndroid Build Coastguard Worker       ProcessMetrics::CreateProcessMetrics(kNullProcessHandle);
719*6777b538SAndroid Build Coastguard Worker #endif
720*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(metrics->GetCumulativeCPUUsage(), ErrorIs(_));
721*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(metrics->GetPlatformIndependentCPUUsage(), ErrorIs(_));
722*6777b538SAndroid Build Coastguard Worker }
723*6777b538SAndroid Build Coastguard Worker 
724*6777b538SAndroid Build Coastguard Worker #endif  // ENABLE_CPU_TESTS
725*6777b538SAndroid Build Coastguard Worker 
726*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_CHROMEOS)
TEST_F(SystemMetricsTest,ParseZramMmStat)727*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, ParseZramMmStat) {
728*6777b538SAndroid Build Coastguard Worker   SwapInfo swapinfo;
729*6777b538SAndroid Build Coastguard Worker 
730*6777b538SAndroid Build Coastguard Worker   const char invalid_input1[] = "aaa";
731*6777b538SAndroid Build Coastguard Worker   const char invalid_input2[] = "1 2 3 4 5 6";
732*6777b538SAndroid Build Coastguard Worker   const char invalid_input3[] = "a 2 3 4 5 6 7";
733*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseZramMmStat(invalid_input1, &swapinfo));
734*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseZramMmStat(invalid_input2, &swapinfo));
735*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseZramMmStat(invalid_input3, &swapinfo));
736*6777b538SAndroid Build Coastguard Worker 
737*6777b538SAndroid Build Coastguard Worker   const char valid_input1[] =
738*6777b538SAndroid Build Coastguard Worker       "17715200 5008166 566062  0 1225715712  127 183842";
739*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseZramMmStat(valid_input1, &swapinfo));
740*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(17715200ULL, swapinfo.orig_data_size);
741*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(5008166ULL, swapinfo.compr_data_size);
742*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(566062ULL, swapinfo.mem_used_total);
743*6777b538SAndroid Build Coastguard Worker }
744*6777b538SAndroid Build Coastguard Worker 
TEST_F(SystemMetricsTest,ParseZramStat)745*6777b538SAndroid Build Coastguard Worker TEST_F(SystemMetricsTest, ParseZramStat) {
746*6777b538SAndroid Build Coastguard Worker   SwapInfo swapinfo;
747*6777b538SAndroid Build Coastguard Worker 
748*6777b538SAndroid Build Coastguard Worker   const char invalid_input1[] = "aaa";
749*6777b538SAndroid Build Coastguard Worker   const char invalid_input2[] = "1 2 3 4 5 6 7 8 9 10";
750*6777b538SAndroid Build Coastguard Worker   const char invalid_input3[] = "a 2 3 4 5 6 7 8 9 10 11";
751*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseZramStat(invalid_input1, &swapinfo));
752*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseZramStat(invalid_input2, &swapinfo));
753*6777b538SAndroid Build Coastguard Worker   EXPECT_FALSE(ParseZramStat(invalid_input3, &swapinfo));
754*6777b538SAndroid Build Coastguard Worker 
755*6777b538SAndroid Build Coastguard Worker   const char valid_input1[] =
756*6777b538SAndroid Build Coastguard Worker       "299    0    2392    0    1    0    8    0    0    0    0";
757*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ParseZramStat(valid_input1, &swapinfo));
758*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(299ULL, swapinfo.num_reads);
759*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(1ULL, swapinfo.num_writes);
760*6777b538SAndroid Build Coastguard Worker }
761*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_CHROMEOS)
762*6777b538SAndroid Build Coastguard Worker 
763*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN) || BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_LINUX) || \
764*6777b538SAndroid Build Coastguard Worker     BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
TEST(SystemMetrics2Test,GetSystemMemoryInfo)765*6777b538SAndroid Build Coastguard Worker TEST(SystemMetrics2Test, GetSystemMemoryInfo) {
766*6777b538SAndroid Build Coastguard Worker   SystemMemoryInfoKB info;
767*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(GetSystemMemoryInfo(&info));
768*6777b538SAndroid Build Coastguard Worker 
769*6777b538SAndroid Build Coastguard Worker   // Ensure each field received a value.
770*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.total, 0);
771*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
772*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.avail_phys, 0);
773*6777b538SAndroid Build Coastguard Worker #else
774*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.free, 0);
775*6777b538SAndroid Build Coastguard Worker #endif
776*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
777*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.buffers, 0);
778*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.cached, 0);
779*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.active_anon + info.inactive_anon, 0);
780*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.active_file + info.inactive_file, 0);
781*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
782*6777b538SAndroid Build Coastguard Worker         // BUILDFLAG(IS_ANDROID)
783*6777b538SAndroid Build Coastguard Worker 
784*6777b538SAndroid Build Coastguard Worker   // All the values should be less than the total amount of memory.
785*6777b538SAndroid Build Coastguard Worker #if !BUILDFLAG(IS_WIN) && !BUILDFLAG(IS_IOS)
786*6777b538SAndroid Build Coastguard Worker   // TODO(crbug.com/711450): re-enable the following assertion on iOS.
787*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.free, info.total);
788*6777b538SAndroid Build Coastguard Worker #endif
789*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
790*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.buffers, info.total);
791*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.cached, info.total);
792*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.active_anon, info.total);
793*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.inactive_anon, info.total);
794*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.active_file, info.total);
795*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.inactive_file, info.total);
796*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
797*6777b538SAndroid Build Coastguard Worker         // BUILDFLAG(IS_ANDROID)
798*6777b538SAndroid Build Coastguard Worker 
799*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_APPLE)
800*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.file_backed, 0);
801*6777b538SAndroid Build Coastguard Worker #endif
802*6777b538SAndroid Build Coastguard Worker 
803*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_CHROMEOS)
804*6777b538SAndroid Build Coastguard Worker   // Chrome OS exposes shmem.
805*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(info.shmem, 0);
806*6777b538SAndroid Build Coastguard Worker   EXPECT_LT(info.shmem, info.total);
807*6777b538SAndroid Build Coastguard Worker #endif
808*6777b538SAndroid Build Coastguard Worker }
809*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_WIN) || BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_LINUX) ||
810*6777b538SAndroid Build Coastguard Worker         // BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
811*6777b538SAndroid Build Coastguard Worker 
812*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
TEST(ProcessMetricsTest,ParseProcStatCPU)813*6777b538SAndroid Build Coastguard Worker TEST(ProcessMetricsTest, ParseProcStatCPU) {
814*6777b538SAndroid Build Coastguard Worker   // /proc/self/stat for a process running "top".
815*6777b538SAndroid Build Coastguard Worker   const char kTopStat[] =
816*6777b538SAndroid Build Coastguard Worker       "960 (top) S 16230 960 16230 34818 960 "
817*6777b538SAndroid Build Coastguard Worker       "4202496 471 0 0 0 "
818*6777b538SAndroid Build Coastguard Worker       "12 16 0 0 "  // <- These are the goods.
819*6777b538SAndroid Build Coastguard Worker       "20 0 1 0 121946157 15077376 314 18446744073709551615 4194304 "
820*6777b538SAndroid Build Coastguard Worker       "4246868 140733983044336 18446744073709551615 140244213071219 "
821*6777b538SAndroid Build Coastguard Worker       "0 0 0 138047495 0 0 0 17 1 0 0 0 0 0";
822*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(12 + 16, ParseProcStatCPU(kTopStat));
823*6777b538SAndroid Build Coastguard Worker 
824*6777b538SAndroid Build Coastguard Worker   // cat /proc/self/stat on a random other machine I have.
825*6777b538SAndroid Build Coastguard Worker   const char kSelfStat[] =
826*6777b538SAndroid Build Coastguard Worker       "5364 (cat) R 5354 5364 5354 34819 5364 "
827*6777b538SAndroid Build Coastguard Worker       "0 142 0 0 0 "
828*6777b538SAndroid Build Coastguard Worker       "0 0 0 0 "  // <- No CPU, apparently.
829*6777b538SAndroid Build Coastguard Worker       "16 0 1 0 1676099790 2957312 114 4294967295 134512640 134528148 "
830*6777b538SAndroid Build Coastguard Worker       "3221224832 3221224344 3086339742 0 0 0 0 0 0 0 17 0 0 0";
831*6777b538SAndroid Build Coastguard Worker 
832*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(0, ParseProcStatCPU(kSelfStat));
833*6777b538SAndroid Build Coastguard Worker 
834*6777b538SAndroid Build Coastguard Worker   // Some weird long-running process with a weird name that I created for the
835*6777b538SAndroid Build Coastguard Worker   // purposes of this test.
836*6777b538SAndroid Build Coastguard Worker   const char kWeirdNameStat[] =
837*6777b538SAndroid Build Coastguard Worker       "26115 (Hello) You ()))  ) R 24614 26115 24614"
838*6777b538SAndroid Build Coastguard Worker       " 34839 26115 4218880 227 0 0 0 "
839*6777b538SAndroid Build Coastguard Worker       "5186 11 0 0 "
840*6777b538SAndroid Build Coastguard Worker       "20 0 1 0 36933953 4296704 90 18446744073709551615 4194304 4196116 "
841*6777b538SAndroid Build Coastguard Worker       "140735857761568 140735857761160 4195644 0 0 0 0 0 0 0 17 14 0 0 0 0 0 "
842*6777b538SAndroid Build Coastguard Worker       "6295056 6295616 16519168 140735857770710 140735857770737 "
843*6777b538SAndroid Build Coastguard Worker       "140735857770737 140735857774557 0";
844*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(5186 + 11, ParseProcStatCPU(kWeirdNameStat));
845*6777b538SAndroid Build Coastguard Worker }
846*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
847*6777b538SAndroid Build Coastguard Worker         // BUILDFLAG(IS_ANDROID)
848*6777b538SAndroid Build Coastguard Worker 
849*6777b538SAndroid Build Coastguard Worker // Disable on Android because base_unittests runs inside a Dalvik VM that
850*6777b538SAndroid Build Coastguard Worker // starts and stop threads (crbug.com/175563).
851*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
852*6777b538SAndroid Build Coastguard Worker // http://crbug.com/396455
TEST(ProcessMetricsTest,DISABLED_GetNumberOfThreads)853*6777b538SAndroid Build Coastguard Worker TEST(ProcessMetricsTest, DISABLED_GetNumberOfThreads) {
854*6777b538SAndroid Build Coastguard Worker   const ProcessHandle current = GetCurrentProcessHandle();
855*6777b538SAndroid Build Coastguard Worker   const int64_t initial_threads = GetNumberOfThreads(current);
856*6777b538SAndroid Build Coastguard Worker   ASSERT_GT(initial_threads, 0);
857*6777b538SAndroid Build Coastguard Worker   const int kNumAdditionalThreads = 10;
858*6777b538SAndroid Build Coastguard Worker   {
859*6777b538SAndroid Build Coastguard Worker     std::unique_ptr<Thread> my_threads[kNumAdditionalThreads];
860*6777b538SAndroid Build Coastguard Worker     for (int i = 0; i < kNumAdditionalThreads; ++i) {
861*6777b538SAndroid Build Coastguard Worker       my_threads[i] = std::make_unique<Thread>("GetNumberOfThreadsTest");
862*6777b538SAndroid Build Coastguard Worker       my_threads[i]->Start();
863*6777b538SAndroid Build Coastguard Worker       ASSERT_EQ(GetNumberOfThreads(current), initial_threads + 1 + i);
864*6777b538SAndroid Build Coastguard Worker     }
865*6777b538SAndroid Build Coastguard Worker   }
866*6777b538SAndroid Build Coastguard Worker   // The Thread destructor will stop them.
867*6777b538SAndroid Build Coastguard Worker   ASSERT_EQ(initial_threads, GetNumberOfThreads(current));
868*6777b538SAndroid Build Coastguard Worker }
869*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
870*6777b538SAndroid Build Coastguard Worker 
871*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_MAC)
872*6777b538SAndroid Build Coastguard Worker namespace {
873*6777b538SAndroid Build Coastguard Worker 
874*6777b538SAndroid Build Coastguard Worker // Keep these in sync so the GetChildOpenFdCount test can refer to correct test
875*6777b538SAndroid Build Coastguard Worker // main.
876*6777b538SAndroid Build Coastguard Worker #define ChildMain ChildFdCount
877*6777b538SAndroid Build Coastguard Worker #define ChildMainString "ChildFdCount"
878*6777b538SAndroid Build Coastguard Worker 
879*6777b538SAndroid Build Coastguard Worker // Command line flag name and file name used for synchronization.
880*6777b538SAndroid Build Coastguard Worker const char kTempDirFlag[] = "temp-dir";
881*6777b538SAndroid Build Coastguard Worker 
882*6777b538SAndroid Build Coastguard Worker const char kSignalReady[] = "ready";
883*6777b538SAndroid Build Coastguard Worker const char kSignalReadyAck[] = "ready-ack";
884*6777b538SAndroid Build Coastguard Worker const char kSignalOpened[] = "opened";
885*6777b538SAndroid Build Coastguard Worker const char kSignalOpenedAck[] = "opened-ack";
886*6777b538SAndroid Build Coastguard Worker const char kSignalClosed[] = "closed";
887*6777b538SAndroid Build Coastguard Worker 
888*6777b538SAndroid Build Coastguard Worker const int kChildNumFilesToOpen = 100;
889*6777b538SAndroid Build Coastguard Worker 
SignalEvent(const FilePath & signal_dir,const char * signal_file)890*6777b538SAndroid Build Coastguard Worker bool SignalEvent(const FilePath& signal_dir, const char* signal_file) {
891*6777b538SAndroid Build Coastguard Worker   File file(signal_dir.AppendASCII(signal_file),
892*6777b538SAndroid Build Coastguard Worker             File::FLAG_CREATE | File::FLAG_WRITE);
893*6777b538SAndroid Build Coastguard Worker   return file.IsValid();
894*6777b538SAndroid Build Coastguard Worker }
895*6777b538SAndroid Build Coastguard Worker 
896*6777b538SAndroid Build Coastguard Worker // Check whether an event was signaled.
CheckEvent(const FilePath & signal_dir,const char * signal_file)897*6777b538SAndroid Build Coastguard Worker bool CheckEvent(const FilePath& signal_dir, const char* signal_file) {
898*6777b538SAndroid Build Coastguard Worker   File file(signal_dir.AppendASCII(signal_file),
899*6777b538SAndroid Build Coastguard Worker             File::FLAG_OPEN | File::FLAG_READ);
900*6777b538SAndroid Build Coastguard Worker   return file.IsValid();
901*6777b538SAndroid Build Coastguard Worker }
902*6777b538SAndroid Build Coastguard Worker 
903*6777b538SAndroid Build Coastguard Worker // Busy-wait for an event to be signaled.
WaitForEvent(const FilePath & signal_dir,const char * signal_file)904*6777b538SAndroid Build Coastguard Worker void WaitForEvent(const FilePath& signal_dir, const char* signal_file) {
905*6777b538SAndroid Build Coastguard Worker   while (!CheckEvent(signal_dir, signal_file)) {
906*6777b538SAndroid Build Coastguard Worker     PlatformThread::Sleep(Milliseconds(10));
907*6777b538SAndroid Build Coastguard Worker   }
908*6777b538SAndroid Build Coastguard Worker }
909*6777b538SAndroid Build Coastguard Worker 
910*6777b538SAndroid Build Coastguard Worker // Subprocess to test the number of open file descriptors.
MULTIPROCESS_TEST_MAIN(ChildMain)911*6777b538SAndroid Build Coastguard Worker MULTIPROCESS_TEST_MAIN(ChildMain) {
912*6777b538SAndroid Build Coastguard Worker   TestChildLauncher::NotifyParent();
913*6777b538SAndroid Build Coastguard Worker 
914*6777b538SAndroid Build Coastguard Worker   CommandLine* command_line = CommandLine::ForCurrentProcess();
915*6777b538SAndroid Build Coastguard Worker   const FilePath temp_path = command_line->GetSwitchValuePath(kTempDirFlag);
916*6777b538SAndroid Build Coastguard Worker   CHECK(DirectoryExists(temp_path));
917*6777b538SAndroid Build Coastguard Worker 
918*6777b538SAndroid Build Coastguard Worker   CHECK(SignalEvent(temp_path, kSignalReady));
919*6777b538SAndroid Build Coastguard Worker   WaitForEvent(temp_path, kSignalReadyAck);
920*6777b538SAndroid Build Coastguard Worker 
921*6777b538SAndroid Build Coastguard Worker   std::vector<File> files;
922*6777b538SAndroid Build Coastguard Worker   for (int i = 0; i < kChildNumFilesToOpen; ++i) {
923*6777b538SAndroid Build Coastguard Worker     files.emplace_back(temp_path.AppendASCII(StringPrintf("file.%d", i)),
924*6777b538SAndroid Build Coastguard Worker                        File::FLAG_CREATE | File::FLAG_WRITE);
925*6777b538SAndroid Build Coastguard Worker   }
926*6777b538SAndroid Build Coastguard Worker 
927*6777b538SAndroid Build Coastguard Worker   CHECK(SignalEvent(temp_path, kSignalOpened));
928*6777b538SAndroid Build Coastguard Worker   WaitForEvent(temp_path, kSignalOpenedAck);
929*6777b538SAndroid Build Coastguard Worker 
930*6777b538SAndroid Build Coastguard Worker   files.clear();
931*6777b538SAndroid Build Coastguard Worker 
932*6777b538SAndroid Build Coastguard Worker   CHECK(SignalEvent(temp_path, kSignalClosed));
933*6777b538SAndroid Build Coastguard Worker 
934*6777b538SAndroid Build Coastguard Worker   // Wait to be terminated.
935*6777b538SAndroid Build Coastguard Worker   while (true) {
936*6777b538SAndroid Build Coastguard Worker     PlatformThread::Sleep(Seconds(1));
937*6777b538SAndroid Build Coastguard Worker   }
938*6777b538SAndroid Build Coastguard Worker }
939*6777b538SAndroid Build Coastguard Worker 
940*6777b538SAndroid Build Coastguard Worker }  // namespace
941*6777b538SAndroid Build Coastguard Worker 
TEST(ProcessMetricsTest,GetChildOpenFdCount)942*6777b538SAndroid Build Coastguard Worker TEST(ProcessMetricsTest, GetChildOpenFdCount) {
943*6777b538SAndroid Build Coastguard Worker   ScopedTempDir temp_dir;
944*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
945*6777b538SAndroid Build Coastguard Worker   const FilePath temp_path = temp_dir.GetPath();
946*6777b538SAndroid Build Coastguard Worker 
947*6777b538SAndroid Build Coastguard Worker   TestChildLauncher child_launcher;
948*6777b538SAndroid Build Coastguard Worker   child_launcher.command_line().AppendSwitchPath(kTempDirFlag, temp_path);
949*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(child_launcher.SpawnChildProcess(ChildMainString));
950*6777b538SAndroid Build Coastguard Worker 
951*6777b538SAndroid Build Coastguard Worker   WaitForEvent(temp_path, kSignalReady);
952*6777b538SAndroid Build Coastguard Worker 
953*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> metrics =
954*6777b538SAndroid Build Coastguard Worker       child_launcher.CreateChildProcessMetrics();
955*6777b538SAndroid Build Coastguard Worker 
956*6777b538SAndroid Build Coastguard Worker   const int fd_count = metrics->GetOpenFdCount();
957*6777b538SAndroid Build Coastguard Worker   EXPECT_GE(fd_count, 0);
958*6777b538SAndroid Build Coastguard Worker 
959*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(SignalEvent(temp_path, kSignalReadyAck));
960*6777b538SAndroid Build Coastguard Worker   WaitForEvent(temp_path, kSignalOpened);
961*6777b538SAndroid Build Coastguard Worker 
962*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(fd_count + kChildNumFilesToOpen, metrics->GetOpenFdCount());
963*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(SignalEvent(temp_path, kSignalOpenedAck));
964*6777b538SAndroid Build Coastguard Worker 
965*6777b538SAndroid Build Coastguard Worker   WaitForEvent(temp_path, kSignalClosed);
966*6777b538SAndroid Build Coastguard Worker 
967*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(fd_count, metrics->GetOpenFdCount());
968*6777b538SAndroid Build Coastguard Worker 
969*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(child_launcher.TerminateChildProcess());
970*6777b538SAndroid Build Coastguard Worker }
971*6777b538SAndroid Build Coastguard Worker 
TEST(ProcessMetricsTest,GetOpenFdCount)972*6777b538SAndroid Build Coastguard Worker TEST(ProcessMetricsTest, GetOpenFdCount) {
973*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> metrics =
974*6777b538SAndroid Build Coastguard Worker       ProcessMetrics::CreateCurrentProcessMetrics();
975*6777b538SAndroid Build Coastguard Worker 
976*6777b538SAndroid Build Coastguard Worker   ScopedTempDir temp_dir;
977*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
978*6777b538SAndroid Build Coastguard Worker 
979*6777b538SAndroid Build Coastguard Worker   int fd_count = metrics->GetOpenFdCount();
980*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(fd_count, 0);
981*6777b538SAndroid Build Coastguard Worker   File file(temp_dir.GetPath().AppendASCII("file"),
982*6777b538SAndroid Build Coastguard Worker             File::FLAG_CREATE | File::FLAG_WRITE);
983*6777b538SAndroid Build Coastguard Worker   int new_fd_count = metrics->GetOpenFdCount();
984*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(new_fd_count, 0);
985*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(new_fd_count, fd_count + 1);
986*6777b538SAndroid Build Coastguard Worker }
987*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_MAC)
988*6777b538SAndroid Build Coastguard Worker 
989*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
990*6777b538SAndroid Build Coastguard Worker 
TEST(ProcessMetricsTestLinux,GetPageFaultCounts)991*6777b538SAndroid Build Coastguard Worker TEST(ProcessMetricsTestLinux, GetPageFaultCounts) {
992*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> process_metrics =
993*6777b538SAndroid Build Coastguard Worker       ProcessMetrics::CreateCurrentProcessMetrics();
994*6777b538SAndroid Build Coastguard Worker 
995*6777b538SAndroid Build Coastguard Worker   PageFaultCounts counts;
996*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(process_metrics->GetPageFaultCounts(&counts));
997*6777b538SAndroid Build Coastguard Worker   ASSERT_GT(counts.minor, 0);
998*6777b538SAndroid Build Coastguard Worker   ASSERT_GE(counts.major, 0);
999*6777b538SAndroid Build Coastguard Worker 
1000*6777b538SAndroid Build Coastguard Worker   // Allocate and touch memory. Touching it is required to make sure that the
1001*6777b538SAndroid Build Coastguard Worker   // page fault count goes up, as memory is typically mapped lazily.
1002*6777b538SAndroid Build Coastguard Worker   {
1003*6777b538SAndroid Build Coastguard Worker     const size_t kMappedSize = 4 << 20;  // 4 MiB.
1004*6777b538SAndroid Build Coastguard Worker 
1005*6777b538SAndroid Build Coastguard Worker     WritableSharedMemoryRegion region =
1006*6777b538SAndroid Build Coastguard Worker         WritableSharedMemoryRegion::Create(kMappedSize);
1007*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(region.IsValid());
1008*6777b538SAndroid Build Coastguard Worker 
1009*6777b538SAndroid Build Coastguard Worker     WritableSharedMemoryMapping mapping = region.Map();
1010*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(mapping.IsValid());
1011*6777b538SAndroid Build Coastguard Worker 
1012*6777b538SAndroid Build Coastguard Worker     memset(mapping.memory(), 42, kMappedSize);
1013*6777b538SAndroid Build Coastguard Worker   }
1014*6777b538SAndroid Build Coastguard Worker 
1015*6777b538SAndroid Build Coastguard Worker   PageFaultCounts counts_after;
1016*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(process_metrics->GetPageFaultCounts(&counts_after));
1017*6777b538SAndroid Build Coastguard Worker   ASSERT_GT(counts_after.minor, counts.minor);
1018*6777b538SAndroid Build Coastguard Worker   ASSERT_GE(counts_after.major, counts.major);
1019*6777b538SAndroid Build Coastguard Worker }
1020*6777b538SAndroid Build Coastguard Worker 
TEST(ProcessMetricsTestLinux,GetCumulativeCPUUsagePerThread)1021*6777b538SAndroid Build Coastguard Worker TEST(ProcessMetricsTestLinux, GetCumulativeCPUUsagePerThread) {
1022*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<ProcessMetrics> metrics =
1023*6777b538SAndroid Build Coastguard Worker       ProcessMetrics::CreateCurrentProcessMetrics();
1024*6777b538SAndroid Build Coastguard Worker 
1025*6777b538SAndroid Build Coastguard Worker   Thread thread1("thread1");
1026*6777b538SAndroid Build Coastguard Worker   thread1.StartAndWaitForTesting();
1027*6777b538SAndroid Build Coastguard Worker   ASSERT_TRUE(thread1.IsRunning());
1028*6777b538SAndroid Build Coastguard Worker 
1029*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> vec1;
1030*6777b538SAndroid Build Coastguard Worker   thread1.task_runner()->PostTask(FROM_HERE, BindOnce(&BusyWork, &vec1));
1031*6777b538SAndroid Build Coastguard Worker 
1032*6777b538SAndroid Build Coastguard Worker   ProcessMetrics::CPUUsagePerThread prev_thread_times;
1033*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(metrics->GetCumulativeCPUUsagePerThread(prev_thread_times));
1034*6777b538SAndroid Build Coastguard Worker 
1035*6777b538SAndroid Build Coastguard Worker   // Should have at least the test runner thread and the thread spawned above.
1036*6777b538SAndroid Build Coastguard Worker   EXPECT_GE(prev_thread_times.size(), 2u);
1037*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ranges::any_of(
1038*6777b538SAndroid Build Coastguard Worker       prev_thread_times,
1039*6777b538SAndroid Build Coastguard Worker       [&thread1](const std::pair<PlatformThreadId, base::TimeDelta>& entry) {
1040*6777b538SAndroid Build Coastguard Worker         return entry.first == thread1.GetThreadId();
1041*6777b538SAndroid Build Coastguard Worker       }));
1042*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ranges::any_of(
1043*6777b538SAndroid Build Coastguard Worker       prev_thread_times,
1044*6777b538SAndroid Build Coastguard Worker       [](const std::pair<PlatformThreadId, base::TimeDelta>& entry) {
1045*6777b538SAndroid Build Coastguard Worker         return entry.first == base::PlatformThread::CurrentId();
1046*6777b538SAndroid Build Coastguard Worker       }));
1047*6777b538SAndroid Build Coastguard Worker 
1048*6777b538SAndroid Build Coastguard Worker   for (const auto& entry : prev_thread_times) {
1049*6777b538SAndroid Build Coastguard Worker     EXPECT_GE(entry.second, base::TimeDelta());
1050*6777b538SAndroid Build Coastguard Worker   }
1051*6777b538SAndroid Build Coastguard Worker 
1052*6777b538SAndroid Build Coastguard Worker   thread1.Stop();
1053*6777b538SAndroid Build Coastguard Worker 
1054*6777b538SAndroid Build Coastguard Worker   ProcessMetrics::CPUUsagePerThread current_thread_times;
1055*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(metrics->GetCumulativeCPUUsagePerThread(current_thread_times));
1056*6777b538SAndroid Build Coastguard Worker 
1057*6777b538SAndroid Build Coastguard Worker   // The stopped thread may still be reported until the kernel cleans it up.
1058*6777b538SAndroid Build Coastguard Worker   EXPECT_GE(prev_thread_times.size(), 1u);
1059*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(ranges::any_of(
1060*6777b538SAndroid Build Coastguard Worker       current_thread_times,
1061*6777b538SAndroid Build Coastguard Worker       [](const std::pair<PlatformThreadId, base::TimeDelta>& entry) {
1062*6777b538SAndroid Build Coastguard Worker         return entry.first == base::PlatformThread::CurrentId();
1063*6777b538SAndroid Build Coastguard Worker       }));
1064*6777b538SAndroid Build Coastguard Worker 
1065*6777b538SAndroid Build Coastguard Worker   // Reported times should not decrease.
1066*6777b538SAndroid Build Coastguard Worker   for (const auto& entry : current_thread_times) {
1067*6777b538SAndroid Build Coastguard Worker     auto prev_it = ranges::find_if(
1068*6777b538SAndroid Build Coastguard Worker         prev_thread_times,
1069*6777b538SAndroid Build Coastguard Worker         [&entry](
1070*6777b538SAndroid Build Coastguard Worker             const std::pair<PlatformThreadId, base::TimeDelta>& prev_entry) {
1071*6777b538SAndroid Build Coastguard Worker           return entry.first == prev_entry.first;
1072*6777b538SAndroid Build Coastguard Worker         });
1073*6777b538SAndroid Build Coastguard Worker 
1074*6777b538SAndroid Build Coastguard Worker     if (prev_it != prev_thread_times.end()) {
1075*6777b538SAndroid Build Coastguard Worker       EXPECT_GE(entry.second, prev_it->second);
1076*6777b538SAndroid Build Coastguard Worker     }
1077*6777b538SAndroid Build Coastguard Worker   }
1078*6777b538SAndroid Build Coastguard Worker }
1079*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_LINUX) ||
1080*6777b538SAndroid Build Coastguard Worker         // BUILDFLAG(IS_CHROMEOS)
1081*6777b538SAndroid Build Coastguard Worker 
1082*6777b538SAndroid Build Coastguard Worker }  // namespace base::debug
1083