package scala.tools.nsc
package backend
package icode
import scala.collection.{ mutable, immutable }
import mutable.{ ArrayBuffer }
import util.{ Position, NoPosition }
import backend.icode.analysis.ProgramPoint
trait BasicBlocks {
self: ICodes =>
import opcodes._
import global.{ settings, log, nme }
import nme.isExceptionResultName
class BasicBlock(val label: Int, val method: IMethod)
extends AnyRef
with ProgramPoint[BasicBlock]
with Seq[Instruction] {
import BBFlags._
def code = method.code
private var flags: Int = 0
def hasFlag(flag: Int): Boolean = (flags & flag) != 0
private def setFlag(flag: Int): Unit = flags |= flag
private def resetFlag(flag: Int) {
flags &= ~flag
}
def closed: Boolean = hasFlag(CLOSED)
def closed_=(b: Boolean) = if (b) setFlag(CLOSED) else resetFlag(CLOSED)
def ignore: Boolean = hasFlag(IGNORING)
def ignore_=(b: Boolean) = if (b) setFlag(IGNORING) else resetFlag(IGNORING)
def loopHeader = hasFlag(LOOP_HEADER)
def loopHeader_=(b: Boolean) =
if (b) setFlag(LOOP_HEADER) else resetFlag(LOOP_HEADER)
def exceptionHandlerStart = hasFlag(EX_HEADER)
def exceptionHandlerStart_=(b: Boolean) =
if (b) setFlag(EX_HEADER) else resetFlag(EX_HEADER)
def touched = hasFlag(DIRTYSUCCS)
def touched_=(b: Boolean) = if (b) {
setFlag(DIRTYSUCCS | DIRTYPREDS)
} else {
resetFlag(DIRTYSUCCS | DIRTYPREDS)
}
setFlag(DIRTYSUCCS | DIRTYPREDS)
var preds: List[BasicBlock] = null
var varsInScope: mutable.Set[Local] = new mutable.LinkedHashSet()
private var instructionList: List[Instruction] = Nil
private var instrs: Array[Instruction] = _
override def toList: List[Instruction] =
if (closed) instrs.toList else instructionList.reverse
def iterator: Iterator[Instruction] =
if (closed) instrs.iterator else instructionList.reverse.iterator
def getArray: Array[Instruction] = {
assert(closed)
instrs
}
def fromList(is: List[Instruction]) {
code.touched = true
instrs = is.toArray
closed = true
}
def indexOf(inst: Instruction): Int = {
assert(closed)
instrs indexWhere (_ eq inst)
}
override def foreach[U](f: Instruction => U) = {
if (!closed) {
method.dump
global.abort("Traversing an open block!: " + label + " in " + method)
}
instrs foreach f
}
def length = if (closed) instrs.length else instructionList.length
def apply(n: Int): Instruction =
if (closed) instrs(n) else instructionList.reverse(n)
def replaceInstruction(pos: Int, instr: Instruction): Boolean = {
assert(closed, "Instructions can be replaced only after the basic block is closed")
instr.setPos(instrs(pos).pos)
instrs(pos) = instr
code.touched = true
true
}
def replaceInstruction(oldInstr: Instruction, newInstr: Instruction): Boolean = {
assert(closed, "Instructions can be replaced only after the basic block is closed")
indexOf(oldInstr) match {
case -1 => false
case idx =>
newInstr setPos oldInstr.pos
instrs(idx) = newInstr
code.touched = true
true
}
}
def replaceInstruction(oldInstr: Instruction, is: List[Instruction]): Boolean = {
assert(closed, "Instructions can be replaced only after the basic block is closed")
indexOf(oldInstr) match {
case -1 => false
case idx =>
instrs = instrs.patch(idx, is, 1)
code.touched = true
true
}
}
def insertAfter(idx: Int, is: List[Instruction]) {
assert(closed, "Instructions can be replaced only after the basic block is closed")
instrs = instrs.patch(idx + 1, is, 0)
code.touched = true
}
def removeInstructionsAt(positions: Int*) {
assert(closed)
instrs = instrs.indices.toArray filterNot positions.toSet map instrs
code.touched = true
}
def removeLastInstruction() {
if (closed)
removeInstructionsAt(size)
else {
instructionList = instructionList.tail
code.touched = true
}
}
def subst(map: Map[Instruction, Instruction]): Unit =
if (!closed)
instructionList = instructionList map (x => map.getOrElse(x, x))
else
instrs.zipWithIndex collect {
case (oldInstr, i) if map contains oldInstr =>
code.touched |= replaceInstruction(i, map(oldInstr))
}
def emit(instr: Instruction) {
val pos = if (instructionList.isEmpty) NoPosition else instructionList.head.pos
emit(instr, pos)
}
def emit(instr: Instruction, pos: Position) {
assert(!closed || ignore, "BasicBlock closed")
if (ignore) {
if (settings.debug.value) {
instr match {
case JUMP(_) | RETURN(_) | THROW(_) | SCOPE_EXIT(_) =>
case STORE_LOCAL(local) if isExceptionResultName(local.sym.name) =>
case x => log("Ignoring instruction, possibly at our peril, at " + pos + ": " + x)
}
}
}
else {
instr.setPos(pos)
instructionList ::= instr
}
}
def emit(instrs: Seq[Instruction]) {
instrs foreach (i => emit(i, i.pos))
}
def emitOnly(instrs: Instruction*) {
instrs foreach (i => if (i.pos == NoPosition) emit(i) else emit(i, i.pos))
this.close
}
def closeWith(instr: Instruction) {
if (closed) () else {
emit(instr)
close
}
}
def closeWith(instr: Instruction, pos: Position) {
if (closed) () else {
emit(instr, pos)
close
}
}
def close() {
assert(!closed || ignore)
assert(instructionList.nonEmpty, "Empty block.")
closed = true
setFlag(DIRTYSUCCS)
instructionList = instructionList.reverse
instrs = instructionList.toArray
}
def open() {
assert(closed)
closed = false
ignore = false
touched = true
instructionList = instructionList.reverse
}
def clear() {
instructionList = Nil
instrs = null
preds = null
}
override def isEmpty: Boolean = instructionList.isEmpty
def enterIgnoreMode() = {
ignore = true
}
def exitIgnoreMode() {
assert(ignore, "Exit ignore mode when not in ignore mode.")
ignore = false
}
def lastInstruction =
if (closed) instrs.last
else instructionList.head
def firstInstruction =
if (closed) instrs(0)
else instructionList.last
def exceptionSuccessorsForBlock(block: BasicBlock): List[BasicBlock] =
method.exh collect { case x if x covers block => x.startBlock }
private var succs: List[BasicBlock] = Nil
private def updateSuccs() {
resetFlag(DIRTYSUCCS)
succs =
if (isEmpty) Nil
else exceptionSuccessors ++ directSuccessors ++ indirectExceptionSuccessors
}
def successors : List[BasicBlock] = {
if (touched) updateSuccs()
succs
}
def directSuccessors: List[BasicBlock] =
if (isEmpty) Nil else lastInstruction match {
case JUMP(whereto) => List(whereto)
case CJUMP(succ, fail, _, _) => fail :: succ :: Nil
case CZJUMP(succ, fail, _, _) => fail :: succ :: Nil
case SWITCH(_, labels) => labels
case RETURN(_) => Nil
case THROW(_) => Nil
case _ =>
if (closed) {
dump
global.abort("The last instruction is not a control flow instruction: " + lastInstruction)
}
else Nil
}
def exceptionSuccessors: List[BasicBlock] =
exceptionSuccessorsForBlock(this)
def indirectExceptionSuccessors: List[BasicBlock] =
directSuccessors flatMap exceptionSuccessorsForBlock distinct
def predecessors: List[BasicBlock] = {
if (hasFlag(DIRTYPREDS)) {
resetFlag(DIRTYPREDS)
preds = code.blocks.iterator filter (_.successors contains this) toList
}
preds
}
override def equals(other: Any): Boolean = other match {
case that: BasicBlock => (that.label == label) && (that.code == code)
case _ => false
}
override def hashCode = label * 41 + code.hashCode
def print() { print(java.lang.System.out) }
def print(out: java.io.PrintStream) {
out.println("block #"+label+" :")
foreach(i => out.println(" " + i))
out.print("Successors: ")
successors.foreach((x: BasicBlock) => out.print(" "+x.label.toString()))
out.println()
}
private def succString = if (successors.isEmpty) "[S: N/A]" else successors.distinct.mkString("[S: ", ", ", "]")
private def predString = if (predecessors.isEmpty) "[P: N/A]" else predecessors.distinct.mkString("[P: ", ", ", "]")
override def toString(): String = "" + label
def blockContents = {
def posStr(p: Position) = if (p.isDefined) p.line.toString else "<??>"
val xs = this.toList map (instr => posStr(instr.pos) + "\t" + instr)
xs.mkString(fullString + " {\n ", "\n ", "\n}")
}
def predContents = predecessors.map(_.blockContents).mkString(predecessors.size + " preds:\n", "\n", "\n")
def succContents = successors.map(_.blockContents).mkString(successors.size + " succs:\n", "\n", "\n")
def fullString: String = List("Block", label, succString, predString, flagsString) mkString " "
def flagsString: String = BBFlags.flagsToString(flags)
}
}
object BBFlags {
val flagMap = Map[Int, String](
LOOP_HEADER -> "loopheader",
IGNORING -> "ignore",
EX_HEADER -> "exheader",
CLOSED -> "closed",
DIRTYSUCCS -> "dirtysuccs",
DIRTYPREDS -> "dirtypreds"
)
def flagsToString(flags: Int) = {
flagMap collect { case (bit, name) if (bit & flags) != 0 => "<" + name + ">" } mkString " "
}
final val LOOP_HEADER = (1 << 0)
final val IGNORING = (1 << 1)
final val EX_HEADER = (1 << 2)
final val CLOSED = (1 << 3)
final val DIRTYSUCCS = (1 << 4)
final val DIRTYPREDS = (1 << 5)
}