package soot.jbco.bafTransformations;
import java.util.*;
import soot.*;
import soot.baf.*;
import soot.jbco.IJbcoTransform;
import soot.toolkits.graph.ExceptionalUnitGraph;
import soot.util.Chain;
/**
* @author Michael Batchelder
*
* Created on 16-Jun-2006
*/
public class RemoveRedundantPushStores extends BodyTransformer implements IJbcoTransform {
public static String dependancies[] = new String[] {"bb.jbco_rrps"};
public String[] getDependancies() {
return dependancies;
}
public static String name = "bb.jbco_rrps";
public String getName() {
return name;
}
public void outputSummary() {}
protected void internalTransform(Body b, String phaseName, Map options)
{
// removes all redundant load-stores
boolean changed = true;
PatchingChain units = b.getUnits();
while (changed) {
changed = false;
Unit prevprevprev = null, prevprev = null, prev = null;
ExceptionalUnitGraph eug = new ExceptionalUnitGraph(b);
Iterator it = units.snapshotIterator();
while (it.hasNext()) {
Unit u = (Unit)it.next();
if (prev != null && prev instanceof PushInst && u instanceof StoreInst) {
if (prevprev != null && prevprev instanceof StoreInst
&& prevprevprev != null && prevprevprev instanceof PushInst) {
Local lprev = ((StoreInst)prevprev).getLocal();
Local l = ((StoreInst)u).getLocal();
if (l == lprev && eug.getSuccsOf(prevprevprev).size() == 1 && eug.getSuccsOf(prevprev).size() == 1) {
fixJumps(prevprevprev, prev, b.getTraps());
fixJumps(prevprev, u, b.getTraps());
units.remove(prevprevprev);
units.remove(prevprev);
changed = true;
break;
}
}
}
prevprevprev = prevprev;
prevprev = prev;
prev = u;
}
} // end while changes have been made
}
private void fixJumps(Unit from, Unit to, Chain t) {
from.redirectJumpsToThisTo(to);
Iterator it = t.iterator();
while (it.hasNext()) {
Trap trap = (Trap)it.next();
if (trap.getBeginUnit() == from)
trap.setBeginUnit(to);
if (trap.getEndUnit() == from)
trap.setEndUnit(to);
if (trap.getHandlerUnit() == from)
trap.setHandlerUnit(to);
}
}
}