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

org.checkerframework.dataflow.analysis.AnalysisResult Maven / Gradle / Ivy

package org.checkerframework.dataflow.analysis;

import com.sun.source.tree.BinaryTree;
import com.sun.source.tree.Tree;
import com.sun.source.tree.UnaryTree;
import java.util.HashMap;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.StringJoiner;
import java.util.concurrent.atomic.AtomicLong;
import javax.lang.model.element.VariableElement;
import org.checkerframework.checker.initialization.qual.UnknownInitialization;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.checkerframework.dataflow.cfg.block.Block;
import org.checkerframework.dataflow.cfg.node.Node;
import org.checkerframework.javacutil.BugInCF;
import org.checkerframework.javacutil.TreeUtils;
import org.plumelib.util.UniqueId;
import org.plumelib.util.UnmodifiableIdentityHashMap;

/**
 * An {@link AnalysisResult} represents the result of a org.checkerframework.dataflow analysis by
 * providing the abstract values given a node or a tree. Note that it does not keep track of custom
 * results computed by some analysis.
 *
 * @param  type of the abstract value that is tracked
 * @param  the store type used in the analysis
 */
public class AnalysisResult, S extends Store> implements UniqueId {

  /**
   * For efficiency, certain maps stored in the result are only copied lazily, when they need to be
   * mutated. This flag tracks if the copying has occurred.
   */
  private boolean mapsCopied = false;

  /** Abstract values of nodes. */
  protected IdentityHashMap nodeValues;

  /**
   * Map from AST {@link Tree}s to sets of {@link Node}s.
   *
   * 

Some of those Nodes might not be keys in {@link #nodeValues}. One reason is that the Node is * unreachable in the control flow graph, so dataflow never gave it a value. */ protected IdentityHashMap> treeLookup; /** * Map from postfix increment or decrement trees that are AST {@link UnaryTree}s to the synthetic * tree that is {@code v + 1} or {@code v - 1}. */ protected IdentityHashMap postfixLookup; /** Map from (effectively final) local variable elements to their abstract value. */ protected final Map finalLocalValues; /** * The transfer inputs of every basic block; assumed to be 'no information' if not present. The * inputs are before blocks in forward analysis, and are after blocks in backward analysis. */ protected final IdentityHashMap> inputs; /** * Caches of the analysis results. It maps from the TransferInput for a Block to a map. The inner * map is from a node within the block to the TransferResult for that node. * * @see #runAnalysisFor(Node, Analysis.BeforeOrAfter, TransferInput, IdentityHashMap, Map) */ protected final Map, IdentityHashMap>> analysisCaches; /** The unique ID for the next-created object. */ private static final AtomicLong nextUid = new AtomicLong(0); /** The unique ID of this object. */ private final transient long uid = nextUid.getAndIncrement(); @Override public long getUid(@UnknownInitialization AnalysisResult this) { return uid; } /** * Initialize with given mappings. * * @param nodeValues {@link #nodeValues} * @param inputs {@link #inputs} * @param treeLookup {@link #treeLookup} * @param postfixLookup {@link #postfixLookup} * @param finalLocalValues {@link #finalLocalValues} * @param analysisCaches {@link #analysisCaches} */ protected AnalysisResult( IdentityHashMap nodeValues, IdentityHashMap> inputs, IdentityHashMap> treeLookup, IdentityHashMap postfixLookup, Map finalLocalValues, Map, IdentityHashMap>> analysisCaches) { this.nodeValues = UnmodifiableIdentityHashMap.wrap(nodeValues); this.treeLookup = UnmodifiableIdentityHashMap.wrap(treeLookup); this.postfixLookup = UnmodifiableIdentityHashMap.wrap(postfixLookup); // TODO: why are inputs and finalLocalValues captured? this.inputs = inputs; this.finalLocalValues = finalLocalValues; this.analysisCaches = analysisCaches; } /** * Initialize with given mappings and empty cache. * * @param nodeValues {@link #nodeValues} * @param inputs {@link #inputs} * @param treeLookup {@link #treeLookup} * @param postfixLookup {@link #postfixLookup} * @param finalLocalValues {@link #finalLocalValues} */ public AnalysisResult( IdentityHashMap nodeValues, IdentityHashMap> inputs, IdentityHashMap> treeLookup, IdentityHashMap postfixLookup, Map finalLocalValues) { this(nodeValues, inputs, treeLookup, postfixLookup, finalLocalValues, new IdentityHashMap<>()); } /** * Initialize empty result with specified cache. * * @param analysisCaches {@link #analysisCaches} */ public AnalysisResult( Map, IdentityHashMap>> analysisCaches) { this( new IdentityHashMap<>(), new IdentityHashMap<>(), new IdentityHashMap<>(), new IdentityHashMap<>(), new HashMap<>(), analysisCaches); } /** * Combine with another analysis result. * * @param other an analysis result to combine with this */ public void combine(AnalysisResult other) { copyMapsIfNeeded(); nodeValues.putAll(other.nodeValues); mergeTreeLookup(treeLookup, other.treeLookup); postfixLookup.putAll(other.postfixLookup); inputs.putAll(other.inputs); finalLocalValues.putAll(other.finalLocalValues); } /** Make copies of certain internal IdentityHashMaps, if they have not been copied already. */ private void copyMapsIfNeeded() { if (!mapsCopied) { nodeValues = new IdentityHashMap<>(nodeValues); treeLookup = new IdentityHashMap<>(treeLookup); postfixLookup = new IdentityHashMap<>(postfixLookup); mapsCopied = true; } } /** * Merge all entries from otherTreeLookup into treeLookup. Merge sets if already present. * * @param treeLookup a map from abstract syntax trees to sets of nodes * @param otherTreeLookup another treeLookup that will be merged into {@code treeLookup} */ private static void mergeTreeLookup( IdentityHashMap> treeLookup, IdentityHashMap> otherTreeLookup) { for (Map.Entry> entry : otherTreeLookup.entrySet()) { Set hit = treeLookup.get(entry.getKey()); if (hit == null) { treeLookup.put(entry.getKey(), entry.getValue()); } else { hit.addAll(entry.getValue()); } } } /** * Returns the value of effectively final local variables. * * @return the value of effectively final local variables */ public Map getFinalLocalValues() { return finalLocalValues; } /** * Returns the abstract value for {@link Node} {@code n}, or {@code null} if no information is * available. Note that if the analysis has not finished yet, this value might not represent the * final value for this node. * * @param n a node * @return the abstract value for {@link Node} {@code n}, or {@code null} if no information is * available */ public @Nullable V getValue(Node n) { return nodeValues.get(n); } /** * Returns the abstract value for {@link Tree} {@code t}, or {@code null} if no information is * available. Note that if the analysis has not finished yet, this value might not represent the * final value for this node. * * @param t a tree * @return the abstract value for {@link Tree} {@code t}, or {@code null} if no information is * available */ public @Nullable V getValue(Tree t) { // This is a set because one Tree might correspond to multiple Nodes. Set nodes = treeLookup.get(t); if (nodes == null) { return null; } V merged = null; for (Node aNode : nodes) { V a = getValue(aNode); if (a != null) { if (merged == null) { merged = a; } else { merged = merged.leastUpperBound(a); } } } return merged; } /** * Returns the {@code Node}s corresponding to a particular {@code Tree}. Multiple {@code Node}s * can correspond to a single {@code Tree} because of several reasons: * *

    *
  1. In a lambda expression such as {@code () -> 5} the {@code 5} is both an {@code * IntegerLiteralNode} and a {@code LambdaResultExpressionNode}. *
  2. Widening and narrowing primitive conversions can result in {@code WideningConversionNode} * and {@code NarrowingConversionNode}. *
  3. Automatic String conversion can result in a {@code StringConversionNode}. *
  4. Trees for {@code finally} blocks are cloned to achieve a precise CFG. Any {@code Tree} * within a finally block can have multiple corresponding {@code Node}s attached to them. *
* * Callers of this method should always iterate through the returned set, possibly ignoring all * {@code Node}s they are not interested in. * * @param tree a tree * @return the set of {@link Node}s for a given {@link Tree} */ public @Nullable Set getNodesForTree(Tree tree) { return treeLookup.get(tree); } /** * Returns the synthetic {@code v + 1} or {@code v - 1} corresponding to the postfix increment or * decrement tree. * * @param postfixTree a postfix increment or decrement tree * @return the synthetic {@code v + 1} or {@code v - 1} corresponding to the postfix increment or * decrement tree */ public BinaryTree getPostfixBinaryTree(UnaryTree postfixTree) { if (!postfixLookup.containsKey(postfixTree)) { throw new BugInCF(postfixTree + " is not in postfixLookup"); } return postfixLookup.get(postfixTree); } /** * Returns the store immediately before a given {@link Tree}. * * @param tree a tree * @return the store immediately before a given {@link Tree} */ public @Nullable S getStoreBefore(Tree tree) { Set nodes = getNodesForTree(tree); if (nodes == null) { return null; } S merged = null; for (Node node : nodes) { S s = getStoreBefore(node); if (merged == null) { merged = s; } else if (s != null) { merged = merged.leastUpperBound(s); } } return merged; } /** * Returns the store immediately before a given {@link Node}. * * @param node a node * @return the store immediately before a given {@link Node} */ public @Nullable S getStoreBefore(Node node) { return runAnalysisFor(node, Analysis.BeforeOrAfter.BEFORE); } /** * Returns the regular store immediately before a given {@link Block}. * * @param block a block * @return the store right before the given block */ public S getStoreBefore(Block block) { TransferInput transferInput = inputs.get(block); assert transferInput != null : "@AssumeAssertion(nullness): transferInput should be non-null"; Analysis analysis = transferInput.analysis; switch (analysis.getDirection()) { case FORWARD: return transferInput.getRegularStore(); case BACKWARD: List nodes = block.getNodes(); if (nodes.isEmpty()) { // This block doesn't contain any node, return the store in the transfer input. return transferInput.getRegularStore(); } else { Node firstNode = nodes.get(0); return analysis.runAnalysisFor( firstNode, Analysis.BeforeOrAfter.BEFORE, transferInput, nodeValues, analysisCaches); } default: throw new BugInCF("Unknown direction: " + analysis.getDirection()); } } /** * Returns the regular store immediately after a given block. * * @param block a block * @return the store after the given block */ public S getStoreAfter(Block block) { TransferInput transferInput = inputs.get(block); assert transferInput != null : "@AssumeAssertion(nullness): transferInput should be non-null"; Analysis analysis = transferInput.analysis; switch (analysis.getDirection()) { case FORWARD: Node lastNode = block.getLastNode(); if (lastNode == null) { // This block doesn't contain any node, return the store in the transfer input. return transferInput.getRegularStore(); } else { return analysis.runAnalysisFor( lastNode, Analysis.BeforeOrAfter.AFTER, transferInput, nodeValues, analysisCaches); } case BACKWARD: return transferInput.getRegularStore(); default: throw new BugInCF("Unknown direction: " + analysis.getDirection()); } } /** * Returns the store immediately after a given {@link Tree}. * * @param tree a tree * @return the store immediately after a given {@link Tree} */ public @Nullable S getStoreAfter(Tree tree) { Set nodes = getNodesForTree(tree); if (nodes == null) { return null; } S merged = null; for (Node node : nodes) { S s = getStoreAfter(node); if (merged == null) { merged = s; } else if (s != null) { merged = merged.leastUpperBound(s); } } return merged; } /** * Returns the store immediately after a given {@link Node}. * * @param node a node * @return the store immediately after a given {@link Node} */ public @Nullable S getStoreAfter(Node node) { return runAnalysisFor(node, Analysis.BeforeOrAfter.AFTER); } /** * Runs the analysis again within the block of {@code node} and returns the store at the location * of {@code node}. If {@code before} is true, then the store immediately before the {@link Node} * {@code node} is returned. Otherwise, the store after {@code node} is returned. * *

If the given {@link Node} cannot be reached (in the control flow graph), then {@code null} * is returned. * * @param node the node to analyze * @param preOrPost which store to return: the store immediately before {@code node} or the store * after {@code node} * @return the store before or after {@code node} (depends on the value of {@code before}) after * running the analysis */ protected @Nullable S runAnalysisFor(Node node, Analysis.BeforeOrAfter preOrPost) { // block is null if node is a formal parameter of a method, or is a field access thereof Block block = node.getBlock(); assert block != null : "@AssumeAssertion(nullness): null block for node " + node; TransferInput transferInput = inputs.get(block); if (transferInput == null) { return null; } // Calling Analysis.runAnalysisFor() may mutate the internal nodeValues map inside an // AbstractAnalysis object, and by default the AnalysisResult constructor just wraps this // map without copying it. So here the AnalysisResult maps must be copied, to preserve // them. copyMapsIfNeeded(); return runAnalysisFor(node, preOrPost, transferInput, nodeValues, analysisCaches); } /** * Runs the analysis again within the block of {@code node} and returns the store at the location * of {@code node}. If {@code before} is true, then the store immediately before the {@link Node} * {@code node} is returned. Otherwise, the store immediately after {@code node} is returned. If * {@code analysisCaches} is not null, this method uses a cache. {@code analysisCaches} is a map * of a block of node to the cached analysis result. If the cache for {@code transferInput} is not * in {@code analysisCaches}, this method creates new cache and stores it in {@code * analysisCaches}. The cache is a map of nodes to the analysis results of the nodes. * * @param the abstract value type to be tracked by the analysis * @param the store type used in the analysis * @param node the node to analyze * @param preOrPost which store to return: the store immediately before {@code node} or the store * after {@code node} * @param transferInput a transfer input * @param nodeValues {@link #nodeValues} * @param analysisCaches {@link #analysisCaches} * @return the store before or after {@code node} (depends on the value of {@code before}) after * running the analysis */ public static , S extends Store> S runAnalysisFor( Node node, Analysis.BeforeOrAfter preOrPost, TransferInput transferInput, IdentityHashMap nodeValues, @Nullable Map, IdentityHashMap>> analysisCaches) { if (transferInput.analysis == null) { throw new BugInCF("Analysis in transferInput cannot be null."); } return transferInput.analysis.runAnalysisFor( node, preOrPost, transferInput, nodeValues, analysisCaches); } /** * Returns the cached TransferResult for a given node. * * @param node the node for which to look up a result * @return the TransferResult at the given node */ public @Nullable TransferResult lookupResult(Node node) { Block block = node.getBlock(); TransferInput blockInput = inputs.get(block); if (blockInput == null) { return null; } IdentityHashMap> cache = analysisCaches.get(blockInput); if (cache == null) { return null; } TransferResult result = cache.get(node); return result; } /** * Returns a verbose string representation of this, useful for debugging. * * @return a string representation of this */ public String toStringDebug() { StringJoiner result = new StringJoiner( String.format("%n "), String.format("AnalysisResult{%n "), String.format("%n}")); result.add("nodeValues = " + nodeValuesToString(nodeValues)); result.add("treeLookup = " + treeLookupToString(treeLookup)); result.add("postfixLookup = " + postfixLookup); result.add("finalLocalValues = " + finalLocalValues); result.add("inputs = " + inputs); result.add("analysisCaches = " + analysisCaches); return result.toString(); } /** * Returns a verbose string representation, useful for debugging. The map has the same type as the * {@code nodeValues} field. * * @param the type of values in the map * @param nodeValues a map to format * @return a printed representation of the given map */ public static String nodeValuesToString(Map nodeValues) { if (nodeValues.isEmpty()) { return "{}"; } StringJoiner result = new StringJoiner(String.format("%n ")); result.add("{"); for (Map.Entry entry : nodeValues.entrySet()) { Node key = entry.getKey(); result.add(String.format("%s => %s", key.toStringDebug(), entry.getValue())); } result.add("}"); return result.toString(); } /** * Returns a verbose string representation of a map, useful for debugging. The map has the same * type as the {@code treeLookup} field. * * @param treeLookup a map to format * @return a printed representation of the given map */ public static String treeLookupToString(Map> treeLookup) { if (treeLookup.isEmpty()) { return "{}"; } StringJoiner result = new StringJoiner(String.format("%n ")); result.add("{"); for (Map.Entry> entry : treeLookup.entrySet()) { Tree key = entry.getKey(); result.add( TreeUtils.toStringTruncated(key, 65) + " => " + Node.nodeCollectionToString(entry.getValue())); } result.add("}"); return result.toString(); } }