170 lines
7.7 KiB
Java
170 lines
7.7 KiB
Java
|
/*
|
||
|
* Copyright (c) 2014, 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.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* @test
|
||
|
* @bug 8035776
|
||
|
* @summary metafactory should fail if arities are mismatched
|
||
|
*/
|
||
|
import java.lang.invoke.*;
|
||
|
import java.util.Arrays;
|
||
|
import static java.lang.invoke.MethodType.methodType;
|
||
|
|
||
|
public class MetafactoryArityTest {
|
||
|
|
||
|
public interface I {}
|
||
|
public static class C { public static String m(int arg) { return ""; } }
|
||
|
|
||
|
static final MethodHandles.Lookup lookup = MethodHandles.lookup();
|
||
|
static final Class<?>[] capInt = { int.class };
|
||
|
static final MethodHandle C_m;
|
||
|
static {
|
||
|
try { C_m = lookup.findStatic(C.class, "m", methodType(String.class, int.class)); }
|
||
|
catch (NoSuchMethodException | IllegalAccessException e) { throw new RuntimeException(e); }
|
||
|
}
|
||
|
|
||
|
public static void main(String... args) {
|
||
|
MethodType unary = methodType(String.class, int.class);
|
||
|
MethodType nullary = methodType(String.class);
|
||
|
MethodType binary = methodType(String.class, int.class, int.class);
|
||
|
MethodType unaryCS = methodType(CharSequence.class, int.class);
|
||
|
MethodType nullaryCS = methodType(CharSequence.class);
|
||
|
MethodType binaryCS = methodType(CharSequence.class, int.class, int.class);
|
||
|
MethodType unaryObj = methodType(Object.class, int.class);
|
||
|
MethodType nullaryObj = methodType(Object.class);
|
||
|
MethodType binaryObj = methodType(Object.class, int.class, int.class);
|
||
|
|
||
|
test(true, C_m, unary, unary);
|
||
|
test(false, C_m, unary, nullary);
|
||
|
test(false, C_m, nullary, unary);
|
||
|
test(false, C_m, unary, binary);
|
||
|
test(false, C_m, binary, unary);
|
||
|
|
||
|
testBridge(true, C_m, unary, unary, unaryCS);
|
||
|
testBridge(false, C_m, unary, unary, nullaryCS);
|
||
|
testBridge(false, C_m, unary, unary, binaryCS);
|
||
|
|
||
|
testBridge(true, C_m, unary, unary, unaryCS, unaryObj);
|
||
|
testBridge(false, C_m, unary, unary, unaryCS, nullaryObj);
|
||
|
testBridge(false, C_m, unary, unary, unaryCS, binaryObj);
|
||
|
|
||
|
testCapture(true, C_m, capInt, nullary, nullary);
|
||
|
testCapture(false, C_m, capInt, binary, binary);
|
||
|
testCapture(false, C_m, capInt, nullary, unary);
|
||
|
testCapture(false, C_m, capInt, nullary, binary);
|
||
|
testCapture(false, C_m, capInt, unary, nullary);
|
||
|
testCapture(false, C_m, capInt, unary, binary);
|
||
|
|
||
|
testCaptureBridge(true, C_m, capInt, nullary, nullary, nullaryCS);
|
||
|
testCaptureBridge(false, C_m, capInt, unary, unary, unaryCS);
|
||
|
testCaptureBridge(false, C_m, capInt, nullary, nullary, unaryCS);
|
||
|
testCaptureBridge(false, C_m, capInt, nullary, nullary, binaryCS);
|
||
|
|
||
|
testCaptureBridge(true, C_m, capInt, nullary, nullary, nullaryCS, nullaryObj);
|
||
|
testCaptureBridge(false, C_m, capInt, unary, unary, unaryCS, unaryObj);
|
||
|
testCaptureBridge(false, C_m, capInt, nullary, nullary, nullaryCS, unaryObj);
|
||
|
testCaptureBridge(false, C_m, capInt, nullary, nullary, nullaryCS, binaryObj);
|
||
|
}
|
||
|
|
||
|
static void test(boolean correct, MethodHandle mh, MethodType instMT, MethodType samMT) {
|
||
|
tryMetafactory(correct, mh, new Class<?>[]{}, instMT, samMT);
|
||
|
tryAltMetafactory(correct, mh, new Class<?>[]{}, instMT, samMT);
|
||
|
}
|
||
|
|
||
|
static void testBridge(boolean correct, MethodHandle mh, MethodType instMT, MethodType samMT, MethodType... bridgeMTs) {
|
||
|
tryAltMetafactory(correct, mh, new Class<?>[]{}, instMT, samMT, bridgeMTs);
|
||
|
}
|
||
|
|
||
|
static void testCapture(boolean correct, MethodHandle mh, Class<?>[] captured, MethodType instMT, MethodType samMT) {
|
||
|
tryMetafactory(correct, mh, captured, instMT, samMT);
|
||
|
tryAltMetafactory(correct, mh, captured, instMT, samMT);
|
||
|
}
|
||
|
|
||
|
static void testCaptureBridge(boolean correct, MethodHandle mh, Class<?>[] captured,
|
||
|
MethodType instMT, MethodType samMT, MethodType... bridgeMTs) {
|
||
|
tryAltMetafactory(correct, mh, captured, instMT, samMT, bridgeMTs);
|
||
|
}
|
||
|
|
||
|
static void tryMetafactory(boolean correct, MethodHandle mh, Class<?>[] captured,
|
||
|
MethodType instMT, MethodType samMT) {
|
||
|
try {
|
||
|
LambdaMetafactory.metafactory(lookup, "run", methodType(I.class, captured),
|
||
|
samMT, mh, instMT);
|
||
|
if (!correct) {
|
||
|
throw new AssertionError("Uncaught linkage error:" +
|
||
|
" impl=" + mh +
|
||
|
", captured=" + Arrays.toString(captured) +
|
||
|
", inst=" + instMT +
|
||
|
", sam=" + samMT);
|
||
|
}
|
||
|
}
|
||
|
catch (LambdaConversionException e) {
|
||
|
if (correct) {
|
||
|
throw new AssertionError("Unexpected linkage error:" +
|
||
|
" e=" + e +
|
||
|
", impl=" + mh +
|
||
|
", captured=" + Arrays.toString(captured) +
|
||
|
", inst=" + instMT +
|
||
|
", sam=" + samMT);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void tryAltMetafactory(boolean correct, MethodHandle mh, Class<?>[] captured,
|
||
|
MethodType instMT, MethodType samMT, MethodType... bridgeMTs) {
|
||
|
boolean bridge = bridgeMTs.length > 0;
|
||
|
Object[] args = new Object[bridge ? 5+bridgeMTs.length : 4];
|
||
|
args[0] = samMT;
|
||
|
args[1] = mh;
|
||
|
args[2] = instMT;
|
||
|
args[3] = bridge ? LambdaMetafactory.FLAG_BRIDGES : 0;
|
||
|
if (bridge) {
|
||
|
args[4] = bridgeMTs.length;
|
||
|
for (int i = 0; i < bridgeMTs.length; i++) args[5+i] = bridgeMTs[i];
|
||
|
}
|
||
|
try {
|
||
|
LambdaMetafactory.altMetafactory(lookup, "run", methodType(I.class, captured), args);
|
||
|
if (!correct) {
|
||
|
throw new AssertionError("Uncaught linkage error:" +
|
||
|
" impl=" + mh +
|
||
|
", captured=" + Arrays.toString(captured) +
|
||
|
", inst=" + instMT +
|
||
|
", sam=" + samMT +
|
||
|
", bridges=" + Arrays.toString(bridgeMTs));
|
||
|
}
|
||
|
}
|
||
|
catch (LambdaConversionException e) {
|
||
|
if (correct) {
|
||
|
throw new AssertionError("Unexpected linkage error:" +
|
||
|
" e=" + e +
|
||
|
", impl=" + mh +
|
||
|
", captured=" + Arrays.toString(captured) +
|
||
|
", inst=" + instMT +
|
||
|
", sam=" + samMT +
|
||
|
", bridges=" + Arrays.toString(bridgeMTs));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|