read/write dependency information;
authorwenzelm
Tue, 24 Jul 2012 23:01:55 +0200
changeset 48494 00eb5be9e76b
parent 48493 142ab4ff8fa8
child 48495 bf5b45870110
read/write dependency information; tuned signature;
src/Pure/General/file.scala
src/Pure/System/build.scala
--- a/src/Pure/General/file.scala	Tue Jul 24 22:00:12 2012 +0200
+++ b/src/Pure/General/file.scala	Tue Jul 24 23:01:55 2012 +0200
@@ -37,10 +37,10 @@
 
   /* write */
 
-  private def write_file(file: JFile, text: CharSequence, zip: Boolean)
+  private def write_file(file: JFile, text: CharSequence, gzip: Boolean)
   {
     val stream1 = new FileOutputStream(file)
-    val stream2 = if (zip) new GZIPOutputStream(new BufferedOutputStream(stream1)) else stream1
+    val stream2 = if (gzip) new GZIPOutputStream(new BufferedOutputStream(stream1)) else stream1
     val writer = new BufferedWriter(new OutputStreamWriter(stream2, Standard_System.charset))
     try { writer.append(text) }
     finally { writer.close }
@@ -49,8 +49,8 @@
   def write(file: JFile, text: CharSequence): Unit = write_file(file, text, false)
   def write(path: Path, text: CharSequence): Unit = write(path.file, text)
 
-  def write_zip(file: JFile, text: CharSequence): Unit = write_file(file, text, true)
-  def write_zip(path: Path, text: CharSequence): Unit = write_zip(path.file, text)
+  def write_gzip(file: JFile, text: CharSequence): Unit = write_file(file, text, true)
+  def write_gzip(path: Path, text: CharSequence): Unit = write_gzip(path.file, text)
 
 
   /* copy */
--- a/src/Pure/System/build.scala	Tue Jul 24 22:00:12 2012 +0200
+++ b/src/Pure/System/build.scala	Tue Jul 24 23:01:55 2012 +0200
@@ -7,7 +7,9 @@
 package isabelle
 
 
-import java.io.{File => JFile}
+import java.io.{File => JFile, BufferedInputStream, FileInputStream,
+  BufferedReader, InputStreamReader, IOException}
+import java.util.zip.GZIPInputStream
 
 import scala.collection.mutable
 import scala.annotation.tailrec
@@ -301,6 +303,18 @@
     def sources(name: String): List[(Path, SHA1.Digest)] = deps(name).sources
   }
 
+  private def read_deps(file: JFile): List[String] =
+    if (file.isFile) {
+      val stream = new GZIPInputStream (new BufferedInputStream(new FileInputStream(file)))
+      val reader = new BufferedReader(new InputStreamReader(stream, Standard_System.charset))
+      try {
+        List(reader.readLine).filter(_.startsWith("sources:")) :::
+        List(reader.readLine).filter(_.startsWith("heap:"))
+      }
+      finally { reader.close }
+    }
+    else Nil
+
   def dependencies(verbose: Boolean, queue: Session.Queue): Deps =
     Deps((Map.empty[String, Node] /: queue.topological_order)(
       { case (deps, (name, info)) =>
@@ -344,7 +358,8 @@
 
   /* jobs */
 
-  private class Job(cwd: JFile, env: Map[String, String], script: String, args: String)
+  private class Job(cwd: JFile, env: Map[String, String], script: String, args: String,
+    val output_path: Option[Path])
   {
     private val args_file = File.tmp_file("args")
     private val env1 = env + ("ARGS_FILE" -> Isabelle_System.posix_path(args_file.getPath))
@@ -358,7 +373,7 @@
     def join: (String, String, Int) = { val res = result.join; args_file.delete; res }
   }
 
-  private def start_job(name: String, info: Session.Info, output: Option[String],
+  private def start_job(name: String, info: Session.Info, output_path: Option[Path],
     options: Options, timing: Boolean, verbose: Boolean, browser_info: Path): Job =
   {
     // global browser info dir
@@ -376,15 +391,18 @@
     val parent = info.parent.getOrElse("")
     val parent_base_name = info.parent_base_name.getOrElse("")
 
+    val output =
+      output_path match { case Some(p) => Isabelle_System.standard_path(p) case None => "" }
+
     val cwd = info.dir.file
-    val env = Map("INPUT" -> parent, "TARGET" -> name, "OUTPUT" -> output.getOrElse(""))
+    val env = Map("INPUT" -> parent, "TARGET" -> name, "OUTPUT" -> output)
     val script =
       if (is_pure(name)) "./build " + name + " \"$OUTPUT\""
       else {
         """
         . "$ISABELLE_HOME/lib/scripts/timestart.bash"
         """ +
-          (if (output.isDefined)
+          (if (output_path.isDefined)
             """ "$ISABELLE_PROCESS" -e "Build.build \"$ARGS_FILE\";" -q -w "$INPUT" "$OUTPUT" """
           else
             """ "$ISABELLE_PROCESS" -e "Build.build \"$ARGS_FILE\";" -r -q "$INPUT" """) +
@@ -405,10 +423,10 @@
       import XML.Encode._
           pair(bool, pair(Options.encode, pair(bool, pair(bool, pair(Path.encode, pair(string,
             pair(string, pair(string, list(pair(Options.encode, list(Path.encode)))))))))))(
-          (output.isDefined, (options, (timing, (verbose, (browser_info, (parent_base_name,
+          (output_path.isDefined, (options, (timing, (verbose, (browser_info, (parent_base_name,
             (name, (info.base_name, info.theories)))))))))
     }
-    new Job(cwd, env, script, YXML.string_of_body(args_xml))
+    new Job(cwd, env, script, YXML.string_of_body(args_xml), output_path)
   }
 
 
@@ -448,7 +466,13 @@
           val sources =
             (queue(name).digest :: deps.sources(name).map(_._2)).map(_.toString).sorted
               .mkString("sources: ", " ", "\n")
-          File.write_zip(log.ext("gz"), sources + out)
+          val heap =
+            job.output_path.map(_.file) match {
+              case Some(file) =>
+                "heap: " + file.length.toString + " " + file.lastModified.toString + "\n"
+              case None => ""
+            }
+          File.write_gzip(log.ext("gz"), sources + heap + out)
         }
         else {
           File.write(log, out)
@@ -469,7 +493,7 @@
             else if (info.parent.map(results(_)).forall(_ == 0)) {
               val output =
                 if (build_images || queue.is_inner(name))
-                  Some(Isabelle_System.standard_path(output_dir + Path.basic(name)))
+                  Some(output_dir + Path.basic(name))
                 else None
               echo((if (output.isDefined) "Building " else "Running ") + name + " ...")
               val job = start_job(name, info, output, info.options, timing, verbose, browser_info)