Browse Source
This new GraalVM feature replaces ConstantFieldFeature and introduces various enhancements: - Leverage the new FieldValueTransformer API - Use GraalVM 22.3 graal-sdk dependency instead of svm one - Avoid using internal GraalVM APIs - No need to configure JPMS exports - Directly integrated in spring-core module - Simplified build configuration Closes gh-29081 Closes gh-29080 Closes gh-29089pull/29364/head
16 changed files with 111 additions and 335 deletions
@ -1,40 +0,0 @@ |
|||||||
description = "Spring Core GraalVM feature" |
|
||||||
|
|
||||||
configurations { |
|
||||||
classesOnlyElements { |
|
||||||
canBeConsumed = true |
|
||||||
canBeResolved = false |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
artifacts { |
|
||||||
classesOnlyElements(compileJava.destinationDirectory) |
|
||||||
classesOnlyElements(sourceSets.main.resources.srcDirs) |
|
||||||
} |
|
||||||
|
|
||||||
tasks.withType(JavaCompile) { |
|
||||||
options.compilerArgs += [ |
|
||||||
"--add-modules", |
|
||||||
"jdk.internal.vm.ci", |
|
||||||
"--add-exports", |
|
||||||
"jdk.internal.vm.ci/jdk.vm.ci.meta=ALL-UNNAMED" |
|
||||||
] |
|
||||||
} |
|
||||||
|
|
||||||
tasks.withType(Javadoc) { |
|
||||||
enabled = false |
|
||||||
} |
|
||||||
|
|
||||||
eclipse.classpath.file { |
|
||||||
whenMerged { |
|
||||||
entries.find{ it.path ==~ '.*JRE_CONTAINER.*' }.each { |
|
||||||
it.entryAttributes['module'] = true |
|
||||||
it.entryAttributes['add-exports'] = 'jdk.internal.vm.ci/jdk.vm.ci.meta=ALL-UNNAMED' |
|
||||||
it.entryAttributes['limit-modules'] = 'java.se,jdk.accessibility,jdk.attach,jdk.compiler,jdk.httpserver,jdk.jartool,jdk.jconsole,jdk.jdi,jdk.management,jdk.sctp,jdk.security.auth,jdk.security.jgss,jdk.unsupported,jdk.dynalink,jdk.incubator.foreign,jdk.incubator.vector,jdk.javadoc,jdk.jfr,jdk.jshell,jdk.management.jfr,jdk.net,jdk.nio.mapmode,jdk.unsupported.desktop,jdk.jsobject,jdk.xml.dom,jdk.internal.vm.ci' |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
dependencies { |
|
||||||
compileOnly("org.graalvm.nativeimage:svm") |
|
||||||
} |
|
||||||
@ -1,51 +0,0 @@ |
|||||||
/* |
|
||||||
* Copyright 2002-2022 the original author or authors. |
|
||||||
* |
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
|
||||||
* you may not use this file except in compliance with the License. |
|
||||||
* You may obtain a copy of the License at |
|
||||||
* |
|
||||||
* https://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
* |
|
||||||
* Unless required by applicable law or agreed to in writing, software |
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS, |
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
||||||
* See the License for the specific language governing permissions and |
|
||||||
* limitations under the License. |
|
||||||
*/ |
|
||||||
|
|
||||||
package org.springframework.aot.graalvm; |
|
||||||
|
|
||||||
import com.oracle.svm.core.annotate.AutomaticFeature; |
|
||||||
import com.oracle.svm.hosted.FeatureImpl.DuringSetupAccessImpl; |
|
||||||
import org.graalvm.compiler.debug.DebugContext; |
|
||||||
import org.graalvm.nativeimage.hosted.Feature; |
|
||||||
|
|
||||||
/** |
|
||||||
* GraalVM {@link Feature} that substitutes field values that match a certain pattern |
|
||||||
* with constants without causing build-time initialization. |
|
||||||
* |
|
||||||
* @author Phillip Webb |
|
||||||
* @author Sebastien Deleuze |
|
||||||
* @since 6.0 |
|
||||||
*/ |
|
||||||
@AutomaticFeature |
|
||||||
class ConstantFieldFeature implements Feature { |
|
||||||
|
|
||||||
@Override |
|
||||||
public void duringSetup(DuringSetupAccess access) { |
|
||||||
duringSetup((DuringSetupAccessImpl) access); |
|
||||||
} |
|
||||||
|
|
||||||
private void duringSetup(DuringSetupAccessImpl access) { |
|
||||||
DebugContext debug = access.getDebugContext(); |
|
||||||
try (DebugContext.Scope scope = debug.scope("ConstantFieldFeature.duringSetup")) { |
|
||||||
debug.log("Installing constant field substitution processor : " + scope); |
|
||||||
ClassLoader classLoader = ConstantFieldFeature.class.getClassLoader(); |
|
||||||
ConstantFieldSubstitutionProcessor substitutionProcessor = |
|
||||||
new ConstantFieldSubstitutionProcessor(debug, classLoader); |
|
||||||
access.registerSubstitutionProcessor(substitutionProcessor); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
@ -1,106 +0,0 @@ |
|||||||
/* |
|
||||||
* Copyright 2002-2022 the original author or authors. |
|
||||||
* |
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
|
||||||
* you may not use this file except in compliance with the License. |
|
||||||
* You may obtain a copy of the License at |
|
||||||
* |
|
||||||
* https://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
* |
|
||||||
* Unless required by applicable law or agreed to in writing, software |
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS, |
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
||||||
* See the License for the specific language governing permissions and |
|
||||||
* limitations under the License. |
|
||||||
*/ |
|
||||||
|
|
||||||
package org.springframework.aot.graalvm; |
|
||||||
|
|
||||||
import java.lang.reflect.Field; |
|
||||||
import java.util.Set; |
|
||||||
import java.util.concurrent.ConcurrentHashMap; |
|
||||||
import java.util.concurrent.ConcurrentMap; |
|
||||||
import java.util.regex.Pattern; |
|
||||||
|
|
||||||
import com.oracle.graal.pointsto.infrastructure.SubstitutionProcessor; |
|
||||||
import com.oracle.svm.core.meta.SubstrateObjectConstant; |
|
||||||
import com.oracle.svm.core.util.UserError; |
|
||||||
import jdk.vm.ci.meta.JavaConstant; |
|
||||||
import jdk.vm.ci.meta.JavaKind; |
|
||||||
import jdk.vm.ci.meta.ResolvedJavaField; |
|
||||||
import jdk.vm.ci.meta.ResolvedJavaType; |
|
||||||
import org.graalvm.compiler.debug.DebugContext; |
|
||||||
|
|
||||||
/** |
|
||||||
* {@link SubstitutionProcessor} to compute at build time the value of the |
|
||||||
* boolean static fields identified by {@link #patterns} in order to allow |
|
||||||
* efficient code shrinking without using class build time initialization. |
|
||||||
* |
|
||||||
* @author Phillip Webb |
|
||||||
* @author Sebastien Deleuze |
|
||||||
* @since 6.0 |
|
||||||
*/ |
|
||||||
class ConstantFieldSubstitutionProcessor extends SubstitutionProcessor { |
|
||||||
|
|
||||||
// Later should be an explicit signal, like an annotation or even a Java keyword
|
|
||||||
private static Pattern[] patterns = { |
|
||||||
Pattern.compile(Pattern.quote("org.springframework.core.NativeDetector#imageCode")), |
|
||||||
Pattern.compile(Pattern.quote("org.springframework.") + ".*#.*Present"), |
|
||||||
Pattern.compile(Pattern.quote("org.springframework.") + ".*#.*PRESENT"), |
|
||||||
Pattern.compile(Pattern.quote("reactor.") + ".*#.*Available") |
|
||||||
}; |
|
||||||
|
|
||||||
private final ThrowawayClassLoader throwawayClassLoader; |
|
||||||
|
|
||||||
private ConcurrentMap<String, ResolvedJavaField> cache = new ConcurrentHashMap<>(); |
|
||||||
|
|
||||||
private Set<String> ignoredFields = ConcurrentHashMap.newKeySet(); |
|
||||||
|
|
||||||
|
|
||||||
ConstantFieldSubstitutionProcessor(DebugContext debug, ClassLoader applicationClassLoader) { |
|
||||||
this.throwawayClassLoader = new ThrowawayClassLoader(applicationClassLoader); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public ResolvedJavaField lookup(ResolvedJavaField field) { |
|
||||||
ResolvedJavaType declaringClass = field.getDeclaringClass(); |
|
||||||
if (field.getType().getJavaKind() == JavaKind.Boolean && field.isStatic()) { |
|
||||||
String fieldIdentifier = declaringClass.toJavaName() + "#" + field.getName(); |
|
||||||
for (Pattern pattern : patterns) { |
|
||||||
if (pattern.matcher(fieldIdentifier).matches() && !this.ignoredFields.contains(fieldIdentifier)) { |
|
||||||
try { |
|
||||||
return this.cache.computeIfAbsent(fieldIdentifier, key -> { |
|
||||||
JavaConstant constant = lookupConstant(declaringClass.toJavaName(), field.getName()); |
|
||||||
// TODO Use proper logging only when --verbose is specified when https://github.com/oracle/graal/issues/4669 will be fixed
|
|
||||||
ConstantReadableJavaField readableJavaField = new ConstantReadableJavaField(field, constant); |
|
||||||
System.out.println("Field " + fieldIdentifier + " set to " + constant.toValueString() + " at build time"); |
|
||||||
return readableJavaField; |
|
||||||
}); |
|
||||||
} |
|
||||||
catch (Throwable ex) { |
|
||||||
this.ignoredFields.add(fieldIdentifier); |
|
||||||
System.out.println("Processing of field " + fieldIdentifier + " skipped due the following error : " + ex.getMessage()); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
return super.lookup(field); |
|
||||||
} |
|
||||||
|
|
||||||
private JavaConstant lookupConstant(String className, String fieldName) { |
|
||||||
try { |
|
||||||
Class<?> throwawayClass = this.throwawayClassLoader.loadClass(className); |
|
||||||
Field field = throwawayClass.getDeclaredField(fieldName); |
|
||||||
field.setAccessible(true); |
|
||||||
Object value = field.get(null); |
|
||||||
if (!(value instanceof Boolean)) { |
|
||||||
throw UserError.abort("Unable to get the value of " + className + "." + fieldName); |
|
||||||
} |
|
||||||
return SubstrateObjectConstant.forBoxedValue(JavaKind.Boolean, value); |
|
||||||
} |
|
||||||
catch (Exception ex) { |
|
||||||
throw new IllegalStateException("Unable to read value from " + className + "." + fieldName, ex); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
@ -1,117 +0,0 @@ |
|||||||
/* |
|
||||||
* Copyright 2002-2022 the original author or authors. |
|
||||||
* |
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
|
||||||
* you may not use this file except in compliance with the License. |
|
||||||
* You may obtain a copy of the License at |
|
||||||
* |
|
||||||
* https://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
* |
|
||||||
* Unless required by applicable law or agreed to in writing, software |
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS, |
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
||||||
* See the License for the specific language governing permissions and |
|
||||||
* limitations under the License. |
|
||||||
*/ |
|
||||||
|
|
||||||
package org.springframework.aot.graalvm; |
|
||||||
|
|
||||||
import java.lang.annotation.Annotation; |
|
||||||
|
|
||||||
import com.oracle.graal.pointsto.infrastructure.WrappedElement; |
|
||||||
import com.oracle.svm.core.meta.ReadableJavaField; |
|
||||||
import jdk.vm.ci.meta.JavaConstant; |
|
||||||
import jdk.vm.ci.meta.JavaType; |
|
||||||
import jdk.vm.ci.meta.MetaAccessProvider; |
|
||||||
import jdk.vm.ci.meta.ResolvedJavaField; |
|
||||||
import jdk.vm.ci.meta.ResolvedJavaType; |
|
||||||
|
|
||||||
/** |
|
||||||
* {@link ReadableJavaField} for a constant value. |
|
||||||
* |
|
||||||
* @author Phillip Webb |
|
||||||
* @since 6.0 |
|
||||||
*/ |
|
||||||
class ConstantReadableJavaField implements ReadableJavaField, WrappedElement { |
|
||||||
|
|
||||||
private final ResolvedJavaField original; |
|
||||||
|
|
||||||
private final JavaConstant constant; |
|
||||||
|
|
||||||
|
|
||||||
public ConstantReadableJavaField(ResolvedJavaField original, JavaConstant constant) { |
|
||||||
this.original = original; |
|
||||||
this.constant = constant; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
@Override |
|
||||||
public <T extends Annotation> T getAnnotation(Class<T> annotationClass) { |
|
||||||
return this.original.getAnnotation(annotationClass); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public Annotation[] getAnnotations() { |
|
||||||
return this.original.getAnnotations(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public Annotation[] getDeclaredAnnotations() { |
|
||||||
return this.original.getDeclaredAnnotations(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public ResolvedJavaType getDeclaringClass() { |
|
||||||
return this.original.getDeclaringClass(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public int getModifiers() { |
|
||||||
return this.original.getModifiers(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public String getName() { |
|
||||||
return this.original.getName(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public int getOffset() { |
|
||||||
return this.original.getOffset(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public JavaType getType() { |
|
||||||
return this.original.getType(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public boolean isInternal() { |
|
||||||
return this.original.isInternal(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public boolean isSynthetic() { |
|
||||||
return this.original.isSynthetic(); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public JavaConstant readValue(MetaAccessProvider metaAccess, JavaConstant receiver) { |
|
||||||
return this.constant; |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public boolean allowConstantFolding() { |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public boolean injectFinalForRuntimeCompilation() { |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public Object getWrapped() { |
|
||||||
return this.original; |
|
||||||
} |
|
||||||
} |
|
||||||
@ -1,5 +0,0 @@ |
|||||||
/** |
|
||||||
* GraalVM implementation specific support which might change at any time, so not considered part of Spring Framework public API. |
|
||||||
*/ |
|
||||||
package org.springframework.aot.graalvm; |
|
||||||
|
|
||||||
@ -0,0 +1,86 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2002-2022 the original author or authors. |
||||||
|
* |
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
* you may not use this file except in compliance with the License. |
||||||
|
* You may obtain a copy of the License at |
||||||
|
* |
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
* |
||||||
|
* Unless required by applicable law or agreed to in writing, software |
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
* See the License for the specific language governing permissions and |
||||||
|
* limitations under the License. |
||||||
|
*/ |
||||||
|
|
||||||
|
package org.springframework.aot.nativex.feature; |
||||||
|
|
||||||
|
import java.lang.reflect.Field; |
||||||
|
import java.lang.reflect.Modifier; |
||||||
|
import java.util.regex.Pattern; |
||||||
|
|
||||||
|
import org.graalvm.nativeimage.hosted.Feature; |
||||||
|
|
||||||
|
/** |
||||||
|
* GraalVM {@link Feature} that substitutes boolean field values that match a certain pattern |
||||||
|
* with values pre-computed AOT without causing class build-time initialization. |
||||||
|
* |
||||||
|
* @author Sebastien Deleuze |
||||||
|
* @author Phillip Webb |
||||||
|
* @since 6.0 |
||||||
|
*/ |
||||||
|
class PreComputeFieldFeature implements Feature { |
||||||
|
|
||||||
|
private static Pattern[] patterns = { |
||||||
|
Pattern.compile(Pattern.quote("org.springframework.core.NativeDetector#imageCode")), |
||||||
|
Pattern.compile(Pattern.quote("org.springframework.") + ".*#.*Present"), |
||||||
|
Pattern.compile(Pattern.quote("org.springframework.") + ".*#.*PRESENT"), |
||||||
|
Pattern.compile(Pattern.quote("reactor.") + ".*#.*Available") |
||||||
|
}; |
||||||
|
|
||||||
|
private final ThrowawayClassLoader throwawayClassLoader = new ThrowawayClassLoader(PreComputeFieldFeature.class.getClassLoader()); |
||||||
|
|
||||||
|
@Override |
||||||
|
public void beforeAnalysis(BeforeAnalysisAccess access) { |
||||||
|
access.registerSubtypeReachabilityHandler(this::iterateFields, Object.class); |
||||||
|
} |
||||||
|
|
||||||
|
/* This method is invoked for every type that is reachable. */ |
||||||
|
private void iterateFields(DuringAnalysisAccess access, Class<?> subtype) { |
||||||
|
try { |
||||||
|
for (Field field : subtype.getDeclaredFields()) { |
||||||
|
int modifiers = field.getModifiers(); |
||||||
|
if (!Modifier.isStatic(modifiers) || !Modifier.isFinal(modifiers) || field.isEnumConstant() || |
||||||
|
(field.getType() != boolean.class && field.getType() != Boolean.class)) { |
||||||
|
continue; |
||||||
|
} |
||||||
|
String fieldIdentifier = field.getDeclaringClass().getName() + "#" + field.getName(); |
||||||
|
for (Pattern pattern : patterns) { |
||||||
|
if (pattern.matcher(fieldIdentifier).matches()) { |
||||||
|
try { |
||||||
|
Object fieldValue = provideFieldValue(field); |
||||||
|
access.registerFieldValueTransformer(field, (receiver, originalValue) -> fieldValue); |
||||||
|
System.out.println("Field " + fieldIdentifier + " set to " + fieldValue + " at build time"); |
||||||
|
} |
||||||
|
catch (Throwable ex) { |
||||||
|
System.out.println("Processing of field " + fieldIdentifier + " skipped due the following error : " + ex.getMessage()); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
catch (NoClassDefFoundError ex) { |
||||||
|
// Skip classes that have not all their field types in the classpath
|
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/* This method is invoked when the field value is written to the image heap or the field is constant folded. */ |
||||||
|
private Object provideFieldValue(Field field) throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException { |
||||||
|
Class<?> throwawayClass = this.throwawayClassLoader.loadClass(field.getDeclaringClass().getName()); |
||||||
|
Field throwawayField = throwawayClass.getDeclaredField(field.getName()); |
||||||
|
throwawayField.setAccessible(true); |
||||||
|
return throwawayField.get(null); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
@ -0,0 +1,9 @@ |
|||||||
|
/** |
||||||
|
* GraalVM native image features, not part of Spring Framework public API. |
||||||
|
*/ |
||||||
|
@NonNullApi |
||||||
|
@NonNullFields |
||||||
|
package org.springframework.aot.nativex.feature; |
||||||
|
|
||||||
|
import org.springframework.lang.NonNullApi; |
||||||
|
import org.springframework.lang.NonNullFields; |
||||||
@ -0,0 +1,9 @@ |
|||||||
|
/** |
||||||
|
* GraalVM native image substitutions, not part of Spring Framework public API. |
||||||
|
*/ |
||||||
|
@NonNullApi |
||||||
|
@NonNullFields |
||||||
|
package org.springframework.aot.nativex.substitution; |
||||||
|
|
||||||
|
import org.springframework.lang.NonNullApi; |
||||||
|
import org.springframework.lang.NonNullFields; |
||||||
@ -1,5 +1,2 @@ |
|||||||
Args = --initialize-at-build-time=org.springframework.aot.graalvm.ThrowawayClassLoader \ |
Args = --initialize-at-build-time=org.springframework.aot.nativex.feature.ThrowawayClassLoader \ |
||||||
--add-exports org.graalvm.nativeimage.builder/com.oracle.svm.hosted=ALL-UNNAMED \ |
--features=org.springframework.aot.nativex.feature.PreComputeFieldFeature |
||||||
--add-exports jdk.internal.vm.compiler/org.graalvm.compiler.debug=ALL-UNNAMED \ |
|
||||||
--add-exports jdk.internal.vm.ci/jdk.vm.ci.meta=ALL-UNNAMED \ |
|
||||||
--add-exports org.graalvm.nativeimage.builder/com.oracle.svm.core.meta=ALL-UNNAMED |
|
||||||
|
|||||||
Loading…
Reference in new issue