8049242: Explicit constructor overload selection should work with StaticClass as well

Reviewed-by: jlaskey, attila
This commit is contained in:
Athijegannathan Sundararajan 2014-07-03 23:03:37 +05:30
parent ade82779ac
commit 14eb01cb47
6 changed files with 128 additions and 10 deletions

View File

@ -390,6 +390,10 @@ abstract class AbstractJavaLinker implements GuardingDynamicLinker {
return new GuardedInvocationComponent(invocation, getClassGuard(type), clazz, ValidationType.EXACT_CLASS); return new GuardedInvocationComponent(invocation, getClassGuard(type), clazz, ValidationType.EXACT_CLASS);
} }
SingleDynamicMethod getConstructorMethod(final String signature) {
return null;
}
private MethodHandle getAssignableGuard(final MethodType type) { private MethodHandle getAssignableGuard(final MethodType type) {
return Guards.asType(assignableGuard, type); return Guards.asType(assignableGuard, type);
} }
@ -412,18 +416,18 @@ abstract class AbstractJavaLinker implements GuardingDynamicLinker {
return inv == null ? null : new GuardedInvocation(inv, getClassGuard(callSiteDescriptor.getMethodType())); return inv == null ? null : new GuardedInvocation(inv, getClassGuard(callSiteDescriptor.getMethodType()));
} }
private static MethodHandle getDynamicMethodInvocation(final CallSiteDescriptor callSiteDescriptor, private MethodHandle getDynamicMethodInvocation(final CallSiteDescriptor callSiteDescriptor,
final LinkerServices linkerServices, final String methodName, final Map<String, DynamicMethod> methodMap) { final LinkerServices linkerServices, final String methodName, final Map<String, DynamicMethod> methodMap) {
final DynamicMethod dynaMethod = getDynamicMethod(methodName, methodMap); final DynamicMethod dynaMethod = getDynamicMethod(methodName, methodMap);
return dynaMethod != null ? dynaMethod.getInvocation(callSiteDescriptor, linkerServices) : null; return dynaMethod != null ? dynaMethod.getInvocation(callSiteDescriptor, linkerServices) : null;
} }
private static DynamicMethod getDynamicMethod(final String methodName, final Map<String, DynamicMethod> methodMap) { private DynamicMethod getDynamicMethod(final String methodName, final Map<String, DynamicMethod> methodMap) {
final DynamicMethod dynaMethod = methodMap.get(methodName); final DynamicMethod dynaMethod = methodMap.get(methodName);
return dynaMethod != null ? dynaMethod : getExplicitSignatureDynamicMethod(methodName, methodMap); return dynaMethod != null ? dynaMethod : getExplicitSignatureDynamicMethod(methodName, methodMap);
} }
private static SingleDynamicMethod getExplicitSignatureDynamicMethod(final String methodName, private SingleDynamicMethod getExplicitSignatureDynamicMethod(final String fullName,
final Map<String, DynamicMethod> methodsMap) { final Map<String, DynamicMethod> methodsMap) {
// What's below is meant to support the "name(type, type, ...)" syntax that programmers can use in a method name // What's below is meant to support the "name(type, type, ...)" syntax that programmers can use in a method name
// to manually pin down an exact overloaded variant. This is not usually required, as the overloaded method // to manually pin down an exact overloaded variant. This is not usually required, as the overloaded method
@ -433,23 +437,33 @@ abstract class AbstractJavaLinker implements GuardingDynamicLinker {
// for performance reasons. // for performance reasons.
// Is the method name lexically of the form "name(types)"? // Is the method name lexically of the form "name(types)"?
final int lastChar = methodName.length() - 1; final int lastChar = fullName.length() - 1;
if(methodName.charAt(lastChar) != ')') { if(fullName.charAt(lastChar) != ')') {
return null; return null;
} }
final int openBrace = methodName.indexOf('('); final int openBrace = fullName.indexOf('(');
if(openBrace == -1) { if(openBrace == -1) {
return null; return null;
} }
final String name = fullName.substring(0, openBrace);
final String signature = fullName.substring(openBrace + 1, lastChar);
// Find an existing method for the "name" part // Find an existing method for the "name" part
final DynamicMethod simpleNamedMethod = methodsMap.get(methodName.substring(0, openBrace)); final DynamicMethod simpleNamedMethod = methodsMap.get(name);
if(simpleNamedMethod == null) { if(simpleNamedMethod == null) {
// explicit signature constructor access
// Java.type("java.awt.Color")["(int,int,int)"]
// will get Color(int,int,int) constructor of Color class.
if (name.isEmpty()) {
return getConstructorMethod(signature);
}
return null; return null;
} }
// Try to get a narrowed dynamic method for the explicit parameter types. // Try to get a narrowed dynamic method for the explicit parameter types.
return simpleNamedMethod.getMethodForExactParamTypes(methodName.substring(openBrace + 1, lastChar)); return simpleNamedMethod.getMethodForExactParamTypes(signature);
} }
private static final MethodHandle IS_METHOD_HANDLE_NOT_NULL = Guards.isNotNull().asType(MethodType.methodType( private static final MethodHandle IS_METHOD_HANDLE_NOT_NULL = Guards.isNotNull().asType(MethodType.methodType(

View File

@ -161,8 +161,9 @@ class StaticClassLinker implements TypeBasedGuardingDynamicLinker {
return null; return null;
} }
DynamicMethod getConstructorMethod(final String signature) { @Override
return constructor.getMethodForExactParamTypes(signature); SingleDynamicMethod getConstructorMethod(final String signature) {
return constructor != null? constructor.getMethodForExactParamTypes(signature) : null;
} }
} }

View File

@ -78,3 +78,15 @@ checkIt(function() new println("hello"));
// call constructor as normal method (without 'new') // call constructor as normal method (without 'new')
checkIt(function() Color()); checkIt(function() Color());
// try constructor on interface
checkIt(function() new java.lang["Runnable()"]);
checkIt(function() new java.lang["Runnable(int)"]);
// try constructor on abstrace class
try {
new java.io["InputStream()"];
throw new Error("should have thrown exception!");
} catch (e) {
print(e);
}

View File

@ -9,3 +9,6 @@ TypeError: Java constructor signature invalid: Object)
TypeError: Java method [jdk.internal.dynalink.beans.OverloadedDynamicMethod java.lang.System.getProperty] cant be used as a constructor. TypeError: Java method [jdk.internal.dynalink.beans.OverloadedDynamicMethod java.lang.System.getProperty] cant be used as a constructor.
TypeError: Java method [jdk.internal.dynalink.beans.OverloadedDynamicMethod java.io.PrintStream.println] cant be used as a constructor. TypeError: Java method [jdk.internal.dynalink.beans.OverloadedDynamicMethod java.io.PrintStream.println] cant be used as a constructor.
TypeError: Constructor [jdk.internal.dynalink.beans.SimpleDynamicMethod Color java.awt.Color.java.awt.Color(int,int,int)] requires new. TypeError: Constructor [jdk.internal.dynalink.beans.SimpleDynamicMethod Color java.awt.Color.java.awt.Color(int,int,int)] requires new.
TypeError: No such Java constructor: Runnable()
TypeError: No such Java constructor: Runnable(int)
java.lang.InstantiationException: java.io.InputStream

View File

@ -0,0 +1,78 @@
/*
* 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.
*/
/**
* JDK-8049242: Explicit constructor overload selection should work with StaticClass as well
*
* @test
* @run
*/
// call explicit constructor
print(new (Java.type("java.awt.Color")["(int,int,int)"])(255,0,255));
// print the constructor itself
print(Java.type("java.awt.Color")["(int,int,int)"]);
// store constructor to call later
var Color = Java.type("java.awt.Color")["(int,int,int)"];
// call stored constructor
print(new Color(33, 233, 2))
// check if default constructor works
var obj = new (Java.type("java.lang.Object")["()"])();
if (obj.class != Java.type("java.lang.Object").class) {
fail("obj is a java.lang.Object");
}
// expected failure cases.
function checkIt(func) {
try {
func();
throw new Error("should have thrown TypeError");
} catch(e) {
if (! (e instanceof TypeError)) {
fail("Expected TypeError, got " + e);
}
print(e);
}
}
// garbage signature string
checkIt(function() new (Java.type("java.lang.Object")["()xxxxx"])());
checkIt(function() new (Java.type("java.lang.Object")["("])());
checkIt(function() new (Java.type("java.lang.Object")[")"])());
// call constructor as normal method (without 'new')
checkIt(function() Color());
// try constructor on interface
checkIt(function() new (Java.type("java.lang.Runnable"))["()"]);
checkIt(function() new (Java.type("java.lang.Runnable"))["(int)"]);
// try constructor on abstrace class
try {
new (Java.type("java.io.InputStream"))["()"];
throw new Error("should have thrown exception!");
} catch (e) {
print(e);
}

View File

@ -0,0 +1,10 @@
java.awt.Color[r=255,g=0,b=255]
[jdk.internal.dynalink.beans.SimpleDynamicMethod Color java.awt.Color.java.awt.Color(int,int,int)]
java.awt.Color[r=33,g=233,b=2]
TypeError: null is not a function
TypeError: null is not a function
TypeError: null is not a function
TypeError: Constructor [jdk.internal.dynalink.beans.SimpleDynamicMethod Color java.awt.Color.java.awt.Color(int,int,int)] requires new.
TypeError: null is not a function
TypeError: null is not a function
java.lang.InstantiationException: java.io.InputStream