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

soot.toolkits.scalar.SmartLocalDefs Maven / Gradle / Ivy

The newest version!
package soot.toolkits.scalar;

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2004 Ondrej Lhotak
 * %%
 * 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.ArrayList;
import java.util.BitSet;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

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

import soot.Local;
import soot.Timers;
import soot.Unit;
import soot.Value;
import soot.ValueBox;
import soot.options.Options;
import soot.toolkits.graph.ExceptionalUnitGraph;
import soot.toolkits.graph.ExceptionalUnitGraph.ExceptionDest;
import soot.toolkits.graph.UnitGraph;
import soot.util.Cons;
import soot.util.LocalBitSetPacker;

/**
 * Analysis that provides an implementation of the LocalDefs interface.
 *
 * This Analysis calculates only the definitions of the locals used by a unit. If you need all definitions of local you
 * should use {@see SimpleLocalDefs}.
 *
 * Be warned: This implementation requires a lot of memory and CPU time, normally {@see SimpleLocalDefs} is much faster.
 */
public class SmartLocalDefs implements LocalDefs {
  private static final Logger logger = LoggerFactory.getLogger(SmartLocalDefs.class);

  private final UnitGraph graph;
  private Map> localToDefs; // for each local, set of units where it's defined
  private Map liveLocalsAfter;
  private final Map, List> answer;

  public SmartLocalDefs(UnitGraph g, LiveLocals live) {
    this.graph = g;
    this.localToDefs = new HashMap>(2 * g.getBody().getLocalCount() + 1);
    this.liveLocalsAfter = new HashMap(2 * g.getBody().getUnits().size() + 1);
    this.answer = new HashMap, List>();

    final Options op = Options.v();
    if (op.verbose()) {
      logger.debug("[" + g.getBody().getMethod().getName() + "]     Constructing SmartLocalDefs...");
    }

    if (op.time()) {
      Timers.v().defsTimer.start();
    }

    final LocalBitSetPacker localPacker = new LocalBitSetPacker(g.getBody());
    localPacker.pack();

    for (Unit u : g) {
      // translate locals to bits
      BitSet set = new BitSet(localPacker.getLocalCount());
      for (Local l : live.getLiveLocalsAfter(u)) {
        set.set(l.getNumber());
      }
      liveLocalsAfter.put(u, set);

      Local l = localDef(u);
      if (l != null) {
        addDefOf(l, u);
      }
    }
    if (op.verbose()) {
      logger.debug("[" + g.getBody().getMethod().getName() + "]        done localToDefs map...");
    }

    LocalDefsAnalysis analysis = new LocalDefsAnalysis(g);
    liveLocalsAfter = null;
    for (Unit u : g) {
      Set s1 = analysis.getFlowBefore(u);
      if (s1 == null || s1.isEmpty()) {
        continue;
      }

      for (ValueBox vb : u.getUseBoxes()) {
        Value v = vb.getValue();
        if (v instanceof Local) {
          Local l = (Local) v;

          Set s2 = defsOf(l);
          if (s2 == null || s2.isEmpty()) {
            continue;
          }

          List lst = intersectionAsList(s1, s2);
          if (!lst.isEmpty()) {
            this.answer.putIfAbsent(new Cons(u, l), lst);
          }
        }
      }
    }
    localToDefs = null;

    localPacker.unpack();

    if (op.time()) {
      Timers.v().defsTimer.end();
    }

    if (op.verbose()) {
      logger.debug("[" + g.getBody().getMethod().getName() + "]     SmartLocalDefs finished.");
    }
  }

  /**
   * Intersects 2 sets and returns the result as a list
   *
   * @param a
   * @param b
   * @return
   */
  private static  List intersectionAsList(Set a, Set b) {
    if (a == null || b == null || a.isEmpty() || b.isEmpty()) {
      return Collections.emptyList();
    } else if (a.size() < b.size()) {
      List c = new ArrayList(a);
      c.retainAll(b);
      return c;
    } else {
      List c = new ArrayList(b);
      c.retainAll(a);
      return c;
    }
  }

  public void printAnswer() {
    System.out.println(answer.toString());
  }

  private Local localDef(Unit u) {
    List defBoxes = u.getDefBoxes();
    switch (defBoxes.size()) {
      case 0:
        return null;
      case 1:
        Value v = defBoxes.get(0).getValue();
        return (v instanceof Local) ? (Local) v : null;
      default:
        throw new RuntimeException();
    }
  }

  private Set defsOf(Local l) {
    Set s = localToDefs.get(l);
    return (s == null) ? Collections.emptySet() : s;
  }

  private void addDefOf(Local l, Unit u) {
    Set s = localToDefs.get(l);
    if (s == null) {
      localToDefs.put(l, s = new HashSet());
    }
    s.add(u);
  }

  class LocalDefsAnalysis extends ForwardFlowAnalysisExtended> {
    LocalDefsAnalysis(UnitGraph g) {
      super(g);
      doAnalysis();
    }

    @Override
    protected void mergeInto(Unit succNode, Set inout, Set in) {
      inout.addAll(in);
    }

    @Override
    protected void merge(Set in1, Set in2, Set out) {
      // mergeInto should be called
      throw new RuntimeException("should never be called");
    }

    @Override
    protected void flowThrough(Set in, Unit u, Unit succ, Set out) {
      out.clear();

      final BitSet liveLocals = liveLocalsAfter.get(u);
      final Local l = localDef(u);
      if (l == null) { // add all units contained in mask
        for (Unit inU : in) {
          if (liveLocals.get(localDef(inU).getNumber())) {
            out.add(inU);
          }
        }
      } else { // check unit whether contained in allDefUnits before add
        // into out set.
        Set allDefUnits = defsOf(l);

        boolean isExceptionalTarget = false;
        if (graph instanceof ExceptionalUnitGraph) {
          for (ExceptionDest ed : ((ExceptionalUnitGraph) graph).getExceptionDests(u)) {
            if (ed.getTrap() != null && ed.getTrap().getHandlerUnit() == succ) {
              isExceptionalTarget = true;
            }
          }
        }

        for (Unit inU : in) {
          if (liveLocals.get(localDef(inU).getNumber())) {
            // If we have a = foo and foo can throw an exception, we
            // must keep the old definition of a.
            if (isExceptionalTarget || !allDefUnits.contains(inU)) {
              out.add(inU);
            }
          }
        }

        assert (isExceptionalTarget || !out.removeAll(allDefUnits));

        if (liveLocals.get(l.getNumber())) {
          if (!isExceptionalTarget) {
            out.add(u);
          }
        }
      }
    }

    @Override
    protected void copy(Set sourceSet, Set destSet) {
      destSet.clear();
      destSet.addAll(sourceSet);
    }

    @Override
    protected Set newInitialFlow() {
      return new HashSet();
    }

    @Override
    protected Set entryInitialFlow() {
      return new HashSet();
    }
  }

  @Override
  public List getDefsOfAt(Local l, Unit s) {
    List lst = answer.get(new Cons(s, l));
    return lst != null ? lst : Collections.emptyList();
  }

  @Override
  public List getDefsOf(Local l) {
    List result = new ArrayList();
    for (Cons cons : answer.keySet()) {
      if (cons.cdr() == l) {
        result.addAll(answer.get(cons));
      }
    }
    return result;
  }

  /**
   * Returns the associated unit graph.
   */
  public UnitGraph getGraph() {
    return graph;
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy