2014-10-24 08:22:17 -07:00
|
|
|
/*
|
2018-09-21 14:50:06 -07:00
|
|
|
* Copyright (c) 2014, 2018, Oracle and/or its affiliates. All rights reserved.
|
2014-10-24 08:22:17 -07:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2017-05-26 22:53:26 -07:00
|
|
|
import sun.invoke.util.Wrapper;
|
|
|
|
import test.java.lang.invoke.lib.Helper;
|
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
import java.io.File;
|
|
|
|
import java.io.Serializable;
|
|
|
|
import java.lang.invoke.MethodHandle;
|
|
|
|
import java.lang.invoke.MethodHandles;
|
|
|
|
import java.lang.invoke.MethodType;
|
|
|
|
import java.lang.invoke.WrongMethodTypeException;
|
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.Random;
|
2014-10-24 08:22:17 -07:00
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @bug 8060483 8066746
|
|
|
|
* @key randomness
|
2018-09-21 14:50:06 -07:00
|
|
|
* @library /test/lib /java/lang/invoke/common
|
2016-01-25 19:01:19 +00:00
|
|
|
* @modules java.base/sun.invoke.util
|
2014-10-24 08:22:17 -07:00
|
|
|
* @summary unit tests for MethodHandles.explicitCastArguments()
|
2015-08-17 12:21:34 +03:00
|
|
|
* @run main ExplicitCastArgumentsTest
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests for MethodHandles.explicitCastArguments().
|
2014-10-24 08:22:17 -07:00
|
|
|
*/
|
|
|
|
public class ExplicitCastArgumentsTest {
|
2015-08-17 12:21:34 +03:00
|
|
|
|
|
|
|
private static final boolean VERBOSE = Helper.IS_VERBOSE;
|
2014-12-09 09:22:07 -08:00
|
|
|
private static final Class<?> THIS_CLASS = ExplicitCastArgumentsTest.class;
|
2015-08-17 12:21:34 +03:00
|
|
|
private static final Random RNG = Helper.RNG;
|
|
|
|
private static final Map<Wrapper, Object> RANDOM_VALUES = new HashMap<>(9);
|
|
|
|
|
|
|
|
static {
|
|
|
|
RANDOM_VALUES.put(Wrapper.BOOLEAN, RNG.nextBoolean());
|
|
|
|
RANDOM_VALUES.put(Wrapper.BYTE, (byte) RNG.nextInt());
|
|
|
|
RANDOM_VALUES.put(Wrapper.SHORT, (short) RNG.nextInt());
|
|
|
|
RANDOM_VALUES.put(Wrapper.CHAR, (char) RNG.nextInt());
|
|
|
|
RANDOM_VALUES.put(Wrapper.INT, RNG.nextInt());
|
|
|
|
RANDOM_VALUES.put(Wrapper.LONG, RNG.nextLong());
|
|
|
|
RANDOM_VALUES.put(Wrapper.FLOAT, RNG.nextFloat());
|
|
|
|
RANDOM_VALUES.put(Wrapper.DOUBLE, RNG.nextDouble());
|
|
|
|
RANDOM_VALUES.put(Wrapper.OBJECT, new Object());
|
|
|
|
}
|
2014-10-24 08:22:17 -07:00
|
|
|
|
|
|
|
public static void main(String[] args) throws Throwable {
|
2014-12-09 09:22:07 -08:00
|
|
|
testVarargsCollector();
|
2015-08-17 12:21:34 +03:00
|
|
|
testNullRef2Prim();
|
2014-12-09 09:22:07 -08:00
|
|
|
testRef2Prim();
|
2015-08-17 12:21:34 +03:00
|
|
|
testPrim2Ref();
|
|
|
|
testPrim2Prim();
|
|
|
|
testNonBCPRef2NonBCPRef();
|
|
|
|
testBCPRef2BCPRef();
|
|
|
|
testNonBCPRef2BCPRef();
|
|
|
|
testReturnAny2Void();
|
|
|
|
testReturnVoid2Any();
|
|
|
|
testMultipleArgs();
|
2014-12-09 09:22:07 -08:00
|
|
|
System.out.println("TEST PASSED");
|
|
|
|
}
|
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Dummy method used in {@link #testVarargsCollector} test to form a method
|
|
|
|
* handle.
|
|
|
|
*
|
|
|
|
* @param args - any args
|
|
|
|
* @return - returns args
|
|
|
|
*/
|
|
|
|
public static String[] f(String... args) {
|
|
|
|
return args;
|
|
|
|
}
|
2014-12-09 09:22:07 -08:00
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Tests that MHs.explicitCastArguments does incorrect type checks for
|
|
|
|
* VarargsCollector. Bug 8066746.
|
|
|
|
*
|
|
|
|
* @throws java.lang.Throwable
|
|
|
|
*/
|
2014-12-09 09:22:07 -08:00
|
|
|
public static void testVarargsCollector() throws Throwable {
|
|
|
|
MethodType mt = MethodType.methodType(String[].class, String[].class);
|
2015-08-17 12:21:34 +03:00
|
|
|
MethodHandle mh = MethodHandles.publicLookup()
|
|
|
|
.findStatic(THIS_CLASS, "f", mt);
|
|
|
|
mh = MethodHandles.explicitCastArguments(mh,
|
|
|
|
MethodType.methodType(Object.class, Object.class));
|
|
|
|
mh.invokeWithArguments((Object) (new String[]{"str1", "str2"}));
|
2014-12-09 09:22:07 -08:00
|
|
|
}
|
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Tests that null wrapper reference is successfully converted to primitive
|
|
|
|
* types. Converted result should be zero for a primitive. Bug 8060483.
|
|
|
|
*/
|
|
|
|
public static void testNullRef2Prim() {
|
2014-10-24 08:22:17 -07:00
|
|
|
for (Wrapper from : Wrapper.values()) {
|
|
|
|
for (Wrapper to : Wrapper.values()) {
|
2015-08-17 12:21:34 +03:00
|
|
|
if (from == Wrapper.VOID || to == Wrapper.VOID) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// MHs.eCA javadoc:
|
|
|
|
// If T0 is a reference and T1 a primitive, and if the reference
|
|
|
|
// is null at runtime, a zero value is introduced.
|
|
|
|
for (TestConversionMode mode : TestConversionMode.values()) {
|
|
|
|
testConversion(mode, from.wrapperType(),
|
|
|
|
to.primitiveType(), null, to.zero(), false, null);
|
|
|
|
}
|
2014-10-24 08:22:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Tests that non-null wrapper reference is successfully converted to
|
|
|
|
* primitive types.
|
|
|
|
*/
|
|
|
|
public static void testRef2Prim() {
|
|
|
|
for (Wrapper from : Wrapper.values()) {
|
|
|
|
for (Wrapper to : Wrapper.values()) {
|
|
|
|
if (from == Wrapper.VOID || to == Wrapper.VOID
|
|
|
|
|| to == Wrapper.OBJECT) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
Object value = RANDOM_VALUES.get(from);
|
|
|
|
for (TestConversionMode mode : TestConversionMode.values()) {
|
|
|
|
if (from != Wrapper.OBJECT) {
|
|
|
|
Object convValue = to.wrap(value);
|
|
|
|
testConversion(mode, from.wrapperType(),
|
|
|
|
to.primitiveType(), value, convValue, false, null);
|
|
|
|
} else {
|
|
|
|
testConversion(mode, from.wrapperType(),
|
|
|
|
to.primitiveType(), value, null,
|
|
|
|
true, ClassCastException.class);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-10-24 08:22:17 -07:00
|
|
|
}
|
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Tests that primitive is successfully converted to wrapper reference
|
|
|
|
* types, to the Number type (if possible) and to the Object type.
|
|
|
|
*/
|
|
|
|
public static void testPrim2Ref() {
|
|
|
|
for (Wrapper from : Wrapper.values()) {
|
|
|
|
for (Wrapper to : Wrapper.values()) {
|
|
|
|
if (from == Wrapper.VOID || from == Wrapper.OBJECT
|
|
|
|
|| to == Wrapper.VOID || to == Wrapper.OBJECT) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
Object value = RANDOM_VALUES.get(from);
|
|
|
|
for (TestConversionMode mode : TestConversionMode.values()) {
|
|
|
|
if (from == to) {
|
|
|
|
testConversion(mode, from.primitiveType(),
|
|
|
|
to.wrapperType(), value, value, false, null);
|
|
|
|
} else {
|
|
|
|
testConversion(mode, from.primitiveType(),
|
|
|
|
to.wrapperType(), value, null, true, ClassCastException.class);
|
|
|
|
}
|
|
|
|
if (from != Wrapper.BOOLEAN && from != Wrapper.CHAR) {
|
|
|
|
testConversion(mode, from.primitiveType(),
|
|
|
|
Number.class, value, value, false, null);
|
|
|
|
} else {
|
|
|
|
testConversion(mode, from.primitiveType(),
|
|
|
|
Number.class, value, null,
|
|
|
|
true, ClassCastException.class);
|
|
|
|
}
|
|
|
|
testConversion(mode, from.primitiveType(),
|
|
|
|
Object.class, value, value, false, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests that primitive is successfully converted to other primitive type.
|
|
|
|
*/
|
|
|
|
public static void testPrim2Prim() {
|
|
|
|
for (Wrapper from : Wrapper.values()) {
|
|
|
|
for (Wrapper to : Wrapper.values()) {
|
|
|
|
if (from == Wrapper.VOID || to == Wrapper.VOID
|
|
|
|
|| from == Wrapper.OBJECT || to == Wrapper.OBJECT) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
Object value = RANDOM_VALUES.get(from);
|
|
|
|
Object convValue = to.wrap(value);
|
|
|
|
for (TestConversionMode mode : TestConversionMode.values()) {
|
|
|
|
testConversion(mode, from.primitiveType(),
|
|
|
|
to.primitiveType(), value, convValue, false, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dummy interface for {@link #testNonBCPRef2Ref} test.
|
|
|
|
*/
|
|
|
|
public static interface TestInterface {}
|
2014-10-24 08:22:17 -07:00
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Dummy class for {@link #testNonBCPRef2Ref} test.
|
|
|
|
*/
|
|
|
|
public static class TestSuperClass implements TestInterface {}
|
2014-10-24 08:22:17 -07:00
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Dummy class for {@link #testNonBCPRef2Ref} test.
|
|
|
|
*/
|
|
|
|
public static class TestSubClass1 extends TestSuperClass {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dummy class for {@link #testNonBCPRef2Ref} test.
|
|
|
|
*/
|
|
|
|
public static class TestSubClass2 extends TestSuperClass {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests non-bootclasspath reference to reference conversions.
|
|
|
|
*
|
|
|
|
* @throws java.lang.Throwable
|
|
|
|
*/
|
|
|
|
public static void testNonBCPRef2NonBCPRef() throws Throwable {
|
|
|
|
Class testInterface = TestInterface.class;
|
|
|
|
Class testSuperClass = TestSuperClass.class;
|
|
|
|
Class testSubClass1 = TestSubClass1.class;
|
|
|
|
Class testSubClass2 = TestSubClass2.class;
|
|
|
|
Object testSuperObj = new TestSuperClass();
|
|
|
|
Object testObj01 = new TestSubClass1();
|
|
|
|
Object testObj02 = new TestSubClass2();
|
|
|
|
Class[] parents = {testInterface, testSuperClass};
|
|
|
|
Class[] children = {testSubClass1, testSubClass2};
|
|
|
|
Object[] childInst = {testObj01, testObj02};
|
|
|
|
for (TestConversionMode mode : TestConversionMode.values()) {
|
|
|
|
for (Class parent : parents) {
|
|
|
|
for (int j = 0; j < children.length; j++) {
|
|
|
|
// Child type to parent type non-null conversion, shoud succeed
|
2017-05-26 22:53:26 -07:00
|
|
|
testConversion(mode, children[j], parent, childInst[j],
|
|
|
|
childInst[j], false, null);
|
2015-08-17 12:21:34 +03:00
|
|
|
// Child type to parent type null conversion, shoud succeed
|
2017-05-26 22:53:26 -07:00
|
|
|
testConversion(mode, children[j], parent, null,
|
|
|
|
null, false, null);
|
2015-08-17 12:21:34 +03:00
|
|
|
// Parent type to child type non-null conversion with parent
|
|
|
|
// type instance, should fail
|
2017-05-26 22:53:26 -07:00
|
|
|
testConversion(mode, parent, children[j], testSuperObj,
|
|
|
|
null, true, ClassCastException.class);
|
2015-08-17 12:21:34 +03:00
|
|
|
// Parent type to child type non-null conversion with child
|
|
|
|
// type instance, should succeed
|
2017-05-26 22:53:26 -07:00
|
|
|
testConversion(mode, parent, children[j], childInst[j],
|
|
|
|
childInst[j], false, null);
|
2015-08-17 12:21:34 +03:00
|
|
|
// Parent type to child type null conversion, should succeed
|
2017-05-26 22:53:26 -07:00
|
|
|
testConversion(mode, parent, children[j], null,
|
|
|
|
null, false, null);
|
2015-08-17 12:21:34 +03:00
|
|
|
}
|
|
|
|
// Parent type to child type non-null conversion with sibling
|
|
|
|
// type instance, should fail
|
2017-05-26 22:53:26 -07:00
|
|
|
testConversion(mode, parent, testSubClass1, testObj02,
|
|
|
|
null, true, ClassCastException.class);
|
2015-08-17 12:21:34 +03:00
|
|
|
}
|
|
|
|
// Sibling type non-null conversion, should fail
|
|
|
|
testConversion(mode, testSubClass1,
|
|
|
|
testSubClass2, testObj01, null, true,
|
|
|
|
ClassCastException.class);
|
|
|
|
// Sibling type null conversion, should succeed
|
|
|
|
testConversion(mode, testSubClass1,
|
|
|
|
testSubClass2, null, null, false, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dummy interface for {@link #testNonBCPRef2BCPRef} test.
|
|
|
|
*/
|
|
|
|
public static interface TestSerializableInterface extends Serializable {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dummy class for {@link #testNonBCPRef2BCPRef} test.
|
|
|
|
*/
|
|
|
|
public static class TestSerializableClass
|
|
|
|
implements TestSerializableInterface {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dummy class for {@link #testNonBCPRef2BCPRef} test.
|
|
|
|
*/
|
|
|
|
public static class TestFileChildClass extends File
|
|
|
|
implements TestSerializableInterface {
|
|
|
|
public TestFileChildClass(String pathname) {
|
|
|
|
super(pathname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests non-bootclasspath reference to bootclasspath reference conversions
|
|
|
|
* and vice-versa.
|
|
|
|
*
|
|
|
|
* @throws java.lang.Throwable
|
|
|
|
*/
|
|
|
|
public static void testNonBCPRef2BCPRef() throws Throwable {
|
|
|
|
Class bcpInterface = Serializable.class;
|
|
|
|
Class bcpSuperClass = File.class;
|
|
|
|
Class nonBcpInterface = TestSerializableInterface.class;
|
|
|
|
Class nonBcpSuperSiblingClass = TestSerializableClass.class;
|
|
|
|
Class nonBcpSubClass = TestFileChildClass.class;
|
|
|
|
Object bcpSuperObj = new File(".");
|
|
|
|
Object testSuperSiblingObj = new TestSerializableClass();
|
|
|
|
Object testSubObj = new TestFileChildClass(".");
|
|
|
|
Class[] parents = {bcpInterface, bcpSuperClass};
|
|
|
|
for (TestConversionMode mode : TestConversionMode.values()) {
|
|
|
|
for (Class parent : parents) {
|
|
|
|
// Child type to parent type non-null conversion, shoud succeed
|
|
|
|
testConversion(mode, nonBcpSubClass, parent, testSubObj,
|
|
|
|
testSubObj, false, null);
|
|
|
|
// Child type to parent type null conversion, shoud succeed
|
|
|
|
testConversion(mode, nonBcpSubClass, parent, null, null,
|
|
|
|
false, null);
|
|
|
|
// Parent type to child type non-null conversion with parent
|
|
|
|
// type instance, should fail
|
|
|
|
testConversion(mode, parent, nonBcpSubClass, bcpSuperObj, null,
|
|
|
|
true, ClassCastException.class);
|
|
|
|
// Parent type to child type non-null conversion with child
|
|
|
|
// type instance, should succeed
|
|
|
|
testConversion(mode, parent, nonBcpSubClass, testSubObj,
|
|
|
|
testSubObj, false, null);
|
|
|
|
// Parent type to child type null conversion, should succeed
|
|
|
|
testConversion(mode, parent, nonBcpSubClass, null, null,
|
|
|
|
false, null);
|
|
|
|
}
|
|
|
|
// Parent type to child type non-null conversion with
|
|
|
|
// super sibling type instance, should fail
|
|
|
|
testConversion(mode, bcpInterface, nonBcpSubClass,
|
|
|
|
testSuperSiblingObj, null, true, ClassCastException.class);
|
|
|
|
Class[] siblings = {nonBcpSubClass, bcpSuperClass};
|
|
|
|
for (Class sibling : siblings) {
|
|
|
|
// Non-bcp class to bcp/non-bcp sibling class non-null
|
|
|
|
// conversion with nonBcpSuperSiblingClass instance, should fail
|
|
|
|
testConversion(mode, nonBcpSuperSiblingClass, sibling,
|
|
|
|
testSuperSiblingObj, null, true, ClassCastException.class);
|
|
|
|
// Non-bcp class to bcp/non-bcp sibling class null conversion,
|
|
|
|
// should succeed
|
|
|
|
testConversion(mode, nonBcpSuperSiblingClass, sibling,
|
|
|
|
null, null, false, null);
|
|
|
|
// Non-bcp interface to bcp/non-bcp sibling class non-null
|
|
|
|
// conversion with nonBcpSubClass instance, should succeed
|
|
|
|
testConversion(mode, nonBcpInterface, sibling, testSubObj,
|
|
|
|
testSubObj, false, null);
|
|
|
|
// Non-bcp interface to bcp/non-bcp sibling class
|
|
|
|
// null conversion, should succeed
|
|
|
|
testConversion(mode, nonBcpInterface, sibling, null, null,
|
|
|
|
false, null);
|
|
|
|
// Non-bcp interface to bcp/non-bcp sibling class non-null
|
|
|
|
// conversion with nonBcpSuperSiblingClass instance, should fail
|
|
|
|
testConversion(mode, nonBcpInterface, sibling,
|
|
|
|
testSuperSiblingObj, testSubObj,
|
|
|
|
true, ClassCastException.class);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests bootclasspath reference to reference conversions.
|
|
|
|
*/
|
|
|
|
public static void testBCPRef2BCPRef() {
|
|
|
|
Class bcpInterface = CharSequence.class;
|
|
|
|
Class bcpSubClass1 = String.class;
|
|
|
|
Class bcpSubClass2 = StringBuffer.class;
|
|
|
|
Object testObj01 = new String("test");
|
|
|
|
Object testObj02 = new StringBuffer("test");
|
|
|
|
Class[] children = {bcpSubClass1, bcpSubClass2};
|
|
|
|
Object[] childInst = {testObj01, testObj02};
|
|
|
|
for (TestConversionMode mode : TestConversionMode.values()) {
|
|
|
|
for (int i = 0; i < children.length; i++) {
|
|
|
|
// Child type to parent type non-null conversion, shoud succeed
|
|
|
|
testConversion(mode, children[i], bcpInterface, childInst[i],
|
|
|
|
childInst[i], false, null);
|
|
|
|
// Child type to parent type null conversion, shoud succeed
|
|
|
|
testConversion(mode, children[i], bcpInterface, null,
|
|
|
|
null, false, null);
|
|
|
|
// Parent type to child type non-null conversion with child
|
|
|
|
// type instance, should succeed
|
|
|
|
testConversion(mode, bcpInterface,
|
|
|
|
children[i], childInst[i], childInst[i], false, null);
|
|
|
|
// Parent type to child type null conversion, should succeed
|
|
|
|
testConversion(mode, bcpInterface,
|
|
|
|
children[i], null, null, false, null);
|
|
|
|
}
|
|
|
|
// Sibling type non-null conversion, should fail
|
|
|
|
testConversion(mode, bcpSubClass1,
|
|
|
|
bcpSubClass2, testObj01, null, true,
|
|
|
|
ClassCastException.class);
|
|
|
|
// Sibling type null conversion, should succeed
|
|
|
|
testConversion(mode, bcpSubClass1,
|
|
|
|
bcpSubClass2, null, null, false, null);
|
|
|
|
// Parent type to child type non-null conversion with sibling
|
|
|
|
// type instance, should fail
|
|
|
|
testConversion(mode, bcpInterface, bcpSubClass1, testObj02,
|
|
|
|
null, true, ClassCastException.class);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dummy method used in {@link #testReturnAny2Void} and
|
|
|
|
* {@link #testReturnVoid2Any} tests to form a method handle.
|
|
|
|
*/
|
|
|
|
public static void retVoid() {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests that non-null any return is successfully converted to non-type
|
|
|
|
* void.
|
|
|
|
*/
|
|
|
|
public static void testReturnAny2Void() {
|
|
|
|
for (Wrapper from : Wrapper.values()) {
|
|
|
|
testConversion(TestConversionMode.RETURN_VALUE, from.wrapperType(),
|
|
|
|
void.class, RANDOM_VALUES.get(from),
|
|
|
|
null, false, null);
|
|
|
|
testConversion(TestConversionMode.RETURN_VALUE, from.primitiveType(),
|
|
|
|
void.class, RANDOM_VALUES.get(from),
|
|
|
|
null, false, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests that void return is successfully converted to primitive and
|
|
|
|
* reference. Result should be zero for primitives and null for references.
|
|
|
|
*/
|
|
|
|
public static void testReturnVoid2Any() {
|
|
|
|
for (Wrapper to : Wrapper.values()) {
|
|
|
|
testConversion(TestConversionMode.RETURN_VALUE, void.class,
|
|
|
|
to.primitiveType(), null,
|
|
|
|
to.zero(), false, null);
|
|
|
|
testConversion(TestConversionMode.RETURN_VALUE, void.class,
|
|
|
|
to.wrapperType(), null,
|
|
|
|
null, false, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void checkForWrongMethodTypeException(MethodHandle mh, MethodType mt) {
|
2014-10-24 08:22:17 -07:00
|
|
|
try {
|
2015-08-17 12:21:34 +03:00
|
|
|
MethodHandles.explicitCastArguments(mh, mt);
|
|
|
|
throw new AssertionError("Expected WrongMethodTypeException is not thrown");
|
|
|
|
} catch (WrongMethodTypeException wmte) {
|
|
|
|
if (VERBOSE) {
|
|
|
|
System.out.printf("Expected exception %s: %s\n",
|
|
|
|
wmte.getClass(), wmte.getMessage());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-10-24 08:22:17 -07:00
|
|
|
|
2015-08-17 12:21:34 +03:00
|
|
|
/**
|
|
|
|
* Tests that MHs.eCA method works correctly with MHs with multiple arguments.
|
|
|
|
* @throws Throwable
|
|
|
|
*/
|
|
|
|
public static void testMultipleArgs() throws Throwable {
|
|
|
|
int arity = 1 + RNG.nextInt(Helper.MAX_ARITY / 2 - 2);
|
|
|
|
int arityMinus = RNG.nextInt(arity);
|
|
|
|
int arityPlus = arity + RNG.nextInt(Helper.MAX_ARITY / 2 - arity) + 1;
|
|
|
|
MethodType mType = Helper.randomMethodTypeGenerator(arity);
|
|
|
|
MethodType mTypeNew = Helper.randomMethodTypeGenerator(arity);
|
|
|
|
MethodType mTypeNewMinus = Helper.randomMethodTypeGenerator(arityMinus);
|
|
|
|
MethodType mTypeNewPlus = Helper.randomMethodTypeGenerator(arityPlus);
|
|
|
|
Class<?> rType = mType.returnType();
|
|
|
|
MethodHandle original;
|
|
|
|
if (rType.equals(void.class)) {
|
|
|
|
MethodType mt = MethodType.methodType(void.class);
|
|
|
|
original = MethodHandles.publicLookup()
|
|
|
|
.findStatic(THIS_CLASS, "retVoid", mt);
|
|
|
|
} else {
|
|
|
|
Object rValue = Helper.castToWrapper(1, rType);
|
|
|
|
original = MethodHandles.constant(rType, rValue);
|
|
|
|
}
|
|
|
|
original = Helper.addTrailingArgs(original, arity, mType.parameterList());
|
|
|
|
MethodHandle target = MethodHandles
|
|
|
|
.explicitCastArguments(original, mTypeNew);
|
|
|
|
Object[] parList = Helper.randomArgs(mTypeNew.parameterList());
|
|
|
|
for (int i = 0; i < parList.length; i++) {
|
|
|
|
if (parList[i] instanceof String) {
|
2022-04-07 14:55:05 +00:00
|
|
|
parList[i] = null; //getting rid of Strings produced by randomArgs
|
2015-08-17 12:21:34 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
target.invokeWithArguments(parList);
|
|
|
|
checkForWrongMethodTypeException(original, mTypeNewMinus);
|
|
|
|
checkForWrongMethodTypeException(original, mTypeNewPlus);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enumeration of test conversion modes.
|
|
|
|
*/
|
|
|
|
public enum TestConversionMode {
|
|
|
|
RETURN_VALUE,
|
|
|
|
ARGUMENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests type and value conversion. Comparing with the given expected result.
|
|
|
|
*
|
|
|
|
* @param mode - test conversion mode. See {@link #TestConversionMode}.
|
|
|
|
* @param from - source type.
|
|
|
|
* @param to - destination type.
|
|
|
|
* @param param - value to be converted.
|
|
|
|
* @param expectedResult - expected value after conversion.
|
|
|
|
* @param failureExpected - true if conversion failure expected.
|
|
|
|
* @param expectedException - expected exception class if
|
|
|
|
* {@code failureExpected} is true.
|
|
|
|
*/
|
|
|
|
public static void testConversion(TestConversionMode mode,
|
|
|
|
Class<?> from, Class<?> to, Object param,
|
|
|
|
Object expectedResult, boolean failureExpected,
|
|
|
|
Class<? extends Throwable> expectedException) {
|
|
|
|
if (VERBOSE) {
|
|
|
|
System.out.printf("Testing return value conversion: "
|
|
|
|
+ "%-10s => %-10s: %5s: ", from.getSimpleName(),
|
|
|
|
to.getSimpleName(), param);
|
|
|
|
}
|
|
|
|
MethodHandle original = null;
|
|
|
|
MethodType newType = null;
|
|
|
|
switch (mode) {
|
|
|
|
case RETURN_VALUE:
|
|
|
|
if (from.equals(void.class)) {
|
|
|
|
MethodType mt = MethodType.methodType(void.class);
|
|
|
|
try {
|
|
|
|
original = MethodHandles.publicLookup()
|
|
|
|
.findStatic(THIS_CLASS, "retVoid", mt);
|
|
|
|
} catch (NoSuchMethodException | IllegalAccessException ex) {
|
|
|
|
throw new Error("Unexpected issue", ex);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
original = MethodHandles.constant(from, param);
|
|
|
|
}
|
|
|
|
newType = original.type().changeReturnType(to);
|
|
|
|
break;
|
|
|
|
case ARGUMENT:
|
|
|
|
if (from.equals(void.class) || to.equals(void.class)) {
|
|
|
|
throw new Error("Test issue: argument conversion does not"
|
|
|
|
+ " work with non-type void");
|
|
|
|
}
|
|
|
|
original = MethodHandles.identity(to);
|
|
|
|
newType = original.type().changeParameterType(0, from);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
String msg = String.format("Test issue: unknown test"
|
|
|
|
+ " convertion mode %s.", mode.name());
|
|
|
|
throw new Error(msg);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
MethodHandle target = MethodHandles
|
|
|
|
.explicitCastArguments(original, newType);
|
|
|
|
Object result;
|
|
|
|
switch (mode) {
|
|
|
|
case RETURN_VALUE:
|
|
|
|
result = target.invokeWithArguments();
|
|
|
|
break;
|
|
|
|
case ARGUMENT:
|
|
|
|
result = target.invokeWithArguments(param);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
String msg = String.format("Test issue: unknown test"
|
|
|
|
+ " convertion mode %s.", mode.name());
|
|
|
|
throw new Error(msg);
|
|
|
|
}
|
|
|
|
if (!failureExpected
|
|
|
|
&& (expectedResult != null && !expectedResult.equals(result)
|
|
|
|
|| expectedResult == null && result != null)) {
|
|
|
|
String msg = String.format("Conversion result %s is not equal"
|
|
|
|
+ " to the expected result %10s",
|
|
|
|
result, expectedResult);
|
|
|
|
throw new AssertionError(msg);
|
|
|
|
}
|
2014-10-24 08:22:17 -07:00
|
|
|
if (VERBOSE) {
|
|
|
|
String resultStr;
|
|
|
|
if (result != null) {
|
2015-08-17 12:21:34 +03:00
|
|
|
resultStr = String.format("Converted value and type are"
|
|
|
|
+ " %10s (%10s)", "'" + result + "'",
|
|
|
|
result.getClass().getSimpleName());
|
2014-10-24 08:22:17 -07:00
|
|
|
} else {
|
2015-08-17 12:21:34 +03:00
|
|
|
resultStr = String.format("Converted value is %10s", result);
|
2014-10-24 08:22:17 -07:00
|
|
|
}
|
|
|
|
System.out.println(resultStr);
|
|
|
|
}
|
|
|
|
if (failureExpected) {
|
2015-08-17 12:21:34 +03:00
|
|
|
String msg = String.format("No exception thrown while testing"
|
|
|
|
+ " return value conversion: %10s => %10s;"
|
|
|
|
+ " parameter: %10s",
|
|
|
|
from, to, param);
|
2014-10-24 08:22:17 -07:00
|
|
|
throw new AssertionError(msg);
|
|
|
|
}
|
|
|
|
} catch (AssertionError e) {
|
|
|
|
throw e; // report test failure
|
|
|
|
} catch (Throwable e) {
|
2015-08-17 12:21:34 +03:00
|
|
|
if (VERBOSE) {
|
|
|
|
System.out.printf("%s: %s\n", e.getClass(), e.getMessage());
|
|
|
|
}
|
|
|
|
if (!failureExpected || !e.getClass().equals(expectedException)) {
|
|
|
|
String msg = String.format("Unexpected exception was thrown"
|
|
|
|
+ " while testing return value conversion:"
|
|
|
|
+ " %s => %s; parameter: %s", from, to, param);
|
2014-10-24 08:22:17 -07:00
|
|
|
throw new AssertionError(msg, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|