0c37008917
Reviewed-by: mseledtsov, sspitsyn, lmesnik
273 lines
9.0 KiB
Java
273 lines
9.0 KiB
Java
/*
|
|
* Copyright (c) 2015, 2022, 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 compiler.jvmci.compilerToVM;
|
|
|
|
import compiler.jvmci.common.CTVMUtilities;
|
|
import compiler.testlibrary.CompilerUtils;
|
|
import jdk.test.lib.util.Pair;
|
|
import jdk.test.lib.Utils;
|
|
import jdk.vm.ci.code.InstalledCode;
|
|
import jdk.vm.ci.meta.MetaAccessProvider;
|
|
import jdk.vm.ci.meta.ResolvedJavaMethod;
|
|
import jdk.vm.ci.runtime.JVMCI;
|
|
import jdk.test.whitebox.WhiteBox;
|
|
import jdk.test.whitebox.code.NMethod;
|
|
|
|
import java.lang.reflect.Constructor;
|
|
import java.lang.reflect.Executable;
|
|
import java.lang.reflect.InvocationTargetException;
|
|
import java.lang.reflect.Method;
|
|
import java.lang.reflect.Modifier;
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.Collections;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
|
|
/**
|
|
* A test case for tests which require compiled code.
|
|
*/
|
|
public class CompileCodeTestCase {
|
|
private static final WhiteBox WB = WhiteBox.getWhiteBox();
|
|
private static final int COMP_LEVEL;
|
|
static {
|
|
int[] levels = CompilerUtils.getAvailableCompilationLevels();
|
|
if (levels.length == 0) {
|
|
throw new Error("TESTBUG: no compilers available");
|
|
}
|
|
COMP_LEVEL = levels[levels.length - 1];
|
|
}
|
|
private static final Class<?>[] CLASSES = {
|
|
Interface.class,
|
|
Dummy.class,
|
|
DummyEx.class};
|
|
private static final Map<Class<?>, Object> RECEIVERS;
|
|
|
|
public final Object receiver;
|
|
public final Executable executable;
|
|
public final int bci;
|
|
private final boolean isOsr;
|
|
|
|
public CompileCodeTestCase(Object receiver, Executable executable,
|
|
int bci) {
|
|
this.receiver = receiver;
|
|
this.executable = executable;
|
|
this.bci = bci;
|
|
isOsr = (bci >= 0);
|
|
}
|
|
|
|
public NMethod compile() {
|
|
return compile(COMP_LEVEL);
|
|
}
|
|
|
|
public Pair<Object, ? extends Throwable> invoke(Object[] args) {
|
|
boolean old = executable.isAccessible();
|
|
executable.setAccessible(true);
|
|
try {
|
|
try {
|
|
if (executable instanceof Method) {
|
|
Method m = (Method) executable;
|
|
return new Pair<>(m.invoke(receiver, args), null);
|
|
}
|
|
|
|
if (executable instanceof Constructor) {
|
|
Constructor c = (Constructor) executable;
|
|
return new Pair<>(c.newInstance(args), null);
|
|
}
|
|
} catch (InvocationTargetException e) {
|
|
return new Pair<>(null, e.getCause());
|
|
} catch (Throwable e) {
|
|
return new Pair<>(null, e);
|
|
}
|
|
} finally {
|
|
executable.setAccessible(old);
|
|
}
|
|
throw new Error(executable + " has unsupported type "
|
|
+ executable.getClass());
|
|
}
|
|
|
|
public NMethod compile(int level) {
|
|
String directive = "[{ match: \"" + executable.getDeclaringClass().getName().replace('.', '/')
|
|
+ "." + (executable instanceof Constructor ? "<init>" : executable.getName())
|
|
+ "\", " + "BackgroundCompilation: false }]";
|
|
if (WB.addCompilerDirective(directive) != 1) {
|
|
throw new Error("Failed to add compiler directive: " + directive);
|
|
}
|
|
boolean enqueued = WB.enqueueMethodForCompilation(executable,
|
|
level, bci);
|
|
if (!enqueued) {
|
|
throw new Error(String.format(
|
|
"%s can't be enqueued for %scompilation on level %d",
|
|
executable, bci >= 0 ? "osr-" : "", level));
|
|
}
|
|
Utils.waitForCondition(() -> WB.isMethodCompiled(executable, isOsr));
|
|
return NMethod.get(executable, isOsr);
|
|
}
|
|
|
|
public static List<CompileCodeTestCase> generate(int bci) {
|
|
ArrayList<CompileCodeTestCase> result = new ArrayList<>();
|
|
for (Class<?> aClass : CLASSES) {
|
|
Object receiver = RECEIVERS.get(aClass);
|
|
if (receiver == null) {
|
|
throw new Error("TESTBUG : no receiver for class " + aClass);
|
|
}
|
|
for (Executable m : aClass.getDeclaredConstructors()) {
|
|
result.add(new CompileCodeTestCase(receiver, m, bci));
|
|
}
|
|
Arrays.stream(aClass.getDeclaredMethods())
|
|
.filter(m -> !Modifier.isAbstract(m.getModifiers()))
|
|
.filter(m -> !Modifier.isNative(m.getModifiers()))
|
|
.map(m -> new CompileCodeTestCase(receiver, m, bci))
|
|
.forEach(result::add);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public NMethod toNMethod() {
|
|
return NMethod.get(executable, isOsr);
|
|
}
|
|
|
|
public InstalledCode toInstalledCode() {
|
|
MetaAccessProvider metaAccess = JVMCI.getRuntime().getHostJVMCIBackend().getMetaAccess();
|
|
ResolvedJavaMethod resolvedJavaMethod = metaAccess.lookupJavaMethod(executable);
|
|
NMethod nmethod = toNMethod();
|
|
long address = nmethod == null ? 0L : nmethod.address;
|
|
long entryPoint = nmethod == null ? 0L : nmethod.entry_point;
|
|
return CTVMUtilities.getInstalledCode(resolvedJavaMethod, executable.getName(), address, entryPoint);
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
return "CompileCodeTestCase{" +
|
|
"executable=" + executable +
|
|
", bci=" + bci +
|
|
'}';
|
|
}
|
|
|
|
public void deoptimize() {
|
|
WB.deoptimizeMethod(executable, isOsr);
|
|
}
|
|
|
|
public NMethod deoptimizeAndCompile() {
|
|
deoptimize();
|
|
return compile();
|
|
}
|
|
|
|
// classes which are used as "input" data in test cases
|
|
private static interface Interface {
|
|
Interface interfaceMethod();
|
|
default Long defaultOverriddenMethod(Interface[] array) {
|
|
return array == null ? 0L : array.length;
|
|
}
|
|
default int defaultMethod(Object o) {
|
|
return o != null ? o.hashCode() : 0;
|
|
}
|
|
}
|
|
|
|
private static abstract class Dummy implements Interface {
|
|
protected Dummy() {
|
|
}
|
|
|
|
private static void staticMethod() {
|
|
}
|
|
|
|
Dummy instanceMethod(int i) {
|
|
return null;
|
|
}
|
|
|
|
abstract Object abstractMethod(double d);
|
|
|
|
@Override
|
|
public Long defaultOverriddenMethod(Interface[] array) {
|
|
return 0L;
|
|
}
|
|
}
|
|
|
|
public static class DummyEx extends Dummy {
|
|
@Override
|
|
public boolean equals(Object o) {
|
|
if (this == o) {
|
|
return true;
|
|
}
|
|
if (o == null || getClass() != o.getClass()) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public int hashCode() {
|
|
return 0;
|
|
}
|
|
|
|
public DummyEx() {
|
|
}
|
|
|
|
protected Dummy instanceMethod(int i) {
|
|
if (i == 0) {
|
|
return this;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
Object abstractMethod(double d) {
|
|
return this;
|
|
}
|
|
|
|
@Override
|
|
public Interface interfaceMethod() {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
static {
|
|
Map<Class<?>, Object> map = new HashMap<>();
|
|
map.put(CompileCodeTestCase.DummyEx.class,
|
|
new CompileCodeTestCase.DummyEx());
|
|
map.put(CompileCodeTestCase.Dummy.class,
|
|
new CompileCodeTestCase.Dummy() {
|
|
@Override
|
|
public CompileCodeTestCase.Interface interfaceMethod() {
|
|
throw new AbstractMethodError();
|
|
}
|
|
|
|
@Override
|
|
Object abstractMethod(double d) {
|
|
throw new AbstractMethodError();
|
|
}
|
|
});
|
|
map.put(CompileCodeTestCase.Interface.class,
|
|
new CompileCodeTestCase.Interface() {
|
|
@Override
|
|
public CompileCodeTestCase.Interface interfaceMethod() {
|
|
throw new AbstractMethodError();
|
|
}
|
|
});
|
|
RECEIVERS = Collections.unmodifiableMap(map);
|
|
}
|
|
|
|
}
|