All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.checkerframework.javacutil.TreeUtils Maven / Gradle / Ivy

package org.checkerframework.javacutil;

import com.sun.source.tree.AnnotatedTypeTree;
import com.sun.source.tree.AnnotationTree;
import com.sun.source.tree.ArrayAccessTree;
import com.sun.source.tree.AssignmentTree;
import com.sun.source.tree.BinaryTree;
import com.sun.source.tree.BlockTree;
import com.sun.source.tree.CaseTree;
import com.sun.source.tree.ClassTree;
import com.sun.source.tree.CompoundAssignmentTree;
import com.sun.source.tree.ExpressionStatementTree;
import com.sun.source.tree.ExpressionTree;
import com.sun.source.tree.IdentifierTree;
import com.sun.source.tree.InstanceOfTree;
import com.sun.source.tree.LambdaExpressionTree;
import com.sun.source.tree.LambdaExpressionTree.BodyKind;
import com.sun.source.tree.LiteralTree;
import com.sun.source.tree.MemberReferenceTree;
import com.sun.source.tree.MemberReferenceTree.ReferenceMode;
import com.sun.source.tree.MemberSelectTree;
import com.sun.source.tree.MethodInvocationTree;
import com.sun.source.tree.MethodTree;
import com.sun.source.tree.NewArrayTree;
import com.sun.source.tree.NewClassTree;
import com.sun.source.tree.ParameterizedTypeTree;
import com.sun.source.tree.ParenthesizedTree;
import com.sun.source.tree.PrimitiveTypeTree;
import com.sun.source.tree.ReturnTree;
import com.sun.source.tree.StatementTree;
import com.sun.source.tree.SwitchTree;
import com.sun.source.tree.Tree;
import com.sun.source.tree.Tree.Kind;
import com.sun.source.tree.TreeVisitor;
import com.sun.source.tree.TypeCastTree;
import com.sun.source.tree.TypeParameterTree;
import com.sun.source.tree.UnionTypeTree;
import com.sun.source.tree.VariableTree;
import com.sun.source.util.SimpleTreeVisitor;
import com.sun.source.util.TreeScanner;
import com.sun.tools.javac.code.Flags;
import com.sun.tools.javac.code.Symbol;
import com.sun.tools.javac.code.Symbol.MethodSymbol;
import com.sun.tools.javac.code.Type;
import com.sun.tools.javac.code.TypeTag;
import com.sun.tools.javac.code.Types;
import com.sun.tools.javac.processing.JavacProcessingEnvironment;
import com.sun.tools.javac.tree.JCTree;
import com.sun.tools.javac.tree.JCTree.JCAnnotatedType;
import com.sun.tools.javac.tree.JCTree.JCAnnotation;
import com.sun.tools.javac.tree.JCTree.JCBinary;
import com.sun.tools.javac.tree.JCTree.JCExpression;
import com.sun.tools.javac.tree.JCTree.JCExpressionStatement;
import com.sun.tools.javac.tree.JCTree.JCFieldAccess;
import com.sun.tools.javac.tree.JCTree.JCLambda;
import com.sun.tools.javac.tree.JCTree.JCLambda.ParameterKind;
import com.sun.tools.javac.tree.JCTree.JCLiteral;
import com.sun.tools.javac.tree.JCTree.JCMemberReference;
import com.sun.tools.javac.tree.JCTree.JCMemberReference.OverloadKind;
import com.sun.tools.javac.tree.JCTree.JCMethodDecl;
import com.sun.tools.javac.tree.JCTree.JCMethodInvocation;
import com.sun.tools.javac.tree.JCTree.JCNewArray;
import com.sun.tools.javac.tree.JCTree.JCNewClass;
import com.sun.tools.javac.tree.JCTree.JCTypeParameter;
import com.sun.tools.javac.tree.TreeInfo;
import com.sun.tools.javac.tree.TreeMaker;
import com.sun.tools.javac.util.Context;
import com.sun.tools.javac.util.Position;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.EnumSet;
import java.util.List;
import java.util.Set;
import java.util.StringJoiner;
import java.util.stream.Collectors;
import javax.annotation.processing.ProcessingEnvironment;
import javax.lang.model.SourceVersion;
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.Name;
import javax.lang.model.element.NestingKind;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.type.ArrayType;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.ExecutableType;
import javax.lang.model.type.TypeKind;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.type.TypeVariable;
import javax.lang.model.util.ElementFilter;
import org.checkerframework.checker.interning.qual.PolyInterned;
import org.checkerframework.checker.nullness.qual.EnsuresNonNullIf;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.checkerframework.checker.signature.qual.FullyQualifiedName;
import org.checkerframework.dataflow.qual.Pure;
import org.checkerframework.javacutil.TreeUtilsAfterJava11.BindingPatternUtils;
import org.checkerframework.javacutil.TreeUtilsAfterJava11.CaseUtils;
import org.checkerframework.javacutil.TreeUtilsAfterJava11.InstanceOfUtils;
import org.checkerframework.javacutil.TreeUtilsAfterJava11.SwitchExpressionUtils;
import org.checkerframework.javacutil.TreeUtilsAfterJava11.YieldUtils;
import org.plumelib.util.CollectionsPlume;
import org.plumelib.util.UniqueIdMap;

/**
 * Utility methods for analyzing a javac {@code Tree}.
 *
 * @see TreePathUtil
 */
public final class TreeUtils {

  // Class cannot be instantiated.
  private TreeUtils() {
    throw new AssertionError("Class TreeUtils cannot be instantiated.");
  }

  /** Unique IDs for trees. Used instead of hash codes, so output is deterministic. */
  public static final UniqueIdMap treeUids = new UniqueIdMap<>();

  /** 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 21. */
  private static final boolean atLeastJava21 = sourceVersionNumber >= 21;

  /**
   * The {@code TreeMaker.Select(JCExpression, Symbol)} method. Return type changes for JDK21+. Only
   * needs to be used while the code is compiled with JDK below 21.
   */
  private static final @Nullable Method TREEMAKER_SELECT;

  /** The value of Flags.RECORD which does not exist in Java 9 or 11. */
  private static final long Flags_RECORD = 2305843009213693952L;

  /** Tree kinds that represent a binary comparison. */
  private static final Set BINARY_COMPARISON_TREE_KINDS =
      EnumSet.of(
          Tree.Kind.EQUAL_TO,
          Tree.Kind.NOT_EQUAL_TO,
          Tree.Kind.LESS_THAN,
          Tree.Kind.GREATER_THAN,
          Tree.Kind.LESS_THAN_EQUAL,
          Tree.Kind.GREATER_THAN_EQUAL);

  static {
    try {
      TREEMAKER_SELECT = TreeMaker.class.getMethod("Select", JCExpression.class, Symbol.class);
    } catch (NoSuchMethodException e) {
      Error err = new AssertionError("Unexpected error in TreeUtils static initializer");
      err.initCause(e);
      throw err;
    }
  }

  /**
   * Checks if the provided method is a constructor method or no.
   *
   * @param tree a tree defining the method
   * @return true iff tree describes a constructor
   */
  public static boolean isConstructor(MethodTree tree) {
    return tree.getName().contentEquals("");
  }

  /**
   * Checks if the method invocation is a call to super.
   *
   * @param tree a tree defining a method invocation
   * @return true iff tree describes a call to super
   */
  public static boolean isSuperConstructorCall(MethodInvocationTree tree) {
    return isNamedMethodCall("super", tree);
  }

  /**
   * Checks if the method invocation is a call to "this".
   *
   * @param tree a tree defining a method invocation
   * @return true iff tree describes a call to this
   */
  public static boolean isThisConstructorCall(MethodInvocationTree tree) {
    return isNamedMethodCall("this", tree);
  }

  /**
   * Checks if the method call is a call to the given method name.
   *
   * @param name a method name
   * @param tree a tree defining a method invocation
   * @return true iff tree is a call to the given method
   */
  private static boolean isNamedMethodCall(String name, MethodInvocationTree tree) {
    return getMethodName(tree.getMethodSelect()).contentEquals(name);
  }

  /**
   * Returns true if the tree is a tree that 'looks like' either an access of a field or an
   * invocation of a method that are owned by the same accessing instance.
   *
   * 

It would only return true if the access tree is of the form: * *

   *   field
   *   this.field
   *
   *   method()
   *   this.method()
   * 
* * It does not perform any semantical check to differentiate between fields and local variables; * local methods or imported static methods. * * @param tree expression tree representing an access to object member * @return {@code true} iff the member is a member of {@code this} instance */ public static boolean isSelfAccess(ExpressionTree tree) { ExpressionTree tr = TreeUtils.withoutParens(tree); // If method invocation check the method select if (tr.getKind() == Tree.Kind.ARRAY_ACCESS) { return false; } if (tree.getKind() == Tree.Kind.METHOD_INVOCATION) { tr = ((MethodInvocationTree) tree).getMethodSelect(); } tr = TreeUtils.withoutParens(tr); if (tr.getKind() == Tree.Kind.TYPE_CAST) { tr = ((TypeCastTree) tr).getExpression(); } tr = TreeUtils.withoutParens(tr); if (tr.getKind() == Tree.Kind.IDENTIFIER) { return true; } if (tr.getKind() == Tree.Kind.MEMBER_SELECT) { tr = ((MemberSelectTree) tr).getExpression(); if (tr.getKind() == Tree.Kind.IDENTIFIER) { Name ident = ((IdentifierTree) tr).getName(); return ident.contentEquals("this") || ident.contentEquals("super"); } } return false; } /** * If the given tree is a parenthesized tree, return the enclosed non-parenthesized tree. * Otherwise, return the same tree. * * @param tree an expression tree * @return the outermost non-parenthesized tree enclosed by the given tree */ @SuppressWarnings("interning:return") // polymorphism implementation public static @PolyInterned ExpressionTree withoutParens(@PolyInterned ExpressionTree tree) { ExpressionTree t = tree; while (t.getKind() == Tree.Kind.PARENTHESIZED) { t = ((ParenthesizedTree) t).getExpression(); } return t; } /** * If the given tree is a parenthesized tree or cast tree, return the enclosed non-parenthesized, * non-cast tree. Otherwise, return the same tree. * * @param tree an expression tree * @return the outermost non-parenthesized non-cast tree enclosed by the given tree */ @SuppressWarnings("interning:return") // polymorphism implementation public static @PolyInterned ExpressionTree withoutParensOrCasts( @PolyInterned ExpressionTree tree) { ExpressionTree t = withoutParens(tree); while (t.getKind() == Tree.Kind.TYPE_CAST) { t = withoutParens(((TypeCastTree) t).getExpression()); } return t; } // Obtaining Elements from Trees. // There are three sets of methods: // * use elementFromDeclaration whenever the tree is a declaration // * use elementFromUse when the tree is a use // * use elementFromTree in other cases; note that it may return null // This section of the file groups methods by their receiver type; that is, it puts all // `elementFrom*(FooTree)` methods together. // TODO: Document when this may return null. /** * Returns the type element corresponding to the given class declaration. * *

This method returns null instead of crashing when no element exists for the class tree, * which can happen for certain kinds of anonymous classes, such as Ordering$1 in * PolyCollectorTypeVar.java in the all-systems test suite and "class MyFileFilter" in * PurgeTxnLog.java. * * @param tree class declaration * @return the element for the given class */ public static @Nullable TypeElement elementFromDeclaration(ClassTree tree) { TypeElement result = (TypeElement) TreeInfo.symbolFor((JCTree) tree); return result; } /** * Returns the type element corresponding to the given class declaration. * *

The TypeElement may be null for an anonymous class. * * @param tree the {@link ClassTree} node to get the element for * @return the {@link TypeElement} for the given tree * @deprecated use {@link #elementFromDeclaration(ClassTree)} */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable TypeElement elementFromTree(ClassTree tree) { return elementFromDeclaration(tree); } /** * Returns the type element corresponding to the given class declaration. * * @param tree the {@link ClassTree} node to get the element for * @return the {@link TypeElement} for the given tree * @deprecated use {@link #elementFromDeclaration(ClassTree)} */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable TypeElement elementFromUse(ClassTree tree) { return elementFromDeclaration(tree); } /** * Returns the fields that are declared within the given class declaration. * * @param tree the {@link ClassTree} node to get the fields for * @return the list of fields that are declared within the given class declaration */ public static List fieldsFromClassTree(ClassTree tree) { return tree.getMembers().stream() .filter(t -> t.getKind() == Kind.VARIABLE) .map(t -> (VariableTree) t) .collect(Collectors.toList()); } /** * Returns the element corresponding to the given tree. * * @param tree the tree corresponding to a use of an element * @return the element for the corresponding declaration, {@code null} otherwise * @deprecated use {@link #elementFromUse(ExpressionTree)} or {@link * #elementFromTree(ExpressionTree)} */ @Pure @Deprecated // not for removal; retain to prevent calls to this overload public static @Nullable Element elementFromDeclaration(ExpressionTree tree) { return TreeUtils.elementFromUse(tree); } /** * Returns the element corresponding to the given tree. * * @param tree the tree corresponding to a use of an element * @return the element for the corresponding declaration, {@code null} otherwise */ @Pure public static @Nullable Element elementFromTree(ExpressionTree tree) { return TreeUtils.elementFromTree((Tree) tree); } /** * Gets the element for the declaration corresponding to this use of an element. To get the * element for a declaration, use {@link #elementFromDeclaration(ClassTree)}, {@link * #elementFromDeclaration(MethodTree)}, or {@link #elementFromDeclaration(VariableTree)} instead. * *

This method is just a wrapper around {@link TreeUtils#elementFromTree(Tree)}, but this class * might be the first place someone looks for this functionality. * * @param tree the tree, which must be a use of an element * @return the element for the corresponding declaration */ @Pure public static Element elementFromUse(ExpressionTree tree) { Element result = TreeUtils.elementFromTree(tree); if (result == null) { throw new BugInCF( "argument to elementFromUse() has no element: %s [%s]", tree, tree.getClass()); } return result; } /** * Returns the VariableElement corresponding to the given use. * * @param tree the tree corresponding to a use of a VariableElement * @return the element for the corresponding declaration */ @Pure public static VariableElement variableElementFromUse(ExpressionTree tree) { VariableElement result = TreeUtils.variableElementFromTree(tree); if (result == null) { throw new BugInCF("null element for %s [%s]", tree, tree.getClass()); } return result; } /** * Returns the element for the given expression. * * @param tree the {@link Tree} node to get the element for * @return the element for the given tree, or null if one could not be found * @deprecated use elementFromUse */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable Element elementFromDeclaration(MemberSelectTree tree) { return TreeUtils.elementFromUse(tree); } /** * Returns the element for the given expression. * * @param tree the {@link Tree} node to get the element for * @return the element for the given tree, or null if one could not be found * @deprecated use elementFromUse */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable Element elementFromTree(MemberSelectTree tree) { return TreeUtils.elementFromUse(tree); } /** * Returns the element for the given expression. * * @param tree a method call * @return the element for the called method */ @Pure public static Element elementFromUse(MemberSelectTree tree) { Element result = TreeInfo.symbolFor((JCTree) tree); if (result == null) { throw new BugInCF("tree = " + tree); } return result; } /** * Returns the ExecutableElement for the called method. * * @param tree the {@link Tree} node to get the element for * @return the Element for the given tree, or null if one could not be found * @deprecated use elementFromUse */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable ExecutableElement elementFromDeclaration(MethodInvocationTree tree) { return TreeUtils.elementFromUse(tree); } /** * Returns the ExecutableElement for the called method. * * @param tree the {@link Tree} node to get the element for * @return the Element for the given tree, or null if one could not be found * @deprecated use elementFromUse */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable ExecutableElement elementFromTree(MethodInvocationTree tree) { return TreeUtils.elementFromUse(tree); } /** * Returns the ExecutableElement for the called method. * * @param tree a method call * @return the ExecutableElement for the called method */ @Pure public static ExecutableElement elementFromUse(MethodInvocationTree tree) { Element result = TreeInfo.symbolFor((JCTree) tree); if (result == null) { throw new BugInCF("tree = %s [%s]", tree, tree.getClass()); } if (!(result instanceof ExecutableElement)) { throw new BugInCF( "Method elements should be ExecutableElement. Found: %s [%s]", result, result.getClass()); } return (ExecutableElement) result; } /** * Returns the ExecutableElement for the method reference. * * @param tree a method reference * @return the ExecutableElement for the method reference */ @Pure public static ExecutableElement elementFromUse(MemberReferenceTree tree) { Element result = elementFromUse((ExpressionTree) tree); if (!(result instanceof ExecutableElement)) { throw new BugInCF( "Method reference elements should be ExecutableElement. Found: %s [%s]", result, result.getClass()); } return (ExecutableElement) result; } /** * Returns the ExecutableElement for the given method declaration. * *

The result can be null, when {@code tree} is a method in an anonymous class and that class * has not been processed yet. To work around this, adapt your processing order. * * @param tree a method declaration * @return the element for the given method */ public static @Nullable ExecutableElement elementFromDeclaration(MethodTree tree) { ExecutableElement result = (ExecutableElement) TreeInfo.symbolFor((JCTree) tree); return result; } /** * Returns the ExecutableElement for the given method declaration. * * @param tree the {@link MethodTree} node to get the element for * @return the Element for the given tree * @deprecated use elementFromDeclaration */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable ExecutableElement elementFromTree(MethodTree tree) { return elementFromDeclaration(tree); } /** * Returns the ExecutableElement for the given method declaration. * * @param tree the {@link MethodTree} node to get the element for * @return the Element for the given tree * @deprecated use elementFromDeclaration */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable ExecutableElement elementFromUse(MethodTree tree) { return elementFromDeclaration(tree); } /** * Returns the ExecutableElement for the given constructor invocation. * * @param tree the {@link NewClassTree} node to get the element for * @return the {@link ExecutableElement} for the given tree, or null if one could not be found * @throws IllegalArgumentException if {@code tree} is null or is not a valid javac-internal tree * (JCTree) * @deprecated use elementFromUse */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static ExecutableElement elementFromDeclaration(NewClassTree tree) { return TreeUtils.elementFromUse(tree); } /** * Gets the ExecutableElement for the called constructor, from a constructor invocation. * * @param tree the {@link NewClassTree} node to get the element for * @return the {@link ExecutableElement} for the given tree, or null if one could not be found * @throws IllegalArgumentException if {@code tree} is null or is not a valid javac-internal tree * (JCTree) * @deprecated use elementFromUse */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static ExecutableElement elementFromTree(NewClassTree tree) { return TreeUtils.elementFromUse(tree); } /** * Gets the ExecutableElement for the called constructor, from a constructor invocation. * * @param tree a constructor invocation * @return the ExecutableElement for the called constructor * @see #elementFromUse(NewClassTree) */ @Pure public static ExecutableElement elementFromUse(NewClassTree tree) { Element result = TreeInfo.symbolFor((JCTree) tree); if (result == null) { throw new BugInCF("null element for %s", tree); } if (!(result instanceof ExecutableElement)) { throw new BugInCF( "Constructor elements should be ExecutableElement. Found: %s [%s]", result, result.getClass()); } return (ExecutableElement) result; } /** * Returns the VariableElement corresponding to the given variable declaration. * * @param tree the variable * @return the element for the given variable */ public static @Nullable VariableElement elementFromDeclaration(VariableTree tree) { VariableElement result = (VariableElement) TreeInfo.symbolFor((JCTree) tree); // `result` can be null, for example for this variable declaration: // PureFunc f1 = TestPure1::myPureMethod; // TODO: check claim above. Initializer expression should have no impact on variable. return result; } /** * Returns the VariableElement corresponding to the given variable declaration. * * @param tree the {@link VariableTree} node to get the element for * @return the {@link VariableElement} for the given tree * @deprecated use elementFromDeclaration */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable VariableElement elementFromTree(VariableTree tree) { return elementFromDeclaration(tree); } /** * Returns the VariableElement corresponding to the given variable declaration. * * @param tree the {@link VariableTree} node to get the element for * @return the {@link VariableElement} for the given tree * @deprecated use elementFromDeclaration */ @Deprecated // not for removal; retain to prevent calls to this overload @Pure public static @Nullable VariableElement elementFromUse(VariableTree tree) { return elementFromDeclaration(tree); } /** * Returns the {@link VariableElement} for the given Tree API node. * * @param tree the {@link Tree} node to get the element for * @return the {@link VariableElement} for the given tree * @throws IllegalArgumentException if {@code tree} is null or is not a valid javac-internal tree * (JCTree) */ @Pure public static VariableElement variableElementFromTree(Tree tree) { VariableElement result = (VariableElement) TreeInfo.symbolFor((JCTree) tree); if (result == null) { throw new BugInCF("null element for %s [%s]", tree, tree.getClass()); } return result; } /** * Returns the {@link Element} for the given Tree API node. For an object instantiation returns * the value of the {@link JCNewClass#constructor} field. * *

Use this only when you do not statically know whether the tree is a declaration or a use of * an element. * * @param tree the {@link Tree} node to get the element for * @return the {@link Element} for the given tree, or null if one could not be found * @throws BugInCF if {@code tree} is null or is not a valid javac-internal tree (JCTree) */ @Pure public static @Nullable Element elementFromTree(Tree tree) { if (tree == null) { throw new BugInCF("TreeUtils.elementFromTree: tree is null"); } if (!(tree instanceof JCTree)) { throw new BugInCF( "TreeUtils.elementFromTree: tree is not a valid Javac tree but a " + tree.getClass()); } if (isExpressionTree(tree)) { tree = withoutParensOrCasts((ExpressionTree) tree); } switch (tree.getKind()) { // symbol() only works on MethodSelects, so we need to get it manually // for method invocations. case METHOD_INVOCATION: return TreeInfo.symbol(((JCMethodInvocation) tree).getMethodSelect()); case ASSIGNMENT: return TreeInfo.symbol((JCTree) ((AssignmentTree) tree).getVariable()); case ARRAY_ACCESS: return elementFromTree(((ArrayAccessTree) tree).getExpression()); case NEW_CLASS: return ((JCNewClass) tree).constructor; case MEMBER_REFERENCE: // TreeInfo.symbol, which is used in the default case, didn't handle // member references until JDK8u20. So handle it here. ExecutableElement memberResult = (ExecutableElement) ((JCMemberReference) tree).sym; return memberResult; default: Element defaultResult; if (isTypeDeclaration(tree) || tree.getKind() == Tree.Kind.VARIABLE || tree.getKind() == Tree.Kind.METHOD) { defaultResult = TreeInfo.symbolFor((JCTree) tree); } else { defaultResult = TreeInfo.symbol((JCTree) tree); } return defaultResult; } } /** * Returns the constructor invoked by {@code newClassTree} unless {@code newClassTree} is creating * an anonymous class. In which case, the super constructor is returned. * * @param newClassTree the constructor invocation * @return the super constructor invoked in the body of the anonymous constructor; or {@link * #elementFromUse(NewClassTree)} if {@code newClassTree} is not creating an anonymous class */ public static ExecutableElement getSuperConstructor(NewClassTree newClassTree) { if (newClassTree.getClassBody() == null) { return elementFromUse(newClassTree); } JCNewClass jcNewClass = (JCNewClass) newClassTree; // Anonymous constructor bodies, which are always synthetic, contain exactly one statement // in the form: // super(arg1, ...) // or // o.super(arg1, ...) // // which is a method invocation of the super constructor. // The method call is guaranteed to return nonnull. JCMethodDecl anonConstructor = (JCMethodDecl) TreeInfo.declarationFor(jcNewClass.constructor, jcNewClass); assert anonConstructor != null; assert anonConstructor.body.stats.size() == 1; JCExpressionStatement stmt = (JCExpressionStatement) anonConstructor.body.stats.head; JCMethodInvocation superInvok = (JCMethodInvocation) stmt.expr; return (ExecutableElement) TreeInfo.symbol(superInvok.meth); } /** * Determine whether the given ExpressionTree has an underlying element. * * @param tree the ExpressionTree to test * @return whether the tree refers to an identifier, member select, or method invocation */ @EnsuresNonNullIf(result = true, expression = "elementFromTree(#1)") @EnsuresNonNullIf(result = true, expression = "elementFromUse(#1)") @Pure public static boolean isUseOfElement(ExpressionTree tree) { ExpressionTree realnode = TreeUtils.withoutParens(tree); switch (realnode.getKind()) { case IDENTIFIER: case MEMBER_SELECT: case METHOD_INVOCATION: case NEW_CLASS: assert elementFromTree(tree) != null : "@AssumeAssertion(nullness): inspection"; assert elementFromUse(tree) != null : "@AssumeAssertion(nullness): inspection"; return true; default: return false; } } /** * Returns true if {@code tree} has a synthetic argument. * *

For some anonymous classes with an explicit enclosing expression, javac creates a synthetic * argument to the constructor that is the enclosing expression of the NewClassTree. Suppose a * programmer writes: * *

{@code class Outer {
   *   class Inner { }
   *     void method() {
   *       this.new Inner(){};
   *     }
   * }}
* * Java 9 javac creates the following synthetic tree for {@code this.new Inner(){}}: * *
{@code new Inner(this) {
   *   (.Outer x0) {
   *     x0.super();
   *   }
   * }}
* * Java 11 javac creates a different tree without the synthetic argument for {@code this.new * Inner(){}}; the first line in the below code differs: * *
{@code this.new Inner() {
   *   (.Outer x0) {
   *     x0.super();
   *   }
   * }}
* * @param tree a new class tree * @return true if {@code tree} has a synthetic argument */ public static boolean hasSyntheticArgument(NewClassTree tree) { if (tree.getClassBody() == null || tree.getEnclosingExpression() != null) { return false; } for (Tree member : tree.getClassBody().getMembers()) { if (member.getKind() == Tree.Kind.METHOD && isConstructor((MethodTree) member)) { MethodTree methodTree = (MethodTree) member; StatementTree f = methodTree.getBody().getStatements().get(0); return TreeUtils.getReceiverTree(((ExpressionStatementTree) f).getExpression()) != null; } } return false; } /** * Returns the name of the invoked method. * * @param tree the method invocation * @return the name of the invoked method */ public static Name methodName(MethodInvocationTree tree) { ExpressionTree expr = tree.getMethodSelect(); if (expr.getKind() == Tree.Kind.IDENTIFIER) { return ((IdentifierTree) expr).getName(); } else if (expr.getKind() == Tree.Kind.MEMBER_SELECT) { return ((MemberSelectTree) expr).getIdentifier(); } throw new BugInCF("TreeUtils.methodName: cannot be here: " + tree); } /** * Returns true if the first statement in the body is a self constructor invocation within a * constructor. * * @param tree the method declaration * @return true if the first statement in the body is a self constructor invocation within a * constructor */ public static boolean containsThisConstructorInvocation(MethodTree tree) { if (!TreeUtils.isConstructor(tree) || tree.getBody().getStatements().isEmpty()) { return false; } StatementTree st = tree.getBody().getStatements().get(0); if (!(st instanceof ExpressionStatementTree) || !(((ExpressionStatementTree) st).getExpression() instanceof MethodInvocationTree)) { return false; } MethodInvocationTree invocation = (MethodInvocationTree) ((ExpressionStatementTree) st).getExpression(); return "this".contentEquals(TreeUtils.methodName(invocation)); } /** * Returns the first statement of the tree if it is a block. If it is not a block or an empty * block, tree is returned. * * @param tree any kind of tree * @return the first statement of the tree if it is a block. If it is not a block or an empty * block, tree is returned. */ public static Tree firstStatement(Tree tree) { Tree first; if (tree.getKind() == Tree.Kind.BLOCK) { BlockTree block = (BlockTree) tree; if (block.getStatements().isEmpty()) { first = block; } else { first = block.getStatements().iterator().next(); } } else { first = tree; } return first; } /** * Determine whether the given class contains an explicit constructor. * * @param tree a class tree * @return true iff there is an explicit constructor */ public static boolean hasExplicitConstructor(ClassTree tree) { TypeElement elem = TreeUtils.elementFromDeclaration(tree); if (elem == null) { return false; } for (ExecutableElement constructorElt : ElementFilter.constructorsIn(elem.getEnclosedElements())) { if (!isSynthetic(constructorElt)) { return true; } } return false; } /** * Returns true if the given method is synthetic. Also returns true if the method is a generated * default constructor, which does not appear in source code but is not considered synthetic. * * @param ee a method or constructor element * @return true iff the given method is synthetic */ public static boolean isSynthetic(ExecutableElement ee) { MethodSymbol ms = (MethodSymbol) ee; long mod = ms.flags(); // GENERATEDCONSTR is for generated constructors, which do not have SYNTHETIC set. return (mod & (Flags.SYNTHETIC | Flags.GENERATEDCONSTR)) != 0; } /** * Returns true if the given method is synthetic. * * @param tree a method declaration tree * @return true iff the given method is synthetic */ public static boolean isSynthetic(MethodTree tree) { ExecutableElement ee = TreeUtils.elementFromDeclaration(tree); return ee != null && isSynthetic(ee); } /** * Returns true if the tree is of a diamond type. In contrast to the implementation in TreeInfo, * this version works on Trees. * * @param tree a tree * @see com.sun.tools.javac.tree.TreeInfo#isDiamond(JCTree) */ public static boolean isDiamondTree(Tree tree) { switch (tree.getKind()) { case ANNOTATED_TYPE: return isDiamondTree(((AnnotatedTypeTree) tree).getUnderlyingType()); case PARAMETERIZED_TYPE: return ((ParameterizedTypeTree) tree).getTypeArguments().isEmpty(); case NEW_CLASS: return isDiamondTree(((NewClassTree) tree).getIdentifier()); default: return false; } } /** * Returns the type arguments to the given new class tree. * * @param tree a new class tree * @return the type arguments to the given new class tree */ public static List getTypeArgumentsToNewClassTree(NewClassTree tree) { Tree typeTree = tree.getIdentifier(); if (typeTree.getKind() == Kind.ANNOTATED_TYPE) { typeTree = ((AnnotatedTypeTree) typeTree).getUnderlyingType(); } if (typeTree.getKind() == Kind.PARAMETERIZED_TYPE) { return ((ParameterizedTypeTree) typeTree).getTypeArguments(); } return Collections.emptyList(); } /** * Returns true if the tree represents a {@code String} concatenation operation. * * @param tree a tree * @return true if the tree represents a {@code String} concatenation operation */ public static boolean isStringConcatenation(Tree tree) { return (tree.getKind() == Tree.Kind.PLUS && TypesUtils.isString(TreeUtils.typeOf(tree))); } /** Returns true if the compound assignment tree is a string concatenation. */ public static boolean isStringCompoundConcatenation(CompoundAssignmentTree tree) { return (tree.getKind() == Tree.Kind.PLUS_ASSIGNMENT && TypesUtils.isString(TreeUtils.typeOf(tree))); } /** * Is this method's declared return type "void"? * * @param tree a method declaration * @return true iff method's declared return type is "void" */ public static boolean isVoidReturn(MethodTree tree) { return typeOf(tree.getReturnType()).getKind() == TypeKind.VOID; } /** * Returns true if the tree is a constant-time expression. * *

A tree is a constant-time expression if it is: * *

    *
  1. a literal tree *
  2. a reference to a final variable initialized with a compile time constant *
  3. a String concatenation of two compile time constants *
* * @param tree the tree to check * @return true if the tree is a constant-time expression */ public static boolean isCompileTimeString(ExpressionTree tree) { tree = TreeUtils.withoutParens(tree); if (tree instanceof LiteralTree) { return true; } if (TreeUtils.isUseOfElement(tree)) { Element elt = TreeUtils.elementFromUse(tree); return ElementUtils.isCompileTimeConstant(elt); } else if (TreeUtils.isStringConcatenation(tree)) { BinaryTree binOp = (BinaryTree) tree; return isCompileTimeString(binOp.getLeftOperand()) && isCompileTimeString(binOp.getRightOperand()); } else { return false; } } /** * Returns the receiver tree of a field access or a method invocation. * * @param expression a field access or a method invocation * @return the expression's receiver tree, or null if it does not have an explicit receiver */ public static @Nullable ExpressionTree getReceiverTree(ExpressionTree expression) { ExpressionTree receiver; switch (expression.getKind()) { case METHOD_INVOCATION: // Trying to handle receiver calls to trees of the form // ((m).getArray()) // returns the type of 'm' in this case receiver = ((MethodInvocationTree) expression).getMethodSelect(); if (receiver.getKind() == Tree.Kind.MEMBER_SELECT) { receiver = ((MemberSelectTree) receiver).getExpression(); } else { // It's a method call "m(foo)" without an explicit receiver return null; } break; case NEW_CLASS: receiver = ((NewClassTree) expression).getEnclosingExpression(); break; case ARRAY_ACCESS: receiver = ((ArrayAccessTree) expression).getExpression(); break; case MEMBER_SELECT: receiver = ((MemberSelectTree) expression).getExpression(); // Avoid int.class if (receiver instanceof PrimitiveTypeTree) { return null; } break; case IDENTIFIER: // It's a field access on implicit this or a local variable/parameter. return null; default: return null; } if (receiver == null) { return null; } return TreeUtils.withoutParens(receiver); } // TODO: What about anonymous classes? // Adding Tree.Kind.NEW_CLASS here doesn't work, because then a // tree gets cast to ClassTree when it is actually a NewClassTree, // for example in enclosingClass above. /** The kinds that represent classes. */ private static final Set classTreeKinds; static { classTreeKinds = EnumSet.noneOf(Tree.Kind.class); for (Tree.Kind kind : Tree.Kind.values()) { if (kind.asInterface() == ClassTree.class) { classTreeKinds.add(kind); } } } /** Kinds that represent a class or method tree. */ private static final Set classAndMethodTreeKinds; static { classAndMethodTreeKinds = EnumSet.copyOf(classTreeKinds()); classAndMethodTreeKinds.add(Tree.Kind.METHOD); } /** * Returns the set of kinds that represent classes and methods. * * @return the set of kinds that represent classes and methods */ public static Set classAndMethodTreeKinds() { return classAndMethodTreeKinds; } /** * Return the set of kinds that represent classes. * * @return the set of kinds that represent classes */ public static Set classTreeKinds() { return classTreeKinds; } /** * Is the given tree kind a class, i.e. a class, enum, interface, or annotation type. * * @param tree the tree to test * @return true, iff the given kind is a class kind */ public static boolean isClassTree(Tree tree) { return classTreeKinds().contains(tree.getKind()); } /** * The kinds that represent declarations that might have {@code @SuppressWarnings} written on * them: classes, methods, and variables. */ private static final Set declarationTreeKinds; static { declarationTreeKinds = EnumSet.noneOf(Tree.Kind.class); declarationTreeKinds.addAll(classTreeKinds); declarationTreeKinds.add(Tree.Kind.METHOD); declarationTreeKinds.add(Tree.Kind.VARIABLE); } /** * Return the set of kinds that represent declarations: classes, methods, and variables. * * @return the set of kinds that represent declarations */ public static Set declarationTreeKinds() { return declarationTreeKinds; } /** * Returns true if the given tree is a declaration. * * @param tree the tree to test * @return true if the given tree is a declaration */ public static boolean isDeclarationTree(Tree tree) { return declarationTreeKinds.contains(tree.getKind()); } /** The kinds that represent types. */ private static final Set typeTreeKinds = EnumSet.of( Tree.Kind.PRIMITIVE_TYPE, Tree.Kind.PARAMETERIZED_TYPE, Tree.Kind.TYPE_PARAMETER, Tree.Kind.ARRAY_TYPE, Tree.Kind.UNBOUNDED_WILDCARD, Tree.Kind.EXTENDS_WILDCARD, Tree.Kind.SUPER_WILDCARD, Tree.Kind.ANNOTATED_TYPE); /** * Return the set of kinds that represent types. * * @return the set of kinds that represent types */ public static Set typeTreeKinds() { return typeTreeKinds; } /** * Is the given tree a type instantiation? * *

TODO: this is an under-approximation: e.g. an identifier could be either a type use or an * expression. How can we distinguish. * * @param tree the tree to test * @return true, iff the given tree is a type */ public static boolean isTypeTree(Tree tree) { return typeTreeKinds().contains(tree.getKind()); } /** * Returns true if the given element is an invocation of the method, or of any method that * overrides that one. */ public static boolean isMethodInvocation( Tree tree, ExecutableElement method, ProcessingEnvironment env) { if (!(tree instanceof MethodInvocationTree)) { return false; } MethodInvocationTree methInvok = (MethodInvocationTree) tree; ExecutableElement invoked = TreeUtils.elementFromUse(methInvok); if (invoked == null) { return false; } return ElementUtils.isMethod(invoked, method, env); } /** * Returns true if the argument is an invocation of one of the given methods, or of any method * that overrides them. * * @param tree a tree that might be a method invocation * @param methods the methods to check for * @param processingEnv the processing environment * @return true if the argument is an invocation of one of the given methods, or of any method * that overrides them */ public static boolean isMethodInvocation( Tree tree, List methods, ProcessingEnvironment processingEnv) { if (!(tree instanceof MethodInvocationTree)) { return false; } MethodInvocationTree methInvok = (MethodInvocationTree) tree; ExecutableElement invoked = TreeUtils.elementFromUse(methInvok); if (invoked == null) { return false; } for (ExecutableElement method : methods) { if (ElementUtils.isMethod(invoked, method, processingEnv)) { return true; } } return false; } /** * Returns the ExecutableElement for a method declaration. Errs if there is not exactly one * matching method. If more than one method takes the same number of formal parameters, then use * {@link #getMethod(String, String, ProcessingEnvironment, String...)}. * * @param type the class that contains the method * @param methodName the name of the method * @param params the number of formal parameters * @param env the processing environment * @return the ExecutableElement for the specified method */ public static ExecutableElement getMethod( Class type, String methodName, int params, ProcessingEnvironment env) { String typeName = type.getCanonicalName(); if (typeName == null) { throw new BugInCF("TreeUtils.getMethod: class %s has no canonical name", type); } return getMethod(typeName, methodName, params, env); } /** * Returns the ExecutableElement for a method declaration. Errs if there is not exactly one * matching method. If more than one method takes the same number of formal parameters, then use * {@link #getMethod(String, String, ProcessingEnvironment, String...)}. * * @param typeName the class that contains the method * @param methodName the name of the method * @param params the number of formal parameters * @param env the processing environment * @return the ExecutableElement for the specified method */ public static ExecutableElement getMethod( @FullyQualifiedName String typeName, String methodName, int params, ProcessingEnvironment env) { List methods = getMethods(typeName, methodName, params, env); if (methods.size() == 1) { return methods.get(0); } throw new BugInCF( "TreeUtils.getMethod(%s, %s, %d): expected 1 match, found %d: %s", typeName, methodName, params, methods.size(), methods); } /** * Returns the ExecutableElement for a method declaration. Returns null there is no matching * method. Errs if there is more than one matching method. If more than one method takes the same * number of formal parameters, then use {@link #getMethod(String, String, ProcessingEnvironment, * String...)}. * * @param typeName the class that contains the method * @param methodName the name of the method * @param params the number of formal parameters * @param env the processing environment * @return the ExecutableElement for the specified method, or null */ public static @Nullable ExecutableElement getMethodOrNull( @FullyQualifiedName String typeName, String methodName, int params, ProcessingEnvironment env) { List methods = getMethods(typeName, methodName, params, env); if (methods.size() == 0) { return null; } else if (methods.size() == 1) { return methods.get(0); } else { throw new BugInCF( "TreeUtils.getMethod(%s, %s, %d): expected 0 or 1 match, found %d", typeName, methodName, params, methods.size()); } } /** * Returns all ExecutableElements for method declarations of methodName, in class typeName, with * params formal parameters. * * @param typeName the class that contains the method * @param methodName the name of the method * @param params the number of formal parameters * @param env the processing environment * @return the ExecutableElements for all matching methods */ public static List getMethods( @FullyQualifiedName String typeName, String methodName, int params, ProcessingEnvironment env) { List methods = new ArrayList<>(1); TypeElement typeElt = env.getElementUtils().getTypeElement(typeName); if (typeElt == null) { throw new UserError("Configuration problem! Could not load type: " + typeName); } for (ExecutableElement exec : ElementFilter.methodsIn(typeElt.getEnclosedElements())) { if (exec.getSimpleName().contentEquals(methodName) && exec.getParameters().size() == params) { methods.add(exec); } } return methods; } /** * Returns the ExecutableElement for a method declaration. Errs if there is no matching method. * * @param type the class that contains the method * @param methodName the name of the method * @param env the processing environment * @param paramTypes the method's formal parameter types * @return the ExecutableElement for the specified method */ public static ExecutableElement getMethod( Class type, String methodName, ProcessingEnvironment env, String... paramTypes) { String typeName = type.getCanonicalName(); if (typeName == null) { throw new BugInCF("TreeUtils.getMethod: class %s has no canonical name", type); } return getMethod(typeName, methodName, env, paramTypes); } /** * Returns the ExecutableElement for a method declaration. Errs if there is no matching method. * * @param typeName the class that contains the method * @param methodName the name of the method * @param env the processing environment * @param paramTypes the method's formal parameter types * @return the ExecutableElement for the specified method */ public static ExecutableElement getMethod( @FullyQualifiedName String typeName, String methodName, ProcessingEnvironment env, String... paramTypes) { TypeElement typeElt = env.getElementUtils().getTypeElement(typeName); for (ExecutableElement exec : ElementFilter.methodsIn(typeElt.getEnclosedElements())) { if (exec.getSimpleName().contentEquals(methodName) && exec.getParameters().size() == paramTypes.length) { boolean typesMatch = true; List params = exec.getParameters(); for (int i = 0; i < paramTypes.length; i++) { VariableElement ve = params.get(i); TypeMirror tm = TypeAnnotationUtils.unannotatedType(ve.asType()); if (!tm.toString().equals(paramTypes[i])) { typesMatch = false; break; } } if (typesMatch) { return exec; } } } // Didn't find an answer. Compose an error message. List candidates = new ArrayList<>(); for (ExecutableElement exec : ElementFilter.methodsIn(typeElt.getEnclosedElements())) { if (exec.getSimpleName().contentEquals(methodName)) { candidates.add(executableElementToString(exec)); } } if (candidates.isEmpty()) { for (ExecutableElement exec : ElementFilter.methodsIn(typeElt.getEnclosedElements())) { candidates.add(executableElementToString(exec)); } } throw new BugInCF( "TreeUtils.getMethod: found no match for %s.%s(%s); candidates: %s", typeName, methodName, Arrays.toString(paramTypes), candidates); } /** * Formats the ExecutableElement in the way that getMethod() expects it. * * @param exec an executable element * @return the ExecutableElement, formatted in the way that getMethod() expects it */ private static String executableElementToString(ExecutableElement exec) { StringJoiner result = new StringJoiner(", ", exec.getSimpleName() + "(", ")"); for (VariableElement param : exec.getParameters()) { result.add(TypeAnnotationUtils.unannotatedType(param.asType()).toString()); } return result.toString(); } /** * Determine whether the given expression is either "this" or an outer "C.this". * *

TODO: Should this also handle "super"? */ public static boolean isExplicitThisDereference(ExpressionTree tree) { if (tree.getKind() == Tree.Kind.IDENTIFIER && ((IdentifierTree) tree).getName().contentEquals("this")) { // Explicit this reference "this" return true; } if (tree.getKind() != Tree.Kind.MEMBER_SELECT) { return false; } MemberSelectTree memSelTree = (MemberSelectTree) tree; if (memSelTree.getIdentifier().contentEquals("this")) { // Outer this reference "C.this" return true; } return false; } /** * Determine whether {@code tree} is a class literal, such as * *

   *   Object . class
   * 
* * @return true iff if tree is a class literal */ public static boolean isClassLiteral(Tree tree) { if (tree.getKind() != Tree.Kind.MEMBER_SELECT) { return false; } return "class".equals(((MemberSelectTree) tree).getIdentifier().toString()); } /** * Determine whether {@code tree} is a field access expression, such as * *
   *   f
   *   obj . f
   * 
* * This method currently also returns true for class literals and qualified this. * * @param tree a tree that might be a field access * @return true iff if tree is a field access expression (implicit or explicit) */ public static boolean isFieldAccess(Tree tree) { return asFieldAccess(tree) != null; } /** * Return the field that {@code tree} is a field access expression for, or null. * *
   *   f
   *   obj . f
   * 
* * This method currently also returns a non-null value for class literals and qualified this. * * @param tree a tree that might be a field access * @return the element if tree is a field access expression (implicit or explicit); null otherwise */ // TODO: fix value for class literals and qualified this, which are not field accesses. public static @Nullable VariableElement asFieldAccess(Tree tree) { if (tree.getKind() == Tree.Kind.MEMBER_SELECT) { // explicit member access (or a class literal or a qualified this) MemberSelectTree memberSelect = (MemberSelectTree) tree; assert isUseOfElement(memberSelect) : "@AssumeAssertion(nullness): tree kind"; Element el = TreeUtils.elementFromUse(memberSelect); if (el.getKind().isField()) { return (VariableElement) el; } } else if (tree.getKind() == Tree.Kind.IDENTIFIER) { // implicit field access IdentifierTree ident = (IdentifierTree) tree; assert isUseOfElement(ident) : "@AssumeAssertion(nullness): tree kind"; Element el = TreeUtils.elementFromUse(ident); if (el.getKind().isField() && !ident.getName().contentEquals("this") && !ident.getName().contentEquals("super")) { return (VariableElement) el; } } return null; } /** * Compute the name of the field that the field access {@code tree} accesses. Requires {@code * tree} to be a field access, as determined by {@code isFieldAccess} (which currently also * returns true for class literals and qualified this). * * @param tree a field access tree * @return the name of the field accessed by {@code tree} */ public static String getFieldName(Tree tree) { assert isFieldAccess(tree); if (tree.getKind() == Tree.Kind.MEMBER_SELECT) { MemberSelectTree mtree = (MemberSelectTree) tree; return mtree.getIdentifier().toString(); } else { IdentifierTree itree = (IdentifierTree) tree; return itree.getName().toString(); } } /** * Determine whether {@code tree} refers to a method element, such as. * *
   *   m(...)
   *   obj . m(...)
   * 
* * @return true iff if tree is a method access expression (implicit or explicit) */ public static boolean isMethodAccess(Tree tree) { if (tree.getKind() == Tree.Kind.MEMBER_SELECT) { // explicit method access MemberSelectTree memberSelect = (MemberSelectTree) tree; assert isUseOfElement(memberSelect) : "@AssumeAssertion(nullness): tree kind"; Element el = TreeUtils.elementFromUse(memberSelect); return el.getKind() == ElementKind.METHOD || el.getKind() == ElementKind.CONSTRUCTOR; } else if (tree.getKind() == Tree.Kind.IDENTIFIER) { // implicit method access IdentifierTree ident = (IdentifierTree) tree; // The field "super" and "this" are also legal methods if (ident.getName().contentEquals("super") || ident.getName().contentEquals("this")) { return true; } assert isUseOfElement(ident) : "@AssumeAssertion(nullness): tree kind"; Element el = TreeUtils.elementFromUse(ident); return el.getKind() == ElementKind.METHOD || el.getKind() == ElementKind.CONSTRUCTOR; } return false; } /** * Compute the name of the method that the method access {@code tree} accesses. Requires {@code * tree} to be a method access, as determined by {@code isMethodAccess}. * * @param tree a method access tree * @return the name of the method accessed by {@code tree} */ public static String getMethodName(Tree tree) { assert isMethodAccess(tree); if (tree.getKind() == Tree.Kind.MEMBER_SELECT) { MemberSelectTree mtree = (MemberSelectTree) tree; return mtree.getIdentifier().toString(); } else { IdentifierTree itree = (IdentifierTree) tree; return itree.getName().toString(); } } /** * Return {@code true} if and only if {@code tree} can have a type annotation. * * @return {@code true} if and only if {@code tree} can have a type annotation */ // TODO: is this implementation precise enough? E.g. does a .class literal work correctly? public static boolean canHaveTypeAnnotation(Tree tree) { return ((JCTree) tree).type != null; } /** * Returns true if and only if the given {@code tree} represents a field access of the given * {@link VariableElement}. */ public static boolean isSpecificFieldAccess(Tree tree, VariableElement var) { if (tree instanceof MemberSelectTree) { MemberSelectTree memSel = (MemberSelectTree) tree; assert isUseOfElement(memSel) : "@AssumeAssertion(nullness): tree kind"; Element field = TreeUtils.elementFromUse(memSel); return field.equals(var); } else if (tree instanceof IdentifierTree) { IdentifierTree idTree = (IdentifierTree) tree; assert isUseOfElement(idTree) : "@AssumeAssertion(nullness): tree kind"; Element field = TreeUtils.elementFromUse(idTree); return field.equals(var); } else { return false; } } /** * Returns the VariableElement for a field declaration. * * @param typeName the class where the field is declared * @param fieldName the name of the field * @param env the processing environment * @return the VariableElement for typeName.fieldName */ public static VariableElement getField( @FullyQualifiedName String typeName, String fieldName, ProcessingEnvironment env) { TypeElement mapElt = env.getElementUtils().getTypeElement(typeName); for (VariableElement var : ElementFilter.fieldsIn(mapElt.getEnclosedElements())) { if (var.getSimpleName().contentEquals(fieldName)) { return var; } } throw new BugInCF("TreeUtils.getField: shouldn't be here"); } /** * Determine whether the given tree represents an ExpressionTree. * * @param tree the Tree to test * @return whether the tree is an ExpressionTree */ public static boolean isExpressionTree(Tree tree) { return tree instanceof ExpressionTree; } /** * Returns true if this is a super call to the {@link Enum} constructor. * * @param tree the method invocation to check * @return true if this is a super call to the {@link Enum} constructor */ public static boolean isEnumSuperCall(MethodInvocationTree tree) { ExecutableElement ex = TreeUtils.elementFromUse(tree); assert ex != null : "@AssumeAssertion(nullness): tree kind"; Name name = ElementUtils.getQualifiedClassName(ex); assert name != null : "@AssumeAssertion(nullness): assumption"; boolean correctClass = "java.lang.Enum".contentEquals(name); boolean correctMethod = "".contentEquals(ex.getSimpleName()); return correctClass && correctMethod; } /** * Determine whether the given tree represents a declaration of a type (including type * parameters). * * @param tree the Tree to test * @return true if the tree is a type declaration */ public static boolean isTypeDeclaration(Tree tree) { return isClassTree(tree) || tree.getKind() == Tree.Kind.TYPE_PARAMETER; } /** * Returns true if tree is an access of array length. * * @param tree tree to check * @return true if tree is an access of array length */ public static boolean isArrayLengthAccess(Tree tree) { if (tree.getKind() == Tree.Kind.MEMBER_SELECT && isFieldAccess(tree) && getFieldName(tree).equals("length")) { ExpressionTree expressionTree = ((MemberSelectTree) tree).getExpression(); if (TreeUtils.typeOf(expressionTree).getKind() == TypeKind.ARRAY) { return true; } } return false; } /** * Returns true if the given {@link MethodTree} is an anonymous constructor (the constructor for * an anonymous class). * * @param method a method tree that may be an anonymous constructor * @return true if the given path points to an anonymous constructor, false if it does not */ public static boolean isAnonymousConstructor(MethodTree method) { Element e = elementFromTree(method); if (e == null || e.getKind() != ElementKind.CONSTRUCTOR) { return false; } TypeElement typeElement = (TypeElement) e.getEnclosingElement(); return typeElement.getNestingKind() == NestingKind.ANONYMOUS; } /** * Returns true if the passed constructor is anonymous and has an explicit enclosing expression. * * @param con an ExecutableElement of a constructor declaration * @param tree the NewClassTree of a constructor declaration * @return true if there is an extra enclosing expression */ public static boolean isAnonymousConstructorWithExplicitEnclosingExpression( ExecutableElement con, NewClassTree tree) { return (tree.getEnclosingExpression() != null) && con.getKind() == ElementKind.CONSTRUCTOR && ((TypeElement) con.getEnclosingElement()).getNestingKind() == NestingKind.ANONYMOUS; } /** * Returns true if the given {@link MethodTree} is a compact canonical constructor (the * constructor for a record where the parameters are implicitly declared and implicitly assigned * to the record's fields). This may be an explicitly declared compact canonical constructor or an * implicitly generated one. * * @param method a method tree that may be a compact canonical constructor * @return true if the given method is a compact canonical constructor */ public static boolean isCompactCanonicalRecordConstructor(MethodTree method) { Symbol s = (Symbol) elementFromTree(method); if (s == null) { throw new BugInCF( "TreeUtils.isCompactCanonicalRecordConstructor: null symbol for method tree: " + method); } return (s.flags() & Flags_RECORD) != 0; } /** * Returns true if the given {@link Tree} is part of a record that has been automatically * generated by the compiler. This can be a field that is derived from the record's header field * list, or an automatically generated canonical constructor. * * @param member the {@link Tree} for a member of a record * @return true if the given path is generated by the compiler */ public static boolean isAutoGeneratedRecordMember(Tree member) { Element e = elementFromTree(member); if (e == null) { throw new BugInCF( "TreeUtils.isAutoGeneratedRecordMember: null element for member tree: " + member); } return ElementUtils.isAutoGeneratedRecordMember(e); } /** * Converts the given AnnotationTrees to AnnotationMirrors. * * @param annoTrees list of annotation trees to convert to annotation mirrors * @return list of annotation mirrors that represent the given annotation trees */ public static List annotationsFromTypeAnnotationTrees( List annoTrees) { return CollectionsPlume.mapList(TreeUtils::annotationFromAnnotationTree, annoTrees); } /** * Converts the given AnnotationTree to an AnnotationMirror. * * @param tree annotation tree to convert to an annotation mirror * @return annotation mirror that represent the given annotation tree */ public static AnnotationMirror annotationFromAnnotationTree(AnnotationTree tree) { return ((JCAnnotation) tree).attribute; } /** * Converts the given AnnotatedTypeTree to a list of AnnotationMirrors. * * @param tree annotated type tree to convert * @return list of AnnotationMirrors from the tree */ public static List annotationsFromTree(AnnotatedTypeTree tree) { return annotationsFromTypeAnnotationTrees(((JCAnnotatedType) tree).annotations); } /** * Converts the given TypeParameterTree to a list of AnnotationMirrors. * * @param tree type parameter tree to convert * @return list of AnnotationMirrors from the tree */ public static List annotationsFromTree(TypeParameterTree tree) { return annotationsFromTypeAnnotationTrees(((JCTypeParameter) tree).annotations); } /** * Converts the given NewArrayTree to a list of AnnotationMirrors. * * @param tree new array tree * @return list of AnnotationMirrors from the tree */ public static List annotationsFromArrayCreation( NewArrayTree tree, int level) { assert tree instanceof JCNewArray; JCNewArray newArray = ((JCNewArray) tree); if (level == -1) { return annotationsFromTypeAnnotationTrees(newArray.annotations); } if (newArray.dimAnnotations.length() > 0 && (level >= 0) && (level < newArray.dimAnnotations.size())) { return annotationsFromTypeAnnotationTrees(newArray.dimAnnotations.get(level)); } return Collections.emptyList(); } /** * Returns true if the tree is the declaration or use of a local variable. * * @param tree the tree to check * @return true if the tree is the declaration or use of a local variable */ public static boolean isLocalVariable(Tree tree) { if (tree.getKind() == Tree.Kind.VARIABLE) { VariableElement varElt = elementFromDeclaration((VariableTree) tree); return varElt != null && ElementUtils.isLocalVariable(varElt); } else if (tree.getKind() == Tree.Kind.IDENTIFIER) { ExpressionTree etree = (ExpressionTree) tree; assert isUseOfElement(etree) : "@AssumeAssertion(nullness): tree kind"; return ElementUtils.isLocalVariable(elementFromUse(etree)); } return false; } /** * Returns the type as a TypeMirror of {@code tree}. To obtain {@code tree}'s AnnotatedTypeMirror, * call {@code AnnotatedTypeFactory.getAnnotatedType()}. * *

Note that for the expression "super", this method returns the type of "this", not "this"'s * superclass. * * @return the type as a TypeMirror of {@code tree} */ public static TypeMirror typeOf(Tree tree) { return ((JCTree) tree).type; } /** * Determines the type for a method invocation at its call site, which has all type variables * substituted with the type arguments at the call site. * *

{@link javax.lang.model.type.TypeVariable} in the returned type should be compared using * {@link TypesUtils#areSame(TypeVariable, TypeVariable)} because the {@code TypeVariable} will be * freshly created by this method and will not be the same using {@link Object#equals(Object)} or * {@link javax.lang.model.util.Types#isSameType(TypeMirror, TypeMirror)}. * * @param tree the method invocation * @return the {@link ExecutableType} corresponding to the method invocation at its call site */ @Pure public static ExecutableType typeFromUse(MethodInvocationTree tree) { TypeMirror type = TreeUtils.typeOf(tree.getMethodSelect()); if (!(type instanceof ExecutableType)) { throw new BugInCF( "TreeUtils.typeFromUse(MethodInvocationTree): type of method select in method" + " invocation should be ExecutableType. Found: %s", type); } ExecutableType executableType = (ExecutableType) type; ExecutableElement element = elementFromUse(tree); if (executableType.getParameterTypes().size() != element.getParameters().size()) { // Sometimes when the method type is viewpoint-adapted, the vararg parameter disappears, // just return the declared type. // For example, // static void call(MethodHandle methodHandle) throws Throwable { // methodHandle.invoke(); // } return (ExecutableType) element.asType(); } return executableType; } /** * Determines the type for a constructor at its call site given an invocation via {@code new}, * which has all type variables substituted with the type arguments at the call site. * * @param tree the constructor invocation * @return the {@link ExecutableType} corresponding to the constructor call (i.e., the given * {@code tree}) at its call site */ @Pure public static ExecutableType typeFromUse(NewClassTree tree) { if (!(tree instanceof JCTree.JCNewClass)) { throw new BugInCF("TreeUtils.typeFromUse(NewClassTree): not a javac internal tree"); } JCNewClass newClassTree = (JCNewClass) tree; TypeMirror type = newClassTree.constructorType; if (!(type instanceof ExecutableType)) { throw new BugInCF( "TreeUtils.typeFromUse(NewClassTree): type of constructor in new class tree" + " should be ExecutableType. Found: %s", type); } return (ExecutableType) type; } /** * Determines the symbol for a constructor given an invocation via {@code new}. * * @see #elementFromUse(NewClassTree) * @param tree the constructor invocation * @return the {@link ExecutableElement} corresponding to the constructor call in {@code tree} * @deprecated use elementFromUse instead */ @Deprecated // 2022-09-12 public static ExecutableElement constructor(NewClassTree tree) { return (ExecutableElement) ((JCNewClass) tree).constructor; } /** * The type of the lambda or method reference tree is a functional interface type. This method * returns the single abstract method declared by that functional interface. (The type of this * method is referred to as the function type.) * * @param tree lambda or member reference tree * @param env the processing environment * @return the single abstract method declared by the type of the tree */ public static ExecutableElement findFunction(Tree tree, ProcessingEnvironment env) { Context ctx = ((JavacProcessingEnvironment) env).getContext(); Types javacTypes = Types.instance(ctx); return (ExecutableElement) javacTypes.findDescriptorSymbol(((Type) typeOf(tree)).asElement()); } /** * Returns true if {@code tree} is an implicitly typed lambda. * *

A lambda expression whose formal type parameters have inferred types is an implicitly typed * lambda. (See JLS 15.27.1) * * @param tree any kind of tree * @return true iff {@code tree} is an implicitly typed lambda */ public static boolean isImplicitlyTypedLambda(Tree tree) { return tree.getKind() == Tree.Kind.LAMBDA_EXPRESSION && ((JCLambda) tree).paramKind == ParameterKind.IMPLICIT; } /** * This is a duplication of {@code * com.sun.tools.javac.tree.JCTree.JCMemberReference.ReferenceKind}, which is not part of the * supported javac API. */ public enum MemberReferenceKind { /** super # instMethod */ SUPER(ReferenceMode.INVOKE, false), /** Type # instMethod */ UNBOUND(ReferenceMode.INVOKE, true), /** Type # staticMethod */ STATIC(ReferenceMode.INVOKE, false), /** Expr # instMethod */ BOUND(ReferenceMode.INVOKE, false), /** Inner # new */ IMPLICIT_INNER(ReferenceMode.NEW, false), /** Toplevel # new */ TOPLEVEL(ReferenceMode.NEW, false), /** ArrayType # new */ ARRAY_CTOR(ReferenceMode.NEW, false); /** Whether this kind is a method reference or a constructor reference. */ final ReferenceMode mode; /** Whether this kind is unbound. */ final boolean unbound; /** * Creates a MemberReferenceKind. * * @param mode whether this kind is a method reference or a constructor reference * @param unbound whether the kind is not bound */ MemberReferenceKind(ReferenceMode mode, boolean unbound) { this.mode = mode; this.unbound = unbound; } /** * Whether this kind is unbound. * * @return Whether this kind is unbound */ public boolean isUnbound() { return unbound; } /** * Returns whether this kind is a constructor reference. * * @return whether this kind is a constructor reference */ public boolean isConstructorReference() { return mode == ReferenceMode.NEW; } /** * Returns the kind of member reference {@code tree} is. * * @param tree a member reference tree * @return the kind of member reference {@code tree} is */ public static MemberReferenceKind getMemberReferenceKind(MemberReferenceTree tree) { JCMemberReference memberTree = (JCMemberReference) tree; switch (memberTree.kind) { case SUPER: return SUPER; case UNBOUND: return UNBOUND; case STATIC: return STATIC; case BOUND: return BOUND; case IMPLICIT_INNER: return IMPLICIT_INNER; case TOPLEVEL: return TOPLEVEL; case ARRAY_CTOR: return ARRAY_CTOR; default: throw new BugInCF("Unexpected ReferenceKind: %s", memberTree.kind); } } } /** * Determine whether an expression {@link ExpressionTree} has the constant value true, according * to the compiler logic. * * @param tree the expression to be checked * @return true if {@code tree} has the constant value true */ public static boolean isExprConstTrue(ExpressionTree tree) { assert tree instanceof JCExpression; if (((JCExpression) tree).type.isTrue()) { return true; } tree = TreeUtils.withoutParens(tree); if (tree instanceof JCTree.JCBinary) { JCBinary binTree = (JCBinary) tree; JCExpression ltree = binTree.lhs; JCExpression rtree = binTree.rhs; switch (binTree.getTag()) { case AND: return isExprConstTrue(ltree) && isExprConstTrue(rtree); case OR: return isExprConstTrue(ltree) || isExprConstTrue(rtree); default: break; } } return false; } /** * Return toString(), but without line separators. * * @param tree a tree * @return a one-line string representation of the tree */ public static String toStringOneLine(Tree tree) { return tree.toString().trim().replaceAll("\\s+", " "); } /** * Return either {@link #toStringOneLine} if it is no more than {@code length} characters, or * {@link #toStringOneLine} quoted and truncated. * * @param tree a tree * @param length the maximum length for the result; must be at least 6 * @return a one-line string representation of the tree that is no longer than {@code length} * characters long */ public static String toStringTruncated(Tree tree, int length) { if (length < 6) { throw new BugInCF("TreeUtils.toStringTruncated: bad length " + length); } String result = toStringOneLine(tree); if (result.length() > length) { // The quoting increases the likelihood that all delimiters are balanced in the result. // That makes it easier to manipulate the result (such as skipping over it) in an // editor. The quoting also makes clear that the value is truncated. result = "\"" + result.substring(0, length - 5) + "...\""; } return result; } /** * Given a javac ExpressionTree representing a fully qualified name such as "java.lang.Object", * creates a String containing the name. * * @param nameExpr an ExpressionTree representing a fully qualified name * @return a String representation of the fully qualified name */ public static String nameExpressionToString(ExpressionTree nameExpr) { TreeVisitor visitor = new SimpleTreeVisitor() { @Override public String visitIdentifier(IdentifierTree tree, Void p) { return tree.toString(); } @Override public String visitMemberSelect(MemberSelectTree tree, Void p) { return tree.getExpression().accept(this, null) + "." + tree.getIdentifier().toString(); } }; return nameExpr.accept(visitor, null); } /** * Returns true if the binary operator may do a widening primitive conversion. See JLS chapter 5. * * @param tree a binary tree * @return true if the tree's operator does numeric promotion on its arguments */ public static boolean isWideningBinary(BinaryTree tree) { switch (tree.getKind()) { case LEFT_SHIFT: case LEFT_SHIFT_ASSIGNMENT: case RIGHT_SHIFT: case RIGHT_SHIFT_ASSIGNMENT: case UNSIGNED_RIGHT_SHIFT: case UNSIGNED_RIGHT_SHIFT_ASSIGNMENT: // Strictly speaking, these operators do unary promotion on each argument // separately. return true; case MULTIPLY: case MULTIPLY_ASSIGNMENT: case DIVIDE: case DIVIDE_ASSIGNMENT: case REMAINDER: case REMAINDER_ASSIGNMENT: case PLUS: case PLUS_ASSIGNMENT: case MINUS: case MINUS_ASSIGNMENT: case LESS_THAN: case LESS_THAN_EQUAL: case GREATER_THAN: case GREATER_THAN_EQUAL: case EQUAL_TO: case NOT_EQUAL_TO: case AND: case XOR: case OR: // These operators do binary promotion on the two arguments together. return true; // TODO: CONDITIONAL_EXPRESSION (?:) sometimes does numeric promotion. default: return false; } } /** * Returns the annotations explicitly written on the given type. * * @param annoTrees annotations written before a variable/method declaration; null if this type is * not from such a location. This might contain type annotations that the Java parser attached * to the declaration rather than to the type. * @param typeTree the type whose annotations to return * @return the annotations explicitly written on the given type */ public static List getExplicitAnnotationTrees( @Nullable List annoTrees, Tree typeTree) { while (true) { switch (typeTree.getKind()) { case IDENTIFIER: case PRIMITIVE_TYPE: if (annoTrees == null) { return Collections.emptyList(); } return annoTrees; case ANNOTATED_TYPE: return ((AnnotatedTypeTree) typeTree).getAnnotations(); case ARRAY_TYPE: case TYPE_PARAMETER: case UNBOUNDED_WILDCARD: case EXTENDS_WILDCARD: case SUPER_WILDCARD: return Collections.emptyList(); case MEMBER_SELECT: if (annoTrees == null) { return Collections.emptyList(); } typeTree = ((MemberSelectTree) typeTree).getExpression(); break; case PARAMETERIZED_TYPE: typeTree = ((ParameterizedTypeTree) typeTree).getType(); break; case UNION_TYPE: List alternatives = ((UnionTypeTree) typeTree).getTypeAlternatives(); List result = new ArrayList<>(alternatives.size()); for (Tree alternative : alternatives) { result.addAll(getExplicitAnnotationTrees(null, alternative)); } return result; default: throw new BugInCF( "TreeUtils.getExplicitAnnotationTrees: what typeTree? %s %s %s", typeTree.getKind(), typeTree.getClass(), typeTree); } } } /** * Return a tree for the default value of the given type. The default value is 0, false, or null. * * @param typeMirror a type * @param processingEnv the processing environment * @return a tree for {@code type}'s default value */ public static LiteralTree getDefaultValueTree( TypeMirror typeMirror, ProcessingEnvironment processingEnv) { typeMirror = TypeAnnotationUtils.unannotatedType(typeMirror); switch (typeMirror.getKind()) { case BYTE: case SHORT: case INT: // Byte should be (byte) 0, but this probably doesn't matter so just use int 0; // Short should be (short) 0, but this probably doesn't matter so just use int 0; return TreeUtils.createLiteral(TypeTag.INT, 0, typeMirror, processingEnv); case CHAR: // Value of a char literal needs to be stored as an integer because // LiteralTree#getValue converts it from an integer to a char before being // returned. return TreeUtils.createLiteral(TypeTag.CHAR, (int) '\u0000', typeMirror, processingEnv); case LONG: return TreeUtils.createLiteral(TypeTag.LONG, 0L, typeMirror, processingEnv); case FLOAT: return TreeUtils.createLiteral(TypeTag.FLOAT, 0.0f, typeMirror, processingEnv); case DOUBLE: return TreeUtils.createLiteral(TypeTag.DOUBLE, 0.0d, typeMirror, processingEnv); case BOOLEAN: // Value of a boolean literal needs to be stored as an integer because // LiteralTree#getValue converts it from an integer to a boolean before being // returned. return TreeUtils.createLiteral(TypeTag.BOOLEAN, 0, typeMirror, processingEnv); default: return TreeUtils.createLiteral( TypeTag.BOT, null, processingEnv.getTypeUtils().getNullType(), processingEnv); } } /** * Creates a LiteralTree for the given value. * * @param typeTag the literal's type tag * @param value a wrapped primitive, null, or a String * @param typeMirror the typeMirror for the literal * @param processingEnv the processing environment * @return a LiteralTree for the given type tag and value */ public static LiteralTree createLiteral( TypeTag typeTag, @Nullable Object value, TypeMirror typeMirror, ProcessingEnvironment processingEnv) { Context context = ((JavacProcessingEnvironment) processingEnv).getContext(); TreeMaker maker = TreeMaker.instance(context); LiteralTree result = maker.Literal(typeTag, value); ((JCLiteral) result).type = (Type) typeMirror; return result; } /** * Returns true if the given tree evaluates to {@code null}. * * @param t a tree * @return true if the given tree evaluates to {@code null} */ public static boolean isNullExpression(Tree t) { while (true) { switch (t.getKind()) { case PARENTHESIZED: t = ((ParenthesizedTree) t).getExpression(); break; case TYPE_CAST: t = ((TypeCastTree) t).getExpression(); break; case NULL_LITERAL: return true; default: return false; } } } /** * Returns true if two expressions originating from the same scope are identical, i.e. they are * syntactically represented in the same way (modulo parentheses) and represent the same value. * *

If the expression includes one or more method calls, assumes the method calls are * deterministic. * * @param expr1 the first expression to compare * @param expr2 the second expression to compare; expr2 must originate from the same scope as * expr1 * @return true if the expressions expr1 and expr2 are syntactically identical */ public static boolean sameTree(ExpressionTree expr1, ExpressionTree expr2) { expr1 = TreeUtils.withoutParens(expr1); expr2 = TreeUtils.withoutParens(expr2); // Converting to a string in order to compare is somewhat inefficient, and it doesn't handle // internal parentheses. We could create a visitor instead. return expr1.getKind() == expr2.getKind() && expr1.toString().equals(expr2.toString()); } /** * Returns true if this is the default case for a switch statement or expression. (Also, returns * true if {@code caseTree} is {@code case null, default:}.) * * @param caseTree a case tree * @return true if {@code caseTree} is the default case for a switch statement or expression * @deprecated use {@link CaseUtils#isDefaultCaseTree(CaseTree)} */ @Deprecated // 2023-09-26 public static boolean isDefaultCaseTree(CaseTree caseTree) { return CaseUtils.isDefaultCaseTree(caseTree); } /** * Returns true if this is a case rule (as opposed to a case statement). * * @param caseTree a case tree * @return true if {@code caseTree} is a case rule * @deprecated use {@link CaseUtils#isCaseRule(CaseTree)} */ @Deprecated // 2023-09-26 public static boolean isCaseRule(CaseTree caseTree) { return CaseUtils.isCaseRule(caseTree); } /** * Get the list of expressions from a case expression. For the default case, this is empty. * Otherwise, in JDK 11 and earlier, this is a singleton list. In JDK 12 onwards, there can be * multiple expressions per case. * * @param caseTree the case expression to get the expressions from * @return the list of expressions in the case * @deprecated use {@link CaseUtils#getExpressions(CaseTree)} */ @Deprecated // 2023-09-26 public static List caseTreeGetExpressions(CaseTree caseTree) { return CaseUtils.getExpressions(caseTree); } /** * Returns the body of the case statement if it is of the form {@code case -> * }. This method should only be called if {@link CaseTree#getStatements()} returns * null. * * @param caseTree the case expression to get the body from * @return the body of the case tree * @deprecated use {@link CaseUtils#getBody(CaseTree)} */ @Deprecated // 2023-09-26 public static @Nullable Tree caseTreeGetBody(CaseTree caseTree) { return CaseUtils.getBody(caseTree); } /** * Returns true if {@code tree} is a {@code BindingPatternTree}. * * @param tree a tree to check * @return true if {@code tree} is a {@code BindingPatternTree} */ public static boolean isBindingPatternTree(Tree tree) { return tree.getKind().name().contentEquals("BINDING_PATTERN"); } /** * Returns the binding variable of {@code bindingPatternTree}. * * @param bindingPatternTree the BindingPatternTree whose binding variable is returned * @return the binding variable of {@code bindingPatternTree} * @deprecated use {@link BindingPatternUtils#getVariable(Tree)} */ @Deprecated // 2023-09-26 public static VariableTree bindingPatternTreeGetVariable(Tree bindingPatternTree) { return BindingPatternUtils.getVariable(bindingPatternTree); } /** * Returns true if {@code tree} is a {@code DeconstructionPatternTree}. * * @param tree a tree to check * @return true if {@code tree} is a {@code DeconstructionPatternTree} */ public static boolean isDeconstructionPatternTree(Tree tree) { return tree.getKind().name().contentEquals("DECONSTRUCTION_PATTERN"); } /** * Returns the pattern of {@code instanceOfTree} tree. Returns null if the instanceof does not * have a pattern, including if the JDK version does not support instance-of patterns. * * @param instanceOfTree the {@link InstanceOfTree} whose pattern is returned * @return the {@code PatternTree} of {@code instanceOfTree} or null if it doesn't exist * @deprecated use {@link InstanceOfUtils#getPattern(InstanceOfTree)} */ @Deprecated // 2023-09-26 public static @Nullable Tree instanceOfTreeGetPattern(InstanceOfTree instanceOfTree) { return InstanceOfUtils.getPattern(instanceOfTree); } /** * Returns the selector expression of {@code switchExpressionTree}. For example * *

   *   switch ( expression ) { ... }
   * 
* * @param switchExpressionTree the switch expression whose selector expression is returned * @return the selector expression of {@code switchExpressionTree} * @deprecated use {@link SwitchExpressionUtils#getExpression(Tree)} */ @Deprecated // 2023-09-26 public static ExpressionTree switchExpressionTreeGetExpression(Tree switchExpressionTree) { return SwitchExpressionUtils.getExpression(switchExpressionTree); } /** * Returns the cases of {@code switchExpressionTree}. For example * *
   *   switch ( expression ) {
   *     cases
   *   }
   * 
* * @param switchExpressionTree the switch expression whose cases are returned * @return the cases of {@code switchExpressionTree} * @deprecated use {@link SwitchExpressionUtils#getCases(Tree)} */ @Deprecated // 2023-09-26 public static List switchExpressionTreeGetCases(Tree switchExpressionTree) { return SwitchExpressionUtils.getCases(switchExpressionTree); } /** * Returns true if {@code switchTree} has a null case label. * * @param switchTree a {@link SwitchTree} or a {@code SwitchExpressionTree} * @return true if {@code switchTree} has a null case label */ public static boolean hasNullCaseLabel(Tree switchTree) { if (!atLeastJava21) { return false; } List cases; if (isSwitchStatement(switchTree)) { cases = ((SwitchTree) switchTree).getCases(); } else { cases = SwitchExpressionUtils.getCases(switchTree); } for (CaseTree caseTree : cases) { List labels = CaseUtils.getLabels(caseTree); for (Tree label : labels) { if (label.getKind() == Kind.NULL_LITERAL) { return true; } } } return false; } /** * Returns true if the given tree is a switch statement (as opposed to a switch expression). * * @param tree the switch statement or expression to check * @return true if the given tree is a switch statement (as opposed to a switch expression) */ public static boolean isSwitchStatement(Tree tree) { return tree.getKind() == Tree.Kind.SWITCH; } /** * Returns true if the given tree is a switch expression. * * @param tree a tree to check * @return true if the given tree is a switch expression */ public static boolean isSwitchExpression(Tree tree) { return tree.getKind().name().equals("SWITCH_EXPRESSION"); } /** * Returns true if the given tree is a yield expression. * * @param tree a tree to check * @return true if the given tree is a yield expression */ public static boolean isYield(Tree tree) { return tree.getKind().name().equals("YIELD"); } /** * Returns true if the given switch statement tree is an enhanced switch statement, as described * in JSL * 14.11.2. * * @param switchTree the switch statement to check * @return true if the given tree is an enhanced switch statement */ public static boolean isEnhancedSwitchStatement(SwitchTree switchTree) { TypeMirror exprType = typeOf(switchTree.getExpression()); // TODO: this should be only char, byte, short, int, Character, Byte, Short, Integer. Is the // over-approximation a problem? Element exprElem = TypesUtils.getTypeElement(exprType); boolean isNotEnum = exprElem == null || exprElem.getKind() != ElementKind.ENUM; if (!TypesUtils.isPrimitiveOrBoxed(exprType) && !TypesUtils.isString(exprType) && isNotEnum) { return true; } for (CaseTree caseTree : switchTree.getCases()) { for (Tree caseLabel : CaseUtils.getLabels(caseTree)) { if (caseLabel.getKind() == Tree.Kind.NULL_LITERAL || TreeUtils.isBindingPatternTree(caseLabel) || TreeUtils.isDeconstructionPatternTree(caseLabel)) { return true; } } } return false; } /** * Returns the value (expression) for {@code yieldTree}. * * @param yieldTree the yield tree * @return the value (expression) for {@code yieldTree} * @deprecated use {@link YieldUtils#getValue(Tree)} */ @Deprecated // 2023-09-26 public static ExpressionTree yieldTreeGetValue(Tree yieldTree) { return YieldUtils.getValue(yieldTree); } /** * Returns true if the {@code variableTree} is declared using the {@code var} Java keyword. * * @param variableTree the variableTree to check * @return true if the variableTree is declared using the {@code var} Java keyword */ public static boolean isVariableTreeDeclaredUsingVar(VariableTree variableTree) { JCExpression type = (JCExpression) variableTree.getType(); return type != null && type.pos == Position.NOPOS; } /** * Returns true if the given method reference has a varargs formal parameter. * * @param methref a method reference * @return if the given method reference has a varargs formal parameter */ public static boolean hasVarargsParameter(MemberReferenceTree methref) { JCMemberReference jcMethoRef = (JCMemberReference) methref; return jcMethoRef.varargsElement != null; } /** * Returns true if the given method/constructor invocation is a varargs invocation. * * @param tree a method/constructor invocation * @return true if the given method/constructor invocation is a varargs invocation */ public static boolean isVarargsCall(Tree tree) { switch (tree.getKind()) { case METHOD_INVOCATION: return isVarargsCall((MethodInvocationTree) tree); case NEW_CLASS: return isVarargsCall((NewClassTree) tree); case MEMBER_REFERENCE: return hasVarargsParameter((MemberReferenceTree) tree); default: return false; } } /** * Returns true if the given method invocation is a varargs invocation. * * @param invok the method invocation * @return true if the given method invocation is a varargs invocation * @deprecated use {@link #isVarargsCall(MethodInvocationTree)} */ @Deprecated // 2024-06-04 public static boolean isVarArgs(MethodInvocationTree invok) { return ((JCMethodInvocation) invok).varargsElement != null; } /** * Returns true if the given method invocation is a varargs invocation. * * @param invok the method invocation * @return true if the given method invocation is a varargs invocation */ public static boolean isVarargsCall(MethodInvocationTree invok) { if (((JCMethodInvocation) invok).varargsElement != null) { return true; } // For some calls the varargsElement element disappears when it should not. This seems to // only be a problem with MethodHandle#invoke and only with no arguments. See // framework/tests/all-systems/Issue6078.java. // So also check for a mismatch between parameter and argument size. // Such a mismatch occurs for every enum constructor: no args, two params (String name, int // ordinal). List parameters = elementFromUse(invok).getParameters(); int numParameters = parameters.size(); if (numParameters != invok.getArguments().size()) { if (numParameters > 0 && parameters.get(numParameters - 1).asType() instanceof ArrayType) { return true; } } return false; } /** * Returns true if the given method invocation is an invocation of a method with a vararg * parameter, and the invocation has zero vararg actuals. * * @param invok the method invocation * @return true if the given method invocation is an invocation of a method with a vararg * parameter, and the invocation has with zero vararg actuals */ public static boolean isCallToVarargsMethodWithZeroVarargsActuals(MethodInvocationTree invok) { if (!TreeUtils.isVarArgs(invok)) { return false; } int numParams = elementFromUse(invok).getParameters().size(); // The comparison of the number of arguments to the number of formals (minus one) checks // whether there are no varargs actuals. return invok.getArguments().size() == numParams - 1; } /** * Returns true if the given constructor invocation is a varargs invocation. * * @param newClassTree the constructor invocation * @return true if the given method invocation is a varargs invocation * @deprecated use {@link #isVarargsCall(NewClassTree)} */ @Deprecated // 2024-06-04 public static boolean isVarArgs(NewClassTree newClassTree) { return isVarargsCall(newClassTree); } /** * Returns true if the given constructor invocation is a varargs invocation. * * @param newClassTree the constructor invocation * @return true if the given method invocation is a varargs invocation */ public static boolean isVarargsCall(NewClassTree newClassTree) { return ((JCNewClass) newClassTree).varargsElement != null; } /** * Determine whether the given tree is of Kind RECORD, in a way that works on all versions of * Java. * * @param tree the tree to get the kind for * @return whether the tree is of the kind RECORD */ public static boolean isRecordTree(Tree tree) { Tree.Kind kind = tree.getKind(); // Must use String comparison because we may be on an older JDK: return kind.name().equals("RECORD"); } /** * Calls getKind() on the given tree, but returns CLASS if the Kind is RECORD. This is needed * because the Checker Framework runs on JDKs before the RECORD item was added, so RECORD can't be * used in case statements, and usually we want to treat them the same as classes. * * @param tree the tree to get the kind for * @return the kind of the tree, but CLASS if the kind was RECORD */ public static Tree.Kind getKindRecordAsClass(Tree tree) { if (isRecordTree(tree)) { return Tree.Kind.CLASS; } return tree.getKind(); } /** * Returns true if the {@code tree} is a binary tree that performs a comparison. * * @param tree the tree to check * @return whether the tree represents a binary comparison */ public static boolean isBinaryComparison(BinaryTree tree) { return BINARY_COMPARISON_TREE_KINDS.contains(tree.getKind()); } /** * Returns the result of {@code treeMaker.Select(base, sym)}. * * @param treeMaker the TreeMaker to use * @param base the expression for the select * @param sym the symbol to select * @return the JCFieldAccess tree to select sym in base */ public static JCFieldAccess Select(TreeMaker treeMaker, Tree base, Symbol sym) { // The return type of TreeMaker.Select changed in // https://github.com/openjdk/jdk/commit/a917fb3fcf0fe1a4c4de86c08ae4041462848b82#diff-0f1b4da56622ccb5ff716ce5a9532819fc5573179a1eb2c803d053196824891aR726 // When the ECF is compiled with Java 21+, even with `--source/target 8`, this will lead to // a java.lang.NoSuchMethodError: 'com.sun.tools.javac.tree.JCTree$JCFieldAccess // com.sun.tools.javac.tree.TreeMaker.Select(com.sun.tools.javac.tree.JCTree$JCExpression, // com.sun.tools.javac.code.Symbol)' // when executed on Java <21. // Therefore, always use reflection to access TreeMaker.Select. // Hopefully, the JVM optimizes the reflective access quickly. try { assert TREEMAKER_SELECT != null : "@AssumeAssertion(nullness): initialization"; JCFieldAccess jfa = (JCFieldAccess) TREEMAKER_SELECT.invoke(treeMaker, base, sym); if (jfa != null) { return jfa; } else { throw new BugInCF("TreeUtils.Select: TreeMaker.Select returned null for tree: %s", base); } } catch (InvocationTargetException | IllegalAccessException e) { throw new BugInCF("TreeUtils.Select: reflection failed for tree: %s", base, e); } } /** * Returns the result of {@code treeMaker.Select(base, name)}. * * @param treeMaker the TreeMaker to use * @param base the expression for the select * @param name the name to select * @return the JCFieldAccess tree to select sym in base */ public static JCFieldAccess Select( TreeMaker treeMaker, Tree base, com.sun.tools.javac.util.Name name) { // There's no need for reflection here. The only reason we even declare this method // is so that callers don't have to remember which overload we provide a wrapper around. return treeMaker.Select((JCExpression) base, name); } /** * Returns true if {@code tree} is an explicitly typed lambda. * *

An lambda whose formal type parameters have declared types or with no parameters is an * explicitly typed lambda. (See JLS 15.27.1) * * @param tree any kind of tree * @return true iff {@code tree} is an implicitly typed lambda */ public static boolean isExplicitlyTypeLambda(Tree tree) { return tree.getKind() == Tree.Kind.LAMBDA_EXPRESSION && ((JCLambda) tree).paramKind == ParameterKind.EXPLICIT; } /** * Returns all expressions that might be the result of {@code lambda}. * * @param lambda a lambda with or without a body * @return a list of expressions that are returned by {@code lambda} */ public static List getReturnedExpressions(LambdaExpressionTree lambda) { if (lambda.getBodyKind() == BodyKind.EXPRESSION) { return Collections.singletonList((ExpressionTree) lambda.getBody()); } List returnExpressions = new ArrayList<>(); TreeScanner scanner = new TreeScanner() { @Override public Void visitReturn(ReturnTree tree, Void o) { if (tree.getExpression() != null) { returnExpressions.add(tree.getExpression()); } return super.visitReturn(tree, o); } @Override public Void visitLambdaExpression(LambdaExpressionTree node, Void unused) { // Don't visit inside anther lambda. return null; } }; scanner.scan(lambda.getBody(), null); return returnExpressions; } /** * Returns whether or not {@code ref} is an exact method reference. * *

From JLS 15.13.1 "If there is only one possible compile-time declaration with only one * possible invocation, it is said to be exact." * * @param ref a method reference * @return whether or not {@code ref} is an exact method reference */ public static boolean isExactMethodReference(MemberReferenceTree ref) { // Seems like overloaded means the same thing as inexact. // overloadKind is set // com.sun.tools.javac.comp.DeferredAttr.DeferredChecker.visitReference() // IsExact: https://docs.oracle.com/javase/specs/jls/se8/html/jls-15.html#jls-15.13.1-400 // Treat OverloadKind.ERROR as overloaded. return ((JCMemberReference) ref).getOverloadKind() == OverloadKind.UNOVERLOADED; } /** * Returns whether or not {@code expression} is a poly expression as defined in JLS 15.2. * * @param expression expression * @return whether or not {@code expression} is a poly expression */ public static boolean isPolyExpression(ExpressionTree expression) { return !isStandaloneExpression(expression); } /** * Returns whether or not {@code expression} is a standalone expression as defined in JLS 15.2. * * @param expression expression * @return whether or not {@code expression} is a standalone expression */ public static boolean isStandaloneExpression(ExpressionTree expression) { if (expression instanceof JCTree.JCExpression) { if (((JCTree.JCExpression) expression).isStandalone()) { return true; } if (expression.getKind() == Tree.Kind.METHOD_INVOCATION) { // This seems to be a bug in at least Java 11. If a method has type arguments, then // it is a standalone expression. return !((MethodInvocationTree) expression).getTypeArguments().isEmpty(); } } return false; } /** * Was applicability by variable arity invocation necessary to determine the method signature? * *

This isn't the same as {@link ExecutableElement#isVarArgs()}. That method returns true if * the method accepts a variable number of arguments. This method returns true if the method * invocation actually used that fact to invoke the method. * * @param methodInvocation a method or constructor invocation * @return whether applicability by variable arity invocation is necessary to determine the method * signature * @deprecated use {@link #isVarargsCall(Tree)} */ @Deprecated // 2024-06-04 public static boolean isVarArgMethodCall(ExpressionTree methodInvocation) { return isVarargsCall(methodInvocation); } /** * Is the tree a reference to a constructor of a generic class whose type argument isn't * specified? For example, {@code HashSet::new)}. * * @param tree may or may not be a {@link MemberReferenceTree} * @return true if tree is a reference to a constructor of a generic class whose type argument * isn't specified */ public static boolean isDiamondMemberReference(ExpressionTree tree) { if (tree.getKind() != Tree.Kind.MEMBER_REFERENCE) { return false; } MemberReferenceTree memRef = (MemberReferenceTree) tree; TypeMirror type = TreeUtils.typeOf(memRef.getQualifierExpression()); if (memRef.getMode() == ReferenceMode.NEW && type.getKind() == TypeKind.DECLARED) { // No need to check array::new because the generic arrays can't be created. TypeElement classElt = (TypeElement) ((Type) type).asElement(); DeclaredType classTypeMirror = (DeclaredType) classElt.asType(); return !classTypeMirror.getTypeArguments().isEmpty() && ((Type) type).getTypeArguments().isEmpty(); } return false; } /** * Return whether {@code tree} is a method reference with a raw type to the left of {@code ::}. * For example, {@code Class::getName}. * * @param tree a tree * @return whether {@code tree} is a method reference with a raw type to the left of {@code ::} */ public static boolean isLikeDiamondMemberReference(ExpressionTree tree) { if (tree.getKind() != Tree.Kind.MEMBER_REFERENCE) { return false; } MemberReferenceTree memberReferenceTree = (MemberReferenceTree) tree; if (TreeUtils.MemberReferenceKind.getMemberReferenceKind(memberReferenceTree).isUnbound()) { TypeMirror preColonTreeType = typeOf(memberReferenceTree.getQualifierExpression()); return TypesUtils.isRaw(preColonTreeType); } return false; } /** * Returns whether the method reference tree needs type argument inference. * * @param memberReferenceTree a method reference tree * @return whether the method reference tree needs type argument inference */ public static boolean needsTypeArgInference(MemberReferenceTree memberReferenceTree) { if (isDiamondMemberReference(memberReferenceTree) || isLikeDiamondMemberReference(memberReferenceTree)) { return true; } ExecutableElement element = TreeUtils.elementFromUse(memberReferenceTree); return !element.getTypeParameters().isEmpty() && (memberReferenceTree.getTypeArguments() == null || memberReferenceTree.getTypeArguments().isEmpty()); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy