jdk-24/test/hotspot/jtreg/gc/arguments/TestMaxMinHeapFreeRatioFlags.java

313 lines
13 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2016, 2019, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package gc.arguments;
/*
* @test TestMaxMinHeapFreeRatioFlags
* @key gc
* @summary Verify that heap size changes according to max and min heap free ratios.
* @requires vm.gc != "Z" & vm.gc != "Shenandoah"
* @library /test/lib
* @library /
* @modules java.base/jdk.internal.misc
* java.management
* @run driver/timeout=240 gc.arguments.TestMaxMinHeapFreeRatioFlags
*/
import java.util.LinkedList;
import java.util.Arrays;
import java.util.Collections;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
import jdk.test.lib.Utils;
import jdk.internal.misc.Unsafe;
public class TestMaxMinHeapFreeRatioFlags {
public static final long M = 1024 * 1024;
public static final long MAX_HEAP_SIZE = 200 * M;
public static final long HEAP_SIZE = 10 * M;
public static final long MAX_NEW_SIZE = 20 * M;
public static final long NEW_SIZE = 5 * M;
public static void main(String args[]) throws Exception {
LinkedList<String> options = new LinkedList<>(
Arrays.asList(Utils.getFilteredTestJavaOpts("-XX:[^ ]*HeapFreeRatio","-XX:\\+ExplicitGCInvokesConcurrent"))
);
negativeTest(20, false, 10, true, options);
negativeTest(100, true, 0, false, options);
negativeTest(101, false, 50, false, options);
negativeTest(49, true, 102, true, options);
negativeTest(-1, false, 50, false, options);
negativeTest(50, true, -1, true, options);
positiveTest(10, false, 90, false, true, options);
positiveTest(10, true, 80, false, true, options);
positiveTest(20, false, 70, true, true, options);
positiveTest(25, true, 65, true, true, options);
positiveTest(40, false, 50, false, true, options);
}
/**
* Verify that heap size will be changed to conform
* min and max heap free ratios.
*
* @param minRatio value of MinHeapFreeRatio option
* @param useXminf used Xminf option instead of MinHeapFreeRatio
* @param maxRatio value of MaxHeapFreeRatio option
* @param useXmaxf used Xmaxf option instead of MaxHeapFreeRatio
* @param options additional options for JVM
*/
public static void positiveTest(int minRatio, boolean useXminf,
int maxRatio, boolean useXmaxf, boolean shrinkHeapInSteps,
LinkedList<String> options) throws Exception {
LinkedList<String> vmOptions = new LinkedList<>(options);
Collections.addAll(vmOptions,
(useXminf ? "-Xminf" + minRatio / 100.0 : "-XX:MinHeapFreeRatio=" + minRatio),
(useXmaxf ? "-Xmaxf" + maxRatio / 100.0 : "-XX:MaxHeapFreeRatio=" + maxRatio),
"-Xmx" + MAX_HEAP_SIZE,
"-Xms" + HEAP_SIZE,
"--add-exports=java.base/jdk.internal.misc=ALL-UNNAMED",
"-XX:NewSize=" + NEW_SIZE,
"-XX:MaxNewSize=" + MAX_NEW_SIZE,
"-XX:" + (shrinkHeapInSteps ? '+' : '-') + "ShrinkHeapInSteps",
RatioVerifier.class.getName(),
Integer.toString(minRatio),
Integer.toString(maxRatio),
Boolean.toString(shrinkHeapInSteps)
);
ProcessBuilder procBuilder = GCArguments.createJavaProcessBuilder(vmOptions.toArray(new String[vmOptions.size()]));
OutputAnalyzer analyzer = new OutputAnalyzer(procBuilder.start());
analyzer.shouldHaveExitValue(0);
}
/**
* Verify that VM will fail to start with specified ratios.
*
* @param minRatio value of MinHeapFreeRatio option
* @param useXminf used Xminf option instead of MinHeapFreeRatio
* @param maxRatio value of MaxHeapFreeRatio option
* @param useXmaxf used Xmaxf option instead of MaxHeapFreeRatio
* @param options additional options for JVM
*/
public static void negativeTest(int minRatio, boolean useXminf,
int maxRatio, boolean useXmaxf,
LinkedList<String> options) throws Exception {
LinkedList<String> vmOptions = new LinkedList<>(options);
Collections.addAll(vmOptions,
(useXminf ? "-Xminf" + minRatio / 100.0 : "-XX:MinHeapFreeRatio=" + minRatio),
(useXmaxf ? "-Xmaxf" + maxRatio / 100.0 : "-XX:MaxHeapFreeRatio=" + maxRatio),
"--add-exports=java.base/jdk.internal.misc=ALL-UNNAMED",
"-version"
);
ProcessBuilder procBuilder = GCArguments.createJavaProcessBuilder(vmOptions.toArray(new String[vmOptions.size()]));
OutputAnalyzer analyzer = new OutputAnalyzer(procBuilder.start());
analyzer.shouldHaveExitValue(1);
analyzer.shouldContain("Error: Could not create the Java Virtual Machine.");
}
/**
* RatioVerifier will be executed in the tested VM.
* It will check that real heap usage after collection lies between MinHeapFreeRatio and MaxHeapFreeRatio.
*/
public static class RatioVerifier {
private static final Unsafe unsafe = Unsafe.getUnsafe();
// Size of byte array that will be allocated
public static final int CHUNK_SIZE = 1024;
// Length of byte array, that will be added to "garbage" list.
public static final int ARRAY_LENGTH = CHUNK_SIZE - Unsafe.ARRAY_BYTE_BASE_OFFSET;
// Amount of tries to force heap shrinking/expansion using GC
public static final int GC_TRIES = 10;
// Value that will be added/substracted from expected min/max heap free ratio
// during memory allocation to make sure that specified limit will be exceeded.
public static final double OVERLOAD = 0.05;
// Acceptable heap free ratio limit exceedance: verification will fail if
// actual ratio is lower than expected min heap free ratio - VARIANCE or
// higher than expected max heap free ratio + VARIANCE.
public static final double VARIANCE = 0.025;
public static LinkedList<Object> garbage = new LinkedList<>();
public static void main(String args[]) throws Exception {
if (args.length != 3) {
throw new IllegalArgumentException("Expected 3 args: <minRatio> <maxRatio> <shrinkHeapInSteps>");
}
if (GCTypes.OldGCType.getOldGCType() == GCTypes.OldGCType.PSOld ||
GCTypes.OldGCType.getOldGCType() == GCTypes.OldGCType.G1) {
System.out.println("Test is not applicable to parallel full GCs");
return;
}
double minRatio = Integer.valueOf(args[0]) / 100.0;
double maxRatio = Integer.valueOf(args[1]) / 100.0;
boolean shrinkHeapInSteps = Boolean.valueOf(args[2]);
long maxHeapSize = getMax();
int gcTries = (shrinkHeapInSteps ? GC_TRIES : 1);
// Initial checks. This also links up everything in these helper methods,
// in case it brings more garbage.
forceGC(gcTries);
verifyRatio(minRatio, maxRatio);
// commit 0.5 of total heap size to have enough space
// to both shink and expand
while (getCommitted() < maxHeapSize / 2) {
garbage.add(new byte[ARRAY_LENGTH]);
}
forceGC(gcTries);
// Verify that current heap free ratio lies between specified limits
verifyRatio(minRatio, maxRatio);
// Estimate how much memory we have to allocate to force expansion
long memoryToFill = (long) (getCommitted() * (1 - minRatio + OVERLOAD))
- getUsed();
long previouslyCommitted = getCommitted();
while (memoryToFill > 0) {
garbage.add(new byte[CHUNK_SIZE]);
memoryToFill -= CHUNK_SIZE;
}
forceGC(gcTries);
// Verify that after memory allocation heap free ratio is still conforming specified limits
verifyRatio(minRatio, maxRatio);
// Verify that heap was actually expanded
if (previouslyCommitted >= getCommitted()) {
throw new RuntimeException("Heap was not expanded.");
}
// Estimate how much memory we have to free to force shrinking
long memoryToFree = getUsed()
- (long) (getCommitted() * (1 - maxRatio - OVERLOAD));
previouslyCommitted = getCommitted();
while (memoryToFree > 0 && garbage.size() > 0) {
garbage.remove(garbage.size() - 1);
memoryToFree -= CHUNK_SIZE;
}
forceGC(gcTries);
// Verify that heap free ratio is still conforming specified limits
verifyRatio(minRatio, maxRatio);
// Verify that heap was actually shrinked
if (previouslyCommitted <= getCommitted()) {
throw new RuntimeException("Heap was not shrinked.");
}
}
public static void forceGC(int gcTries) {
for (int i = 0; i < gcTries; i++) {
System.gc();
try {
Thread.sleep(10);
} catch (InterruptedException ie) {
}
}
}
/**
* Verify that heap free ratio is conforming specified limits.
* Actual heap free ratio may be very close to one of specified limits,
* but exceed for more then VARIANCE.
* Verification will also pass if actual ratio is not conforming limits,
* but it is not possible to shrink/expand heap.
*/
public static void verifyRatio(double minRatio, double maxRatio) {
double ratio = getHeapFreeRatio();
System.out.println(minRatio + " " + ratio + " " + maxRatio);
if (minRatio - ratio > VARIANCE
&& getCommitted() < getMax()) {
throw new RuntimeException("Current heap free ratio is lower than "
+ "MinHeapFreeRatio (" + ratio + " vs " + minRatio + ").");
}
if (ratio - maxRatio > VARIANCE
&& getUsed() > getInit()) {
throw new RuntimeException("Current heap free ratio is higher than "
+ "MaxHeapFreeRatio (" + ratio + " vs " + maxRatio + ").");
}
}
/*
* Obtain information about heap size.
*
* For G1 information summed up for all type of regions,
* because tested options affect overall heap sizing.
*
* For all other GCs return information only for old gen.
*/
public static long getMax() {
return HeapRegionUsageTool.getOldUsage().getMax();
}
public static long getInit() {
if (GCTypes.OldGCType.getOldGCType() == GCTypes.OldGCType.G1) {
return HeapRegionUsageTool.getEdenUsage().getInit()
+ HeapRegionUsageTool.getSurvivorUsage().getInit()
+ HeapRegionUsageTool.getOldUsage().getInit();
} else {
return HeapRegionUsageTool.getOldUsage().getInit();
}
}
public static long getUsed() {
if (GCTypes.OldGCType.getOldGCType() == GCTypes.OldGCType.G1) {
return HeapRegionUsageTool.getEdenUsage().getUsed()
+ HeapRegionUsageTool.getSurvivorUsage().getUsed()
+ HeapRegionUsageTool.getOldUsage().getUsed();
} else {
return HeapRegionUsageTool.getOldUsage().getUsed();
}
}
public static long getCommitted() {
if (GCTypes.OldGCType.getOldGCType() == GCTypes.OldGCType.G1) {
return HeapRegionUsageTool.getEdenUsage().getCommitted()
+ HeapRegionUsageTool.getSurvivorUsage().getCommitted()
+ HeapRegionUsageTool.getOldUsage().getCommitted();
} else {
return HeapRegionUsageTool.getOldUsage().getCommitted();
}
}
public static long getFree() {
return getCommitted() - getUsed();
}
public static double getHeapFreeRatio() {
return getFree() / (double) getCommitted();
}
}
}