jdk-24/test/hotspot/jtreg/runtime/InvocationTests/shared/AbstractGenerator.java
2023-10-16 08:00:48 +00:00

177 lines
6.0 KiB
Java

/*
* Copyright (c) 2009, 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 shared;
import java.io.File;
import java.io.FileOutputStream;
import java.util.Arrays;
import java.util.Map;
import java.util.List;
import java.util.ArrayList;
public abstract class AbstractGenerator {
protected final boolean dumpClasses;
protected final boolean executeTests;
private static int testNum = 0;
private static int classesBeforeGC = 0;
protected AbstractGenerator(String[] args) {
List<String> params = new ArrayList<String>(Arrays.asList(args));
if (params.contains("--help")) {
Utils.printHelp();
System.exit(0);
}
dumpClasses = params.contains("--dump");
executeTests = !params.contains("--noexecute");
params.remove("--dump");
params.remove("--noexecute");
Utils.init(params);
}
/*******************************************************************/
public static void writeToFile(File dir, Map<String, byte[]> classes) {
for (String name : classes.keySet()) {
try {
writeToFile(dir, name, classes.get(name));
} catch (Exception e) {
throw new RuntimeException(e);
}
}
}
/*******************************************************************/
public static void writeToFile(File dir, String fullName, byte[] classBytecode) {
if (!dir.isDirectory()) {
throw new RuntimeException("Invalid parameter: dir doesn't point to an existing directory");
}
File classFile =
new File(
dir.getPath() + File.separator
+ fullName.replaceAll("\\.", File.separator)
+ ".class"
);
classFile.getParentFile().mkdirs();
try {
FileOutputStream fos = new FileOutputStream(classFile);
try {
fos.write(classBytecode);
} finally {
fos.close();
}
} catch (Exception e) {
throw new RuntimeException(e);
}
}
protected boolean exec(Map<String, byte[]> classes, String description, String calleeClassName, String classNameC, String[] callSites) throws ClassNotFoundException {
boolean isPassed = true;
testNum++;
// Every N-th classes, force a GC to kick out the loaded classes from previous tests.
// Different tests come in with different number of classes, so testNum is not reliable.
classesBeforeGC -= classes.size();
if (classesBeforeGC <= 0) {
System.gc();
classesBeforeGC = 3000;
}
String caseDescription = String.format("%4d| %s", testNum, description);
// Create test executor for a single case
classes.put(
ExecutorGenerator.className
, new ExecutorGenerator(
caseDescription
, calleeClassName
, classNameC
).generateExecutor(callSites)
);
// Dump generated set to disk, if needed
if (dumpClasses) {
File dir = new File("classes" + File.separator + String.format("%04d", testNum));
dir.mkdirs();
writeToFile(dir, classes);
}
ByteArrayClassLoader loader = new ByteArrayClassLoader(classes);
Class paramClass;
Class targetClass;
Checker checker;
System.out.printf(caseDescription);
try {
paramClass = loader.loadClass(calleeClassName);
targetClass = loader.loadClass(classNameC);
checker = getChecker(paramClass, targetClass);
if (executeTests) {
// Check runtime behavior
Caller caller = new Caller(loader, checker, paramClass, targetClass);
for (String site : callSites) {
String callResult = caller.call(site);
System.out.printf(" %7s", callResult);
if (!caller.isPassed()) {
String result = checker.check(loader.loadClass(site));
System.out.printf("/%s", Checker.abbreviateResult(result));
isPassed = false;
}
}
if (!caller.isPassed()) {
System.out.print(" | FAILED");
}
} else {
for (String site : callSites) {
String result = checker.check(loader.loadClass(site));
System.out.printf(" %7s", Checker.abbreviateResult(result));
}
}
} catch (Throwable e) {
String result = Checker.abbreviateResult(e.getClass().getName());
for (String site : callSites) {
System.out.printf(" %7s", result);
}
}
System.out.println();
return isPassed;
}
protected abstract Checker getChecker(Class paramClass, Class targetClass);
}