blob: 1f62f8da0a7e80e0da0542947235e6bd2da46467 [file] [log] [blame]
// Copyright 2024 The Bazel Authors. All rights reserved.
//
// 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.
package com.google.devtools.build.lib.sandbox;
import static com.google.common.base.Preconditions.checkArgument;
import static java.nio.charset.StandardCharsets.UTF_8;
import com.google.common.base.Joiner;
import com.google.common.base.Splitter;
import com.google.common.io.Files;
import java.io.File;
import java.io.IOException;
import java.util.List;
import javax.annotation.Nullable;
/** Represents a v2 cgroup. */
public class CgroupsInfoV2 extends CgroupsInfo {
public CgroupsInfoV2(Type type, @Nullable File cgroupDir) {
super(type, Version.V2, cgroupDir);
}
@Override
public CgroupsInfo createBlazeSpawnsCgroup(String procSelfCgroupPath) {
checkArgument(
type == Type.ROOT, "Should only be creating the Blaze spawns cgroup from the root cgroup.");
File blazeProcessCgroupDir;
File blazeSpawnsDir;
try {
blazeProcessCgroupDir = getBlazeProcessCgroupDir(cgroupDir, procSelfCgroupPath);
// In cgroups v2, we need to step back from the leaf node to make a further hierarchy.
blazeSpawnsDir =
new File(
blazeProcessCgroupDir.getParentFile(),
"blaze_" + ProcessHandle.current().pid() + "_spawns.slice");
blazeSpawnsDir.mkdirs();
blazeSpawnsDir.deleteOnExit();
setSubtreeControllers(blazeSpawnsDir);
} catch (Exception e) {
return new InvalidCgroupsInfo(Type.BLAZE_SPAWNS, getVersion(), e);
}
return new CgroupsInfoV2(Type.BLAZE_SPAWNS, blazeSpawnsDir);
}
@Override
public CgroupsInfo createIndividualSpawnCgroup(String dirName, int memoryLimitMb) {
checkArgument(
type == Type.BLAZE_SPAWNS,
"Should only be creating the individual spawn's cgroup from the Blaze spawns cgroup.");
if (!canWrite()) {
return new InvalidCgroupsInfo(
Type.SPAWN,
getVersion(),
String.format("Cgroup %s is invalid, unable to create spawn's cgroup here.", cgroupDir));
}
File spawnCgroupDir = new File(cgroupDir, dirName + ".scope");
spawnCgroupDir.mkdirs();
spawnCgroupDir.deleteOnExit();
try {
if (memoryLimitMb > 0) {
// In cgroups v2, we need to propagate the controllers into new subdirs.
Files.asCharSink(new File(spawnCgroupDir, "memory.oom.group"), UTF_8).write("1\n");
Files.asCharSink(new File(spawnCgroupDir, "memory.max"), UTF_8)
.write(Long.toString(memoryLimitMb * 1024L * 1024L));
}
} catch (Exception e) {
return new InvalidCgroupsInfo(Type.SPAWN, getVersion(), e);
}
return new CgroupsInfoV2(Type.SPAWN, spawnCgroupDir);
}
@Override
public int getMemoryUsageInKb() {
return getMemoryUsageInKbFromFile("memory.current");
}
/**
* Returns the path to the cgroup containing the Blaze process.
*
* <p>In v2, there is only one entry, and it looks something like this:
*
* <pre>
* 0::/user.slice/user-123.slice/session-1.scope
* </pre>
*
* @param mountPoint the directory where the cgroup hierarchy is mounted.
* @param procSelfCgroupPath path for the /proc/self/cgroup file.
* @throws IOException if there are errors reading the given procs cgroup file.
*/
private static File getBlazeProcessCgroupDir(File mountPoint, String procSelfCgroupPath)
throws IOException {
List<String> contents = Files.readLines(new File(procSelfCgroupPath), UTF_8);
if (contents.isEmpty()) {
throw new IllegalStateException(
"Found no memory cgroup entries in '" + procSelfCgroupPath + "'");
}
List<String> parts = Splitter.on(":").limit(3).splitToList(contents.get(0));
return new File(mountPoint, parts.get(2));
}
/**
* Sets the subtree controllers we need. This also checks that the controllers are available.
*
* @param blazeDir A directory in the cgroups hierarchy.
* @throws IOException If reading or writing the {@code cgroup.controllers} or {@code
* cgroup.subtree_control} file fails.
* @throws IllegalStateException if the {@code memory} and {code pids} controllers are either not
* available or cannot be set for subtrees.
*/
private static void setSubtreeControllers(File blazeDir) throws IOException {
var controllers =
Joiner.on(' ').join(Files.readLines(new File(blazeDir, "cgroup.controllers"), UTF_8));
if (!(controllers.contains("memory") && controllers.contains("pids"))) {
throw new IllegalStateException(
String.format(
"Required controllers 'memory' and 'pids' not found in %s/cgroup.controllers",
blazeDir));
}
var subtreeControllers =
Joiner.on(' ').join(Files.readLines(new File(blazeDir, "cgroup.subtree_control"), UTF_8));
if (!subtreeControllers.contains("memory") || !subtreeControllers.contains("pids")) {
Files.asCharSink(new File(blazeDir, "cgroup.subtree_control"), UTF_8)
.write("+memory +pids\n");
}
}
}