/*
* Copyright (c) 2013, 2018, 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 org.graalvm.compiler.replacements.processor;
import static org.graalvm.compiler.processor.AbstractProcessor.getAnnotationValue;
import static org.graalvm.compiler.processor.AbstractProcessor.getSimpleName;
import static org.graalvm.compiler.replacements.processor.ClassSubstitutionHandler.CLASS_SUBSTITUTION_CLASS_NAME;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import javax.annotation.processing.Messager;
import javax.lang.model.element.AnnotationMirror;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.Name;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.type.TypeKind;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.ElementFilter;
import javax.tools.Diagnostic.Kind;
import org.graalvm.compiler.processor.AbstractProcessor;
/**
* Handler for the {@value #METHOD_SUBSTITUTION_CLASS_NAME} annotation.
*/
public final class MethodSubstitutionHandler extends AnnotationHandler {
private static final String METHOD_SUBSTITUTION_CLASS_NAME = "org.graalvm.compiler.api.replacements.MethodSubstitution";
private static final boolean DEBUG = false;
private static final String ORIGINAL_METHOD_NAME = "value";
private static final String ORIGINAL_IS_STATIC = "isStatic";
private static final String ORIGINAL_SIGNATURE = "signature";
private static final String ORIGINAL_METHOD_NAME_DEFAULT = "";
private static final String ORIGINAL_SIGNATURE_DEFAULT = "";
public MethodSubstitutionHandler(AbstractProcessor processor) {
super(processor, METHOD_SUBSTITUTION_CLASS_NAME);
}
@SuppressWarnings("unused")
@Override
public void process(Element element, AnnotationMirror annotation, PluginGenerator generator) {
if (element.getKind() != ElementKind.METHOD) {
assert false : "Element is guaranteed to be a method.";
return;
}
ExecutableElement substitutionMethod = (ExecutableElement) element;
TypeElement substitutionType = findEnclosingClass(substitutionMethod);
assert substitutionType != null;
Messager messager = processor.env().getMessager();
AnnotationMirror substitutionClassAnnotation = processor.getAnnotation(substitutionType, processor.getType(CLASS_SUBSTITUTION_CLASS_NAME));
if (substitutionClassAnnotation == null) {
messager.printMessage(Kind.ERROR, String.format("A @%s annotation is required on the enclosing class.", getSimpleName(CLASS_SUBSTITUTION_CLASS_NAME)), element, annotation);
return;
}
boolean optional = getAnnotationValue(substitutionClassAnnotation, "optional", Boolean.class);
if (optional) {
return;
}
TypeElement originalType = ClassSubstitutionHandler.resolveOriginalType(processor, substitutionType, substitutionClassAnnotation);
if (originalType == null) {
messager.printMessage(Kind.ERROR, String.format("The @%s annotation is invalid on the enclosing class.", getSimpleName(CLASS_SUBSTITUTION_CLASS_NAME)), element, annotation);
return;
}
if (!substitutionMethod.getModifiers().contains(Modifier.STATIC)) {
messager.printMessage(Kind.ERROR, String.format("A @%s method must be static.", getSimpleName(METHOD_SUBSTITUTION_CLASS_NAME)), element, annotation);
}
if (substitutionMethod.getModifiers().contains(Modifier.ABSTRACT) || substitutionMethod.getModifiers().contains(Modifier.NATIVE)) {
messager.printMessage(Kind.ERROR, String.format("A @%s method must not be native or abstract.", getSimpleName(METHOD_SUBSTITUTION_CLASS_NAME)), element, annotation);
}
String originalName = originalName(substitutionMethod, annotation);
boolean isStatic = getAnnotationValue(annotation, ORIGINAL_IS_STATIC, Boolean.class);
TypeMirror[] originalSignature = originalSignature(originalType, substitutionMethod, annotation, isStatic);
if (originalSignature == null) {
return;
}
ExecutableElement originalMethod = originalMethod(substitutionMethod, annotation, originalType, originalName, originalSignature, isStatic);
if (DEBUG && originalMethod != null) {
messager.printMessage(Kind.NOTE, String.format("Found original method %s in type %s.", originalMethod, findEnclosingClass(originalMethod)));
}
}
private TypeMirror[] originalSignature(TypeElement originalType, ExecutableElement method, AnnotationMirror annotation, boolean isStatic) {
String signatureString = getAnnotationValue(annotation, ORIGINAL_SIGNATURE, String.class);
List<TypeMirror> parameters = new ArrayList<>();
Messager messager = processor.env().getMessager();
if (signatureString.equals(ORIGINAL_SIGNATURE_DEFAULT)) {
for (int i = 0; i < method.getParameters().size(); i++) {
parameters.add(method.getParameters().get(i).asType());
}
if (!isStatic) {
if (parameters.isEmpty()) {
messager.printMessage(Kind.ERROR, "Method signature must be a static method with the 'this' object as its first parameter", method, annotation);
return null;
} else {
TypeMirror thisParam = parameters.remove(0);
if (!isSubtype(originalType.asType(), thisParam)) {
Name thisName = method.getParameters().get(0).getSimpleName();
messager.printMessage(Kind.ERROR, String.format("The type of %s must assignable from %s", thisName, originalType), method, annotation);
}
}
}
parameters.add(0, method.getReturnType());
} else {
try {
APHotSpotSignature signature = new APHotSpotSignature(signatureString);
parameters.add(signature.getReturnType(processor.env()));
for (int i = 0; i < signature.getParameterCount(false); i++) {
parameters.add(signature.getParameterType(processor.env(), i));
}
} catch (Exception e) {
/*
* That's not good practice and should be changed after APHotSpotSignature has
* received a cleanup.
*/
messager.printMessage(Kind.ERROR, String.format("Parsing the signature failed: %s", e.getMessage() != null ? e.getMessage() : e.toString()), method, annotation);
return null;
}
}
return parameters.toArray(new TypeMirror[parameters.size()]);
}
private static String originalName(ExecutableElement substituteMethod, AnnotationMirror substitution) {
String originalMethodName = getAnnotationValue(substitution, ORIGINAL_METHOD_NAME, String.class);
if (originalMethodName.equals(ORIGINAL_METHOD_NAME_DEFAULT)) {
originalMethodName = substituteMethod.getSimpleName().toString();
}
return originalMethodName;
}
private ExecutableElement originalMethod(ExecutableElement substitutionMethod, AnnotationMirror substitutionAnnotation, TypeElement originalType, String originalName,
TypeMirror[] originalSignature, boolean isStatic) {
TypeMirror signatureReturnType = originalSignature[0];
TypeMirror[] signatureParameters = Arrays.copyOfRange(originalSignature, 1, originalSignature.length);
List<ExecutableElement> searchElements;
if (originalName.equals("<init>")) {
searchElements = ElementFilter.constructorsIn(originalType.getEnclosedElements());
} else {
searchElements = ElementFilter.methodsIn(originalType.getEnclosedElements());
}
Messager messager = processor.env().getMessager();
ExecutableElement originalMethod = null;
outer: for (ExecutableElement searchElement : searchElements) {
if (searchElement.getSimpleName().toString().equals(originalName) && searchElement.getParameters().size() == signatureParameters.length) {
for (int i = 0; i < signatureParameters.length; i++) {
VariableElement parameter = searchElement.getParameters().get(i);
if (!isTypeCompatible(parameter.asType(), signatureParameters[i])) {
continue outer;
}
}
originalMethod = searchElement;
break;
}
}
if (originalMethod == null) {
boolean optional = getAnnotationValue(substitutionAnnotation, "optional", Boolean.class);
if (!optional) {
messager.printMessage(Kind.ERROR,
String.format("Could not find the original method with name '%s' and parameters '%s'.", originalName, Arrays.toString(signatureParameters)),
substitutionMethod, substitutionAnnotation);
}
return null;
}
if (originalMethod.getModifiers().contains(Modifier.STATIC) != isStatic) {
boolean optional = getAnnotationValue(substitutionAnnotation, "optional", Boolean.class);
if (!optional) {
messager.printMessage(Kind.ERROR, String.format("The %s element must be set to %s.", ORIGINAL_IS_STATIC, !isStatic), substitutionMethod, substitutionAnnotation);
}
return null;
}
if (!isTypeCompatible(originalMethod.getReturnType(), signatureReturnType)) {
messager.printMessage(
Kind.ERROR,
String.format("The return type of the substitution method '%s' must match with the return type of the original method '%s'.", signatureReturnType,
originalMethod.getReturnType()),
substitutionMethod, substitutionAnnotation);
return null;
}
return originalMethod;
}
private boolean isTypeCompatible(TypeMirror originalType, TypeMirror substitutionType) {
TypeMirror original = originalType;
TypeMirror substitution = substitutionType;
if (needsErasure(original)) {
original = processor.env().getTypeUtils().erasure(original);
}
if (needsErasure(substitution)) {
substitution = processor.env().getTypeUtils().erasure(substitution);
}
return processor.env().getTypeUtils().isSameType(original, substitution);
}
/**
* Tests whether one type is a subtype of another. Any type is considered to be a subtype of
* itself.
*
* @param t1 the first type
* @param t2 the second type
/**代码未完, 请加载全部代码(NowJava.com).**/