Skip to content

Commit 1b04ffe

Browse files
author
Jonathan Dowland
committedDec 16, 2022
8254001: [Metrics] Enhance parsing of cgroup interface files for version detection
Reviewed-by: sgehwolf Backport-of: a50725db2ab621e1a17cb5505f78e4bc73972a27
1 parent b2619cf commit 1b04ffe

File tree

5 files changed

+453
-274
lines changed

5 files changed

+453
-274
lines changed
 

‎jdk/src/linux/classes/jdk/internal/platform/CgroupInfo.java

+55-5
Original file line numberDiff line numberDiff line change
@@ -26,36 +26,86 @@
2626
package jdk.internal.platform;
2727

2828
/**
29-
* Data structure to hold info from /proc/self/cgroup
29+
* Data structure to hold info from /proc/self/cgroup,
30+
* /proc/cgroups and /proc/self/mountinfo
3031
*
3132
* man 7 cgroups
3233
*
3334
* @see CgroupSubsystemFactory
3435
*/
35-
class CgroupInfo {
36+
public class CgroupInfo {
3637

3738
private final String name;
3839
private final int hierarchyId;
3940
private final boolean enabled;
41+
private String mountPoint;
42+
private String mountRoot;
43+
private String cgroupPath;
4044

4145
private CgroupInfo(String name, int hierarchyId, boolean enabled) {
4246
this.name = name;
4347
this.hierarchyId = hierarchyId;
4448
this.enabled = enabled;
4549
}
4650

47-
String getName() {
51+
public String getName() {
4852
return name;
4953
}
5054

51-
int getHierarchyId() {
55+
public int getHierarchyId() {
5256
return hierarchyId;
5357
}
5458

55-
boolean isEnabled() {
59+
public boolean isEnabled() {
5660
return enabled;
5761
}
5862

63+
public String getMountPoint() {
64+
return mountPoint;
65+
}
66+
67+
public void setMountPoint(String mountPoint) {
68+
this.mountPoint = mountPoint;
69+
}
70+
71+
public String getMountRoot() {
72+
return mountRoot;
73+
}
74+
75+
public void setMountRoot(String mountRoot) {
76+
this.mountRoot = mountRoot;
77+
}
78+
79+
public String getCgroupPath() {
80+
return cgroupPath;
81+
}
82+
83+
public void setCgroupPath(String cgroupPath) {
84+
this.cgroupPath = cgroupPath;
85+
}
86+
87+
/*
88+
* Creates a CgroupInfo instance from a line in /proc/cgroups.
89+
* Comment token (hash) is handled by the caller.
90+
*
91+
* Example (annotated):
92+
*
93+
* #subsys_name hierarchy num_cgroups enabled
94+
* cpuset 10 1 1 (a)
95+
* cpu 7 8 1 (b)
96+
* [...]
97+
*
98+
* Line (a) would yield:
99+
* info = new CgroupInfo("cpuset", 10, true);
100+
* return info;
101+
* Line (b) results in:
102+
* info = new CgroupInfo("cpu", 7, true);
103+
* return info;
104+
*
105+
*
106+
* See CgroupSubsystemFactory.determineType()
107+
*
108+
*/
59109
static CgroupInfo fromCgroupsLine(String line) {
60110
String[] tokens = line.split("\\s+");
61111
if (tokens.length != 4) {

‎jdk/src/linux/classes/jdk/internal/platform/CgroupSubsystemFactory.java

+215-38
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,15 @@
2626
package jdk.internal.platform;
2727

2828
import java.io.IOException;
29+
import java.io.UncheckedIOException;
30+
import java.nio.file.Path;
2931
import java.nio.file.Paths;
32+
import java.util.Collections;
3033
import java.util.HashMap;
3134
import java.util.List;
3235
import java.util.Map;
3336
import java.util.Optional;
37+
import java.util.function.Consumer;
3438
import java.util.regex.Matcher;
3539
import java.util.regex.Pattern;
3640
import java.util.stream.Stream;
@@ -66,17 +70,19 @@ public class CgroupSubsystemFactory {
6670
*/
6771
private static final Pattern MOUNTINFO_PATTERN = Pattern.compile(
6872
"^[^\\s]+\\s+[^\\s]+\\s+[^\\s]+\\s+" + // (1), (2), (3)
69-
"[^\\s]+\\s+([^\\s]+)\\s+" + // (4), (5) - group 1: mount point
73+
"([^\\s]+)\\s+([^\\s]+)\\s+" + // (4), (5) - group 1, 2: root, mount point
7074
"[^-]+-\\s+" + // (6), (7), (8)
71-
"([^\\s]+)\\s+" + // (9) - group 2: filesystem type
75+
"([^\\s]+)\\s+" + // (9) - group 3: filesystem type
7276
".*$"); // (10), (11)
7377

7478
static CgroupMetrics create() {
7579
Optional<CgroupTypeResult> optResult = null;
7680
try {
77-
optResult = determineType("/proc/self/mountinfo", "/proc/cgroups");
81+
optResult = determineType("/proc/self/mountinfo", "/proc/cgroups", "/proc/self/cgroup");
7882
} catch (IOException e) {
7983
return null;
84+
} catch (UncheckedIOException e) {
85+
return null;
8086
}
8187

8288
if (optResult.equals(Optional.empty())) {
@@ -95,17 +101,37 @@ static CgroupMetrics create() {
95101
return null;
96102
}
97103

104+
Map<String, CgroupInfo> infos = result.getInfos();
98105
if (result.isCgroupV2()) {
99-
CgroupSubsystem subsystem = CgroupV2Subsystem.getInstance();
106+
// For unified it doesn't matter which controller we pick.
107+
CgroupInfo anyController = infos.get(MEMORY_CTRL);
108+
CgroupSubsystem subsystem = CgroupV2Subsystem.getInstance(anyController);
100109
return subsystem != null ? new CgroupMetrics(subsystem) : null;
101110
} else {
102-
CgroupV1Subsystem subsystem = CgroupV1Subsystem.getInstance();
111+
CgroupV1Subsystem subsystem = CgroupV1Subsystem.getInstance(infos);
103112
return subsystem != null ? new CgroupV1MetricsImpl(subsystem) : null;
104113
}
105114
}
106115

107-
public static Optional<CgroupTypeResult> determineType(String mountInfo, String cgroups) throws IOException {
108-
Map<String, CgroupInfo> infos = new HashMap<>();
116+
/*
117+
* Determine the type of the cgroup system (v1 - legacy or hybrid - or, v2 - unified)
118+
* based on three files:
119+
*
120+
* (1) mountInfo (i.e. /proc/self/mountinfo)
121+
* (2) cgroups (i.e. /proc/cgroups)
122+
* (3) selfCgroup (i.e. /proc/self/cgroup)
123+
*
124+
* File 'cgroups' is inspected for the hierarchy ID of the mounted cgroup pseudo
125+
* filesystem. The hierarchy ID, in turn, helps us distinguish cgroups v2 and
126+
* cgroup v1. For a system with zero hierarchy ID, but with >= 1 relevant cgroup
127+
* controllers mounted in 'mountInfo' we can infer it's cgroups v2. Anything else
128+
* will be cgroup v1 (hybrid or legacy). File 'selfCgroup' is being used for
129+
* figuring out the mount path of the controller in the cgroup hierarchy.
130+
*/
131+
public static Optional<CgroupTypeResult> determineType(String mountInfo,
132+
String cgroups,
133+
String selfCgroup) throws IOException {
134+
final Map<String, CgroupInfo> infos = new HashMap<>();
109135
List<String> lines = CgroupUtil.readAllLinesPrivileged(Paths.get(cgroups));
110136
for (String line : lines) {
111137
if (line.startsWith("#")) {
@@ -136,60 +162,207 @@ public static Optional<CgroupTypeResult> determineType(String mountInfo, String
136162
anyControllersEnabled = anyControllersEnabled || info.isEnabled();
137163
}
138164

139-
// If there are no mounted, relevant cgroup controllers in mountinfo and only
140-
// 0 hierarchy IDs in /proc/cgroups have been seen, we are on a cgroups v1 system.
165+
// If there are no mounted, relevant cgroup controllers in 'mountinfo' and only
166+
// 0 hierarchy IDs in file 'cgroups' have been seen, we are on a cgroups v1 system.
141167
// However, continuing in that case does not make sense as we'd need
142168
// information from mountinfo for the mounted controller paths which we wouldn't
143169
// find anyway in that case.
144-
try (Stream<String> mntInfo = CgroupUtil.readFilePrivileged(Paths.get(mountInfo))) {
145-
boolean anyCgroupMounted = mntInfo.anyMatch(CgroupSubsystemFactory::isRelevantControllerMount);
146-
if (!anyCgroupMounted && isCgroupsV2) {
147-
return Optional.empty();
170+
lines = CgroupUtil.readAllLinesPrivileged(Paths.get(mountInfo));
171+
boolean anyCgroupMounted = false;
172+
for (String line: lines) {
173+
boolean cgroupsControllerFound = amendCgroupInfos(line, infos, isCgroupsV2);
174+
anyCgroupMounted = anyCgroupMounted || cgroupsControllerFound;
175+
}
176+
if (!anyCgroupMounted) {
177+
return Optional.empty();
178+
}
179+
180+
// Map a cgroup version specific 'action' to a line in 'selfCgroup' (i.e.
181+
// /proc/self/cgroups) , split on the ':' token, so as to set the appropriate
182+
// path to the cgroup controller in cgroup data structures 'infos'.
183+
// See:
184+
// setCgroupV1Path() for the action run for cgroups v1 systems
185+
// setCgroupV2Path() for the action run for cgroups v2 systems
186+
try (Stream<String> selfCgroupLines =
187+
CgroupUtil.readFilePrivileged(Paths.get(selfCgroup))) {
188+
Consumer<String[]> action = (tokens -> setCgroupV1Path(infos, tokens));
189+
if (isCgroupsV2) {
190+
action = (tokens -> setCgroupV2Path(infos, tokens));
148191
}
192+
// The limit value of 3 is because /proc/self/cgroup contains three
193+
// colon-separated tokens per line. The last token, cgroup path, might
194+
// contain a ':'.
195+
selfCgroupLines.map(line -> line.split(":", 3)).forEach(action);
149196
}
150-
CgroupTypeResult result = new CgroupTypeResult(isCgroupsV2, anyControllersEnabled, anyCgroupsV2Controller, anyCgroupsV1Controller);
197+
198+
CgroupTypeResult result = new CgroupTypeResult(isCgroupsV2,
199+
anyControllersEnabled,
200+
anyCgroupsV2Controller,
201+
anyCgroupsV1Controller,
202+
Collections.unmodifiableMap(infos));
151203
return Optional.of(result);
152204
}
153205

154-
private static boolean isRelevantControllerMount(String line) {
155-
Matcher lineMatcher = MOUNTINFO_PATTERN.matcher(line.trim());
156-
if (lineMatcher.matches()) {
157-
String mountPoint = lineMatcher.group(1);
158-
String fsType = lineMatcher.group(2);
159-
if (fsType.equals("cgroup")) {
160-
String filename = Paths.get(mountPoint).getFileName().toString();
161-
for (String fn: filename.split(",")) {
162-
switch (fn) {
163-
case MEMORY_CTRL: // fall through
164-
case CPU_CTRL:
165-
case CPUSET_CTRL:
166-
case CPUACCT_CTRL:
167-
case BLKIO_CTRL:
168-
return true;
169-
default: break; // ignore not recognized controllers
170-
}
171-
}
172-
} else if (fsType.equals("cgroup2")) {
173-
return true;
174-
}
175-
}
176-
return false;
206+
/*
207+
* Sets the path to the cgroup controller for cgroups v2 based on a line
208+
* in /proc/self/cgroup file (represented as the 'tokens' array).
209+
*
210+
* Example:
211+
*
212+
* 0::/
213+
*
214+
* => tokens = [ "0", "", "/" ]
215+
*/
216+
private static void setCgroupV2Path(Map<String, CgroupInfo> infos,
217+
String[] tokens) {
218+
int hierarchyId = Integer.parseInt(tokens[0]);
219+
String cgroupPath = tokens[2];
220+
for (CgroupInfo info: infos.values()) {
221+
assert hierarchyId == info.getHierarchyId() && hierarchyId == 0;
222+
info.setCgroupPath(cgroupPath);
223+
}
224+
}
225+
226+
/*
227+
* Sets the path to the cgroup controller for cgroups v1 based on a line
228+
* in /proc/self/cgroup file (represented as the 'tokens' array).
229+
*
230+
* Note that multiple controllers might be joined at a single path.
231+
*
232+
* Example:
233+
*
234+
* 7:cpu,cpuacct:/system.slice/docker-74ad896fb40bbefe0f181069e4417505fffa19052098f27edf7133f31423bc0b.scope
235+
*
236+
* => tokens = [ "7", "cpu,cpuacct", "/system.slice/docker-74ad896fb40bbefe0f181069e4417505fffa19052098f27edf7133f31423bc0b.scope" ]
237+
*/
238+
private static void setCgroupV1Path(Map<String, CgroupInfo> infos,
239+
String[] tokens) {
240+
String controllerName = tokens[1];
241+
String cgroupPath = tokens[2];
242+
if (controllerName != null && cgroupPath != null) {
243+
for (String cName: controllerName.split(",")) {
244+
switch (cName) {
245+
case MEMORY_CTRL: // fall through
246+
case CPUSET_CTRL:
247+
case CPUACCT_CTRL:
248+
case CPU_CTRL:
249+
case BLKIO_CTRL:
250+
CgroupInfo info = infos.get(cName);
251+
info.setCgroupPath(cgroupPath);
252+
break;
253+
// Ignore not recognized controllers
254+
default:
255+
break;
256+
}
257+
}
258+
}
259+
}
260+
261+
/**
262+
* Amends cgroup infos with mount path and mount root. The passed in
263+
* 'mntInfoLine' represents a single line in, for example,
264+
* /proc/self/mountinfo. Each line is matched with MOUNTINFO_PATTERN
265+
* (see above), so as to extract the relevant tokens from the line.
266+
*
267+
* Host example cgroups v1:
268+
*
269+
* 44 30 0:41 / /sys/fs/cgroup/devices rw,nosuid,nodev,noexec,relatime shared:16 - cgroup cgroup rw,seclabel,devices
270+
*
271+
* Container example cgroups v1:
272+
*
273+
* 1901 1894 0:37 /system.slice/docker-2291eeb92093f9d761aaf971782b575e9be56bd5930d4b5759b51017df3c1387.scope /sys/fs/cgroup/cpu,cpuacct ro,nosuid,nodev,noexec,relatime master:12 - cgroup cgroup rw,seclabel,cpu,cpuacct
274+
*
275+
* Container example cgroups v2:
276+
*
277+
* 1043 1034 0:27 / /sys/fs/cgroup ro,nosuid,nodev,noexec,relatime - cgroup2 cgroup2 rw,seclabel,nsdelegate
278+
*
279+
*
280+
* @return {@code true} iff a relevant controller has been found at the
281+
* given line
282+
*/
283+
private static boolean amendCgroupInfos(String mntInfoLine,
284+
Map<String, CgroupInfo> infos,
285+
boolean isCgroupsV2) {
286+
Matcher lineMatcher = MOUNTINFO_PATTERN.matcher(mntInfoLine.trim());
287+
boolean cgroupv1ControllerFound = false;
288+
boolean cgroupv2ControllerFound = false;
289+
if (lineMatcher.matches()) {
290+
String mountRoot = lineMatcher.group(1);
291+
String mountPath = lineMatcher.group(2);
292+
String fsType = lineMatcher.group(3);
293+
if (fsType.equals("cgroup")) {
294+
Path p = Paths.get(mountPath);
295+
String[] controllerNames = p.getFileName().toString().split(",");
296+
for (String controllerName: controllerNames) {
297+
switch (controllerName) {
298+
case MEMORY_CTRL: // fall-through
299+
case CPU_CTRL:
300+
case CPUACCT_CTRL:
301+
case BLKIO_CTRL: {
302+
CgroupInfo info = infos.get(controllerName);
303+
assert info.getMountPoint() == null;
304+
assert info.getMountRoot() == null;
305+
info.setMountPoint(mountPath);
306+
info.setMountRoot(mountRoot);
307+
cgroupv1ControllerFound = true;
308+
break;
309+
}
310+
case CPUSET_CTRL: {
311+
CgroupInfo info = infos.get(controllerName);
312+
if (info.getMountPoint() != null) {
313+
// On some systems duplicate cpuset controllers get mounted in addition to
314+
// the main cgroup controllers most likely under /sys/fs/cgroup. In that
315+
// case pick the one under /sys/fs/cgroup and discard others.
316+
if (!info.getMountPoint().startsWith("/sys/fs/cgroup")) {
317+
info.setMountPoint(mountPath);
318+
info.setMountRoot(mountRoot);
319+
}
320+
} else {
321+
info.setMountPoint(mountPath);
322+
info.setMountRoot(mountRoot);
323+
}
324+
cgroupv1ControllerFound = true;
325+
break;
326+
}
327+
default:
328+
// Ignore controllers which we don't recognize
329+
break;
330+
}
331+
}
332+
} else if (fsType.equals("cgroup2")) {
333+
if (isCgroupsV2) { // will be false for hybrid
334+
// All controllers have the same mount point and root mount
335+
// for unified hierarchy.
336+
for (CgroupInfo info: infos.values()) {
337+
assert info.getMountPoint() == null;
338+
assert info.getMountRoot() == null;
339+
info.setMountPoint(mountPath);
340+
info.setMountRoot(mountRoot);
341+
}
342+
}
343+
cgroupv2ControllerFound = true;
344+
}
345+
}
346+
return cgroupv1ControllerFound || cgroupv2ControllerFound;
177347
}
178348

179349
public static final class CgroupTypeResult {
180350
private final boolean isCgroupV2;
181351
private final boolean anyControllersEnabled;
182352
private final boolean anyCgroupV2Controllers;
183353
private final boolean anyCgroupV1Controllers;
354+
private final Map<String, CgroupInfo> infos;
184355

185356
private CgroupTypeResult(boolean isCgroupV2,
186357
boolean anyControllersEnabled,
187358
boolean anyCgroupV2Controllers,
188-
boolean anyCgroupV1Controllers) {
359+
boolean anyCgroupV1Controllers,
360+
Map<String, CgroupInfo> infos) {
189361
this.isCgroupV2 = isCgroupV2;
190362
this.anyControllersEnabled = anyControllersEnabled;
191363
this.anyCgroupV1Controllers = anyCgroupV1Controllers;
192364
this.anyCgroupV2Controllers = anyCgroupV2Controllers;
365+
this.infos = infos;
193366
}
194367

195368
public boolean isCgroupV2() {
@@ -207,5 +380,9 @@ public boolean isAnyCgroupV2Controllers() {
207380
public boolean isAnyCgroupV1Controllers() {
208381
return anyCgroupV1Controllers;
209382
}
383+
384+
public Map<String, CgroupInfo> getInfos() {
385+
return infos;
386+
}
210387
}
211388
}

‎jdk/src/linux/classes/jdk/internal/platform/cgroupv1/CgroupV1Subsystem.java

+83-181
Original file line numberDiff line numberDiff line change
@@ -25,15 +25,11 @@
2525

2626
package jdk.internal.platform.cgroupv1;
2727

28-
import java.io.IOException;
29-
import java.io.UncheckedIOException;
30-
import java.nio.file.Path;
31-
import java.nio.file.Paths;
32-
import java.util.stream.Stream;
28+
import java.util.Map;
3329

30+
import jdk.internal.platform.CgroupInfo;
3431
import jdk.internal.platform.CgroupSubsystem;
3532
import jdk.internal.platform.CgroupSubsystemController;
36-
import jdk.internal.platform.CgroupUtil;
3733
import jdk.internal.platform.CgroupV1Metrics;
3834

3935
public class CgroupV1Subsystem implements CgroupSubsystem, CgroupV1Metrics {
@@ -42,173 +38,107 @@ public class CgroupV1Subsystem implements CgroupSubsystem, CgroupV1Metrics {
4238
private CgroupV1SubsystemController cpuacct;
4339
private CgroupV1SubsystemController cpuset;
4440
private CgroupV1SubsystemController blkio;
45-
private boolean activeSubSystems;
4641

47-
private static final CgroupV1Subsystem INSTANCE = initSubSystem();
42+
private static volatile CgroupV1Subsystem INSTANCE;
4843

4944
private static final String PROVIDER_NAME = "cgroupv1";
5045

51-
private CgroupV1Subsystem() {
52-
activeSubSystems = false;
53-
}
46+
private CgroupV1Subsystem() {}
5447

55-
public static CgroupV1Subsystem getInstance() {
48+
/**
49+
* Get a singleton instance of CgroupV1Subsystem. Initially, it creates a new
50+
* object by retrieving the pre-parsed information from cgroup interface
51+
* files from the provided 'infos' map.
52+
*
53+
* See CgroupSubsystemFactory.determineType() where the actual parsing of
54+
* cgroup interface files happens.
55+
*
56+
* @return A singleton CgroupV1Subsystem instance, never null
57+
*/
58+
public static CgroupV1Subsystem getInstance(Map<String, CgroupInfo> infos) {
59+
if (INSTANCE == null) {
60+
CgroupV1Subsystem tmpSubsystem = initSubSystem(infos);
61+
synchronized (CgroupV1Subsystem.class) {
62+
if (INSTANCE == null) {
63+
INSTANCE = tmpSubsystem;
64+
}
65+
}
66+
}
5667
return INSTANCE;
5768
}
5869

59-
private static CgroupV1Subsystem initSubSystem() {
70+
private static CgroupV1Subsystem initSubSystem(Map<String, CgroupInfo> infos) {
6071
CgroupV1Subsystem subsystem = new CgroupV1Subsystem();
6172

62-
/**
63-
* Find the cgroup mount points for subsystems
64-
* by reading /proc/self/mountinfo
65-
*
66-
* Example for docker MemorySubSystem subsystem:
67-
* 219 214 0:29 /docker/7208cebd00fa5f2e342b1094f7bed87fa25661471a4637118e65f1c995be8a34 /sys/fs/cgroup/MemorySubSystem ro,nosuid,nodev,noexec,relatime - cgroup cgroup rw,MemorySubSystem
68-
*
69-
* Example for host:
70-
* 34 28 0:29 / /sys/fs/cgroup/MemorySubSystem rw,nosuid,nodev,noexec,relatime shared:16 - cgroup cgroup rw,MemorySubSystem
73+
boolean anyActiveControllers = false;
74+
/*
75+
* Find the cgroup mount points for subsystem controllers
76+
* by looking up relevant data in the infos map
7177
*/
72-
try (Stream<String> lines =
73-
CgroupUtil.readFilePrivileged(Paths.get("/proc/self/mountinfo"))) {
74-
75-
lines.filter(line -> line.contains(" - cgroup "))
76-
.map(line -> line.split(" "))
77-
.forEach(entry -> createSubSystemController(subsystem, entry));
78-
79-
} catch (UncheckedIOException e) {
80-
return null;
81-
} catch (IOException e) {
82-
return null;
83-
}
84-
85-
/**
86-
* Read /proc/self/cgroup and map host mount point to
87-
* local one via /proc/self/mountinfo content above
88-
*
89-
* Docker example:
90-
* 5:memory:/docker/6558aed8fc662b194323ceab5b964f69cf36b3e8af877a14b80256e93aecb044
91-
*
92-
* Host example:
93-
* 5:memory:/user.slice
94-
*
95-
* Construct a path to the process specific memory and cpuset
96-
* cgroup directory.
97-
*
98-
* For a container running under Docker from memory example above
99-
* the paths would be:
100-
*
101-
* /sys/fs/cgroup/memory
102-
*
103-
* For a Host from memory example above the path would be:
104-
*
105-
* /sys/fs/cgroup/memory/user.slice
106-
*
107-
*/
108-
try (Stream<String> lines =
109-
CgroupUtil.readFilePrivileged(Paths.get("/proc/self/cgroup"))) {
110-
111-
// The limit value of 3 is because /proc/self/cgroup contains three
112-
// colon-separated tokens per line. The last token, cgroup path, might
113-
// contain a ':'.
114-
lines.map(line -> line.split(":", 3))
115-
.filter(line -> (line.length >= 3))
116-
.forEach(line -> setSubSystemControllerPath(subsystem, line));
117-
118-
} catch (UncheckedIOException e) {
119-
return null;
120-
} catch (IOException e) {
121-
return null;
78+
for (CgroupInfo info: infos.values()) {
79+
switch (info.getName()) {
80+
case "memory": {
81+
if (info.getMountRoot() != null && info.getMountPoint() != null) {
82+
CgroupV1MemorySubSystemController controller = new CgroupV1MemorySubSystemController(info.getMountRoot(), info.getMountPoint());
83+
controller.setPath(info.getCgroupPath());
84+
boolean isHierarchial = getHierarchical(controller);
85+
controller.setHierarchical(isHierarchial);
86+
boolean isSwapEnabled = getSwapEnabled(controller);
87+
controller.setSwapEnabled(isSwapEnabled);
88+
subsystem.setMemorySubSystem(controller);
89+
anyActiveControllers = true;
90+
}
91+
break;
92+
}
93+
case "cpuset": {
94+
if (info.getMountRoot() != null && info.getMountPoint() != null) {
95+
CgroupV1SubsystemController controller = new CgroupV1SubsystemController(info.getMountRoot(), info.getMountPoint());
96+
controller.setPath(info.getCgroupPath());
97+
subsystem.setCpuSetController(controller);
98+
anyActiveControllers = true;
99+
}
100+
break;
101+
}
102+
case "cpuacct": {
103+
if (info.getMountRoot() != null && info.getMountPoint() != null) {
104+
CgroupV1SubsystemController controller = new CgroupV1SubsystemController(info.getMountRoot(), info.getMountPoint());
105+
controller.setPath(info.getCgroupPath());
106+
subsystem.setCpuAcctController(controller);
107+
anyActiveControllers = true;
108+
}
109+
break;
110+
}
111+
case "cpu": {
112+
if (info.getMountRoot() != null && info.getMountPoint() != null) {
113+
CgroupV1SubsystemController controller = new CgroupV1SubsystemController(info.getMountRoot(), info.getMountPoint());
114+
controller.setPath(info.getCgroupPath());
115+
subsystem.setCpuController(controller);
116+
anyActiveControllers = true;
117+
}
118+
break;
119+
}
120+
case "blkio": {
121+
if (info.getMountRoot() != null && info.getMountPoint() != null) {
122+
CgroupV1SubsystemController controller = new CgroupV1SubsystemController(info.getMountRoot(), info.getMountPoint());
123+
controller.setPath(info.getCgroupPath());
124+
subsystem.setBlkIOController(controller);
125+
anyActiveControllers = true;
126+
}
127+
break;
128+
}
129+
default:
130+
throw new AssertionError("Unrecognized controller in infos: " + info.getName());
131+
}
122132
}
123133

124134
// Return Metrics object if we found any subsystems.
125-
if (subsystem.activeSubSystems()) {
135+
if (anyActiveControllers) {
126136
return subsystem;
127137
}
128138

129139
return null;
130140
}
131141

132-
/**
133-
* createSubSystem objects and initialize mount points
134-
*/
135-
private static void createSubSystemController(CgroupV1Subsystem subsystem, String[] mountentry) {
136-
if (mountentry.length < 5) return;
137-
138-
Path p = Paths.get(mountentry[4]);
139-
String[] subsystemNames = p.getFileName().toString().split(",");
140-
141-
for (String subsystemName: subsystemNames) {
142-
switch (subsystemName) {
143-
case "memory":
144-
subsystem.setMemorySubSystem(new CgroupV1MemorySubSystemController(mountentry[3], mountentry[4]));
145-
break;
146-
case "cpuset":
147-
subsystem.setCpuSetController(new CgroupV1SubsystemController(mountentry[3], mountentry[4]));
148-
break;
149-
case "cpuacct":
150-
subsystem.setCpuAcctController(new CgroupV1SubsystemController(mountentry[3], mountentry[4]));
151-
break;
152-
case "cpu":
153-
subsystem.setCpuController(new CgroupV1SubsystemController(mountentry[3], mountentry[4]));
154-
break;
155-
case "blkio":
156-
subsystem.setBlkIOController(new CgroupV1SubsystemController(mountentry[3], mountentry[4]));
157-
break;
158-
default:
159-
// Ignore subsystems that we don't support
160-
break;
161-
}
162-
}
163-
}
164-
165-
/**
166-
* setSubSystemPath based on the contents of /proc/self/cgroup
167-
*/
168-
private static void setSubSystemControllerPath(CgroupV1Subsystem subsystem, String[] entry) {
169-
String controller = entry[1];
170-
String base = entry[2];
171-
if (controller != null && base != null) {
172-
for (String cName: controller.split(",")) {
173-
switch (cName) {
174-
case "memory":
175-
setPath(subsystem, subsystem.memoryController(), base);
176-
break;
177-
case "cpuset":
178-
setPath(subsystem, subsystem.cpuSetController(), base);
179-
break;
180-
case "cpuacct":
181-
setPath(subsystem, subsystem.cpuController(), base);
182-
break;
183-
case "cpu":
184-
setPath(subsystem, subsystem.cpuAcctController(), base);
185-
break;
186-
case "blkio":
187-
setPath(subsystem, subsystem.blkIOController(), base);
188-
break;
189-
// Ignore subsystems that we don't support
190-
default:
191-
break;
192-
}
193-
}
194-
}
195-
}
196-
197-
private static void setPath(CgroupV1Subsystem subsystem, CgroupV1SubsystemController controller, String base) {
198-
if (controller != null) {
199-
controller.setPath(base);
200-
if (controller instanceof CgroupV1MemorySubSystemController) {
201-
CgroupV1MemorySubSystemController memorySubSystem = (CgroupV1MemorySubSystemController)controller;
202-
boolean isHierarchial = getHierarchical(memorySubSystem);
203-
memorySubSystem.setHierarchical(isHierarchial);
204-
boolean isSwapEnabled = getSwapEnabled(memorySubSystem);
205-
memorySubSystem.setSwapEnabled(isSwapEnabled);
206-
}
207-
subsystem.setActiveSubSystems();
208-
}
209-
}
210-
211-
212142
private static boolean getSwapEnabled(CgroupV1MemorySubSystemController controller) {
213143
long retval = getLongValue(controller, "memory.memsw.limit_in_bytes");
214144
return retval > 0;
@@ -220,14 +150,6 @@ private static boolean getHierarchical(CgroupV1MemorySubSystemController control
220150
return hierarchical > 0;
221151
}
222152

223-
private void setActiveSubSystems() {
224-
activeSubSystems = true;
225-
}
226-
227-
private boolean activeSubSystems() {
228-
return activeSubSystems;
229-
}
230-
231153
private void setMemorySubSystem(CgroupV1MemorySubSystemController memory) {
232154
this.memory = memory;
233155
}
@@ -248,26 +170,6 @@ private void setBlkIOController(CgroupV1SubsystemController blkio) {
248170
this.blkio = blkio;
249171
}
250172

251-
private CgroupV1SubsystemController memoryController() {
252-
return memory;
253-
}
254-
255-
private CgroupV1SubsystemController cpuController() {
256-
return cpu;
257-
}
258-
259-
private CgroupV1SubsystemController cpuAcctController() {
260-
return cpuacct;
261-
}
262-
263-
private CgroupV1SubsystemController cpuSetController() {
264-
return cpuset;
265-
}
266-
267-
private CgroupV1SubsystemController blkIOController() {
268-
return blkio;
269-
}
270-
271173
private static long getLongValue(CgroupSubsystemController controller,
272174
String parm) {
273175
return CgroupSubsystemController.getLongValue(controller,

‎jdk/src/linux/classes/jdk/internal/platform/cgroupv2/CgroupV2Subsystem.java

+22-42
Original file line numberDiff line numberDiff line change
@@ -28,19 +28,18 @@
2828
import java.io.IOException;
2929
import java.io.UncheckedIOException;
3030
import java.nio.file.Paths;
31-
import java.util.List;
3231
import java.util.concurrent.TimeUnit;
3332
import java.util.function.Function;
3433
import java.util.stream.Collectors;
35-
import java.util.stream.Stream;
3634

35+
import jdk.internal.platform.CgroupInfo;
3736
import jdk.internal.platform.CgroupSubsystem;
3837
import jdk.internal.platform.CgroupSubsystemController;
3938
import jdk.internal.platform.CgroupUtil;
4039

4140
public class CgroupV2Subsystem implements CgroupSubsystem {
4241

43-
private static final CgroupV2Subsystem INSTANCE = initSubsystem();
42+
private static volatile CgroupV2Subsystem INSTANCE;
4443
private static final long[] LONG_ARRAY_NOT_SUPPORTED = null;
4544
private static final int[] INT_ARRAY_UNAVAILABLE = null;
4645
private final CgroupSubsystemController unified;
@@ -65,48 +64,29 @@ private long getLongVal(String file) {
6564
return getLongVal(file, CgroupSubsystem.LONG_RETVAL_UNLIMITED);
6665
}
6766

68-
private static CgroupV2Subsystem initSubsystem() {
69-
// read mountinfo so as to determine root mount path
70-
String mountPath = null;
71-
try (Stream<String> lines =
72-
CgroupUtil.readFilePrivileged(Paths.get("/proc/self/mountinfo"))) {
73-
74-
String l = lines.filter(line -> line.contains(" - cgroup2 "))
75-
.collect(Collectors.joining());
76-
String[] tokens = l.split(" ");
77-
mountPath = tokens[4];
78-
} catch (UncheckedIOException e) {
79-
return null;
80-
} catch (IOException e) {
81-
return null;
82-
}
83-
String cgroupPath = null;
84-
try {
85-
List<String> lines = CgroupUtil.readAllLinesPrivileged(Paths.get("/proc/self/cgroup"));
86-
for (String line: lines) {
87-
String[] tokens = line.split(":");
88-
if (tokens.length != 3) {
89-
return null; // something is not right.
90-
}
91-
if (!"0".equals(tokens[0])) {
92-
// hierarchy must be zero for cgroups v2
93-
return null;
67+
/**
68+
* Get the singleton instance of a cgroups v2 subsystem. On initialization,
69+
* a new object from the given cgroup information 'anyController' is being
70+
* created. Note that the cgroup information has been parsed from cgroup
71+
* interface files ahead of time.
72+
*
73+
* See CgroupSubsystemFactory.determineType() for the cgroup interface
74+
* files parsing logic.
75+
*
76+
* @return A singleton CgroupSubsystem instance, never null.
77+
*/
78+
public static CgroupSubsystem getInstance(CgroupInfo anyController) {
79+
if (INSTANCE == null) {
80+
CgroupSubsystemController unified = new CgroupV2SubsystemController(
81+
anyController.getMountPoint(),
82+
anyController.getCgroupPath());
83+
CgroupV2Subsystem tmpCgroupSystem = new CgroupV2Subsystem(unified);
84+
synchronized (CgroupV2Subsystem.class) {
85+
if (INSTANCE == null) {
86+
INSTANCE = tmpCgroupSystem;
9487
}
95-
cgroupPath = tokens[2];
96-
break;
9788
}
98-
} catch (UncheckedIOException e) {
99-
return null;
100-
} catch (IOException e) {
101-
return null;
10289
}
103-
CgroupSubsystemController unified = new CgroupV2SubsystemController(
104-
mountPath,
105-
cgroupPath);
106-
return new CgroupV2Subsystem(unified);
107-
}
108-
109-
public static CgroupSubsystem getInstance() {
11090
return INSTANCE;
11191
}
11292

‎jdk/test/jdk/internal/platform/cgroup/TestCgroupSubsystemFactory.java

+78-8
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
* or visit www.oracle.com if you need additional information or have any
2121
* questions.
2222
*/
23+
import static org.junit.Assert.assertEquals;
2324
import static org.junit.Assert.assertFalse;
2425
import static org.junit.Assert.assertTrue;
2526

@@ -34,6 +35,7 @@
3435
import org.junit.Before;
3536
import org.junit.Test;
3637

38+
import jdk.internal.platform.CgroupInfo;
3739
import jdk.internal.platform.CgroupSubsystemFactory;
3840
import jdk.internal.platform.CgroupSubsystemFactory.CgroupTypeResult;
3941
import jdk.testlibrary.Utils;
@@ -61,6 +63,9 @@ public class TestCgroupSubsystemFactory {
6163
private Path cgroupv1MntInfoSystemdOnly;
6264
private Path cgroupv1MntInfoDoubleCpusets;
6365
private Path cgroupv1MntInfoDoubleCpusets2;
66+
private Path cgroupv1SelfCgroup;
67+
private Path cgroupv2SelfCgroup;
68+
private Path cgroupv1SelfCgroupJoinCtrl;
6469
private String mntInfoEmpty = "";
6570
private String cgroupsNonZeroJoinControllers =
6671
"#subsys_name hierarchy num_cgroups enabled\n" +
@@ -77,6 +82,17 @@ public class TestCgroupSubsystemFactory {
7782
"hugetlb\t4\t153\t1\n" +
7883
"pids\t5\t95\t1\n" +
7984
"rdma\t8\t1\t1\n";
85+
private String selfCgroupNonZeroJoinControllers =
86+
"9:cpuset:/\n" +
87+
"8:perf_event:/\n" +
88+
"7:rdma:/\n" +
89+
"6:freezer:/\n" +
90+
"5:blkio:/user.slice\n" +
91+
"4:pids:/user.slice/user-1000.slice/session-3.scope\n" +
92+
"3:devices:/user.slice\n" +
93+
"2:cpu,cpuacct,memory,net_cls,net_prio,hugetlb:/user.slice/user-1000.slice/session-3.scope\n" +
94+
"1:name=systemd:/user.slice/user-1000.slice/session-3.scope\n" +
95+
"0::/user.slice/user-1000.slice/session-3.scope\n";
8096
private String cgroupsZeroHierarchy =
8197
"#subsys_name hierarchy num_cgroups enabled\n" +
8298
"cpuset 0 1 1\n" +
@@ -136,6 +152,19 @@ public class TestCgroupSubsystemFactory {
136152
private String mntInfoCgroupv1MoreCpusetLine = "121 32 0:37 / /cpuset rw,relatime shared:69 - cgroup none rw,cpuset\n";
137153
private String mntInfoCgroupsV1DoubleCpuset = mntInfoHybrid + mntInfoCgroupv1MoreCpusetLine;
138154
private String mntInfoCgroupsV1DoubleCpuset2 = mntInfoCgroupv1MoreCpusetLine + mntInfoHybrid;
155+
private String cgroupv1SelfCgroupContent = "11:memory:/user.slice/user-1000.slice/user@1000.service\n" +
156+
"10:hugetlb:/\n" +
157+
"9:cpuset:/\n" +
158+
"8:pids:/user.slice/user-1000.slice/user@1000.service\n" +
159+
"7:freezer:/\n" +
160+
"6:blkio:/\n" +
161+
"5:net_cls,net_prio:/\n" +
162+
"4:devices:/user.slice\n" +
163+
"3:perf_event:/\n" +
164+
"2:cpu,cpuacct:/\n" +
165+
"1:name=systemd:/user.slice/user-1000.slice/user@1000.service/apps.slice/apps-org.gnome.Terminal.slice/vte-spawn-3c00b338-5b65-439f-8e97-135e183d135d.scope\n" +
166+
"0::/user.slice/user-1000.slice/user@1000.service/apps.slice/apps-org.gnome.Terminal.slice/vte-spawn-3c00b338-5b65-439f-8e97-135e183d135d.scope\n";
167+
private String cgroupv2SelfCgroupContent = "0::/user.slice/user-1000.slice/session-2.scope";
139168

140169
@Before
141170
public void setup() {
@@ -171,6 +200,15 @@ public void setup() {
171200

172201
cgroupv1MountInfoJoinControllers = Paths.get(existingDirectory.toString(), "mntinfo_cgv1_join_controllers");
173202
Files.write(cgroupv1MountInfoJoinControllers, mntInfoCgroupv1JoinControllers.getBytes());
203+
204+
cgroupv1SelfCgroup = Paths.get(existingDirectory.toString(), "self_cgroup_cgv1");
205+
Files.write(cgroupv1SelfCgroup, cgroupv1SelfCgroupContent.getBytes());
206+
207+
cgroupv2SelfCgroup = Paths.get(existingDirectory.toString(), "self_cgroup_cgv2");
208+
Files.write(cgroupv2SelfCgroup, cgroupv2SelfCgroupContent.getBytes());
209+
210+
cgroupv1SelfCgroupJoinCtrl = Paths.get(existingDirectory.toString(), "self_cgroup_cgv1_join_controllers");
211+
Files.write(cgroupv1SelfCgroupJoinCtrl, selfCgroupNonZeroJoinControllers.getBytes());
174212
} catch (IOException e) {
175213
throw new RuntimeException(e);
176214
}
@@ -189,18 +227,22 @@ public void teardown() {
189227
public void testCgroupv1JoinControllerCombo() throws IOException {
190228
String cgroups = cgroupv1CgroupsJoinControllers.toString();
191229
String mountInfo = cgroupv1MountInfoJoinControllers.toString();
192-
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups);
230+
String selfCgroup = cgroupv1SelfCgroupJoinCtrl.toString();
231+
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
193232

194233
assertTrue("Expected non-empty cgroup result", result.isPresent());
195234
CgroupTypeResult res = result.get();
196235
assertFalse("Join controller combination expected as cgroups v1", res.isCgroupV2());
236+
CgroupInfo memoryInfo = res.getInfos().get("memory");
237+
assertEquals("/user.slice/user-1000.slice/session-3.scope", memoryInfo.getCgroupPath());
197238
}
198239

199240
@Test
200241
public void testCgroupv1SystemdOnly() throws IOException {
201242
String cgroups = cgroupv1CgInfoZeroHierarchy.toString();
202243
String mountInfo = cgroupv1MntInfoSystemdOnly.toString();
203-
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups);
244+
String selfCgroup = cgroupv1SelfCgroup.toString(); // Content doesn't matter
245+
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
204246

205247
assertTrue("zero hierarchy ids with no *relevant* controllers mounted", Optional.empty().equals(result));
206248
}
@@ -214,29 +256,39 @@ public void testCgroupv1MultipleCpusetMounts() throws IOException {
214256
private void doMultipleCpusetMountsTest(Path info) throws IOException {
215257
String cgroups = cgroupv1CgInfoNonZeroHierarchy.toString();
216258
String mountInfo = info.toString();
217-
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups);
259+
String selfCgroup = cgroupv1SelfCgroup.toString();
260+
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
218261

219262
assertTrue("Expected non-empty cgroup result", result.isPresent());
220263
CgroupTypeResult res = result.get();
221264
assertFalse("Duplicate cpusets should not influence detection heuristic", res.isCgroupV2());
265+
CgroupInfo cpuSetInfo = res.getInfos().get("cpuset");
266+
assertEquals("/sys/fs/cgroup/cpuset", cpuSetInfo.getMountPoint());
267+
assertEquals("/", cpuSetInfo.getMountRoot());
222268
}
223269

224270
@Test
225271
public void testHybridCgroupsV1() throws IOException {
226272
String cgroups = cgroupv1CgInfoNonZeroHierarchy.toString();
227273
String mountInfo = cgroupv1MntInfoNonZeroHierarchy.toString();
228-
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups);
274+
String selfCgroup = cgroupv1SelfCgroup.toString();
275+
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
229276

230277
assertTrue("Expected non-empty cgroup result", result.isPresent());
231278
CgroupTypeResult res = result.get();
232279
assertFalse("hybrid hierarchy expected as cgroups v1", res.isCgroupV2());
280+
CgroupInfo memoryInfo = res.getInfos().get("memory");
281+
assertEquals("/user.slice/user-1000.slice/user@1000.service", memoryInfo.getCgroupPath());
282+
assertEquals("/", memoryInfo.getMountRoot());
283+
assertEquals("/sys/fs/cgroup/memory", memoryInfo.getMountPoint());
233284
}
234285

235286
@Test
236287
public void testZeroHierarchyCgroupsV1() throws IOException {
237288
String cgroups = cgroupv1CgInfoZeroHierarchy.toString();
238289
String mountInfo = cgroupv1MntInfoZeroHierarchy.toString();
239-
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups);
290+
String selfCgroup = cgroupv1SelfCgroup.toString(); // Content doesn't matter
291+
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
240292

241293
assertTrue("zero hierarchy ids with no mounted controllers => empty result", Optional.empty().equals(result));
242294
}
@@ -245,26 +297,44 @@ public void testZeroHierarchyCgroupsV1() throws IOException {
245297
public void testZeroHierarchyCgroupsV2() throws IOException {
246298
String cgroups = cgroupv2CgInfoZeroHierarchy.toString();
247299
String mountInfo = cgroupv2MntInfoZeroHierarchy.toString();
248-
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups);
300+
String selfCgroup = cgroupv2SelfCgroup.toString();
301+
Optional<CgroupTypeResult> result = CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
249302

250303
assertTrue("Expected non-empty cgroup result", result.isPresent());
251304
CgroupTypeResult res = result.get();
252305

253306
assertTrue("zero hierarchy ids with mounted controllers expected cgroups v2", res.isCgroupV2());
307+
CgroupInfo memoryInfo = res.getInfos().get("memory");
308+
assertEquals("/user.slice/user-1000.slice/session-2.scope", memoryInfo.getCgroupPath());
309+
CgroupInfo cpuInfo = res.getInfos().get("cpu");
310+
assertEquals(memoryInfo.getCgroupPath(), cpuInfo.getCgroupPath());
311+
assertEquals(memoryInfo.getMountPoint(), cpuInfo.getMountPoint());
312+
assertEquals(memoryInfo.getMountRoot(), cpuInfo.getMountRoot());
313+
assertEquals("/sys/fs/cgroup", cpuInfo.getMountPoint());
254314
}
255315

256316
@Test(expected = IOException.class)
257317
public void mountInfoFileNotFound() throws IOException {
258318
String cgroups = cgroupv1CgInfoZeroHierarchy.toString(); // any existing file
319+
String selfCgroup = cgroupv1SelfCgroup.toString(); // any existing file
259320
String mountInfo = Paths.get(existingDirectory.toString(), "not-existing-mountinfo").toString();
260321

261-
CgroupSubsystemFactory.determineType(mountInfo, cgroups);
322+
CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
262323
}
263324

264325
@Test(expected = IOException.class)
265326
public void cgroupsFileNotFound() throws IOException {
266327
String cgroups = Paths.get(existingDirectory.toString(), "not-existing-cgroups").toString();
267328
String mountInfo = cgroupv2MntInfoZeroHierarchy.toString(); // any existing file
268-
CgroupSubsystemFactory.determineType(mountInfo, cgroups);
329+
String selfCgroup = cgroupv2SelfCgroup.toString(); // any existing file
330+
CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
331+
}
332+
333+
@Test(expected = IOException.class)
334+
public void selfCgroupsFileNotFound() throws IOException {
335+
String cgroups = cgroupv1CgInfoZeroHierarchy.toString(); // any existing file
336+
String mountInfo = cgroupv2MntInfoZeroHierarchy.toString(); // any existing file
337+
String selfCgroup = Paths.get(existingDirectory.toString(), "not-existing-self-cgroups").toString();
338+
CgroupSubsystemFactory.determineType(mountInfo, cgroups, selfCgroup);
269339
}
270340
}

0 commit comments

Comments
 (0)
Please sign in to comment.