mirror of
https://github.com/openjdk/jdk.git
synced 2025-08-27 06:45:07 +02:00
8198698: Support Lambda proxy classes in dynamic CDS archive
Added archving of hidden classes of type lambda proxy classes. Co-authored-by: Ioi Lam <ioi.lam@oracle.com> Reviewed-by: mchung, iklam, dholmes
This commit is contained in:
parent
268d870187
commit
847a3baca8
57 changed files with 3724 additions and 39 deletions
|
@ -1,5 +1,5 @@
|
|||
/*
|
||||
* Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2012, 2020, 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
|
||||
|
@ -244,6 +244,47 @@ import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
|||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Spins the lambda proxy class.
|
||||
*
|
||||
* This first checks if a lambda proxy class can be loaded from CDS archive.
|
||||
* Otherwise, generate the lambda proxy class. If CDS dumping is enabled, it
|
||||
* registers the lambda proxy class for including into the CDS archive.
|
||||
*/
|
||||
private Class<?> spinInnerClass() throws LambdaConversionException {
|
||||
// include lambda proxy class in CDS archive at dump time
|
||||
if (LambdaProxyClassArchive.isDumpArchive()) {
|
||||
Class<?> innerClass = generateInnerClass();
|
||||
LambdaProxyClassArchive.register(targetClass,
|
||||
samMethodName,
|
||||
invokedType,
|
||||
samMethodType,
|
||||
implMethod,
|
||||
instantiatedMethodType,
|
||||
isSerializable,
|
||||
markerInterfaces,
|
||||
additionalBridges,
|
||||
innerClass);
|
||||
return innerClass;
|
||||
}
|
||||
|
||||
// load from CDS archive if present
|
||||
Class<?> innerClass = LambdaProxyClassArchive.find(targetClass,
|
||||
samMethodName,
|
||||
invokedType,
|
||||
samMethodType,
|
||||
implMethod,
|
||||
instantiatedMethodType,
|
||||
isSerializable,
|
||||
markerInterfaces,
|
||||
additionalBridges,
|
||||
!disableEagerInitialization);
|
||||
if (innerClass == null) {
|
||||
innerClass = generateInnerClass();
|
||||
}
|
||||
return innerClass;
|
||||
}
|
||||
|
||||
/**
|
||||
* Generate a class file which implements the functional
|
||||
* interface, define and return the class.
|
||||
|
@ -259,7 +300,7 @@ import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
|||
* @throws LambdaConversionException If properly formed functional interface
|
||||
* is not found
|
||||
*/
|
||||
private Class<?> spinInnerClass() throws LambdaConversionException {
|
||||
private Class<?> generateInnerClass() throws LambdaConversionException {
|
||||
String[] interfaces;
|
||||
String samIntf = samBase.getName().replace('.', '/');
|
||||
boolean accidentallySerializable = !isSerializable && Serializable.class.isAssignableFrom(samBase);
|
||||
|
|
|
@ -0,0 +1,136 @@
|
|||
/*
|
||||
* Copyright (c) 2020, 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.
|
||||
*
|
||||
*/
|
||||
package java.lang.invoke;
|
||||
|
||||
import jdk.internal.loader.BuiltinClassLoader;
|
||||
import jdk.internal.misc.VM;
|
||||
|
||||
final class LambdaProxyClassArchive {
|
||||
private static final boolean dumpArchive;
|
||||
private static final boolean sharingEnabled;
|
||||
|
||||
static {
|
||||
dumpArchive = VM.isCDSDumpingEnabled();
|
||||
sharingEnabled = VM.isCDSSharingEnabled();
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if CDS dynamic dump is enabled.
|
||||
*/
|
||||
static boolean isDumpArchive() {
|
||||
return dumpArchive;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if CDS sharing is enabled.
|
||||
*/
|
||||
static boolean isSharingEnabled() {
|
||||
return sharingEnabled;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if the class is loaded by a built-in class loader.
|
||||
*/
|
||||
static boolean loadedByBuiltinLoader(Class<?> cls) {
|
||||
ClassLoader cl = cls.getClassLoader();
|
||||
return (cl == null || (cl instanceof BuiltinClassLoader)) ? true : false;
|
||||
}
|
||||
|
||||
private static native void addToArchive(Class<?> caller,
|
||||
String invokedName,
|
||||
MethodType invokedType,
|
||||
MethodType samMethodType,
|
||||
MemberName implMethod,
|
||||
MethodType instantiatedMethodType,
|
||||
Class<?> lambdaProxyClass);
|
||||
|
||||
private static native Class<?> findFromArchive(Class<?> caller,
|
||||
String invokedName,
|
||||
MethodType invokedType,
|
||||
MethodType samMethodType,
|
||||
MemberName implMethod,
|
||||
MethodType instantiatedMethodType,
|
||||
boolean initialize);
|
||||
|
||||
/**
|
||||
* Registers the lambdaProxyClass into CDS archive.
|
||||
* The VM will store the lambdaProxyClass into a hash table
|
||||
* using the first six argumennts as the key.
|
||||
*
|
||||
* CDS only archives lambda proxy class if it's not serializable
|
||||
* and no marker interfaces and no additional bridges, and if it is
|
||||
* loaded by a built-in class loader.
|
||||
*/
|
||||
static boolean register(Class<?> caller,
|
||||
String invokedName,
|
||||
MethodType invokedType,
|
||||
MethodType samMethodType,
|
||||
MethodHandle implMethod,
|
||||
MethodType instantiatedMethodType,
|
||||
boolean isSerializable,
|
||||
Class<?>[] markerInterfaces,
|
||||
MethodType[] additionalBridges,
|
||||
Class<?> lambdaProxyClass) {
|
||||
if (!isDumpArchive())
|
||||
throw new IllegalStateException("should only register lambda proxy class at dump time");
|
||||
|
||||
if (loadedByBuiltinLoader(caller) &&
|
||||
!isSerializable && markerInterfaces.length == 0 && additionalBridges.length == 0) {
|
||||
addToArchive(caller, invokedName, invokedType, samMethodType,
|
||||
implMethod.internalMemberName(), instantiatedMethodType,
|
||||
lambdaProxyClass);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Lookup a lambda proxy class from the CDS archive using the first
|
||||
* six arguments as the key.
|
||||
*
|
||||
* CDS only archives lambda proxy class if it's not serializable
|
||||
* and no marker interfaces and no additional bridges, and if it is
|
||||
* loaded by a built-in class loader.
|
||||
*/
|
||||
static Class<?> find(Class<?> caller,
|
||||
String invokedName,
|
||||
MethodType invokedType,
|
||||
MethodType samMethodType,
|
||||
MethodHandle implMethod,
|
||||
MethodType instantiatedMethodType,
|
||||
boolean isSerializable,
|
||||
Class<?>[] markerInterfaces,
|
||||
MethodType[] additionalBridges,
|
||||
boolean initialize) {
|
||||
if (isDumpArchive())
|
||||
throw new IllegalStateException("cannot load class from CDS archive at dump time");
|
||||
|
||||
if (!loadedByBuiltinLoader(caller) ||
|
||||
!isSharingEnabled() || isSerializable || markerInterfaces.length > 0 || additionalBridges.length > 0)
|
||||
return null;
|
||||
|
||||
return findFromArchive(caller, invokedName, invokedType, samMethodType,
|
||||
implMethod.internalMemberName(), instantiatedMethodType, initialize);
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue