org.checkerframework.javacutil.Resolver Maven / Gradle / Ivy
package org.checkerframework.javacutil;
import com.sun.source.util.TreePath;
import com.sun.source.util.Trees;
import com.sun.tools.javac.api.JavacScope;
import com.sun.tools.javac.code.Kinds;
import com.sun.tools.javac.code.Kinds.KindSelector;
import com.sun.tools.javac.code.Symbol;
import com.sun.tools.javac.code.Symbol.ClassSymbol;
import com.sun.tools.javac.code.Symbol.PackageSymbol;
import com.sun.tools.javac.code.Symbol.TypeSymbol;
import com.sun.tools.javac.code.Type;
import com.sun.tools.javac.comp.AttrContext;
import com.sun.tools.javac.comp.DeferredAttr;
import com.sun.tools.javac.comp.Env;
import com.sun.tools.javac.comp.Resolve;
import com.sun.tools.javac.processing.JavacProcessingEnvironment;
import com.sun.tools.javac.util.Context;
import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition;
import com.sun.tools.javac.util.List;
import com.sun.tools.javac.util.Log;
import com.sun.tools.javac.util.Name;
import com.sun.tools.javac.util.Names;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Arrays;
import javax.annotation.processing.ProcessingEnvironment;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.type.TypeMirror;
import org.checkerframework.checker.nullness.qual.NonNull;
import org.checkerframework.checker.nullness.qual.Nullable;
/** A utility class to find symbols corresponding to string references (identifiers). */
// This class reflectively accesses jdk.compiler/com.sun.tools.javac.comp.
// This is why --add-opens=jdk.compiler/com.sun.tools.javac.comp=ALL-UNNAMED is required when
// running the Checker Framework. If this class is re-written, then that --add-opens should be
// removed.
public class Resolver {
private final Resolve resolve;
private final Names names;
private final Trees trees;
private final Log log;
private static final Method FIND_METHOD;
private static final Method FIND_VAR;
private static final Method FIND_IDENT;
private static final Method FIND_IDENT_IN_TYPE;
private static final Method FIND_IDENT_IN_PACKAGE;
private static final Method FIND_TYPE;
private static final Class> ACCESSERROR;
// Note that currently access(...) is defined in InvalidSymbolError, a superclass of AccessError
private static final Method ACCESSERROR_ACCESS;
/** The latest source version supported by this compiler. */
private static final int sourceVersionNumber =
Integer.parseInt(SourceVersion.latest().toString().substring("RELEASE_".length()));
/** Whether we are running on at least Java 13. */
private static final boolean atLeastJava13 = sourceVersionNumber >= 13;
/** Whether we are running on at least Java 23. */
private static final boolean atLeastJava23 = sourceVersionNumber >= 23;
static {
try {
FIND_METHOD =
Resolve.class.getDeclaredMethod(
"findMethod",
Env.class,
Type.class,
Name.class,
List.class,
List.class,
boolean.class,
boolean.class);
FIND_METHOD.setAccessible(true);
if (atLeastJava23) {
FIND_VAR =
Resolve.class.getDeclaredMethod(
"findVar", DiagnosticPosition.class, Env.class, Name.class);
} else {
FIND_VAR = Resolve.class.getDeclaredMethod("findVar", Env.class, Name.class);
}
FIND_VAR.setAccessible(true);
if (atLeastJava13) {
FIND_IDENT =
Resolve.class.getDeclaredMethod(
"findIdent", DiagnosticPosition.class, Env.class, Name.class, KindSelector.class);
} else {
FIND_IDENT =
Resolve.class.getDeclaredMethod("findIdent", Env.class, Name.class, KindSelector.class);
}
FIND_IDENT.setAccessible(true);
if (atLeastJava13) {
FIND_IDENT_IN_TYPE =
Resolve.class.getDeclaredMethod(
"findIdentInType",
DiagnosticPosition.class,
Env.class,
Type.class,
Name.class,
KindSelector.class);
} else {
FIND_IDENT_IN_TYPE =
Resolve.class.getDeclaredMethod(
"findIdentInType", Env.class, Type.class, Name.class, KindSelector.class);
}
FIND_IDENT_IN_TYPE.setAccessible(true);
if (atLeastJava13) {
FIND_IDENT_IN_PACKAGE =
Resolve.class.getDeclaredMethod(
"findIdentInPackage",
DiagnosticPosition.class,
Env.class,
TypeSymbol.class,
Name.class,
KindSelector.class);
} else {
FIND_IDENT_IN_PACKAGE =
Resolve.class.getDeclaredMethod(
"findIdentInPackage", Env.class, TypeSymbol.class, Name.class, KindSelector.class);
}
FIND_IDENT_IN_PACKAGE.setAccessible(true);
FIND_TYPE = Resolve.class.getDeclaredMethod("findType", Env.class, Name.class);
FIND_TYPE.setAccessible(true);
} catch (Exception e) {
Error err =
new AssertionError("Compiler 'Resolve' class doesn't contain required 'find*' method");
err.initCause(e);
throw err;
}
try {
ACCESSERROR = Class.forName("com.sun.tools.javac.comp.Resolve$AccessError");
ACCESSERROR_ACCESS = ACCESSERROR.getMethod("access", Name.class, TypeSymbol.class);
ACCESSERROR_ACCESS.setAccessible(true);
} catch (ClassNotFoundException e) {
throw new BugInCF("Compiler 'Resolve$AccessError' class could not be retrieved.", e);
} catch (NoSuchMethodException e) {
throw new BugInCF(
"Compiler 'Resolve$AccessError' class doesn't contain required 'access' method", e);
}
}
public Resolver(ProcessingEnvironment env) {
Context context = ((JavacProcessingEnvironment) env).getContext();
this.resolve = Resolve.instance(context);
this.names = Names.instance(context);
this.trees = Trees.instance(env);
this.log = Log.instance(context);
}
/**
* Determine the environment for the given path.
*
* @param path the tree path to the local scope
* @return the corresponding attribution environment
*/
public Env getEnvForPath(TreePath path) {
TreePath iter = path;
JavacScope scope = null;
while (scope == null && iter != null) {
try {
scope = (JavacScope) trees.getScope(iter);
} catch (NullPointerException t) {
// This statement fixes https://github.com/typetools/checker-framework/issues/1059 .
// It work around the crash by skipping through the TreePath until something doesn't
// crash. This probably returns the class scope, so users might not get the
// variables they expect. But that is better than crashing.
iter = iter.getParentPath();
}
}
if (scope != null) {
return scope.getEnv();
} else {
throw new BugInCF("Could not determine any possible scope for path: " + path.getLeaf());
}
}
/**
* Finds the package with name {@code name}.
*
* @param name the name of the package
* @param path the tree path to the local scope
* @return the {@code PackageSymbol} for the package if it is found, {@code null} otherwise
*/
public @Nullable PackageSymbol findPackage(String name, TreePath path) {
Log.DiagnosticHandler discardDiagnosticHandler = new Log.DiscardDiagnosticHandler(log);
try {
Env env = getEnvForPath(path);
final Element res;
if (atLeastJava13) {
res =
wrapInvocationOnResolveInstance(
FIND_IDENT, null, env, names.fromString(name), Kinds.KindSelector.PCK);
} else {
res =
wrapInvocationOnResolveInstance(
FIND_IDENT, env, names.fromString(name), Kinds.KindSelector.PCK);
}
// findIdent will return a PackageSymbol even for a symbol that is not a package,
// such as a.b.c.MyClass.myStaticField. "exists()" must be called on it to ensure
// that it exists.
if (res.getKind() == ElementKind.PACKAGE) {
PackageSymbol ps = (PackageSymbol) res;
return ps.exists() ? ps : null;
} else {
return null;
}
} finally {
log.popDiagnosticHandler(discardDiagnosticHandler);
}
}
/**
* Finds the field with name {@code name} in {@code type} or a superclass or superinterface of
* {@code type}.
*
* The method adheres to all the rules of Java's scoping (while also considering the imports)
* for name resolution.
*
* @param name the name of the field
* @param type the type of the receiver (i.e., the type in which to look for the field)
* @param path the tree path to the local scope
* @return the element for the field, {@code null} otherwise
*/
public @Nullable VariableElement findField(String name, TypeMirror type, TreePath path) {
Log.DiagnosticHandler discardDiagnosticHandler = new Log.DiscardDiagnosticHandler(log);
try {
Env env = getEnvForPath(path);
final Element res;
if (atLeastJava13) {
res =
wrapInvocationOnResolveInstance(
FIND_IDENT_IN_TYPE,
null,
env,
type,
names.fromString(name),
Kinds.KindSelector.VAR);
} else {
res =
wrapInvocationOnResolveInstance(
FIND_IDENT_IN_TYPE, env, type, names.fromString(name), Kinds.KindSelector.VAR);
}
if (res.getKind().isField()) {
return (VariableElement) res;
} else if (res.getKind() == ElementKind.OTHER && ACCESSERROR.isInstance(res)) {
// Return the inaccessible field that was found
return (VariableElement) wrapInvocation(res, ACCESSERROR_ACCESS, null, null);
} else {
// Most likely didn't find the field and the Element is a SymbolNotFoundError
return null;
}
} finally {
log.popDiagnosticHandler(discardDiagnosticHandler);
}
}
/**
* Finds the local variable (including formal parameters) with name {@code name} in the given
* scope.
*
* @param name the name of the local variable
* @param path the tree path to the local scope
* @return the element for the local variable, {@code null} otherwise
*/
public @Nullable VariableElement findLocalVariableOrParameter(String name, TreePath path) {
Log.DiagnosticHandler discardDiagnosticHandler = new Log.DiscardDiagnosticHandler(log);
try {
Env env = getEnvForPath(path);
// Either a VariableElement or a SymbolNotFoundError.
Element res;
if (atLeastJava23) {
res = wrapInvocationOnResolveInstance(FIND_VAR, null, env, names.fromString(name));
} else {
res = wrapInvocationOnResolveInstance(FIND_VAR, env, names.fromString(name));
}
// Every kind in the documentation of Element.getKind() is explicitly tested, possibly
// in the "default:" case.
switch (res.getKind()) {
case EXCEPTION_PARAMETER:
case LOCAL_VARIABLE:
case PARAMETER:
case RESOURCE_VARIABLE:
return (VariableElement) res;
case ENUM_CONSTANT:
case FIELD:
return null;
default:
if (ElementUtils.isBindingVariable(res)) {
return (VariableElement) res;
}
if (res instanceof VariableElement) {
throw new BugInCF("unhandled variable ElementKind " + res.getKind());
}
// The Element might be a SymbolNotFoundError.
return null;
}
} finally {
log.popDiagnosticHandler(discardDiagnosticHandler);
}
}
/**
* Finds the class literal with name {@code name}.
*
* The method adheres to all the rules of Java's scoping (while also considering the imports)
* for name resolution.
*
* @param name the name of the class
* @param path the tree path to the local scope
* @return the element for the class
*/
public Element findClass(String name, TreePath path) {
Log.DiagnosticHandler discardDiagnosticHandler = new Log.DiscardDiagnosticHandler(log);
try {
Env env = getEnvForPath(path);
return wrapInvocationOnResolveInstance(FIND_TYPE, env, names.fromString(name));
} finally {
log.popDiagnosticHandler(discardDiagnosticHandler);
}
}
/**
* Finds the class with name {@code name} in a given package.
*
* @param name the name of the class
* @param pck the PackageSymbol for the package
* @param path the tree path to the local scope
* @return the {@code ClassSymbol} for the class if it is found, {@code null} otherwise
*/
public @Nullable ClassSymbol findClassInPackage(String name, PackageSymbol pck, TreePath path) {
Log.DiagnosticHandler discardDiagnosticHandler = new Log.DiscardDiagnosticHandler(log);
try {
Env env = getEnvForPath(path);
final Element res;
if (atLeastJava13) {
res =
wrapInvocationOnResolveInstance(
FIND_IDENT_IN_PACKAGE,
null,
env,
pck,
names.fromString(name),
Kinds.KindSelector.TYP);
} else {
res =
wrapInvocationOnResolveInstance(
FIND_IDENT_IN_PACKAGE, env, pck, names.fromString(name), Kinds.KindSelector.TYP);
}
if (ElementUtils.isTypeElement(res)) {
return (ClassSymbol) res;
} else {
return null;
}
} finally {
log.popDiagnosticHandler(discardDiagnosticHandler);
}
}
/**
* Finds the method element for a given name and list of expected parameter types.
*
* The method adheres to all the rules of Java's scoping (while also considering the imports)
* for name resolution.
*
*
(This method takes into account autoboxing.)
*
*
This method is a wrapper around {@code com.sun.tools.javac.comp.Resolve.findMethod}.
*
* @param methodName name of the method to find
* @param receiverType type of the receiver of the method
* @param path tree path
* @param argumentTypes types of arguments passed to the method call
* @return the method element (if found)
*/
public @Nullable ExecutableElement findMethod(
String methodName,
TypeMirror receiverType,
TreePath path,
java.util.List argumentTypes) {
Log.DiagnosticHandler discardDiagnosticHandler = new Log.DiscardDiagnosticHandler(log);
try {
Env env = getEnvForPath(path);
Type site = (Type) receiverType;
Name name = names.fromString(methodName);
List argtypes = List.nil();
for (TypeMirror a : argumentTypes) {
argtypes = argtypes.append((Type) a);
}
List typeargtypes = List.nil();
boolean allowBoxing = true;
boolean useVarargs = false;
try {
// For some reason we have to set our own method context, which is rather ugly.
// TODO: find a nicer way to do this.
Object methodContext = buildMethodContext();
Object oldContext = getField(resolve, "currentResolutionContext");
setField(resolve, "currentResolutionContext", methodContext);
Element resolveResult =
wrapInvocationOnResolveInstance(
FIND_METHOD, env, site, name, argtypes, typeargtypes, allowBoxing, useVarargs);
setField(resolve, "currentResolutionContext", oldContext);
ExecutableElement methodResult;
if (resolveResult.getKind() == ElementKind.METHOD
|| resolveResult.getKind() == ElementKind.CONSTRUCTOR) {
methodResult = (ExecutableElement) resolveResult;
} else if (resolveResult.getKind() == ElementKind.OTHER
&& ACCESSERROR.isInstance(resolveResult)) {
// Return the inaccessible method that was found.
methodResult =
(ExecutableElement) wrapInvocation(resolveResult, ACCESSERROR_ACCESS, null, null);
} else {
methodResult = null;
}
return methodResult;
} catch (Throwable t) {
Error err =
new AssertionError(
String.format(
"Unexpected reflection error in findMethod(%s, %s, ..., %s)",
methodName,
receiverType,
// path
argumentTypes));
err.initCause(t);
throw err;
}
} finally {
log.popDiagnosticHandler(discardDiagnosticHandler);
}
}
/**
* Build an instance of {@code Resolve$MethodResolutionContext}.
*
* @return a MethodResolutionContext
* @throws ClassNotFoundException if there is trouble constructing the instance
* @throws InstantiationException if there is trouble constructing the instance
* @throws IllegalAccessException if there is trouble constructing the instance
* @throws InvocationTargetException if there is trouble constructing the instance
* @throws NoSuchFieldException if there is trouble constructing the instance
*/
protected Object buildMethodContext()
throws ClassNotFoundException,
InstantiationException,
IllegalAccessException,
InvocationTargetException,
NoSuchFieldException {
// Class is not accessible, instantiate reflectively.
Class> methCtxClss =
Class.forName("com.sun.tools.javac.comp.Resolve$MethodResolutionContext");
Constructor> constructor = methCtxClss.getDeclaredConstructors()[0];
constructor.setAccessible(true);
Object methodContext = constructor.newInstance(resolve);
// we need to also initialize the fields attrMode and step
setField(methodContext, "attrMode", DeferredAttr.AttrMode.CHECK);
@SuppressWarnings("rawtypes")
List> phases = (List) getField(resolve, "methodResolutionSteps");
assert phases != null : "@AssumeAssertion(nullness): assumption";
setField(methodContext, "step", phases.get(1));
return methodContext;
}
/**
* Reflectively set a field.
*
* @param receiver the receiver in which to set the field
* @param fieldName name of field to set
* @param value new value for field
* @throws NoSuchFieldException if the field does not exist in the receiver
* @throws IllegalAccessException if the field is not accessible
*/
@SuppressWarnings({
"nullness:argument",
"interning:argument"
}) // assume that the fields all accept null and uninterned values
private void setField(Object receiver, String fieldName, @Nullable Object value)
throws NoSuchFieldException, IllegalAccessException {
Field f = receiver.getClass().getDeclaredField(fieldName);
f.setAccessible(true);
f.set(receiver, value);
}
/** Reflectively get the value of a field. */
private @Nullable Object getField(Object receiver, String fieldName)
throws NoSuchFieldException, IllegalAccessException {
Field f = receiver.getClass().getDeclaredField(fieldName);
f.setAccessible(true);
return f.get(receiver);
}
/**
* Wrap a method invocation on the {@code resolve} object.
*
* @param method the method to called
* @param args the arguments to the call
* @return the result of invoking the method on {@code resolve} (as the receiver) and the
* arguments
*/
private Symbol wrapInvocationOnResolveInstance(Method method, @Nullable Object... args) {
return wrapInvocation(resolve, method, args);
}
/**
* Invoke a method reflectively. This is like {@code Method.invoke()}, but it throws no checked
* exceptions.
*
* @param receiver the receiver
* @param method the method to called
* @param args the arguments to the call
* @return the result of invoking the method on the receiver and arguments
*/
private Symbol wrapInvocation(Object receiver, Method method, @Nullable Object... args) {
try {
@SuppressWarnings("nullness") // assume arguments are OK
@NonNull Symbol res = (Symbol) method.invoke(receiver, args);
return res;
} catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException e) {
throw new BugInCF(
e,
"Unexpected reflection error in wrapInvocation(%s, %s, %s)",
receiver,
method,
Arrays.toString(args));
}
}
}