136 lines
6.9 KiB
Java
Raw Normal View History

/*
* 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 metaspace.stressHierarchy.common.generateHierarchy;
import java.util.*;
import vm.share.InMemoryJavaCompiler;
public class GenerateHierarchyHelper {
public static enum Type {
CLASSES, INTERFACES, MIXED;
}
// Class-container that represents generated source file
private static class ClassDescriptor {public String fullName; public CharSequence sourceCode; }
private static enum Inheritance {
CLASS_EXTENDS_CLASS, CLASS_IMPLEMENTS_INTERFACE, INTERFACE_EXTENDS_INTERFACE
}
private static final int EDGE_IN_MIXED_CASE = 30;
private static Random random = new Random();
public static TreeDescriptor generateHierarchy(int depth, int minLevelSize, int maxLevelSize, Type type) {
TreeDescriptor tree = new TreeDescriptor();
Map<String, CharSequence> sourceMap = new HashMap<String, CharSequence>();
int numberOfNodesInPrevLevel = 1;
// generate root
String packageName = composePackageName(0, 0);
String className = packageName + ".Dummy";
switch (type) {
case CLASSES:
sourceMap.put(className, "package " + packageName +";\n public class Dummy { " +
"public int calculate2() {return hashCode();} " +
"public double calculate() {return hashCode() + 0.1;} " +
"public String composeString() {return \"_root_\";}" +
"}");
break;
default:
sourceMap.put(className, "package " + packageName + ";\n public interface Dummy {}");
}
tree.addNode(0, 0, 0, className);
for (int level = 1; level < depth; level++) {
int nodesInLevel = minLevelSize + random.nextInt(maxLevelSize - minLevelSize);
for (int nodeIndex = 0; nodeIndex < nodesInLevel; nodeIndex++) {
int parent = random.nextInt(numberOfNodesInPrevLevel);
Inheritance inheritance = null;
switch (type) {
case CLASSES:
inheritance = Inheritance.CLASS_EXTENDS_CLASS;
break;
case INTERFACES:
inheritance = Inheritance.INTERFACE_EXTENDS_INTERFACE;
break;
case MIXED:
inheritance = level < EDGE_IN_MIXED_CASE ? Inheritance.INTERFACE_EXTENDS_INTERFACE
: (level == EDGE_IN_MIXED_CASE ? Inheritance.CLASS_IMPLEMENTS_INTERFACE
: Inheritance.CLASS_EXTENDS_CLASS);
break;
}
ClassDescriptor classDescriptor = generateClassCode(level, nodeIndex, parent, inheritance);
sourceMap.put(classDescriptor.fullName, classDescriptor.sourceCode);
tree.addNode(level, nodeIndex, parent, classDescriptor.fullName);
}
numberOfNodesInPrevLevel = nodesInLevel;
}
Map<String, byte[]> bytecodeMap = InMemoryJavaCompiler.compile(sourceMap);
for (NodeDescriptor nodeDescriptor : tree.nodeDescriptorList) {
nodeDescriptor.bytecode = bytecodeMap.get(nodeDescriptor.className);
}
return tree;
}
private static String composePackageName(int level, int nodeIndex) {
return "package_level" + level + "_num" + nodeIndex;
}
private static ClassDescriptor generateClassCode(int level, int nodeIndex, int parent, Inheritance inheritance) {
ClassDescriptor result = new ClassDescriptor();
String packageName = composePackageName(level, nodeIndex);
String parentPackage = composePackageName(level - 1, parent);
result.fullName = packageName + ".Dummy";
StringBuffer source = new StringBuffer("package ");
source.append(packageName + ";\n\n");
switch (inheritance) {
case INTERFACE_EXTENDS_INTERFACE:
source.append(" public interface Dummy extends " + parentPackage + ".Dummy {}");
break;
case CLASS_EXTENDS_CLASS:
source.append(" public class Dummy extends " + parentPackage + ".Dummy { " +
"public int calculate2() {return (super.calculate2() + hashCode() % 1000);} " +
"public double calculate() {return (super.calculate() + hashCode() + 0.1);} " +
"public String composeString() {return super.composeString() + \"_" + packageName + "_\";}" +
"}");
break;
case CLASS_IMPLEMENTS_INTERFACE:
source.append(" public class Dummy implements " + parentPackage + ".Dummy { " +
"public int calculate2() {return hashCode();} " +
"public double calculate() {return hashCode() + 0.1;} " +
"public String composeString() {return \"_ancestor_class_\";}" +
"}");
break;
}
result.sourceCode = source;
return result;
}
}