08855df46a
Reviewed-by: vlivanov, erikj, mseledtsov, gthornbr
261 lines
8.4 KiB
Java
261 lines
8.4 KiB
Java
/*
|
|
* Copyright (c) 2013, 2018, 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 vm.share.vmstresser;
|
|
|
|
import java.util.*;
|
|
import java.util.concurrent.locks.*;
|
|
|
|
import nsk.share.*;
|
|
import nsk.share.classload.*;
|
|
import nsk.share.test.*;
|
|
|
|
/**
|
|
* Stresser that load classes until OOME, then unload some of them and continue loading.
|
|
*/
|
|
public class MetaspaceStresser extends Thread {
|
|
|
|
/**
|
|
* Capacity of class containers.
|
|
* This amount of classes will be unloaded on reset call.
|
|
*/
|
|
public static final int DEFAULT_BUCKET_SIZE = 4000;
|
|
|
|
public static final int DEFAULT_PAUSE_TIME = 0;
|
|
|
|
/*
|
|
* Loaded classes stored in ClassContainer instances.
|
|
* Such instances organized in array-based stack as it is
|
|
* one of the simplest way to minimize possibility
|
|
* to get OOME and guarntee that after replacing
|
|
* reference to class container by null there will be
|
|
* no cached refereces and container will be reclaimed by
|
|
* GC and classes will become unloadable.
|
|
*/
|
|
// Maximum available amount of arrays with class containers.
|
|
private static final int CONTAINERS_ARRAY_LENGTH = 1000;
|
|
// Maximum length array with class containers.
|
|
private static final int CONTAINER_ARRAYS_COUNT = 100;
|
|
|
|
private ClassContainersStack containersStack = new ClassContainersStack(CONTAINER_ARRAYS_COUNT * CONTAINERS_ARRAY_LENGTH,
|
|
CONTAINERS_ARRAY_LENGTH);
|
|
private ClassContainer newContainer = null;
|
|
|
|
private ExecutionController controller = null;
|
|
private int bucketSize = DEFAULT_BUCKET_SIZE;
|
|
private int pauseTime = DEFAULT_PAUSE_TIME;
|
|
|
|
private ReentrantLock lock = new ReentrantLock();
|
|
|
|
/**
|
|
* Construct MetaspaceStrresser with default bucket size
|
|
* and pause time.
|
|
* @param c controller to control execution time.
|
|
*/
|
|
public MetaspaceStresser(ExecutionController c) {
|
|
controller = c;
|
|
}
|
|
|
|
/**
|
|
* Construct MetaspaceStrresser with custom bucket size
|
|
* and pause time.
|
|
* @param c controller to control execution time.
|
|
* @param bucketSize classes to be unloaded on reset.
|
|
* @param pauseTime pause after reset.
|
|
*/
|
|
public MetaspaceStresser(ExecutionController c, int bucketSize, int pauseTime) {
|
|
this(c);
|
|
this.bucketSize = bucketSize;
|
|
this.pauseTime = pauseTime;
|
|
}
|
|
|
|
/**
|
|
* Fill Metaspace with classes.
|
|
* Classes will be loaded until OOME, then some of them will be unloaded.
|
|
*/
|
|
public synchronized void prepare() {
|
|
while (controller.continueExecution()) {
|
|
try {
|
|
fillContainerStack();
|
|
} catch (OutOfMemoryError oome) {
|
|
unloadLastClassBucket();
|
|
return;
|
|
} catch (ClassNotFoundException cnfe) {
|
|
throw new TestBug("Unexpected exception in stresser.", cnfe);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Load new class to container, fill containerStack.
|
|
* Classes will be loaded until OOME
|
|
* @throws ClassNotFoundException
|
|
*/
|
|
private void fillContainerStack() throws ClassNotFoundException {
|
|
newContainer = new ClassContainer();
|
|
while (newContainer.size() < bucketSize && controller.continueExecution()) {
|
|
newContainer.loadClass();
|
|
}
|
|
containersStack.push(newContainer);
|
|
newContainer = null;
|
|
}
|
|
|
|
/**
|
|
* Run stresser.
|
|
* Stresser will load classes until OOME, then bucketSize classes
|
|
* will be unloaded and stresser will wait pauseTime millisiconds
|
|
* before continuing class loading.
|
|
*/
|
|
public void run() {
|
|
try {
|
|
while (controller.continueExecution()) {
|
|
try {
|
|
fillContainerStack();
|
|
} catch (OutOfMemoryError oome) {
|
|
unloadLastClassBucket();
|
|
try {
|
|
Thread.sleep(pauseTime);
|
|
} catch (InterruptedException ie) {
|
|
}
|
|
}
|
|
}
|
|
} catch (Throwable e) {
|
|
throw new TestBug("Unexpected exception in stresser.", e);
|
|
} finally {
|
|
containersStack.free();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Unload most recently loaded bucket of classes.
|
|
*/
|
|
public void unloadLastClassBucket() {
|
|
while (controller.continueExecution()) {
|
|
try {
|
|
containersStack.pop();
|
|
System.gc();
|
|
break;
|
|
} catch (OutOfMemoryError oome) {
|
|
oome.printStackTrace();
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Array-based stack for ClassContainer's.
|
|
*/
|
|
private class ClassContainersStack {
|
|
|
|
private int arrayLength = 0;
|
|
private int arraysCount = 0;
|
|
private int arrayIndex = 0;
|
|
private int elemIndex = 0;
|
|
|
|
private ClassContainer data[][];
|
|
|
|
/**
|
|
* Create ClassContainersStack that will be able
|
|
* to store size classes in arrays of segmentSize length.
|
|
*/
|
|
public ClassContainersStack(int size, int segementSize) {
|
|
arrayLength = segementSize;
|
|
arraysCount = size / arrayLength;
|
|
data = new ClassContainer[arraysCount][];
|
|
data[0] = new ClassContainer[arrayLength];
|
|
}
|
|
|
|
/**
|
|
* Push ClassContainer c into stack.
|
|
*/
|
|
public synchronized void push(ClassContainer c) {
|
|
data[arrayIndex][elemIndex] = c;
|
|
elemIndex++;
|
|
if (elemIndex == arrayLength) {
|
|
if (arrayIndex == arraysCount) {
|
|
throw new TestBug("ClassContainersStack ran out of available slots");
|
|
}
|
|
data[arrayIndex + 1] = new ClassContainer[arrayLength];
|
|
arrayIndex++;
|
|
elemIndex = 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove reference to top ClassContainer.
|
|
*/
|
|
public synchronized void pop() {
|
|
data[arrayIndex][elemIndex] = null;
|
|
if (elemIndex > 0) {
|
|
elemIndex--;
|
|
} else if (arrayIndex > 0) {
|
|
data[arrayIndex] = null;
|
|
arrayIndex--;
|
|
elemIndex = arrayLength - 1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove all stored ClassContainers.
|
|
*/
|
|
public synchronized void free() {
|
|
data = null;
|
|
System.gc();
|
|
data = new ClassContainer[arraysCount][];
|
|
data[0] = new ClassContainer[arrayLength];
|
|
arrayIndex = 0;
|
|
elemIndex = 0;
|
|
}
|
|
|
|
}
|
|
|
|
/// Variable used to create uniqe name for generated classes.
|
|
private static long lastClass = 0;
|
|
|
|
/**
|
|
* Class container consists of classes and their ClassLoader, so
|
|
* if there will be no references to container and classes inside it then
|
|
* it could be easely collected by GC.
|
|
*/
|
|
private class ClassContainer {
|
|
|
|
private List<Class> classes = new LinkedList<Class>();
|
|
private GeneratingClassLoader loader = new GeneratingClassLoader();
|
|
private String prefix = loader.getPrefix();
|
|
private int length = loader.getNameLength();
|
|
|
|
public void loadClass() throws ClassNotFoundException {
|
|
String newName = prefix + "c" + lastClass;
|
|
lastClass++;
|
|
while (newName.length() < length) {
|
|
newName = newName + "c";
|
|
}
|
|
classes.add(loader.loadClass(newName));
|
|
}
|
|
|
|
public int size() {
|
|
return classes.size();
|
|
}
|
|
}
|
|
|
|
}
|