package scala.tools.nsc
package backend
package icode
import java.io.PrintWriter
import scala.collection.mutable
import scala.tools.nsc.symtab._
import analysis.{ Liveness, ReachingDefinitions }
import scala.tools.nsc.symtab.classfile.ICodeReader
abstract class ICodes extends AnyRef
with Members
with BasicBlocks
with Opcodes
with TypeStacks
with TypeKinds
with ExceptionHandlers
with Primitives
with Linearizers
with Printers
with Repository
{
val global: Global
import global.{ log, definitions, settings }
val classes = new mutable.HashMap[global.Symbol, IClass]
def shouldCheckIcode = settings.check contains global.genicode.phaseName
def checkerDebug(msg: String) = if (shouldCheckIcode && global.opt.debug) println(msg)
val linearizer: Linearizer = settings.Xlinearizer.value match {
case "rpo" => new ReversePostOrderLinearizer()
case "dfs" => new DepthFirstLinerizer()
case "normal" => new NormalLinearizer()
case "dump" => new DumpLinearizer()
case x => global.abort("Unknown linearizer: " + x)
}
private var alreadyDumping = false
def dump() {
if (alreadyDumping) return
else alreadyDumping = true
val printer = new TextPrinter(new PrintWriter(Console.out, true),
new DumpLinearizer)
classes.values foreach printer.printClass
}
def checkValid(m: IMethod) {
val bs = m.code.blocks.toList
for (b <- bs ; if !b.closed) {
if (b.isEmpty) {
log("!!! Found open but empty block while inlining " + m + ": removing from block list.")
m.code removeBlock b
}
else {
Console.println("Fatal bug in inliner: found open block when inlining " + m)
m.dump
global.abort("Open block was: " + b + " " + b.flagsString)
}
}
}
object liveness extends Liveness {
val global: ICodes.this.global.type = ICodes.this.global
}
object reachingDefinitions extends ReachingDefinitions {
val global: ICodes.this.global.type = ICodes.this.global
}
lazy val AnyRefReference: TypeKind = REFERENCE(definitions.AnyRefClass)
lazy val BoxedUnitReference: TypeKind = REFERENCE(definitions.BoxedUnitClass)
lazy val NothingReference: TypeKind = REFERENCE(definitions.NothingClass)
lazy val NullReference: TypeKind = REFERENCE(definitions.NullClass)
lazy val ObjectReference: TypeKind = REFERENCE(definitions.ObjectClass)
lazy val StringReference: TypeKind = REFERENCE(definitions.StringClass)
lazy val ThrowableReference: TypeKind = REFERENCE(definitions.ThrowableClass)
object icodeReader extends ICodeReader {
lazy val global: ICodes.this.global.type = ICodes.this.global
}
abstract class ICodePhase(prev: Phase) extends global.GlobalPhase(prev) {
override def erasedTypes = true
override def apply(unit: global.CompilationUnit): Unit =
unit.icode foreach apply
def apply(cls: global.icodes.IClass): Unit
}
}