1d1cd32bc3
Reviewed-by: tschatzl, aboldtch
269 lines
9.2 KiB
Java
269 lines
9.2 KiB
Java
/*
|
|
* Copyright (c) 2016, 2023, 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.stress;
|
|
|
|
/*
|
|
* @test
|
|
* @key stress
|
|
* @summary Stress G1 by humongous allocations in situation near OOM
|
|
* @requires vm.gc.G1
|
|
* @requires !vm.flightRecorder
|
|
* @library /test/lib
|
|
* @modules java.base/jdk.internal.misc
|
|
* @run driver/timeout=180 gc.stress.TestStressG1Humongous 4 3 1.1 120
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @requires vm.gc.G1
|
|
* @requires !vm.flightRecorder
|
|
* @library /test/lib
|
|
* @modules java.base/jdk.internal.misc
|
|
* @run driver/timeout=180 gc.stress.TestStressG1Humongous 16 5 2.1 120
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @requires vm.gc.G1
|
|
* @requires !vm.flightRecorder
|
|
* @library /test/lib
|
|
* @modules java.base/jdk.internal.misc
|
|
* @run driver/timeout=180 gc.stress.TestStressG1Humongous 32 4 0.6 120
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @requires vm.gc.G1
|
|
* @requires !vm.flightRecorder
|
|
* @library /test/lib
|
|
* @modules java.base/jdk.internal.misc
|
|
* @run driver/timeout=900 gc.stress.TestStressG1Humongous 1 7 0.6 600
|
|
*/
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.Collections;
|
|
import java.util.concurrent.CountDownLatch;
|
|
import java.util.concurrent.atomic.AtomicInteger;
|
|
|
|
import jdk.test.lib.Platform;
|
|
import jdk.test.lib.Utils;
|
|
import jdk.test.lib.process.ProcessTools;
|
|
import jdk.test.lib.process.OutputAnalyzer;
|
|
|
|
public class TestStressG1Humongous{
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
if (args.length != 4) {
|
|
throw new IllegalArgumentException("Test expects 4 arguments");
|
|
}
|
|
|
|
// Limit heap size on 32-bit platforms
|
|
int heapSize = Platform.is32bit() ? 512 : 1024;
|
|
|
|
// Region size, threads, humongous size, and timeout passed as @run arguments
|
|
int regionSize = Integer.parseInt(args[0]);
|
|
int threads = Integer.parseInt(args[1]);
|
|
double humongousSize = Double.parseDouble(args[2]);
|
|
int timeout = Integer.parseInt(args[3]);
|
|
|
|
ArrayList<String> options = new ArrayList<>();
|
|
Collections.addAll(options, Utils.getTestJavaOpts());
|
|
Collections.addAll(options,
|
|
"-Xlog:gc=debug",
|
|
"-Xmx" + heapSize + "m",
|
|
"-XX:+UseG1GC",
|
|
"-XX:G1HeapRegionSize=" + regionSize + "m",
|
|
"-Dtimeout=" + timeout,
|
|
"-Dthreads=" + threads,
|
|
"-Dhumongoussize=" + humongousSize,
|
|
"-Dregionsize=" + regionSize,
|
|
TestStressG1HumongousImpl.class.getName()
|
|
);
|
|
OutputAnalyzer output = ProcessTools.executeLimitedTestJava(options);
|
|
output.shouldHaveExitValue(0);
|
|
}
|
|
}
|
|
|
|
class TestStressG1HumongousImpl {
|
|
// Timeout in seconds
|
|
private static final int TIMEOUT = Integer.getInteger("timeout", 60);
|
|
private static final int THREAD_COUNT = Integer.getInteger("threads", 2);
|
|
private static final int REGION_SIZE = Integer.getInteger("regionsize", 1) * 1024 * 1024;
|
|
private static final int HUMONGOUS_SIZE = (int) (REGION_SIZE * Double.parseDouble(System.getProperty("humongoussize", "1.5")));
|
|
private static final int NUMBER_OF_FREE_REGIONS = 2;
|
|
|
|
private volatile boolean isRunning;
|
|
private final Thread[] threads;
|
|
private final AtomicInteger alocatedObjectsCount;
|
|
private CountDownLatch countDownLatch;
|
|
public static final List<Object> GARBAGE = Collections.synchronizedList(new ArrayList<>());
|
|
|
|
public static void main(String[] args) throws InterruptedException {
|
|
new TestStressG1HumongousImpl().run();
|
|
}
|
|
|
|
public TestStressG1HumongousImpl() {
|
|
isRunning = true;
|
|
threads = new Thread[THREAD_COUNT];
|
|
alocatedObjectsCount = new AtomicInteger(0);
|
|
}
|
|
|
|
private void run() throws InterruptedException {
|
|
new Thread(new Timer()).start();
|
|
int checkedAmountOfHObjects = getExpectedAmountOfObjects();
|
|
while (isRunning()) {
|
|
countDownLatch = new CountDownLatch(THREAD_COUNT);
|
|
startAllocationThreads(checkedAmountOfHObjects);
|
|
countDownLatch.await();
|
|
GARBAGE.clear();
|
|
System.out.println("Allocated " + alocatedObjectsCount.get() + " objects.");
|
|
alocatedObjectsCount.set(0);
|
|
}
|
|
System.out.println("Done!");
|
|
}
|
|
|
|
/**
|
|
* Tries to fill available memory with humongous objects to get expected amount.
|
|
* @return expected amount of humongous objects
|
|
*/
|
|
private int getExpectedAmountOfObjects() {
|
|
long maxMem = Runtime.getRuntime().maxMemory();
|
|
int expectedHObjects = (int) (maxMem / HUMONGOUS_SIZE);
|
|
// Will allocate NUMBER_OF_FREE_REGIONS region less to give some free space for VM.
|
|
int checkedAmountOfHObjects = checkHeapCapacity(expectedHObjects) - NUMBER_OF_FREE_REGIONS;
|
|
if (checkedAmountOfHObjects <= 0) {
|
|
throw new RuntimeException("Cannot start testing because selected maximum heap "
|
|
+ "is not large enough to contain more than " + NUMBER_OF_FREE_REGIONS + " regions");
|
|
}
|
|
return checkedAmountOfHObjects;
|
|
}
|
|
|
|
/**
|
|
* Starts several threads to allocate the requested amount of humongous objects.
|
|
* @param totalObjects total amount of object that will be created
|
|
*/
|
|
private void startAllocationThreads(int totalObjects) {
|
|
int objectsPerThread = totalObjects / THREAD_COUNT;
|
|
int objectsForLastThread = objectsPerThread + totalObjects % THREAD_COUNT;
|
|
for (int i = 0; i < THREAD_COUNT - 1; ++i) {
|
|
threads[i] = new Thread(new AllocationThread(countDownLatch, objectsPerThread, alocatedObjectsCount));
|
|
}
|
|
threads[THREAD_COUNT - 1] = new Thread(new AllocationThread(countDownLatch, objectsForLastThread, alocatedObjectsCount));
|
|
for (int i = 0; i < THREAD_COUNT; ++i) {
|
|
threads[i].start();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates a humongous object of the predefined size.
|
|
*/
|
|
private void createObject() {
|
|
GARBAGE.add(new byte[HUMONGOUS_SIZE]);
|
|
}
|
|
|
|
/**
|
|
* Tries to create the requested amount of humongous objects.
|
|
* In case of OOME, stops creating and cleans the created garbage.
|
|
* @param expectedObjects amount of objects based on heap size
|
|
* @return amount of created objects
|
|
*/
|
|
private int checkHeapCapacity(int expectedObjects) {
|
|
int allocated = 0;
|
|
try {
|
|
while (isRunning() && allocated < expectedObjects) {
|
|
createObject();
|
|
++allocated;
|
|
}
|
|
} catch (OutOfMemoryError oome) {
|
|
GARBAGE.clear();
|
|
}
|
|
return allocated;
|
|
}
|
|
|
|
private void setDone() {
|
|
isRunning = false;
|
|
}
|
|
|
|
private boolean isRunning() {
|
|
return isRunning;
|
|
}
|
|
|
|
/**
|
|
* Thread which allocates requested amount of humongous objects.
|
|
*/
|
|
private class AllocationThread implements Runnable {
|
|
|
|
private final int totalObjects;
|
|
private final CountDownLatch cdl;
|
|
private final AtomicInteger allocationCounter;
|
|
|
|
/**
|
|
* Creates allocation thread
|
|
* @param cdl CountDownLatch
|
|
* @param objects amount of objects to allocate
|
|
* @param counter
|
|
*/
|
|
public AllocationThread(CountDownLatch cdl, int objects, AtomicInteger counter) {
|
|
totalObjects = objects;
|
|
this.cdl = cdl;
|
|
allocationCounter = counter;
|
|
}
|
|
|
|
@Override
|
|
public void run() {
|
|
int allocatedObjects = 0;
|
|
try {
|
|
while (isRunning && allocatedObjects < totalObjects) {
|
|
createObject();
|
|
allocatedObjects++;
|
|
allocationCounter.incrementAndGet();
|
|
}
|
|
|
|
} catch (OutOfMemoryError oome) {
|
|
GARBAGE.clear();
|
|
System.out.print("OOME was caught.");
|
|
System.out.println(" Allocated in thread: " + allocatedObjects + " . Totally allocated: " + allocationCounter.get() + ".");
|
|
} finally {
|
|
cdl.countDown();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Simple Runnable which waits TIMEOUT and sets isRunning to false.
|
|
*/
|
|
class Timer implements Runnable {
|
|
|
|
@Override
|
|
public void run() {
|
|
try {
|
|
Thread.sleep(TIMEOUT * 1000);
|
|
} catch (InterruptedException ex) {
|
|
}
|
|
setDone();
|
|
}
|
|
}
|
|
}
|