# HG changeset patch
# Parent 271ef464fb3a4ae80c345a7234eaa4f90a32aeea
8186216: Use ldc + condy instead of invokedynamic for constant lambdas
diff -r 271ef464fb3a make/CompileJavaModules.gmk
--- a/make/CompileJavaModules.gmk Thu Mar 22 09:07:08 2018 -0700
+++ b/make/CompileJavaModules.gmk Thu Mar 22 17:39:26 2018 -0400
@@ -268,7 +268,7 @@
################################################################################
-java.rmi_ADD_JAVAC_FLAGS += -Xdoclint:all/protected '-Xdoclint/package:java.*,javax.*'
+java.rmi_ADD_JAVAC_FLAGS += -Xdoclint:all/protected '-Xdoclint/package:java.*,javax.*' -XDforNonCapturingLambda=generateIndy
java.rmi_CLEAN_FILES += $(wildcard \
$(TOPDIR)/src/java.rmi/share/classes/sun/rmi/registry/resources/*.properties \
$(TOPDIR)/src/java.rmi/share/classes/sun/rmi/server/resources/*.properties)
diff -r 271ef464fb3a src/java.base/share/classes/java/lang/invoke/AbstractValidatingLambdaMetafactory.java
--- a/src/java.base/share/classes/java/lang/invoke/AbstractValidatingLambdaMetafactory.java Thu Mar 22 09:07:08 2018 -0700
+++ b/src/java.base/share/classes/java/lang/invoke/AbstractValidatingLambdaMetafactory.java Thu Mar 22 17:39:26 2018 -0400
@@ -187,6 +187,8 @@
c.getName()));
}
}
+
+ validateMetafactoryArgs();
}
/**
@@ -203,7 +205,7 @@
* Check the meta-factory arguments for errors
* @throws LambdaConversionException if there are improper conversions
*/
- void validateMetafactoryArgs() throws LambdaConversionException {
+ private void validateMetafactoryArgs() throws LambdaConversionException {
// Check arity: captured + SAM == impl
final int implArity = implMethodType.parameterCount();
final int capturedArity = invokedType.parameterCount();
diff -r 271ef464fb3a src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java
--- a/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java Thu Mar 22 09:07:08 2018 -0700
+++ b/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java Thu Mar 22 17:39:26 2018 -0400
@@ -101,6 +101,7 @@
private final String[] argNames; // Generated names for the constructor arguments
private final String[] argDescs; // Type descriptors for the constructor arguments
private final String lambdaClassName; // Generated name for the generated class "X$$Lambda$1"
+ private final Class> innerClass; // Lambda proxy class
/**
* General meta-factory constructor, supporting both standard cases and
@@ -152,6 +153,7 @@
super(caller, invokedType, samMethodName, samMethodType,
implMethod, instantiatedMethodType,
isSerializable, markerInterfaces, additionalBridges);
+
implMethodClassName = implClass.getName().replace('.', '/');
implMethodName = implInfo.getName();
implMethodDesc = implInfo.getMethodType().toMethodDescriptorString();
@@ -169,6 +171,43 @@
} else {
argNames = argDescs = EMPTY_STRING_ARRAY;
}
+
+ innerClass = spinInnerClass();
+ }
+
+ /**
+ * Returns an instance of the lambda object for a non-capturing lambda
+ * or method reference
+ *
+ * @return the lambda object
+ * @throws LambdaConversionException If there was an error creating the
+ * lambda object
+ */
+ Object buildInstance() throws LambdaConversionException {
+ final Constructor>[] ctrs = AccessController.doPrivileged(
+ new PrivilegedAction<>() {
+ @Override
+ public Constructor>[] run() {
+ Constructor>[] ctrs = innerClass.getDeclaredConstructors();
+ if (ctrs.length == 1) {
+ // The lambda implementing inner class constructor is private, set
+ // it accessible (by us) before creating the constant sole instance
+ ctrs[0].setAccessible(true);
+ }
+ return ctrs;
+ }
+ });
+ if (ctrs.length != 1) {
+ throw new LambdaConversionException("Expected one lambda constructor for "
+ + innerClass.getCanonicalName() + ", got " + ctrs.length);
+ }
+
+ try {
+ return ctrs[0].newInstance();
+ }
+ catch (ReflectiveOperationException e) {
+ throw new LambdaConversionException("Exception instantiating lambda object", e);
+ }
}
/**
@@ -179,39 +218,13 @@
*
* @return a CallSite, which, when invoked, will return an instance of the
* functional interface
- * @throws ReflectiveOperationException
- * @throws LambdaConversionException If properly formed functional interface
- * is not found
+ * @throws LambdaConversionException If there was an error creating the
+ * lambda factory call site
*/
@Override
CallSite buildCallSite() throws LambdaConversionException {
- final Class> innerClass = spinInnerClass();
if (invokedType.parameterCount() == 0) {
- final Constructor>[] ctrs = AccessController.doPrivileged(
- new PrivilegedAction<>() {
- @Override
- public Constructor>[] run() {
- Constructor>[] ctrs = innerClass.getDeclaredConstructors();
- if (ctrs.length == 1) {
- // The lambda implementing inner class constructor is private, set
- // it accessible (by us) before creating the constant sole instance
- ctrs[0].setAccessible(true);
- }
- return ctrs;
- }
- });
- if (ctrs.length != 1) {
- throw new LambdaConversionException("Expected one lambda constructor for "
- + innerClass.getCanonicalName() + ", got " + ctrs.length);
- }
-
- try {
- Object inst = ctrs[0].newInstance();
- return new ConstantCallSite(MethodHandles.constant(samBase, inst));
- }
- catch (ReflectiveOperationException e) {
- throw new LambdaConversionException("Exception instantiating lambda object", e);
- }
+ return new ConstantCallSite(MethodHandles.constant(samBase, buildInstance()));
} else {
try {
UNSAFE.ensureClassInitialized(innerClass);
@@ -240,7 +253,7 @@
* @throws LambdaConversionException If properly formed functional interface
* is not found
*/
- private Class> spinInnerClass() throws LambdaConversionException {
+ private Class> spinInnerClass() {
String[] interfaces;
String samIntf = samBase.getName().replace('.', '/');
boolean accidentallySerializable = !isSerializable && Serializable.class.isAssignableFrom(samBase);
diff -r 271ef464fb3a src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java
--- a/src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java Thu Mar 22 09:07:08 2018 -0700
+++ b/src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java Thu Mar 22 17:39:26 2018 -0400
@@ -314,13 +314,74 @@
MethodHandle implMethod,
MethodType instantiatedMethodType)
throws LambdaConversionException {
- AbstractValidatingLambdaMetafactory mf;
- mf = new InnerClassLambdaMetafactory(caller, invokedType,
- invokedName, samMethodType,
- implMethod, instantiatedMethodType,
- false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY);
- mf.validateMetafactoryArgs();
- return mf.buildCallSite();
+ return new InnerClassLambdaMetafactory(caller, invokedType,
+ invokedName, samMethodType,
+ implMethod, instantiatedMethodType,
+ false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY)
+ .buildCallSite();
+ }
+
+ /**
+ * Special-case case version of {@link LambdaMetafactory#metafactory(MethodHandles.Lookup, String, Class, MethodType, MethodHandle, MethodType)}
+ * that is restricted to non-capturing lambdas. Rather than returning a
+ * {@link CallSite} that is linked to a factory for lambda objects, the
+ * lambda object itself is returned.
+ *
+ *
Typically used as a bootstrap method for {@code Constant_Dynamic}
+ * constants, to support the lambda expression and method
+ * reference expression features of the Java Programming Language.
+ *
+ *
The function object returned is an instance of a class which
+ * implements the interface named by {@code functionalInterface},
+ * declares a method with the name given by {@code invokedName} and the
+ * signature given by {@code samMethodType}. It may also override additional
+ * methods from {@code Object}.
+ *
+ * @param caller Represents a lookup context with the accessibility
+ * privileges of the caller. When used with {@code invokedynamic},
+ * this is stacked automatically by the VM.
+ * @param invokedName The name of the method to implement. When used with
+ * {@code Dynamic} constants, this is provided by the
+ * {@code NameAndType} of the {@code InvokeDynamic}
+ * structure and is stacked automatically by the VM.
+ * @param functionalInterface The functional interface the function object
+ * should implement. When used with {@code invokedynamic},
+ * this is provided by the {@code NameAndType} of
+ * the {@code InvokeDynamic} structure and is
+ * stacked automatically by the VM. In the event
+ * that the implementation method is an instance
+ * method and this signature has any parameters,
+ * the first parameter in the invocation signature
+ * must correspond to the receiver.
+ * @param samMethodType Signature and return type of method to be implemented
+ * by the function object.
+ * @param implMethod A direct method handle describing the implementation
+ * method which should be called (with suitable adaptation
+ * of argument types, return types, and with captured
+ * arguments prepended to the invocation arguments) at
+ * invocation time.
+ * @param instantiatedMethodType The signature and return type that should
+ * be enforced dynamically at invocation time.
+ * This may be the same as {@code samMethodType},
+ * or may be a specialization of it.
+ * @return a CallSite whose target can be used to perform capture, generating
+ * instances of the interface named by {@code invokedType}
+ * @throws LambdaConversionException If any of the linkage invariants
+ * described {@link LambdaMetafactory above}
+ * are violated
+ */
+ public static Object metafactory(MethodHandles.Lookup caller,
+ String invokedName,
+ Class> functionalInterface,
+ MethodType samMethodType,
+ MethodHandle implMethod,
+ MethodType instantiatedMethodType)
+ throws LambdaConversionException {
+ return new InnerClassLambdaMetafactory(caller, MethodType.methodType(functionalInterface),
+ invokedName, samMethodType,
+ implMethod, instantiatedMethodType,
+ false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY)
+ .buildInstance();
}
/**
@@ -487,14 +548,12 @@
}
}
- AbstractValidatingLambdaMetafactory mf
- = new InnerClassLambdaMetafactory(caller, invokedType,
- invokedName, samMethodType,
- implMethod,
- instantiatedMethodType,
- isSerializable,
- markerInterfaces, bridges);
- mf.validateMetafactoryArgs();
- return mf.buildCallSite();
+ return new InnerClassLambdaMetafactory(caller, invokedType,
+ invokedName, samMethodType,
+ implMethod,
+ instantiatedMethodType,
+ isSerializable,
+ markerInterfaces, bridges)
+ .buildCallSite();
}
}
diff -r 271ef464fb3a src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java
--- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java Thu Mar 22 09:07:08 2018 -0700
+++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java Thu Mar 22 17:39:26 2018 -0400
@@ -181,7 +181,8 @@
DIAMOND_WITH_ANONYMOUS_CLASS_CREATION(JDK9, Fragments.FeatureDiamondAndAnonClass, DiagKind.NORMAL),
UNDERSCORE_IDENTIFIER(MIN, JDK8),
PRIVATE_INTERFACE_METHODS(JDK9, Fragments.FeaturePrivateIntfMethods, DiagKind.PLURAL),
- LOCAL_VARIABLE_TYPE_INFERENCE(JDK10);
+ LOCAL_VARIABLE_TYPE_INFERENCE(JDK10),
+ CONDY_FOR_LAMBDA(JDK11);
enum DiagKind {
NORMAL,
diff -r 271ef464fb3a src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symbol.java
--- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symbol.java Thu Mar 22 09:07:08 2018 -0700
+++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symbol.java Thu Mar 22 17:39:26 2018 -0400
@@ -1631,6 +1631,10 @@
public R accept(Symbol.Visitor v, P p) {
return v.visitVarSymbol(this, p);
}
+
+ public boolean isDynamic() {
+ return false;
+ }
}
/** A class for method symbols.
@@ -2005,6 +2009,26 @@
}
}
+ /** A class for condy.
+ */
+ public static class DynamicVarSymbol extends VarSymbol {
+ public Object[] staticArgs;
+ public MethodSymbol bsm;
+ public int bsmKind;
+
+ public DynamicVarSymbol(Name name, Symbol owner, int bsmKind, MethodSymbol bsm, Type type, Object[] staticArgs) {
+ super(0, name, type, owner);
+ this.bsm = bsm;
+ this.bsmKind = bsmKind;
+ this.staticArgs = staticArgs;
+ }
+
+ @Override
+ public boolean isDynamic() {
+ return true;
+ }
+ }
+
/** A class for predefined operators.
*/
public static class OperatorSymbol extends MethodSymbol {
diff -r 271ef464fb3a src/jdk.compiler/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java
--- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java Thu Mar 22 09:07:08 2018 -0700
+++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java Thu Mar 22 17:39:26 2018 -0400
@@ -35,6 +35,7 @@
import com.sun.tools.javac.code.Symbol;
import com.sun.tools.javac.code.Symbol.ClassSymbol;
import com.sun.tools.javac.code.Symbol.DynamicMethodSymbol;
+import com.sun.tools.javac.code.Symbol.DynamicVarSymbol;
import com.sun.tools.javac.code.Symbol.MethodSymbol;
import com.sun.tools.javac.code.Symbol.TypeSymbol;
import com.sun.tools.javac.code.Symbol.VarSymbol;
@@ -70,7 +71,8 @@
import javax.lang.model.element.ElementKind;
import javax.lang.model.type.TypeKind;
-import com.sun.tools.javac.main.Option;
+import com.sun.tools.javac.code.Source;
+import com.sun.tools.javac.code.Source.Feature;
/**
* This pass desugars lambda expressions into static methods
@@ -149,6 +151,12 @@
dumpLambdaToMethodStats = options.isSet("debug.dumpLambdaToMethodStats");
attr = Attr.instance(context);
forceSerializable = options.isSet("forceSerializable");
+ Source source = Source.instance(context);
+ // format: -XDforNonCapturingLambda=generateCondy, which is the default, or -XDforNonCapturingLambda=generateIndy
+ String condyOp = options.get("forNonCapturingLambda");
+ condyForLambda = condyOp != null ?
+ condyOp.equals("generateCondy") :
+ Feature.CONDY_FOR_LAMBDA.allowedInSource(source);
}
//
@@ -1108,8 +1116,53 @@
}
}
}
+ return makeDynamicCall(tree, syms.lambdaMetafactory,
+ metafactoryName, staticArgs, tree.type, indyType, indy_args, samSym.name);
+ }
- return makeIndyCall(tree, syms.lambdaMetafactory, metafactoryName, staticArgs, indyType, indy_args, samSym.name);
+ private JCExpression makeDynamicCall(DiagnosticPosition pos, Type site, Name bsmName,
+ List