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

soot.jimple.toolkits.annotation.purity.DirectedCallGraph Maven / Gradle / Ivy

package soot.jimple.toolkits.annotation.purity;

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2005 Antoine Mine
 * %%
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation, either version 2.1 of the
 * License, or (at your option) any later version.
 * 
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Lesser Public License for more details.
 * 
 * You should have received a copy of the GNU General Lesser Public
 * License along with this program.  If not, see
 * .
 * #L%
 */

import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import soot.SootMethod;
import soot.jimple.toolkits.callgraph.CallGraph;
import soot.jimple.toolkits.callgraph.Edge;
import soot.toolkits.graph.DirectedGraph;
import soot.util.HashMultiMap;
import soot.util.MultiMap;

/**
 * Builds a DirectedGraph from a CallGraph and SootMethodFilter.
 *
 * This is used in AbstractInterproceduralAnalysis to construct a reverse pseudo topological order on which to iterate. You
 * can specify a SootMethodFilter to trim the graph by cutting call edges.
 *
 * Methods filtered-out by the SootMethodFilter will not appear in the DirectedGraph!
 */
public class DirectedCallGraph implements DirectedGraph {
  private static final Logger logger = LoggerFactory.getLogger(DirectedCallGraph.class);

  protected Set nodes;
  protected Map> succ;
  protected Map> pred;
  protected List heads;
  protected List tails;
  protected int size;

  /**
   * The constructor does all the work here. After constructed, you can safely use all interface methods. Moreover, these
   * methods should perform very fast...
   *
   * The DirectedGraph will only contain methods in call paths from a method in head and comprising only methods wanted by
   * filter. Moreover, only concrete methods are put in the graph...
   *
   * @param cg
   * @param filter
   * @param heads
   *          is a List of SootMethod
   * @param verbose
   */
  public DirectedCallGraph(CallGraph cg, SootMethodFilter filter, Iterator heads, boolean verbose) {
    // filter heads by filter
    List filteredHeads = new LinkedList();
    while (heads.hasNext()) {
      SootMethod m = heads.next();
      if (m.isConcrete() && filter.want(m)) {
        filteredHeads.add(m);
      }
    }

    this.nodes = new HashSet(filteredHeads);

    MultiMap s = new HashMultiMap();
    MultiMap p = new HashMultiMap();

    // simple breadth-first visit
    Set remain = new HashSet(filteredHeads);
    int nb = 0;
    if (verbose) {
      logger.debug("[AM] dumping method dependencies");
    }
    while (!remain.isEmpty()) {
      Set newRemain = new HashSet();
      for (SootMethod m : remain) {
        if (verbose) {
          logger.debug(" |- " + m.toString() + " calls");
        }

        for (Iterator itt = cg.edgesOutOf(m); itt.hasNext();) {
          Edge edge = itt.next();
          SootMethod mm = edge.tgt();
          boolean keep = mm.isConcrete() && filter.want(mm);
          if (verbose) {
            logger.debug(" |  |- " + mm.toString() + (keep ? "" : " (filtered out)"));
          }
          if (keep) {
            if (this.nodes.add(mm)) {
              newRemain.add(mm);
            }
            s.put(m, mm);
            p.put(mm, m);
          }
        }
        nb++;
      }
      remain = newRemain;
    }
    logger.debug("[AM] number of methods to be analysed: " + nb);

    // MultiMap -> Map of List
    this.succ = new HashMap>();
    this.pred = new HashMap>();
    this.tails = new LinkedList();
    this.heads = new LinkedList();
    for (SootMethod x : this.nodes) {
      Set ss = s.get(x);
      Set pp = p.get(x);
      this.succ.put(x, new LinkedList(ss));
      this.pred.put(x, new LinkedList(pp));
      if (ss.isEmpty()) {
        this.tails.add(x);
      }
      if (pp.isEmpty()) {
        this.heads.add(x);
      }
    }

    this.size = this.nodes.size();
  }

  /**
   * You get a List of SootMethod.
   *
   * @return
   */
  @Override
  public List getHeads() {
    return heads;
  }

  /**
   * You get a List of SootMethod.
   *
   * @return
   */
  @Override
  public List getTails() {
    return tails;
  }

  /**
   * You get an Iterator on SootMethod.
   *
   * @return
   */
  @Override
  public Iterator iterator() {
    return nodes.iterator();
  }

  /**
   *
   * @return
   */
  @Override
  public int size() {
    return size;
  }

  /**
   * You get a List of SootMethod.
   *
   * @param s
   *
   * @return
   */
  @Override
  public List getSuccsOf(SootMethod s) {
    return succ.get(s);
  }

  /**
   * You get a List of SootMethod.
   *
   * @param s
   *
   * @return
   */
  @Override
  public List getPredsOf(SootMethod s) {
    return pred.get(s);
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy