2b00ac0d02
Reviewed-by: ihse, mchung, vromero
385 lines
15 KiB
Java
385 lines
15 KiB
Java
/*
|
|
* Copyright (c) 2021, 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.
|
|
*/
|
|
|
|
import java.io.Serializable;
|
|
import java.lang.Enum.EnumDesc;
|
|
import java.lang.constant.ClassDesc;
|
|
import java.lang.constant.ConstantDescs;
|
|
import java.lang.constant.MethodTypeDesc;
|
|
import java.lang.invoke.CallSite;
|
|
import java.lang.invoke.MethodHandle;
|
|
import java.lang.invoke.MethodHandles;
|
|
import java.lang.invoke.MethodType;
|
|
import java.lang.reflect.AccessFlag;
|
|
import java.lang.runtime.SwitchBootstraps;
|
|
import java.util.concurrent.atomic.AtomicBoolean;
|
|
import java.lang.classfile.ClassFile;
|
|
|
|
import org.testng.annotations.Test;
|
|
|
|
import static org.testng.Assert.assertEquals;
|
|
import static org.testng.Assert.assertFalse;
|
|
import static org.testng.Assert.assertTrue;
|
|
import static org.testng.Assert.fail;
|
|
|
|
/**
|
|
* @test
|
|
* @bug 8318144
|
|
* @enablePreview
|
|
* @compile SwitchBootstrapsTest.java
|
|
* @run testng/othervm SwitchBootstrapsTest
|
|
*/
|
|
@Test
|
|
public class SwitchBootstrapsTest {
|
|
|
|
public static final MethodHandle BSM_TYPE_SWITCH;
|
|
public static final MethodHandle BSM_ENUM_SWITCH;
|
|
|
|
static {
|
|
try {
|
|
BSM_TYPE_SWITCH = MethodHandles.lookup().findStatic(SwitchBootstraps.class, "typeSwitch",
|
|
MethodType.methodType(CallSite.class, MethodHandles.Lookup.class, String.class, MethodType.class, Object[].class));
|
|
BSM_ENUM_SWITCH = MethodHandles.lookup().findStatic(SwitchBootstraps.class, "enumSwitch",
|
|
MethodType.methodType(CallSite.class, MethodHandles.Lookup.class, String.class, MethodType.class, Object[].class));
|
|
}
|
|
catch (ReflectiveOperationException e) {
|
|
throw new AssertionError("Should not happen", e);
|
|
}
|
|
}
|
|
|
|
private void testType(Object target, int start, int result, Object... labels) throws Throwable {
|
|
MethodType switchType = MethodType.methodType(int.class, Object.class, int.class);
|
|
MethodHandle indy = ((CallSite) BSM_TYPE_SWITCH.invoke(MethodHandles.lookup(), "", switchType, labels)).dynamicInvoker();
|
|
assertEquals((int) indy.invoke(target, start), result);
|
|
assertEquals(-1, (int) indy.invoke(null, start));
|
|
}
|
|
|
|
private void testEnum(Enum<?> target, int start, int result, Object... labels) throws Throwable {
|
|
testEnum(target.getClass(), target, start, result, labels);
|
|
}
|
|
|
|
private void testEnum(Class<?> targetClass, Enum<?> target, int start, int result, Object... labels) throws Throwable {
|
|
MethodType switchType = MethodType.methodType(int.class, targetClass, int.class);
|
|
MethodHandle indy = ((CallSite) BSM_ENUM_SWITCH.invoke(MethodHandles.lookup(), "", switchType, labels)).dynamicInvoker();
|
|
assertEquals((int) indy.invoke(target, start), result);
|
|
assertEquals(-1, (int) indy.invoke(null, start));
|
|
}
|
|
|
|
public enum E1 {
|
|
A,
|
|
B;
|
|
}
|
|
|
|
public enum E2 {
|
|
C;
|
|
}
|
|
|
|
public void testTypes() throws Throwable {
|
|
testType("", 0, 0, String.class, Object.class);
|
|
testType("", 0, 0, Object.class);
|
|
testType("", 0, 1, Integer.class);
|
|
testType("", 0, 1, Integer.class, Serializable.class);
|
|
testType(E1.A, 0, 0, E1.class, Object.class);
|
|
testType(E2.C, 0, 1, E1.class, Object.class);
|
|
testType(new Serializable() { }, 0, 1, Comparable.class, Serializable.class);
|
|
testType("", 0, 0, "", String.class);
|
|
testType("", 1, 1, "", String.class);
|
|
testType("a", 0, 1, "", String.class);
|
|
testType(1, 0, 0, 1, Integer.class);
|
|
testType(2, 0, 1, 1, Integer.class);
|
|
testType(Byte.valueOf((byte) 1), 0, 0, 1, Integer.class);
|
|
testType(Short.valueOf((short) 1), 0, 0, 1, Integer.class);
|
|
testType(Character.valueOf((char) 1), 0, 0, 1, Integer.class);
|
|
testType(Integer.valueOf((int) 1), 0, 0, 1, Integer.class);
|
|
try {
|
|
testType(1, 0, 1, 1.0, Integer.class);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK
|
|
}
|
|
testType("", 0, 0, String.class, String.class, String.class, String.class, String.class);
|
|
testType("", 1, 1, String.class, String.class, String.class, String.class, String.class);
|
|
testType("", 2, 2, String.class, String.class, String.class, String.class, String.class);
|
|
testType("", 3, 3, String.class, String.class, String.class, String.class, String.class);
|
|
testType("", 3, 3, String.class, String.class, String.class, String.class, String.class);
|
|
testType("", 4, 4, String.class, String.class, String.class, String.class, String.class);
|
|
testType("", 0, 0);
|
|
}
|
|
|
|
public void testEnums() throws Throwable {
|
|
testEnum(E1.A, 0, 2, "B", "C", "A", E1.class);
|
|
testEnum(E1.B, 0, 0, "B", "C", "A", E1.class);
|
|
testEnum(E1.B, 1, 3, "B", "C", "A", E1.class);
|
|
try {
|
|
testEnum(E1.B, 1, 3, "B", "C", "A", E2.class);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK
|
|
}
|
|
try {
|
|
testEnum(E1.B, 1, 3, "B", "C", "A", String.class);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK
|
|
}
|
|
testEnum(E1.B, 0, 0, "B", "A");
|
|
testEnum(E1.A, 0, 1, "B", "A");
|
|
testEnum(E1.A, 0, 0, "A", "A", "B");
|
|
testEnum(E1.A, 1, 1, "A", "A", "B");
|
|
testEnum(E1.A, 2, 3, "A", "A", "B");
|
|
testEnum(E1.A, 0, 0);
|
|
}
|
|
|
|
public void testEnumsWithConstants() throws Throwable {
|
|
enum E {
|
|
A {},
|
|
B {},
|
|
C {}
|
|
}
|
|
ClassDesc eDesc = E.class.describeConstable().get();
|
|
Object[] typeParams = new Object[] {
|
|
EnumDesc.of(eDesc, "A"),
|
|
EnumDesc.of(eDesc, "B"),
|
|
EnumDesc.of(eDesc, "C"),
|
|
"a",
|
|
String.class
|
|
};
|
|
testType(E.A, 0, 0, typeParams);
|
|
testType(E.B, 0, 1, typeParams);
|
|
testType(E.C, 0, 2, typeParams);
|
|
testType("a", 0, 3, typeParams);
|
|
testType("x", 0, 4, typeParams);
|
|
testType('a', 0, 5, typeParams);
|
|
testEnum(E.class, E.A, 0, 0, "A", "B", "C");
|
|
testEnum(E.class, E.B, 0, 1, "A", "B", "C");
|
|
testEnum(E.class, E.C, 0, 2, "A", "B", "C");
|
|
}
|
|
|
|
public void testWrongSwitchTypes() throws Throwable {
|
|
MethodType[] switchTypes = new MethodType[] {
|
|
MethodType.methodType(int.class, Object.class),
|
|
MethodType.methodType(int.class, double.class, int.class),
|
|
MethodType.methodType(int.class, Object.class, Integer.class)
|
|
};
|
|
for (MethodType switchType : switchTypes) {
|
|
try {
|
|
BSM_TYPE_SWITCH.invoke(MethodHandles.lookup(), "", switchType);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK, expected
|
|
}
|
|
}
|
|
MethodType[] enumSwitchTypes = new MethodType[] {
|
|
MethodType.methodType(int.class, Enum.class),
|
|
MethodType.methodType(int.class, Object.class, int.class),
|
|
MethodType.methodType(int.class, double.class, int.class),
|
|
MethodType.methodType(int.class, Enum.class, Integer.class)
|
|
};
|
|
for (MethodType enumSwitchType : enumSwitchTypes) {
|
|
try {
|
|
BSM_ENUM_SWITCH.invoke(MethodHandles.lookup(), "", enumSwitchType);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK, expected
|
|
}
|
|
}
|
|
}
|
|
|
|
public void testSwitchLabelTypes() throws Throwable {
|
|
enum E {A}
|
|
try {
|
|
testType(E.A, 0, -1, E.A);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK, expected
|
|
}
|
|
}
|
|
|
|
public void testSwitchQualifiedEnum() throws Throwable {
|
|
enum E {A, B, C}
|
|
Object[] labels = new Object[] {
|
|
EnumDesc.of(ClassDesc.of(E.class.getName()), "A"),
|
|
EnumDesc.of(ClassDesc.of(E.class.getName()), "B"),
|
|
EnumDesc.of(ClassDesc.of(E.class.getName()), "C")
|
|
};
|
|
testType(E.A, 0, 0, labels);
|
|
testType(E.B, 0, 1, labels);
|
|
testType(E.C, 0, 2, labels);
|
|
}
|
|
|
|
public void testNullLabels() throws Throwable {
|
|
MethodType switchType = MethodType.methodType(int.class, Object.class, int.class);
|
|
try {
|
|
BSM_TYPE_SWITCH.invoke(MethodHandles.lookup(), "", switchType, (Object[]) null);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (NullPointerException ex) {
|
|
//OK
|
|
}
|
|
try {
|
|
BSM_TYPE_SWITCH.invoke(MethodHandles.lookup(), "", switchType,
|
|
new Object[] {1, null, String.class});
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK
|
|
}
|
|
MethodType enumSwitchType = MethodType.methodType(int.class, E1.class, int.class);
|
|
try {
|
|
BSM_TYPE_SWITCH.invoke(MethodHandles.lookup(), "", enumSwitchType, (Object[]) null);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (NullPointerException ex) {
|
|
//OK
|
|
}
|
|
try {
|
|
BSM_TYPE_SWITCH.invoke(MethodHandles.lookup(), "", enumSwitchType,
|
|
new Object[] {1, null, String.class});
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK
|
|
}
|
|
}
|
|
|
|
private static AtomicBoolean enumInitialized = new AtomicBoolean();
|
|
public void testEnumInitialization1() throws Throwable {
|
|
enumInitialized.set(false);
|
|
|
|
enum E {
|
|
A;
|
|
|
|
static {
|
|
enumInitialized.set(true);
|
|
}
|
|
}
|
|
|
|
MethodType enumSwitchType = MethodType.methodType(int.class, E.class, int.class);
|
|
|
|
CallSite invocation = (CallSite) BSM_ENUM_SWITCH.invoke(MethodHandles.lookup(), "", enumSwitchType, new Object[] {"A"});
|
|
assertFalse(enumInitialized.get());
|
|
assertEquals(invocation.dynamicInvoker().invoke(null, 0), -1);
|
|
assertFalse(enumInitialized.get());
|
|
E e = E.A;
|
|
assertTrue(enumInitialized.get());
|
|
assertEquals(invocation.dynamicInvoker().invoke(e, 0), 0);
|
|
}
|
|
|
|
public void testEnumInitialization2() throws Throwable {
|
|
enumInitialized.set(false);
|
|
|
|
enum E {
|
|
A;
|
|
|
|
static {
|
|
enumInitialized.set(true);
|
|
}
|
|
}
|
|
|
|
MethodType switchType = MethodType.methodType(int.class, Object.class, int.class);
|
|
Object[] labels = new Object[] {
|
|
EnumDesc.of(ClassDesc.of(E.class.getName()), "A"),
|
|
"test"
|
|
};
|
|
CallSite invocation = (CallSite) BSM_TYPE_SWITCH.invoke(MethodHandles.lookup(), "", switchType, labels);
|
|
assertFalse(enumInitialized.get());
|
|
assertEquals(invocation.dynamicInvoker().invoke(null, 0), -1);
|
|
assertFalse(enumInitialized.get());
|
|
assertEquals(invocation.dynamicInvoker().invoke("test", 0), 1);
|
|
assertFalse(enumInitialized.get());
|
|
E e = E.A;
|
|
assertTrue(enumInitialized.get());
|
|
assertEquals(invocation.dynamicInvoker().invoke(e, 0), 0);
|
|
}
|
|
|
|
public void testIncorrectEnumLabels() throws Throwable {
|
|
try {
|
|
testEnum(E1.B, 0, -1, "B", 1);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK
|
|
}
|
|
try {
|
|
testEnum(E1.B, 0, -1, "B", null);
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IllegalArgumentException ex) {
|
|
//OK
|
|
}
|
|
}
|
|
|
|
public void testIncorrectEnumStartIndex() throws Throwable {
|
|
try {
|
|
testEnum(E1.B, -1, -1, "B");
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IndexOutOfBoundsException ex) {
|
|
//OK
|
|
}
|
|
try {
|
|
testEnum(E1.B, 2, -1, "B");
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IndexOutOfBoundsException ex) {
|
|
//OK
|
|
}
|
|
}
|
|
|
|
public void testIncorrectTypeStartIndex() throws Throwable {
|
|
try {
|
|
testType("", -1, -1, "");
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IndexOutOfBoundsException ex) {
|
|
//OK
|
|
}
|
|
try {
|
|
testType("", 2, -1, "");
|
|
fail("Didn't get the expected exception.");
|
|
} catch (IndexOutOfBoundsException ex) {
|
|
//OK
|
|
}
|
|
}
|
|
|
|
public void testHiddenClassAsCaseLabel() throws Throwable {
|
|
MethodHandles.Lookup lookup = MethodHandles.lookup();
|
|
byte[] classBytes = createClass();
|
|
Class<?> classA = lookup.defineHiddenClass(classBytes, false).lookupClass();
|
|
Class<?> classB = lookup.defineHiddenClass(classBytes, false).lookupClass();
|
|
Object[] labels = new Object[] {
|
|
classA,
|
|
classB,
|
|
};
|
|
testType(classA.getConstructor().newInstance(), 0, 0, labels);
|
|
testType(classB.getConstructor().newInstance(), 0, 1, labels);
|
|
}
|
|
|
|
private static byte[] createClass() {
|
|
return ClassFile.of().build(ClassDesc.of("C"), clb -> {
|
|
clb.withFlags(AccessFlag.SYNTHETIC)
|
|
.withMethodBody("<init>",
|
|
MethodTypeDesc.of(ConstantDescs.CD_void),
|
|
ClassFile.ACC_PUBLIC,
|
|
cb -> {
|
|
cb.aload(0);
|
|
cb.invokespecial(ConstantDescs.CD_Object,
|
|
"<init>",
|
|
MethodTypeDesc.of(ConstantDescs.CD_void));
|
|
cb.return_();
|
|
});
|
|
});
|
|
}
|
|
}
|