提交 94b75cc1 编写于 作者: R rfield

8028739: javac generates incorrect descriptor for MethodHandle::invoke

Summary: introduce special handling for signature polymorphic methods
Reviewed-by: jjg
上级 4f4e3796
...@@ -357,9 +357,11 @@ public class LambdaToMethod extends TreeTranslator { ...@@ -357,9 +357,11 @@ public class LambdaToMethod extends TreeTranslator {
//first determine the method symbol to be used to generate the sam instance //first determine the method symbol to be used to generate the sam instance
//this is either the method reference symbol, or the bridged reference symbol //this is either the method reference symbol, or the bridged reference symbol
Symbol refSym = localContext.needsBridge() ? Symbol refSym = localContext.needsBridge()
localContext.bridgeSym : ? localContext.bridgeSym
tree.sym; : localContext.isSignaturePolymorphic()
? localContext.sigPolySym
: tree.sym;
//build the bridge method, if needed //build the bridge method, if needed
if (localContext.needsBridge()) { if (localContext.needsBridge()) {
...@@ -1995,6 +1997,7 @@ public class LambdaToMethod extends TreeTranslator { ...@@ -1995,6 +1997,7 @@ public class LambdaToMethod extends TreeTranslator {
final boolean isSuper; final boolean isSuper;
final Symbol bridgeSym; final Symbol bridgeSym;
final Symbol sigPolySym;
ReferenceTranslationContext(JCMemberReference tree) { ReferenceTranslationContext(JCMemberReference tree) {
super(tree); super(tree);
...@@ -2004,6 +2007,12 @@ public class LambdaToMethod extends TreeTranslator { ...@@ -2004,6 +2007,12 @@ public class LambdaToMethod extends TreeTranslator {
referenceBridgeName(), null, referenceBridgeName(), null,
owner.enclClass()) owner.enclClass())
: null; : null;
this.sigPolySym = isSignaturePolymorphic()
? makePrivateSyntheticMethod(tree.sym.flags(),
tree.sym.name,
bridgedRefSig(),
tree.sym.enclClass())
: null;
if (dumpLambdaToMethodStats) { if (dumpLambdaToMethodStats) {
String key = bridgeSym == null ? String key = bridgeSym == null ?
"mref.stat" : "mref.stat.1"; "mref.stat" : "mref.stat.1";
...@@ -2105,6 +2114,15 @@ public class LambdaToMethod extends TreeTranslator { ...@@ -2105,6 +2114,15 @@ public class LambdaToMethod extends TreeTranslator {
types.erasure(owner.enclClass().asType())); types.erasure(owner.enclClass().asType()));
} }
/**
* Signature polymorphic methods need special handling.
* e.g. MethodHandle.invoke() MethodHandle.invokeExact()
*/
final boolean isSignaturePolymorphic() {
return tree.sym.kind == MTH &&
types.isSignaturePolymorphic((MethodSymbol)tree.sym);
}
/** /**
* Does this reference needs a bridge (i.e. var args need to be * Does this reference needs a bridge (i.e. var args need to be
* expanded or "super" is used) * expanded or "super" is used)
......
/*
* Copyright (c) 2013, 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. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* 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 8028739
* @summary javac generates incorrect descriptor for MethodHandle::invoke
* @run testng MethodReferenceTestMethodHandle
*/
import java.lang.invoke.*;
import java.util.*;
import org.testng.annotations.Test;
import static org.testng.Assert.assertEquals;
@Test
public class MethodReferenceTestMethodHandle {
MethodHandles.Lookup lookup = MethodHandles.lookup();
interface ReplaceItf {
Object apply(String a, char b, char c) throws Throwable;
}
interface FormatItf {
Object apply(String a, Object... args) throws Throwable;
}
interface AddItf {
void apply(List st, int idx, Object v) throws Throwable;
}
public void testVirtual() throws Throwable {
MethodType mt = MethodType.methodType(String.class, char.class, char.class);
MethodHandle ms = lookup.findVirtual(String.class, "replace", mt);
// --- String.replace(String, char, char) ---
assertEquals("oome otring to oearch", ms.invoke("some string to search", 's', 'o'));
ReplaceItf f1 = (a, b, c) -> ms.invoke(a,b,c);
assertEquals("oome otring to oearch", f1.apply("some string to search", 's', 'o'));
ReplaceItf f2 = ms::invoke;
assertEquals("oome otring to oearch", f2.apply("some string to search", 's', 'o'));
assertEquals("oome otring to oearch", f2.apply("some string to search", new Character('s'), 'o'));
assertEquals("oome otring to oearch", ((ReplaceItf) ms::invoke).apply("some string to search", 's', 'o'));
}
public void testStatic() throws Throwable {
MethodType fmt = MethodType.methodType(String.class, String.class, (new Object[1]).getClass());
MethodHandle fms = lookup.findStatic(String.class, "format", fmt);
// --- String.format(String, Object...) ---
assertEquals("Testing One 2 3", fms.invoke("Testing %s %d %x", "One", new Integer(2), 3));
FormatItf ff2 = fms::invoke;
assertEquals("Testing One 2 3", ff2.apply("Testing %s %d %x", "One", new Integer(2), 3));
assertEquals("Testing One 2 3", ((FormatItf) fms::invoke).apply("Testing %s %d %x", "One", new Integer(2), 3));
assertEquals("Testing One 2 3 four", ff2.apply("Testing %s %d %x %s", "One", new Integer(2), 3, "four"));
}
public void testVoid() throws Throwable {
MethodType pmt = MethodType.methodType(void.class, int.class, Object.class);
MethodHandle pms = lookup.findVirtual(List.class, "add", pmt);
List<String> list = new ArrayList<>();
// --- List.add(int,String) ---
pms.invoke(list, 0, "Hi");
AddItf pf2 = pms::invoke;
pf2.apply(list, 1, "there");
AddItf pf3 = pms::invokeExact;
pf3.apply(list, 2, "you");
assertEquals("Hi", list.get(0));
assertEquals("there", list.get(1));
assertEquals("you", list.get(2));
}
}
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册