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

soot.jimple.spark.pag.PagToDotDumper Maven / Gradle / Ivy

package soot.jimple.spark.pag;

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2007 Manu Sridharan
 * %%
 * 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.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.PrintStream;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;

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

import soot.SootField;
import soot.SootMethod;
import soot.jimple.spark.ondemand.genericutil.Predicate;
import soot.jimple.spark.sets.P2SetVisitor;

/**
 * Utilities for dumping dot representations of parts of a {@link PAG}.
 *
 * @author msridhar
 *
 */
public class PagToDotDumper {
  private static final Logger logger = LoggerFactory.getLogger(PagToDotDumper.class);

  public static final int TRACE_MAX_LVL = 99;
  private PAG pag;

  private HashMap vmatches;

  private HashMap invVmatches;

  public PagToDotDumper(PAG pag) {
    this.pag = pag;
    this.vmatches = new HashMap();
    this.invVmatches = new HashMap();
  }

  /**
   * Build vmatchEdges and store them in vmatches field
   *
   */
  private void buildVmatchEdges() {
    // for each store and load pair
    for (Iterator iter = pag.loadSourcesIterator(); iter.hasNext();) {
      final FieldRefNode frn1 = (FieldRefNode) iter.next();
      for (Iterator iter2 = pag.storeInvSourcesIterator(); iter2.hasNext();) {
        final FieldRefNode frn2 = (FieldRefNode) iter2.next();
        VarNode base1 = frn1.getBase();
        VarNode base2 = frn2.getBase();

        // debug(frn1, frn2, base1, base2);

        // if they store & load the same field
        if (frn1.getField().equals(frn2.getField())) {

          if (base1.getP2Set().hasNonEmptyIntersection(base2.getP2Set())) {

            // System.err.println("srcs:");
            Node[] src = pag.loadLookup(frn1);
            Node[] dst = pag.storeInvLookup(frn2);

            for (int i = 0; i < src.length; i++) {
              // System.err.println(src[i]);
              vmatches.put(src[i], dst);
            }
            // System.err.println("dst:");
            for (int i = 0; i < dst.length; i++) {
              // System.err.println(dst[i]);
              invVmatches.put(dst[i], src);
            }

          }
        }

      }
    }
  }

  /**
   * @param frn1
   * @param frn2
   * @param base1
   * @param base2
   * @param lvn
   * @param mName
   *          TODO
   */
  @SuppressWarnings("unused")
  private void debug(final FieldRefNode frn1, final FieldRefNode frn2, VarNode base1, VarNode base2) {
    if (base1 instanceof LocalVarNode && base2 instanceof LocalVarNode) {
      LocalVarNode lvn1 = (LocalVarNode) base1;
      LocalVarNode lvn2 = (LocalVarNode) base2;
      if (lvn1.getMethod().getDeclaringClass().getName().equals("java.util.Hashtable$ValueCollection")
          && lvn1.getMethod().getName().equals("contains")
          && lvn2.getMethod().getDeclaringClass().getName().equals("java.util.Hashtable$ValueCollection")
          && lvn2.getMethod().getName().equals("")) {
        System.err.println("Method: " + lvn1.getMethod().getName());
        System.err.println(makeLabel(frn1));
        System.err.println("Base: " + base1.getVariable());
        System.err.println("Field: " + frn1.getField());
        System.err.println(makeLabel(frn2));
        System.err.println("Base: " + base2.getVariable());
        System.err.println("Field: " + frn2.getField());

        if (frn1.getField().equals(frn2.getField())) {
          System.err.println("field match");
          if (base1.getP2Set().hasNonEmptyIntersection(base2.getP2Set())) {
            System.err.println("non empty");
          } else {
            System.err.println("b1: " + base1.getP2Set());
            System.err.println("b2: " + base2.getP2Set());
          }
        }
      }
    }

  }

  /**
   * @param lvNode
   * @param node
   * @return
   */
  private static String translateEdge(Node src, Node dest, String label) {
    return makeNodeName(src) + " -> " + makeNodeName(dest) + " [label=\"" + label + "\"];";
  }

  private final static Predicate emptyP2SetPred = new Predicate() {
    public boolean test(Node n) {
      return !(n instanceof AllocNode) && n.getP2Set().isEmpty();
    }
  };

  /**
   * Generate a node declaration for a dot file.
   *
   * @param node
   *          the node
   * @param p
   *          a predicate over nodes, which, if true, will cause the node to appear red
   * @return the appropriate {@link String} for the dot file
   */
  public static String makeDotNodeLabel(Node n, Predicate p) {
    String color = "";
    String label;

    if (p.test(n)) {
      color = ", color=red";
    }
    if (n instanceof LocalVarNode) {
      label = makeLabel((LocalVarNode) n);
    } else if (n instanceof AllocNode) {
      label = makeLabel((AllocNode) n);
    } else if (n instanceof FieldRefNode) {
      label = makeLabel((FieldRefNode) n);
    } else {
      label = n.toString();
    }
    return makeNodeName(n) + "[label=\"" + label + "\"" + color + "];";

  }

  private static String translateLabel(Node n) {
    return makeDotNodeLabel(n, emptyP2SetPred);
  }

  /**
   * @param lvNode
   * @param cName
   * @param mName
   * @return
   */
  private boolean isDefinedIn(LocalVarNode lvNode, String cName, String mName) {
    return lvNode.getMethod() != null && lvNode.getMethod().getDeclaringClass().getName().equals(cName)
        && lvNode.getMethod().getName().equals(mName);
  }

  private void printOneNode(VarNode node) {
    PrintStream ps = System.err;

    ps.println(makeLabel(node));
    Node[] succs = pag.simpleInvLookup(node);
    ps.println("assign");
    ps.println("======");
    for (int i = 0; i < succs.length; i++) {
      ps.println(succs[i]);
    }

    succs = pag.allocInvLookup(node);
    ps.println("new");
    ps.println("======");
    for (int i = 0; i < succs.length; i++) {
      ps.println(succs[i]);

    }

    succs = pag.loadInvLookup(node);
    ps.println("load");
    ps.println("======");
    for (int i = 0; i < succs.length; i++) {
      ps.println(succs[i]);
    }

    succs = pag.storeLookup(node);
    ps.println("store");
    ps.println("======");
    for (int i = 0; i < succs.length; i++) {
      ps.println(succs[i]);
    }

  }

  /**
   * dumps the points-to sets for all locals in a method in a dot representation. The graph has edges from each local to all
   * {@link AllocNode}s in its points-to set
   *
   * @param fName
   *          a name for the output file
   * @param mName
   *          the name of the method whose locals should be dumped
   * @throws FileNotFoundException
   *           if unable to output to specified file
   */
  public void dumpP2SetsForLocals(String fName, String mName) throws FileNotFoundException {

    FileOutputStream fos = new FileOutputStream(new File(fName));
    PrintStream ps = new PrintStream(fos);
    ps.println("digraph G {");

    dumpLocalP2Set(mName, ps);

    ps.print("}");

  }

  private void dumpLocalP2Set(String mName, final PrintStream ps) {

    for (Iterator iter = pag.getVarNodeNumberer().iterator(); iter.hasNext();) {
      VarNode vNode = (VarNode) iter.next();

      if (vNode instanceof LocalVarNode) {
        final LocalVarNode lvNode = (LocalVarNode) vNode;
        if (lvNode.getMethod() != null && lvNode.getMethod().getName().equals(mName)) {

          ps.println("\t" + makeNodeName(lvNode) + " [label=\"" + makeLabel(lvNode) + "\"];");
          lvNode.getP2Set().forall(new P2SetToDotPrinter(lvNode, ps));

        }
      }
    }
  }

  /**
   * Dump the PAG for some method in the program in dot format
   *
   * @param fName
   *          The filename for the output
   * @param cName
   *          The name of the declaring class for the method
   * @param mName
   *          The name of the method
   * @throws FileNotFoundException
   *           if output file cannot be written
   */
  public void dumpPAGForMethod(String fName, String cName, String mName) throws FileNotFoundException {
    FileOutputStream fos = new FileOutputStream(new File(fName));
    PrintStream ps = new PrintStream(fos);
    ps.println("digraph G {");
    ps.println("\trankdir=LR;");
    dumpLocalPAG(cName, mName, ps);
    ps.print("}");
    try {
      fos.close();
    } catch (IOException e) {
      logger.error(e.getMessage(), e);
    }
    ps.close();
  }

  private void dumpLocalPAG(String cName, String mName, final PrintStream ps) {
    // this.queryMethod = mName;
    // iterate over all variable nodes
    for (Iterator iter = pag.getVarNodeNumberer().iterator(); iter.hasNext();) {
      final Node node = (Node) iter.next();

      if (!(node instanceof LocalVarNode)) {
        continue;
      }

      final LocalVarNode lvNode = (LocalVarNode) node;

      // nodes that is defined in the specified class and method
      if (isDefinedIn(lvNode, cName, mName)) {

        dumpForwardReachableNodesFrom(lvNode, ps);

      }
    }

    // for (Iterator iter = pag.getFieldRefNodeNumberer().iterator(); iter
    // .hasNext();) {
    // final FieldRefNode frNode = (FieldRefNode) iter.next();
    //
    // if (frNode.getBase().)
    // Node[] succs = pag.storeInvLookup(frNode);
    // for (int i = 0; i < succs.length; i++) {
    // ps.println("\t" + translateLabel(succs[i]));
    // // print edge
    // ps.println("\t" + translateEdge(frNode, succs[i], "store"));
    // }
    // }

  }

  /**
   * @param lvNode
   * @param ps
   */
  private void dumpForwardReachableNodesFrom(final LocalVarNode lvNode, final PrintStream ps) {
    ps.println("\t" + translateLabel(lvNode));

    Node[] succs = pag.simpleInvLookup(lvNode);
    for (int i = 0; i < succs.length; i++) {
      ps.println("\t" + translateLabel(succs[i]));
      // print edge
      ps.println("\t" + translateEdge(lvNode, succs[i], "assign"));
    }

    succs = pag.allocInvLookup(lvNode);
    for (int i = 0; i < succs.length; i++) {
      ps.println("\t" + translateLabel(succs[i]));
      // print edge
      ps.println("\t" + translateEdge(lvNode, succs[i], "new"));
    }

    succs = pag.loadInvLookup(lvNode);
    for (int i = 0; i < succs.length; i++) {
      final FieldRefNode frNode = (FieldRefNode) succs[i];
      ps.println("\t" + translateLabel(frNode));
      ps.println("\t" + translateLabel(frNode.getBase()));

      // print edge
      ps.println("\t" + translateEdge(lvNode, frNode, "load"));
      ps.println("\t" + translateEdge(frNode, frNode.getBase(), "getBase"));

    }

    succs = pag.storeLookup(lvNode);

    for (int i = 0; i < succs.length; i++) {
      final FieldRefNode frNode = (FieldRefNode) succs[i];
      ps.println("\t" + translateLabel(frNode));
      ps.println("\t" + translateLabel(frNode.getBase()));
      // print edge
      ps.println("\t" + translateEdge(frNode, lvNode, "store"));
      ps.println("\t" + translateEdge(frNode, frNode.getBase(), "getBase"));
    }
  }

  public void traceNode(int id) {
    buildVmatchEdges();
    String fName = "trace." + id + ".dot";
    try {
      FileOutputStream fos = new FileOutputStream(new File(fName));
      PrintStream ps = new PrintStream(fos);
      ps.println("digraph G {");

      // iterate over all variable nodes
      for (Iterator iter = pag.getVarNodeNumberer().iterator(); iter.hasNext();) {
        final VarNode n = (VarNode) iter.next();

        if (n.getNumber() == id) {
          LocalVarNode lvn = (LocalVarNode) n;

          printOneNode(lvn);

          trace(lvn, ps, new HashSet(), TRACE_MAX_LVL);
        }
      }

      ps.print("}");
      ps.close();
      fos.close();
    } catch (IOException e) {
      logger.error(e.getMessage(), e);
    }
  }

  public void traceNode(String cName, String mName, String varName) {
    String mName2 = mName;
    if (mName.indexOf('<') == 0) {
      mName2 = mName.substring(1, mName.length() - 1);
    }

    traceLocalVarNode("trace." + cName + "." + mName2 + "." + varName + ".dot", cName, mName, varName);
  }

  public void traceLocalVarNode(String fName, String cName, String mName, String varName) {
    PrintStream ps;

    buildVmatchEdges();

    try {
      FileOutputStream fos = new FileOutputStream(new File(fName));
      ps = new PrintStream(fos);
      ps.println("digraph G {");

      // iterate over all variable nodes
      for (Iterator iter = pag.getVarNodeNumberer().iterator(); iter.hasNext();) {
        final VarNode n = (VarNode) iter.next();

        if (!(n instanceof LocalVarNode)) {
          continue;
        }
        LocalVarNode lvn = (LocalVarNode) n;
        // HACK
        if (lvn.getMethod() == null) {
          continue;
        }
        if (isDefinedIn(lvn, cName, mName)) {
          // System.err.println("class match");
          // System.err.println(lvn.getVariable());
          if (lvn.getVariable().toString().equals(varName)) {
            // System.err.println(lvn);

            trace(lvn, ps, new HashSet(), 10);
          }
        }

      }
      ps.print("}");
      ps.close();
      fos.close();
    } catch (IOException e) {
      logger.error(e.getMessage(), e);
    }
  }

  /**
   *
   * Do a DFS traversal
   *
   * @param name
   * @param name2
   * @param ps2
   */
  private void trace(VarNode node, PrintStream ps, HashSet visitedNodes, int level) {
    if (level < 1) {
      return;
    }
    ps.println("\t" + translateLabel(node));

    // // assign value to others
    // Node[] preds = pag.storeLookup(node);
    // for (int i = 0; i < preds.length; i++) {
    // if (visitedNodes.contains(preds[i]))
    // continue;
    // ps.println("\t" + translateLabel(preds[i]));
    // // print edge
    // ps.println("\t" + translateEdge(preds[i], node, "store"));
    // visitedNodes.add(preds[i]);
    // // trace((VarNode) preds[i], ps, visitedNodes);
    // }

    // get other's value
    Node[] succs = pag.simpleInvLookup(node);
    for (int i = 0; i < succs.length; i++) {
      if (visitedNodes.contains(succs[i])) {
        continue;
      }
      ps.println("\t" + translateLabel(succs[i]));
      // print edge
      ps.println("\t" + translateEdge(node, succs[i], "assign"));
      visitedNodes.add(succs[i]);
      trace((VarNode) succs[i], ps, visitedNodes, level - 1);
    }

    succs = pag.allocInvLookup(node);
    for (int i = 0; i < succs.length; i++) {
      if (visitedNodes.contains(succs[i])) {
        continue;
      }
      ps.println("\t" + translateLabel(succs[i]));
      // print edge
      ps.println("\t" + translateEdge(node, succs[i], "new"));
    }

    succs = vmatches.get(node);
    if (succs != null) {
      // System.err.println(succs.length);
      for (int i = 0; i < succs.length; i++) {
        // System.err.println(succs[i]);
        if (visitedNodes.contains(succs[i])) {
          continue;
        }
        ps.println("\t" + translateLabel(succs[i]));
        // print edge
        ps.println("\t" + translateEdge(node, succs[i], "vmatch"));
        trace((VarNode) succs[i], ps, visitedNodes, level - 1);
      }
    }
  }

  public static String makeNodeName(Node n) {
    return "node_" + n.getNumber();
  }

  public static String makeLabel(AllocNode n) {
    return n.getNewExpr().toString();
  }

  public static String makeLabel(LocalVarNode n) {
    SootMethod sm = n.getMethod();
    return "LV " + n.getVariable().toString() + " " + n.getNumber() + "\\n" + sm.getDeclaringClass() + "\\n" + sm.getName();
  }

  /**
   * @param node
   * @return
   */
  public static String makeLabel(FieldRefNode node) {
    if (node.getField() instanceof SootField) {
      final SootField sf = (SootField) node.getField();
      return "FNR " + makeLabel(node.getBase()) + "." + sf.getName();
    } else {
      return "FNR " + makeLabel(node.getBase()) + "." + node.getField();
    }
  }

  /**
   * @param base
   * @return
   */
  public static String makeLabel(VarNode base) {
    if (base instanceof LocalVarNode) {
      return makeLabel((LocalVarNode) base);
    } else {
      return base.toString();
    }
  }

  class P2SetToDotPrinter extends P2SetVisitor {

    private final Node curNode;

    private final PrintStream ps;

    P2SetToDotPrinter(Node curNode, PrintStream ps) {
      this.curNode = curNode;
      this.ps = ps;
    }

    public void visit(Node n) {
      ps.println("\t" + makeNodeName(n) + " [label=\"" + makeLabel((AllocNode) n) + "\"];");

      ps.print("\t" + makeNodeName(curNode) + " -> ");
      ps.println(makeNodeName(n) + ";");
    }

  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy