Skip to content

Commit 9a0d1e7

Browse files
Jonathan Dowlandjerboaa
Jonathan Dowland
authored andcommittedAug 26, 2022
8292541: [Metrics] Reported memory limit may exceed physical machine memory
Reviewed-by: stuefe, sgehwolf
1 parent c74b6d4 commit 9a0d1e7

File tree

3 files changed

+52
-16
lines changed

3 files changed

+52
-16
lines changed
 

‎src/java.base/linux/classes/jdk/internal/platform/CgroupMetrics.java

+8-1
Original file line numberDiff line numberDiff line change
@@ -121,7 +121,13 @@ public long getMemoryFailCount() {
121121

122122
@Override
123123
public long getMemoryLimit() {
124-
return subsystem.getMemoryLimit();
124+
long subsMem = subsystem.getMemoryLimit();
125+
// Catch the cgroup memory limit exceeding host physical memory.
126+
// Treat this as unlimited.
127+
if (subsMem >= getTotalMemorySize0()) {
128+
return CgroupSubsystem.LONG_RETVAL_UNLIMITED;
129+
}
130+
return subsMem;
125131
}
126132

127133
@Override
@@ -178,5 +184,6 @@ public static Metrics getInstance() {
178184
}
179185

180186
private static native boolean isUseContainerSupport();
187+
private static native long getTotalMemorySize0();
181188

182189
}

‎src/java.base/linux/native/libjava/CgroupMetrics.c

+8
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
* or visit www.oracle.com if you need additional information or have any
2323
* questions.
2424
*/
25+
#include <unistd.h>
2526

2627
#include "jni.h"
2728
#include "jvm.h"
@@ -33,3 +34,10 @@ Java_jdk_internal_platform_CgroupMetrics_isUseContainerSupport(JNIEnv *env, jcla
3334
{
3435
return JVM_IsUseContainerSupport();
3536
}
37+
38+
JNIEXPORT jlong JNICALL
39+
Java_jdk_internal_platform_CgroupMetrics_getTotalMemorySize0
40+
(JNIEnv *env, jclass ignored)
41+
{
42+
return sysconf(_SC_PHYS_PAGES) * sysconf(_SC_PAGESIZE);
43+
}

‎test/hotspot/jtreg/containers/docker/TestMemoryAwareness.java

+36-15
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,13 @@
4747
public class TestMemoryAwareness {
4848
private static final String imageName = Common.imageName("memory");
4949

50+
private static String getHostMaxMemory() throws Exception {
51+
DockerRunOptions opts = Common.newOpts(imageName);
52+
String goodMem = Common.run(opts).firstMatch("total physical memory: (\\d+)", 1);
53+
assertNotNull(goodMem, "no match for 'total physical memory' in trace output");
54+
return goodMem;
55+
}
56+
5057
public static void main(String[] args) throws Exception {
5158
if (!DockerTestUtils.canTestDocker()) {
5259
return;
@@ -79,7 +86,10 @@ public static void main(String[] args) throws Exception {
7986
"1G", Integer.toString(((int) Math.pow(2, 20)) * 1024),
8087
"1500M", Integer.toString(((int) Math.pow(2, 20)) * (1500 - 1024))
8188
);
82-
testContainerMemExceedsPhysical();
89+
final String hostMaxMem = getHostMaxMemory();
90+
testOperatingSystemMXBeanIgnoresMemLimitExceedingPhysicalMemory(hostMaxMem);
91+
testMetricsIgnoresMemLimitExceedingPhysicalMemory(hostMaxMem);
92+
testContainerMemExceedsPhysical(hostMaxMem);
8393
} finally {
8494
if (!DockerTestUtils.RETAIN_IMAGE_AFTER_TEST) {
8595
DockerTestUtils.removeDockerImage(imageName);
@@ -102,24 +112,16 @@ private static void testMemoryLimit(String valueToSet, String expectedTraceValue
102112

103113
// JDK-8292083
104114
// Ensure that Java ignores container memory limit values above the host's physical memory.
105-
private static void testContainerMemExceedsPhysical()
115+
private static void testContainerMemExceedsPhysical(final String hostMaxMem)
106116
throws Exception {
107-
108117
Common.logNewTestCase("container memory limit exceeds physical memory");
109-
110-
DockerRunOptions opts = Common.newOpts(imageName);
111-
112-
// first run: establish physical memory in test environment and derive
113-
// a bad value one power of ten larger
114-
String goodMem = Common.run(opts).firstMatch("total physical memory: (\\d+)", 1);
115-
assertNotNull(goodMem, "no match for 'total physical memory' in trace output");
116-
String badMem = goodMem + "0";
117-
118-
// second run: set a container memory limit to the bad value
119-
opts = Common.newOpts(imageName)
118+
String badMem = hostMaxMem + "0";
119+
// set a container memory limit to the bad value
120+
DockerRunOptions opts = Common.newOpts(imageName)
120121
.addDockerOpts("--memory", badMem);
122+
121123
Common.run(opts)
122-
.shouldMatch("container memory limit (ignored: " + badMem + "|unlimited: -1), using host value " + goodMem);
124+
.shouldMatch("container memory limit (ignored: " + badMem + "|unlimited: -1), using host value " + hostMaxMem);
123125
}
124126

125127

@@ -200,4 +202,23 @@ private static void testOperatingSystemMXBeanAwareness(String memoryAllocation,
200202
}
201203
}
202204

205+
206+
// JDK-8292541: Ensure OperatingSystemMXBean ignores container memory limits above the host's physical memory.
207+
private static void testOperatingSystemMXBeanIgnoresMemLimitExceedingPhysicalMemory(final String hostMaxMem)
208+
throws Exception {
209+
String badMem = hostMaxMem + "0";
210+
testOperatingSystemMXBeanAwareness(badMem, hostMaxMem, badMem, hostMaxMem);
211+
}
212+
213+
// JDK-8292541: Ensure Metrics ignores container memory limits above the host's physical memory.
214+
private static void testMetricsIgnoresMemLimitExceedingPhysicalMemory(final String hostMaxMem)
215+
throws Exception {
216+
Common.logNewTestCase("Metrics ignore container memory limit exceeding physical memory");
217+
String badMem = hostMaxMem + "0";
218+
DockerRunOptions opts = Common.newOpts(imageName)
219+
.addJavaOpts("-XshowSettings:system")
220+
.addDockerOpts("--memory", badMem);
221+
222+
DockerTestUtils.dockerRunJava(opts).shouldMatch("Memory Limit: Unlimited");
223+
}
203224
}

0 commit comments

Comments
 (0)
Please sign in to comment.