sbt/compile/integration/Compiler.scala

127 lines
5.1 KiB
Scala
Raw Normal View History

2010-09-04 14:46:36 +02:00
/* sbt -- Simple Build Tool
* Copyright 2010 Mark Harrah
*/
package sbt
import xsbti.{Logger => _,_}
2011-03-02 12:46:28 +01:00
import compiler._
2010-09-04 14:46:36 +02:00
import inc._
import Locate.DefinesClass
2010-09-04 14:46:36 +02:00
import java.io.File
object Compiler
2010-09-04 14:46:36 +02:00
{
2010-09-05 17:12:44 +02:00
val DefaultMaxErrors = 100
2011-04-12 04:12:03 +02:00
final case class Inputs(compilers: Compilers, config: Options, incSetup: IncSetup)
final case class Options(classpath: Seq[File], sources: Seq[File], classesDirectory: File, options: Seq[String], javacOptions: Seq[String], maxErrors: Int, order: CompileOrder.Value)
final case class IncSetup(analysisMap: Map[File, Analysis], definesClass: DefinesClass, skip: Boolean, cacheFile: File)
2011-04-12 04:12:03 +02:00
final case class Compilers(scalac: AnalyzingCompiler, javac: JavaCompiler)
2010-09-04 14:46:36 +02:00
@deprecated("Use the other inputs variant.", "0.12.0")
def inputs(classpath: Seq[File], sources: Seq[File], outputDirectory: File, options: Seq[String], javacOptions: Seq[String], definesClass: DefinesClass, maxErrors: Int, order: CompileOrder.Value)(implicit compilers: Compilers, log: Logger): Inputs =
2010-09-04 14:46:36 +02:00
{
import Path._
2010-09-04 14:46:36 +02:00
val classesDirectory = outputDirectory / "classes"
val cacheFile = outputDirectory / "cache_old_style"
2010-09-04 14:46:36 +02:00
val augClasspath = classesDirectory.asFile +: classpath
val incSetup = IncSetup(Map.empty, definesClass, false, cacheFile)
inputs(augClasspath, sources, classesDirectory, options, javacOptions, maxErrors, order)(compilers, incSetup, log)
2010-09-04 14:46:36 +02:00
}
def inputs(classpath: Seq[File], sources: Seq[File], classesDirectory: File, options: Seq[String], javacOptions: Seq[String], maxErrors: Int, order: CompileOrder.Value)(implicit compilers: Compilers, incSetup: IncSetup, log: Logger): Inputs =
2010-09-04 14:46:36 +02:00
new Inputs(
compilers,
2011-04-04 03:15:35 +02:00
new Options(classpath, sources, classesDirectory, options, javacOptions, maxErrors, order),
incSetup
2010-09-04 14:46:36 +02:00
)
2010-09-05 17:12:44 +02:00
2011-05-26 04:44:22 +02:00
def compilers(cpOptions: ClasspathOptions)(implicit app: AppConfiguration, log: Logger): Compilers =
2010-09-05 17:12:44 +02:00
{
val scalaProvider = app.provider.scalaProvider
2011-05-26 04:44:22 +02:00
compilers(ScalaInstance(scalaProvider.version, scalaProvider.launcher), cpOptions)
2010-09-05 17:12:44 +02:00
}
2011-05-26 04:44:22 +02:00
def compilers(instance: ScalaInstance, cpOptions: ClasspathOptions)(implicit app: AppConfiguration, log: Logger): Compilers =
compilers(instance, cpOptions, None)
2010-09-04 14:46:36 +02:00
def compilers(instance: ScalaInstance, cpOptions: ClasspathOptions, javaHome: Option[File])(implicit app: AppConfiguration, log: Logger): Compilers =
2010-09-04 14:46:36 +02:00
{
val javac = directOrFork(instance, cpOptions, javaHome)
2010-09-04 14:46:36 +02:00
compilers(instance, cpOptions, javac)
}
def compilers(instance: ScalaInstance, cpOptions: ClasspathOptions, javac: JavaCompiler.Fork)(implicit app: AppConfiguration, log: Logger): Compilers =
2010-09-04 14:46:36 +02:00
{
val javaCompiler = JavaCompiler.fork(cpOptions, instance)(javac)
compilers(instance, cpOptions, javaCompiler)
}
def compilers(instance: ScalaInstance, cpOptions: ClasspathOptions, javac: JavaCompiler)(implicit app: AppConfiguration, log: Logger): Compilers =
{
val scalac = scalaCompiler(instance, cpOptions)
new Compilers(scalac, javac)
}
def scalaCompiler(instance: ScalaInstance, cpOptions: ClasspathOptions)(implicit app: AppConfiguration, log: Logger): AnalyzingCompiler =
{
val launcher = app.provider.scalaProvider.launcher
val componentManager = new ComponentManager(launcher.globalLock, app.provider.components, Option(launcher.ivyHome), log)
2010-09-04 14:46:36 +02:00
new AnalyzingCompiler(instance, componentManager, cpOptions, log)
}
def directOrFork(instance: ScalaInstance, cpOptions: ClasspathOptions, javaHome: Option[File]): JavaCompiler =
if(javaHome.isDefined)
JavaCompiler.fork(cpOptions, instance)(forkJavac(javaHome))
else
JavaCompiler.directOrFork(cpOptions, instance)(forkJavac(None))
def forkJavac(javaHome: Option[File]): JavaCompiler.Fork =
{
import Path._
def exec(jc: JavacContract) = javaHome match { case None => jc.name; case Some(jh) => (jh / "bin" / jc.name).absolutePath }
(contract: JavacContract, args: Seq[String], log: Logger) => {
log.debug("Forking " + contract.name + ": " + exec(contract) + " " + args.mkString(" "))
val javacLogger = new JavacLogger(log)
var exitCode = -1
try {
exitCode = Process(exec(contract), args) ! javacLogger
} finally {
javacLogger.flush(exitCode)
}
exitCode
}
}
2010-09-04 14:46:36 +02:00
def apply(in: Inputs, log: Logger): Analysis =
2010-09-04 14:46:36 +02:00
{
import in.compilers._
import in.config._
import in.incSetup._
val agg = new AggressiveCompile(cacheFile)
agg(scalac, javac, sources, classpath, classesDirectory, options, javacOptions, analysisMap, definesClass, maxErrors, order, skip)(log)
2010-09-04 14:46:36 +02:00
}
}
private[sbt] class JavacLogger(log: Logger) extends ProcessLogger {
import scala.collection.mutable.ListBuffer
import Level.{Info, Warn, Error, Value => LogLevel}
private val msgs: ListBuffer[(LogLevel, String)] = new ListBuffer()
def info(s: => String): Unit =
synchronized { msgs += ((Info, s)) }
def error(s: => String): Unit =
synchronized { msgs += ((Error, s)) }
def buffer[T](f: => T): T = f
private def print(desiredLevel: LogLevel)(t: (LogLevel, String)) = t match {
case (Info, msg) => log.info(msg)
case (Error, msg) => log.log(desiredLevel, msg)
}
def flush(exitCode: Int): Unit = {
val level = if (exitCode == 0) Warn else Error
msgs foreach print(level)
msgs.clear()
}
}