package scala.tools.nsc
package backend.icode.analysis
import scala.collection.{ mutable, immutable }
trait DataFlowAnalysis[L <: SemiLattice] {
type P <: ProgramPoint[P]
val lattice: L
val worklist: mutable.Set[P] = new mutable.LinkedHashSet
val in: mutable.Map[P, lattice.Elem] = new mutable.HashMap
val out: mutable.Map[P, lattice.Elem] = new mutable.HashMap
val visited: mutable.HashSet[P] = new mutable.HashSet
var stat = true
var iterations = 0
def init(f: => Unit): Unit = {
iterations = 0
in.clear; out.clear; worklist.clear; visited.clear;
f
}
def reinit(f: => Unit): Unit = {
iterations = 0
worklist.clear; visited.clear;
f
}
def run(): Unit
def forwardAnalysis(f: (P, lattice.Elem) => lattice.Elem): Unit = try {
while (!worklist.isEmpty) {
if (stat) iterations += 1
val point = worklist.iterator.next; worklist -= point; visited += point;
val output = f(point, in(point))
if ((lattice.bottom == out(point)) || output != out(point)) {
out(point) = output
val succs = point.successors
succs foreach { p =>
if (!worklist(p))
worklist += p;
if (!in.isDefinedAt(p))
assert(false, "Invalid successor for: " + point + " successor " + p + " does not exist")
in(p) = lattice.lub(in(p) :: (p.predecessors map out.apply), p.exceptionHandlerStart)
}
}
}
} catch {
case e: NoSuchElementException =>
Console.println("in: " + in.mkString("", "\n", ""))
Console.println("out: " + out.mkString("", "\n", ""))
e.printStackTrace
sys.error("Could not find element " + e.getMessage)
}
def backwardAnalysis(f: (P, lattice.Elem) => lattice.Elem): Unit =
while (worklist.nonEmpty) {
if (stat) iterations += 1
val point = worklist.head
worklist -= point
out(point) = lattice.lub(point.successors map in.apply, false)
val input = f(point, out(point))
if ((lattice.bottom == in(point)) || input != in(point)) {
in(point) = input
worklist ++= point.predecessors
}
}
}