blob: 9a4c2442fee3993ed68279fd08e179ace4e673fe [file] [log] [blame]
// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifdef UNSAFE_BUFFERS_BUILD
// TODO(crbug.com/40285824): Remove this and convert code to safer constructs.
#pragma allow_unsafe_buffers
#endif
#include "components/system_cpu/cpu_probe_linux.h"
#include <memory>
#include <optional>
#include <string>
#include "base/files/file.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/files/scoped_temp_dir.h"
#include "base/test/task_environment.h"
#include "base/test/test_timeouts.h"
#include "base/threading/platform_thread.h"
#include "components/system_cpu/cpu_probe.h"
#include "components/system_cpu/cpu_sample.h"
#include "components/system_cpu/pressure_test_support.h"
#include "components/system_cpu/procfs_stat_cpu_parser.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace system_cpu {
class CpuProbeLinuxTest : public testing::Test {
public:
CpuProbeLinuxTest() = default;
~CpuProbeLinuxTest() override = default;
void SetUp() override {
ASSERT_TRUE(temp_dir_.CreateUniqueTempDir());
fake_stat_path_ = temp_dir_.GetPath().AppendASCII("stat");
// Create the /proc/stat file before creating the parser, in case the parser
// implementation keeps an open handle to the file indefinitely.
stat_file_ = base::File(fake_stat_path_, base::File::FLAG_CREATE_ALWAYS |
base::File::FLAG_WRITE);
probe_ =
std::make_unique<FakePlatformCpuProbe<CpuProbeLinux>>(fake_stat_path_);
}
[[nodiscard]] bool WriteFakeStat(const std::string& contents) {
if (!stat_file_.SetLength(0)) {
return false;
}
if (contents.size() > 0) {
if (!stat_file_.Write(0, contents.data(), contents.size())) {
return false;
}
}
return true;
}
protected:
base::test::TaskEnvironment task_environment_;
base::ScopedTempDir temp_dir_;
base::FilePath fake_stat_path_;
base::File stat_file_;
std::unique_ptr<FakePlatformCpuProbe<CpuProbeLinux>> probe_;
};
TEST_F(CpuProbeLinuxTest, ProductionDataNoCrash) {
// Overwrite the fake probe with one that reads the production stat path.
probe_ = std::make_unique<FakePlatformCpuProbe<CpuProbeLinux>>(
base::FilePath(ProcfsStatCpuParser::kProcfsStatPath));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
base::PlatformThread::Sleep(TestTimeouts::tiny_timeout());
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_GE(sample->cpu_utilization, 0.0);
EXPECT_LE(sample->cpu_utilization, 1.0);
}
TEST_F(CpuProbeLinuxTest, OneCoreFullInfo) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 0 0 0 0 0 0 0 0 0 0
intr 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217
ctxt 23456789012
btime 1234567890
processes 6789012
procs_running 700
procs_blocked 600
softirq 900 901 902 903 904 905 906 907 908 909 910
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 100 0 0 300 0 0 0 0 0 0
intr 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217
ctxt 23456789012
btime 1234567890
processes 6789012
procs_running 700
procs_blocked 600
softirq 900 901 902 903 904 905 906 907 908 909 910
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.25);
}
TEST_F(CpuProbeLinuxTest, RepeatedSamples) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 0 0 0 0 0 0 0 0 0 0
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 100 0 0 300 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.25);
// Second sample should have the difference since the first sample.
// delta: user = 200-100, sys = 100-0, idle = 500-300
// (user+sys) / (idle+user+sys) = 200/400
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 200 100 0 500 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample2 = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample2.has_value());
EXPECT_EQ(sample2->cpu_utilization, 0.5);
}
TEST_F(CpuProbeLinuxTest, TwoCoresFullInfo) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 0 0 0 0 0 0 0 0 0 0
cpu1 0 0 0 0 0 0 0 0 0 0
intr 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217
ctxt 23456789012
btime 1234567890
processes 6789012
procs_running 700
procs_blocked 600
softirq 900 901 902 903 904 905 906 907 908 909 910
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// cpu0: user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
// cpu1: user = 100, sys = 100, idle = 200
// (user+sys) / (idle+user+sys) = 200/400
// total (user+sys) / total (idle+user+sys) = 300/800
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 100 0 0 300 0 0 0 0 0 0
cpu1 100 100 0 200 0 0 0 0 0 0
intr 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217
ctxt 23456789012
btime 1234567890
processes 6789012
procs_running 700
procs_blocked 600
softirq 900 901 902 903 904 905 906 907 908 909 910
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.375);
}
TEST_F(CpuProbeLinuxTest, TwoCoresDifferentBaselines) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 30 0 0 0 0 0 0
cpu1 40 50 0 60 0 0 0 0 0 0
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// cpu0 delta: user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
// cpu1 delta: user = 100, sys = 100, idle = 200
// (user+sys) / (idle+user+sys) = 200/400
// total (user+sys) / total (idle+user+sys) = 300/800
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 110 20 0 330 0 0 0 0 0 0
cpu1 140 150 0 260 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.375);
}
TEST_F(CpuProbeLinuxTest, TwoCoresSecondCoreMissingStat) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 30 0 0 0 0 0 0
cpu1 40 50 0 60 0 0 0 0 0 0
intr 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217
ctxt 23456789012
btime 1234567890
processes 6789012
procs_running 700
procs_blocked 600
softirq 900 901 902 903 904 905 906 907 908 909 910
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// cpu0 delta: user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 110 20 0 330 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.25);
// Second core reappears. Delta should be based on original baseline, not 0:
// cpu0 delta: user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
// cpu1 delta: user = 100, sys = 100, idle = 200
// (user+sys) / (idle+user+sys) = 200/400
// total (user+sys) / total (idle+user+sys) = 300/800
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 210 20 0 630 0 0 0 0 0 0
cpu1 140 150 0 260 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample2 = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample2.has_value());
EXPECT_EQ(sample2->cpu_utilization, 0.375);
}
TEST_F(CpuProbeLinuxTest, TwoCoresSecondCoreAddedStat) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 30 0 0 0 0 0 0
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// cpu0 delta: user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
// cpu1 delta (vs 0): user = 100, sys = 100, idle = 200
// (user+sys) / (idle+user+sys) = 200/400
// But second core isn't included since it wasn't in the baseline, so:
// total (user+sys) / total (idle+user+sys) = 100/400
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 110 20 0 330 0 0 0 0 0 0
cpu1 100 100 0 200 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.25);
// cpu0 delta: user = 100, sys = 100, idle = 200
// (user+sys) / (idle+user+sys) = 200/400
// cpu1 delta: user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
// Now the second core is included, with the last measurement as a baseline:
// total (user+sys) / total (idle+user+sys) = 300/800
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 210 120 0 530 0 0 0 0 0 0
cpu1 200 100 0 500 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample2 = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample2.has_value());
EXPECT_EQ(sample2->cpu_utilization, 0.375);
}
TEST_F(CpuProbeLinuxTest, ParseErrorInBaseline) {
ASSERT_TRUE(WriteFakeStat(R"(
parse error
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// Update should be ignored as baseline is unreliable.
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 100 0 0 300 0 0 0 0 0 0
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt);
// Next update can use the last update as a baseline.
// cpu0 delta: user = 100, sys = 0, idle = 300
// (user+sys) / (idle+user+sys) = 100/400
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 200 0 0 600 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.25);
}
TEST_F(CpuProbeLinuxTest, ParseErrorInSample) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 30 0 0 0 0 0 0
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
ASSERT_TRUE(WriteFakeStat(R"(
bad stat file
)"));
EXPECT_FALSE(probe_->UpdateAndWaitForSample().has_value());
// Second sample should have the difference since the baseline, since the
// first sample was ignored.
// delta: user = 100, sys = 100, idle = 200
// (user+sys) / (idle+user+sys) = 200/400
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 110 120 0 230 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample2 = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample2.has_value());
EXPECT_EQ(sample2->cpu_utilization, 0.5);
}
TEST_F(CpuProbeLinuxTest, ZeroAndNegativeDeltas) {
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 30 0 0 0 0 0 0
cpu1 40 50 0 60 0 0 0 0 0 0
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt)
<< "No baseline on first Update()";
// cpu0: active delta 0, idle delta 100 (0% usage)
// cpu1: active delta 100, idle delta 0 (100% usage)
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 130 0 0 0 0 0 0
cpu1 140 50 0 60 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample.has_value());
EXPECT_EQ(sample->cpu_utilization, 0.5);
// cpu0: active delta 0, idle delta 100 (0% usage)
// cpu1: active delta 0, idle delta 0 (no time passed - ignore)
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 230 0 0 0 0 0 0
cpu1 140 50 0 60 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample2 = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample2.has_value());
EXPECT_EQ(sample2->cpu_utilization, 0.0);
// Both cores: active delta 0, idle delta 0 (no time passed - ignore)
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 10 20 0 230 0 0 0 0 0 0
cpu1 140 50 0 60 0 0 0 0 0 0
)"));
EXPECT_EQ(probe_->UpdateAndWaitForSample(), std::nullopt);
// cpu0: user time increases, sys decreases - ignore only the decreasing
// values, include the others.
// cpu0 delta: user = 100, sys = -10 (clamped to 0), idle = 300
// (user+sys) / (idle+user+sys) = 100/400
// cpu1: all times decrease - ignore this core completely.
ASSERT_TRUE(WriteFakeStat(R"(
cpu 0 0 0 0 0 0 0 0 0 0
cpu0 110 10 0 530 0 0 0 0 0 0
cpu1 130 40 0 50 0 0 0 0 0 0
)"));
std::optional<CpuSample> sample3 = probe_->UpdateAndWaitForSample();
ASSERT_TRUE(sample3.has_value());
EXPECT_EQ(sample3->cpu_utilization, 0.25);
}
} // namespace system_cpu