summaryrefslogtreecommitdiffhomepage
path: root/test/syscalls/linux/cgroup.cc
diff options
context:
space:
mode:
Diffstat (limited to 'test/syscalls/linux/cgroup.cc')
-rw-r--r--test/syscalls/linux/cgroup.cc421
1 files changed, 0 insertions, 421 deletions
diff --git a/test/syscalls/linux/cgroup.cc b/test/syscalls/linux/cgroup.cc
deleted file mode 100644
index a1006a978..000000000
--- a/test/syscalls/linux/cgroup.cc
+++ /dev/null
@@ -1,421 +0,0 @@
-// Copyright 2021 The gVisor Authors.
-//
-// Licensed under the Apache License, Version 2.0 (the "License");
-// you may not use this file except in compliance with the License.
-// You may obtain a copy of the License at
-//
-// http://www.apache.org/licenses/LICENSE-2.0
-//
-// Unless required by applicable law or agreed to in writing, software
-// distributed under the License is distributed on an "AS IS" BASIS,
-// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-// See the License for the specific language governing permissions and
-// limitations under the License.
-
-// All tests in this file rely on being about to mount and unmount cgroupfs,
-// which isn't expected to work, or be safe on a general linux system.
-
-#include <sys/mount.h>
-#include <unistd.h>
-
-#include "gtest/gtest.h"
-#include "absl/container/flat_hash_map.h"
-#include "absl/container/flat_hash_set.h"
-#include "absl/strings/str_split.h"
-#include "test/util/capability_util.h"
-#include "test/util/cgroup_util.h"
-#include "test/util/temp_path.h"
-#include "test/util/test_util.h"
-
-namespace gvisor {
-namespace testing {
-namespace {
-
-using ::testing::_;
-using ::testing::Ge;
-using ::testing::Gt;
-
-std::vector<std::string> known_controllers = {"cpu", "cpuset", "cpuacct",
- "memory"};
-
-bool CgroupsAvailable() {
- return IsRunningOnGvisor() && !IsRunningWithVFS1() &&
- TEST_CHECK_NO_ERRNO_AND_VALUE(HaveCapability(CAP_SYS_ADMIN));
-}
-
-TEST(Cgroup, MountSucceeds) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs(""));
- EXPECT_NO_ERRNO(c.ContainsCallingProcess());
-}
-
-TEST(Cgroup, SeparateMounts) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
-
- for (const auto& ctl : known_controllers) {
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs(ctl));
- EXPECT_NO_ERRNO(c.ContainsCallingProcess());
- }
-}
-
-TEST(Cgroup, AllControllersImplicit) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs(""));
-
- absl::flat_hash_map<std::string, CgroupsEntry> cgroups_entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- for (const auto& ctl : known_controllers) {
- EXPECT_TRUE(cgroups_entries.contains(ctl))
- << absl::StreamFormat("ctl=%s", ctl);
- }
- EXPECT_EQ(cgroups_entries.size(), known_controllers.size());
-}
-
-TEST(Cgroup, AllControllersExplicit) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("all"));
-
- absl::flat_hash_map<std::string, CgroupsEntry> cgroups_entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- for (const auto& ctl : known_controllers) {
- EXPECT_TRUE(cgroups_entries.contains(ctl))
- << absl::StreamFormat("ctl=%s", ctl);
- }
- EXPECT_EQ(cgroups_entries.size(), known_controllers.size());
-}
-
-TEST(Cgroup, ProcsAndTasks) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs(""));
- absl::flat_hash_set<pid_t> pids = ASSERT_NO_ERRNO_AND_VALUE(c.Procs());
- absl::flat_hash_set<pid_t> tids = ASSERT_NO_ERRNO_AND_VALUE(c.Tasks());
-
- EXPECT_GE(tids.size(), pids.size()) << "Found more processes than threads";
-
- // Pids should be a strict subset of tids.
- for (auto it = pids.begin(); it != pids.end(); ++it) {
- EXPECT_TRUE(tids.contains(*it))
- << absl::StreamFormat("Have pid %d, but no such tid", *it);
- }
-}
-
-TEST(Cgroup, ControllersMustBeInUniqueHierarchy) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- // Hierarchy #1: all controllers.
- Cgroup all = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs(""));
- // Hierarchy #2: memory.
- //
- // This should conflict since memory is already in hierarchy #1, and the two
- // hierarchies have different sets of controllers, so this mount can't be a
- // view into hierarchy #1.
- EXPECT_THAT(m.MountCgroupfs("memory"), PosixErrorIs(EBUSY, _))
- << "Memory controller mounted on two hierarchies";
- EXPECT_THAT(m.MountCgroupfs("cpu"), PosixErrorIs(EBUSY, _))
- << "CPU controller mounted on two hierarchies";
-}
-
-TEST(Cgroup, UnmountFreesControllers) {
- SKIP_IF(!CgroupsAvailable());
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup all = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs(""));
- // All controllers are now attached to all's hierarchy. Attempting new mount
- // with any individual controller should fail.
- EXPECT_THAT(m.MountCgroupfs("memory"), PosixErrorIs(EBUSY, _))
- << "Memory controller mounted on two hierarchies";
-
- // Unmount the "all" hierarchy. This should enable any controller to be
- // mounted on a new hierarchy again.
- ASSERT_NO_ERRNO(m.Unmount(all));
- EXPECT_NO_ERRNO(m.MountCgroupfs("memory"));
- EXPECT_NO_ERRNO(m.MountCgroupfs("cpu"));
-}
-
-TEST(Cgroup, OnlyContainsControllerSpecificFiles) {
- SKIP_IF(!CgroupsAvailable());
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup mem = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("memory"));
- EXPECT_THAT(Exists(mem.Relpath("memory.usage_in_bytes")),
- IsPosixErrorOkAndHolds(true));
- // CPU files shouldn't exist in memory cgroups.
- EXPECT_THAT(Exists(mem.Relpath("cpu.cfs_period_us")),
- IsPosixErrorOkAndHolds(false));
- EXPECT_THAT(Exists(mem.Relpath("cpu.cfs_quota_us")),
- IsPosixErrorOkAndHolds(false));
- EXPECT_THAT(Exists(mem.Relpath("cpu.shares")), IsPosixErrorOkAndHolds(false));
-
- Cgroup cpu = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpu"));
- EXPECT_THAT(Exists(cpu.Relpath("cpu.cfs_period_us")),
- IsPosixErrorOkAndHolds(true));
- EXPECT_THAT(Exists(cpu.Relpath("cpu.cfs_quota_us")),
- IsPosixErrorOkAndHolds(true));
- EXPECT_THAT(Exists(cpu.Relpath("cpu.shares")), IsPosixErrorOkAndHolds(true));
- // Memory files shouldn't exist in cpu cgroups.
- EXPECT_THAT(Exists(cpu.Relpath("memory.usage_in_bytes")),
- IsPosixErrorOkAndHolds(false));
-}
-
-TEST(Cgroup, InvalidController) {
- SKIP_IF(!CgroupsAvailable());
-
- TempPath mountpoint = ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir());
- std::string mopts = "this-controller-is-invalid";
- EXPECT_THAT(
- mount("none", mountpoint.path().c_str(), "cgroup", 0, mopts.c_str()),
- SyscallFailsWithErrno(EINVAL));
-}
-
-TEST(Cgroup, MoptAllMustBeExclusive) {
- SKIP_IF(!CgroupsAvailable());
-
- TempPath mountpoint = ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir());
- std::string mopts = "all,cpu";
- EXPECT_THAT(
- mount("none", mountpoint.path().c_str(), "cgroup", 0, mopts.c_str()),
- SyscallFailsWithErrno(EINVAL));
-}
-
-TEST(MemoryCgroup, MemoryUsageInBytes) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("memory"));
- EXPECT_THAT(c.ReadIntegerControlFile("memory.usage_in_bytes"),
- IsPosixErrorOkAndHolds(Gt(0)));
-}
-
-TEST(CPUCgroup, ControlFilesHaveDefaultValues) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpu"));
- EXPECT_THAT(c.ReadIntegerControlFile("cpu.cfs_quota_us"),
- IsPosixErrorOkAndHolds(-1));
- EXPECT_THAT(c.ReadIntegerControlFile("cpu.cfs_period_us"),
- IsPosixErrorOkAndHolds(100000));
- EXPECT_THAT(c.ReadIntegerControlFile("cpu.shares"),
- IsPosixErrorOkAndHolds(1024));
-}
-
-TEST(CPUAcctCgroup, CPUAcctUsage) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpuacct"));
-
- const int64_t usage =
- ASSERT_NO_ERRNO_AND_VALUE(c.ReadIntegerControlFile("cpuacct.usage"));
- const int64_t usage_user =
- ASSERT_NO_ERRNO_AND_VALUE(c.ReadIntegerControlFile("cpuacct.usage_user"));
- const int64_t usage_sys =
- ASSERT_NO_ERRNO_AND_VALUE(c.ReadIntegerControlFile("cpuacct.usage_sys"));
-
- EXPECT_GE(usage, 0);
- EXPECT_GE(usage_user, 0);
- EXPECT_GE(usage_sys, 0);
-
- EXPECT_GE(usage_user + usage_sys, usage);
-}
-
-TEST(CPUAcctCgroup, CPUAcctStat) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpuacct"));
-
- std::string stat =
- ASSERT_NO_ERRNO_AND_VALUE(c.ReadControlFile("cpuacct.stat"));
-
- // We're expecting the contents of "cpuacct.stat" to look similar to this:
- //
- // user 377986
- // system 220662
-
- std::vector<absl::string_view> lines =
- absl::StrSplit(stat, '\n', absl::SkipEmpty());
- ASSERT_EQ(lines.size(), 2);
-
- std::vector<absl::string_view> user_tokens =
- StrSplit(lines[0], absl::ByChar(' '));
- EXPECT_EQ(user_tokens[0], "user");
- EXPECT_THAT(Atoi<int64_t>(user_tokens[1]), IsPosixErrorOkAndHolds(Ge(0)));
-
- std::vector<absl::string_view> sys_tokens =
- StrSplit(lines[1], absl::ByChar(' '));
- EXPECT_EQ(sys_tokens[0], "system");
- EXPECT_THAT(Atoi<int64_t>(sys_tokens[1]), IsPosixErrorOkAndHolds(Ge(0)));
-}
-
-TEST(ProcCgroups, Empty) {
- SKIP_IF(!CgroupsAvailable());
-
- absl::flat_hash_map<std::string, CgroupsEntry> entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- // No cgroups mounted yet, we should have no entries.
- EXPECT_TRUE(entries.empty());
-}
-
-TEST(ProcCgroups, ProcCgroupsEntries) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
-
- Cgroup mem = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("memory"));
- absl::flat_hash_map<std::string, CgroupsEntry> entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- EXPECT_EQ(entries.size(), 1);
- ASSERT_TRUE(entries.contains("memory"));
- CgroupsEntry mem_e = entries["memory"];
- EXPECT_EQ(mem_e.subsys_name, "memory");
- EXPECT_GE(mem_e.hierarchy, 1);
- // Expect a single root cgroup.
- EXPECT_EQ(mem_e.num_cgroups, 1);
- // Cgroups are currently always enabled when mounted.
- EXPECT_TRUE(mem_e.enabled);
-
- // Add a second cgroup, and check for new entry.
-
- Cgroup cpu = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpu"));
- entries = ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- EXPECT_EQ(entries.size(), 2);
- EXPECT_TRUE(entries.contains("memory")); // Still have memory entry.
- ASSERT_TRUE(entries.contains("cpu"));
- CgroupsEntry cpu_e = entries["cpu"];
- EXPECT_EQ(cpu_e.subsys_name, "cpu");
- EXPECT_GE(cpu_e.hierarchy, 1);
- EXPECT_EQ(cpu_e.num_cgroups, 1);
- EXPECT_TRUE(cpu_e.enabled);
-
- // Separate hierarchies, since controllers were mounted separately.
- EXPECT_NE(mem_e.hierarchy, cpu_e.hierarchy);
-}
-
-TEST(ProcCgroups, UnmountRemovesEntries) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup cg = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpu,memory"));
- absl::flat_hash_map<std::string, CgroupsEntry> entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- EXPECT_EQ(entries.size(), 2);
-
- ASSERT_NO_ERRNO(m.Unmount(cg));
-
- entries = ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- EXPECT_TRUE(entries.empty());
-}
-
-TEST(ProcPIDCgroup, Empty) {
- SKIP_IF(!CgroupsAvailable());
-
- absl::flat_hash_map<std::string, PIDCgroupEntry> entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcPIDCgroupEntries(getpid()));
- EXPECT_TRUE(entries.empty());
-}
-
-TEST(ProcPIDCgroup, Entries) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("memory"));
-
- absl::flat_hash_map<std::string, PIDCgroupEntry> entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcPIDCgroupEntries(getpid()));
- EXPECT_EQ(entries.size(), 1);
- PIDCgroupEntry mem_e = entries["memory"];
- EXPECT_GE(mem_e.hierarchy, 1);
- EXPECT_EQ(mem_e.controllers, "memory");
- EXPECT_EQ(mem_e.path, "/");
-
- Cgroup c1 = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpu"));
- entries = ASSERT_NO_ERRNO_AND_VALUE(ProcPIDCgroupEntries(getpid()));
- EXPECT_EQ(entries.size(), 2);
- EXPECT_TRUE(entries.contains("memory")); // Still have memory entry.
- PIDCgroupEntry cpu_e = entries["cpu"];
- EXPECT_GE(cpu_e.hierarchy, 1);
- EXPECT_EQ(cpu_e.controllers, "cpu");
- EXPECT_EQ(cpu_e.path, "/");
-
- // Separate hierarchies, since controllers were mounted separately.
- EXPECT_NE(mem_e.hierarchy, cpu_e.hierarchy);
-}
-
-TEST(ProcPIDCgroup, UnmountRemovesEntries) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup all = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs(""));
-
- absl::flat_hash_map<std::string, PIDCgroupEntry> entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcPIDCgroupEntries(getpid()));
- EXPECT_GT(entries.size(), 0);
-
- ASSERT_NO_ERRNO(m.Unmount(all));
-
- entries = ASSERT_NO_ERRNO_AND_VALUE(ProcPIDCgroupEntries(getpid()));
- EXPECT_TRUE(entries.empty());
-}
-
-TEST(ProcCgroup, PIDCgroupMatchesCgroups) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("memory"));
- Cgroup c1 = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("cpu"));
-
- absl::flat_hash_map<std::string, CgroupsEntry> cgroups_entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
- absl::flat_hash_map<std::string, PIDCgroupEntry> pid_entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcPIDCgroupEntries(getpid()));
-
- CgroupsEntry cgroup_mem = cgroups_entries["memory"];
- PIDCgroupEntry pid_mem = pid_entries["memory"];
-
- EXPECT_EQ(cgroup_mem.hierarchy, pid_mem.hierarchy);
-
- CgroupsEntry cgroup_cpu = cgroups_entries["cpu"];
- PIDCgroupEntry pid_cpu = pid_entries["cpu"];
-
- EXPECT_EQ(cgroup_cpu.hierarchy, pid_cpu.hierarchy);
- EXPECT_NE(cgroup_mem.hierarchy, cgroup_cpu.hierarchy);
- EXPECT_NE(pid_mem.hierarchy, pid_cpu.hierarchy);
-}
-
-TEST(ProcCgroup, MultiControllerHierarchy) {
- SKIP_IF(!CgroupsAvailable());
-
- Mounter m(ASSERT_NO_ERRNO_AND_VALUE(TempPath::CreateDir()));
- Cgroup c = ASSERT_NO_ERRNO_AND_VALUE(m.MountCgroupfs("memory,cpu"));
-
- absl::flat_hash_map<std::string, CgroupsEntry> cgroups_entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcCgroupsEntries());
-
- CgroupsEntry mem_e = cgroups_entries["memory"];
- CgroupsEntry cpu_e = cgroups_entries["cpu"];
-
- // Both controllers should have the same hierarchy ID.
- EXPECT_EQ(mem_e.hierarchy, cpu_e.hierarchy);
-
- absl::flat_hash_map<std::string, PIDCgroupEntry> pid_entries =
- ASSERT_NO_ERRNO_AND_VALUE(ProcPIDCgroupEntries(getpid()));
-
- // Expecting an entry listing both controllers, that matches the previous
- // hierarchy ID. Note that the controllers are listed in alphabetical order.
- PIDCgroupEntry pid_e = pid_entries["cpu,memory"];
- EXPECT_EQ(pid_e.hierarchy, mem_e.hierarchy);
-}
-
-} // namespace
-} // namespace testing
-} // namespace gvisor