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

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

Go to download

The Checker Framework enhances Java's type system to make it more powerful and useful. This lets software developers detect and prevent errors in their Java programs. The Checker Framework includes compiler plug-ins ("checkers") that find bugs or verify their absence. It also permits you to write your own compiler plug-ins.

There is a newer version: 3.48.3
Show newest version
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 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.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 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()}. * * @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/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 isVarArgs(Tree tree) { switch (tree.getKind()) { case METHOD_INVOCATION: return isVarArgs((MethodInvocationTree) tree); case NEW_CLASS: return isVarArgs((NewClassTree) tree); default: throw new BugInCF("TreeUtils.isVarArgs: unexpected kind of tree: " + tree); } } /** * 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 isVarArgs(MethodInvocationTree invok) { return isVarArgs(elementFromUse(invok), invok.getArguments()); } /** * 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 */ public static boolean isVarArgs(NewClassTree newClassTree) { return isVarArgs(elementFromUse(newClassTree), newClassTree.getArguments()); } /** * Returns true if a method/constructor invocation is a varargs invocation. * * @param method the method or constructor * @param args the arguments passed at the invocation * @return true if the given method/constructor invocation is a varargs invocation */ private static boolean isVarArgs(ExecutableElement method, List args) { if (!method.isVarArgs()) { return false; } List parameters = method.getParameters(); if (parameters.size() != args.size()) { return true; } TypeMirror lastArgType = typeOf(args.get(args.size() - 1)); if (lastArgType.getKind() == TypeKind.NULL) { return false; } if (lastArgType.getKind() != TypeKind.ARRAY) { return true; } TypeMirror varargsParamType = parameters.get(parameters.size() - 1).asType(); return TypesUtils.getArrayDepth(varargsParamType) != TypesUtils.getArrayDepth(lastArgType); } /** * 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 */ public static boolean isVarArgMethodCall(ExpressionTree methodInvocation) { if (methodInvocation.getKind() == Tree.Kind.METHOD_INVOCATION) { return ((JCMethodInvocation) methodInvocation).varargsElement != null; } else if (methodInvocation.getKind() == Tree.Kind.NEW_CLASS) { return ((JCNewClass) methodInvocation).varargsElement != null; } else { return false; } } /** * 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