soot.jimple.toolkits.scalar.pre.DelayabilityAnalysis Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of soot Show documentation
Show all versions of soot Show documentation
A Java Optimization Framework
package soot.jimple.toolkits.scalar.pre;
/*-
* #%L
* Soot - a J*va Optimization Framework
* %%
* Copyright (C) 2002 Florian Loitsch
* %%
* 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.Map;
import soot.EquivalentValue;
import soot.Unit;
import soot.toolkits.graph.DirectedGraph;
import soot.toolkits.scalar.ArrayPackedSet;
import soot.toolkits.scalar.BoundedFlowSet;
import soot.toolkits.scalar.CollectionFlowUniverse;
import soot.toolkits.scalar.FlowSet;
import soot.toolkits.scalar.ForwardFlowAnalysis;
/**
* Performs a Delayability-analysis on the given graph. This analysis is the third analysis in the PRE (lazy code motion) and
* has little (no?) sense if used alone. Basically it tries to push the computations we would insert in the Busy Code Motion
* as far down as possible, to decrease life-time ranges (clearly this is not true, if the computation "uses" two variables
* and produces only one temporary).
*/
public class DelayabilityAnalysis extends ForwardFlowAnalysis> {
private final EarliestnessComputation earliest;
private final Map unitToKillValue;
private final BoundedFlowSet set;
/**
* this constructor should not be used, and will throw a runtime-exception!
*/
@Deprecated
public DelayabilityAnalysis(DirectedGraph dg) {
/* we have to add super(dg). otherwise Javac complains. */
super(dg);
throw new RuntimeException("Don't use this Constructor!");
}
/**
* Automatically performs the Delayability-analysis on the graph dg
and the Earliest-computation
* earliest
.
* the equivRhsMap
is only here to avoid doing these things again...
*
* @param dg
* a ExceptionalUnitGraph
* @param earliest
* the earliest-computation of the same graph.
* @param equivRhsMap
* the rhs of each unit (if assignment-stmt).
*/
public DelayabilityAnalysis(DirectedGraph dg, EarliestnessComputation earliest,
Map equivRhsMap) {
this(dg, earliest, equivRhsMap,
new ArrayPackedSet(new CollectionFlowUniverse(equivRhsMap.values())));
}
/**
* Automatically performs the Delayability-analysis on the graph dg
and the Earliest-computation
* earliest
.
* the equivRhsMap
is only here to avoid doing these things again...
* as set-operations are usually more efficient, if the sets come from one source, sets should be shared around analyses,
* if the analyses are to be combined.
*
* @param dg
* a ExceptionalUnitGraph
* @param earliest
* the earliest-computation of the same graph.
* @param equivRhsMap
* the rhs of each unit (if assignment-stmt).
* @param set
* the shared set.
*/
public DelayabilityAnalysis(DirectedGraph dg, EarliestnessComputation earliest,
Map equivRhsMap, BoundedFlowSet set) {
super(dg);
this.set = set;
this.unitToKillValue = equivRhsMap;
this.earliest = earliest;
doAnalysis();
// finally add the genSet to each BeforeFlow
for (Unit currentUnit : dg) {
getFlowBefore(currentUnit).union(earliest.getFlowBefore(currentUnit));
}
}
@Override
protected FlowSet newInitialFlow() {
return set.topSet();
}
@Override
protected FlowSet entryInitialFlow() {
return set.emptySet();
}
@Override
protected void flowThrough(FlowSet in, Unit u, FlowSet out) {
in.copy(out);
// Perform generation
out.union(earliest.getFlowBefore(u));
/* Perform kill */
EquivalentValue equiVal = (EquivalentValue) unitToKillValue.get(u);
if (equiVal != null) {
out.remove(equiVal);
}
}
@Override
protected void merge(FlowSet inSet1, FlowSet inSet2, FlowSet outSet) {
inSet1.intersection(inSet2, outSet);
}
protected void copy(FlowSet sourceSet, FlowSet destSet) {
sourceSet.copy(destSet);
}
}