| author | bulwahn | 
| Sat, 28 Jan 2012 12:05:26 +0100 | |
| changeset 46352 | 73b03235799a | 
| parent 46121 | 30a69cd8a9a0 | 
| child 46548 | c54a4a22501c | 
| permissions | -rw-r--r-- | 
| 43283 | 1 | /* Title: Pure/System/isabelle_process.scala | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 2 | Author: Makarius | 
| 27963 | 3 | Options: :folding=explicit:collapseFolds=1: | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 4 | |
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 5 | Isabelle process management -- always reactive due to multi-threaded I/O. | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 6 | */ | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 7 | |
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 8 | package isabelle | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 9 | |
| 43520 
cec9b95fa35d
explicit import java.lang.System to prevent odd scope problems;
 wenzelm parents: 
43514diff
changeset | 10 | import java.lang.System | 
| 27955 
4c32c5e75eca
use java.util.concurrent.LinkedBlockingQueue, which blocks as required;
 wenzelm parents: 
27949diff
changeset | 11 | import java.util.concurrent.LinkedBlockingQueue | 
| 28045 | 12 | import java.io.{BufferedReader, BufferedWriter, InputStreamReader, OutputStreamWriter,
 | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 13 | InputStream, OutputStream, BufferedOutputStream, IOException} | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 14 | |
| 32474 
0818e6b1c8a6
Isabelle_Process: receiver as Actor, not EventBus;
 wenzelm parents: 
32448diff
changeset | 15 | import scala.actors.Actor | 
| 
0818e6b1c8a6
Isabelle_Process: receiver as Actor, not EventBus;
 wenzelm parents: 
32448diff
changeset | 16 | import Actor._ | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 17 | |
| 27973 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 18 | |
| 32474 
0818e6b1c8a6
Isabelle_Process: receiver as Actor, not EventBus;
 wenzelm parents: 
32448diff
changeset | 19 | object Isabelle_Process | 
| 
0818e6b1c8a6
Isabelle_Process: receiver as Actor, not EventBus;
 wenzelm parents: 
32448diff
changeset | 20 | {
 | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 21 | /* results */ | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 22 | |
| 39525 | 23 | object Kind | 
| 24 |   {
 | |
| 25 | val message_markup = Map( | |
| 45666 | 26 |       ('A' : Int) -> Isabelle_Markup.INIT,
 | 
| 27 |       ('B' : Int) -> Isabelle_Markup.STATUS,
 | |
| 28 |       ('C' : Int) -> Isabelle_Markup.REPORT,
 | |
| 29 |       ('D' : Int) -> Isabelle_Markup.WRITELN,
 | |
| 30 |       ('E' : Int) -> Isabelle_Markup.TRACING,
 | |
| 31 |       ('F' : Int) -> Isabelle_Markup.WARNING,
 | |
| 32 |       ('G' : Int) -> Isabelle_Markup.ERROR,
 | |
| 33 |       ('H' : Int) -> Isabelle_Markup.RAW)
 | |
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 34 | } | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 35 | |
| 44733 
329320fc88df
buffer prover messages to prevent overloading of session_actor input channel -- which is critical due to synchronous messages wrt. GUI thread;
 wenzelm parents: 
44732diff
changeset | 36 | sealed abstract class Message | 
| 43721 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 37 | |
| 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 38 | class Input(name: String, args: List[String]) extends Message | 
| 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 39 |   {
 | 
| 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 40 | override def toString: String = | 
| 45666 | 41 | XML.Elem(Markup(Isabelle_Markup.PROVER_COMMAND, List((Markup.NAME, name))), | 
| 43721 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 42 | args.map(s => | 
| 45666 | 43 |           List(XML.Text("\n"),
 | 
| 44 | XML.elem(Isabelle_Markup.PROVER_ARG, YXML.parse_body(s)))).flatten).toString | |
| 43721 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 45 | } | 
| 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 46 | |
| 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 47 | class Result(val message: XML.Elem) extends Message | 
| 34100 | 48 |   {
 | 
| 43747 | 49 | def kind: String = message.markup.name | 
| 43780 | 50 | def properties: Properties.T = message.markup.properties | 
| 43747 | 51 | def body: XML.Body = message.body | 
| 37689 
628eabe2213a
simplified Isabelle_Process.Result: use markup directly;
 wenzelm parents: 
37132diff
changeset | 52 | |
| 45666 | 53 | def is_init = kind == Isabelle_Markup.INIT | 
| 54 | def is_exit = kind == Isabelle_Markup.EXIT | |
| 55 | def is_stdout = kind == Isabelle_Markup.STDOUT | |
| 56 | def is_stderr = kind == Isabelle_Markup.STDERR | |
| 57 | def is_system = kind == Isabelle_Markup.SYSTEM | |
| 58 | def is_status = kind == Isabelle_Markup.STATUS | |
| 59 | def is_report = kind == Isabelle_Markup.REPORT | |
| 60 | def is_raw = kind == Isabelle_Markup.RAW | |
| 46121 | 61 | def is_syslog = is_init || is_exit || is_system || is_stderr | 
| 34100 | 62 | |
| 63 | override def toString: String = | |
| 64 |     {
 | |
| 29522 
793766d4c1b5
moved message markup into Scala layer -- reduced redundancy;
 wenzelm parents: 
29506diff
changeset | 65 | val res = | 
| 38236 
d8c7be27e01d
explicitly distinguish Output.status (essential feedback) vs. Output.report (useful markup);
 wenzelm parents: 
38231diff
changeset | 66 | if (is_status || is_report) message.body.map(_.toString).mkString | 
| 43748 | 67 | else if (is_raw) "..." | 
| 37689 
628eabe2213a
simplified Isabelle_Process.Result: use markup directly;
 wenzelm parents: 
37132diff
changeset | 68 | else Pretty.string_of(message.body) | 
| 38230 
ed147003de4b
simplified type XML.Tree: embed Markup directly, avoid slightly odd triple;
 wenzelm parents: 
37712diff
changeset | 69 | if (properties.isEmpty) | 
| 29572 
e3a99d957392
replaced java.util.Properties by plain association list;
 wenzelm parents: 
29522diff
changeset | 70 | kind.toString + " [[" + res + "]]" | 
| 
e3a99d957392
replaced java.util.Properties by plain association list;
 wenzelm parents: 
29522diff
changeset | 71 | else | 
| 
e3a99d957392
replaced java.util.Properties by plain association list;
 wenzelm parents: 
29522diff
changeset | 72 | kind.toString + " " + | 
| 38230 
ed147003de4b
simplified type XML.Tree: embed Markup directly, avoid slightly odd triple;
 wenzelm parents: 
37712diff
changeset | 73 |           (for ((x, y) <- properties) yield x + "=" + y).mkString("{", ",", "}") + " [[" + res + "]]"
 | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 74 | } | 
| 27973 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 75 | } | 
| 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 76 | } | 
| 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 77 | |
| 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 78 | |
| 45055 
55274f7e306b
explicit option for socket vs. fifo communication;
 wenzelm parents: 
45027diff
changeset | 79 | class Isabelle_Process( | 
| 45075 | 80 | timeout: Time = Time.seconds(25), | 
| 45055 
55274f7e306b
explicit option for socket vs. fifo communication;
 wenzelm parents: 
45027diff
changeset | 81 | receiver: Isabelle_Process.Message => Unit = Console.println(_), | 
| 
55274f7e306b
explicit option for socket vs. fifo communication;
 wenzelm parents: 
45027diff
changeset | 82 | args: List[String] = Nil) | 
| 29192 | 83 | {
 | 
| 31797 | 84 | import Isabelle_Process._ | 
| 29194 | 85 | |
| 27973 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 86 | |
| 39626 
a5d0bcfb95a3
manage persistent syslog via Session, not Isabelle_Process;
 wenzelm parents: 
39625diff
changeset | 87 | /* results */ | 
| 39573 | 88 | |
| 89 | private def system_result(text: String) | |
| 90 |   {
 | |
| 45666 | 91 | receiver(new Result(XML.Elem(Markup(Isabelle_Markup.SYSTEM, Nil), List(XML.Text(text))))) | 
| 39573 | 92 | } | 
| 93 | ||
| 43745 | 94 | private val xml_cache = new XML.Cache() | 
| 39573 | 95 | |
| 43780 | 96 | private def put_result(kind: String, props: Properties.T, body: XML.Body) | 
| 39573 | 97 |   {
 | 
| 45666 | 98 | if (kind == Isabelle_Markup.INIT) system_channel.accepted() | 
| 99 | if (kind == Isabelle_Markup.RAW) | |
| 44732 | 100 | receiver(new Result(XML.Elem(Markup(kind, props), body))) | 
| 43746 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 101 |     else {
 | 
| 45709 
87017fcbad83
clarified modules (again) -- NB: both Document and Protocol are specific to this particular prover;
 wenzelm parents: 
45672diff
changeset | 102 | val msg = XML.Elem(Markup(kind, props), Protocol.clean_message(body)) | 
| 44732 | 103 | receiver(new Result(xml_cache.cache_tree(msg).asInstanceOf[XML.Elem])) | 
| 43746 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 104 | } | 
| 39573 | 105 | } | 
| 106 | ||
| 107 | private def put_result(kind: String, text: String) | |
| 108 |   {
 | |
| 43695 
5130dfe1b7be
simplified Symbol based on lazy Symbol.Interpretation -- reduced odd "functorial style";
 wenzelm parents: 
43661diff
changeset | 109 | put_result(kind, Nil, List(XML.Text(Symbol.decode(text)))) | 
| 39573 | 110 | } | 
| 111 | ||
| 112 | ||
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 113 | /* input actors */ | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 114 | |
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 115 | private case class Input_Text(text: String) | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 116 | private case class Input_Chunks(chunks: List[Array[Byte]]) | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 117 | |
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 118 | private case object Close | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 119 | private def close(p: (Thread, Actor)) | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 120 |   {
 | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 121 |     if (p != null && p._1.isAlive) {
 | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 122 | p._2 ! Close | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 123 | p._1.join | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 124 | } | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 125 | } | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 126 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 127 | @volatile private var standard_input: (Thread, Actor) = null | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 128 | @volatile private var command_input: (Thread, Actor) = null | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 129 | |
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 130 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 131 | /** process manager **/ | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 132 | |
| 45158 | 133 | private val system_channel = System_Channel() | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 134 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 135 | private val process = | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 136 |     try {
 | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 137 | val cmdline = | 
| 45027 
f459e93a038e
more abstract wrapping of fifos as System_Channel;
 wenzelm parents: 
44775diff
changeset | 138 |         Isabelle_System.getenv_strict("ISABELLE_PROCESS") ::
 | 
| 45055 
55274f7e306b
explicit option for socket vs. fifo communication;
 wenzelm parents: 
45027diff
changeset | 139 | (system_channel.isabelle_args ::: args) | 
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 140 | new Isabelle_System.Managed_Process(false, cmdline: _*) | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 141 | } | 
| 45027 
f459e93a038e
more abstract wrapping of fifos as System_Channel;
 wenzelm parents: 
44775diff
changeset | 142 |     catch { case e: IOException => system_channel.accepted(); throw(e) }
 | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 143 | |
| 39587 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 144 | val process_result = | 
| 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 145 |     Simple_Thread.future("process_result") { process.join }
 | 
| 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 146 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 147 | private def terminate_process() | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 148 |   {
 | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 149 |     try { process.terminate }
 | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 150 |     catch { case e: IOException => system_result("Failed to terminate Isabelle: " + e.getMessage) }
 | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 151 | } | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 152 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 153 |   private val process_manager = Simple_Thread.fork("process_manager")
 | 
| 39572 | 154 |   {
 | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 155 | val (startup_failed, startup_output) = | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 156 |     {
 | 
| 40848 
8662b9b1f123
more abstract/uniform handling of time, preferring seconds as Double;
 wenzelm parents: 
39731diff
changeset | 157 | val expired = System.currentTimeMillis() + timeout.ms | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 158 | val result = new StringBuilder(100) | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 159 | |
| 39587 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 160 | var finished: Option[Boolean] = None | 
| 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 161 |       while (finished.isEmpty && System.currentTimeMillis() <= expired) {
 | 
| 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 162 |         while (finished.isEmpty && process.stdout.ready) {
 | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 163 | val c = process.stdout.read | 
| 39587 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 164 | if (c == 2) finished = Some(true) | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 165 | else result += c.toChar | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 166 | } | 
| 39587 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 167 | if (process_result.is_finished) finished = Some(false) | 
| 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 168 | else Thread.sleep(10) | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 169 | } | 
| 39623 | 170 | (finished.isEmpty || !finished.get, result.toString.trim) | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 171 | } | 
| 39572 | 172 | system_result(startup_output) | 
| 173 | ||
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 174 |     if (startup_failed) {
 | 
| 45666 | 175 | put_result(Isabelle_Markup.EXIT, "Return code: 127") | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 176 | process.stdin.close | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 177 | Thread.sleep(300) | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 178 | terminate_process() | 
| 39587 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 179 | process_result.join | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 180 | } | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 181 |     else {
 | 
| 45027 
f459e93a038e
more abstract wrapping of fifos as System_Channel;
 wenzelm parents: 
44775diff
changeset | 182 | val (command_stream, message_stream) = system_channel.rendezvous() | 
| 39530 
16adc476348f
Isabelle_Process: more robust rendezvous, even without proper blocking on open (Cygwin);
 wenzelm parents: 
39528diff
changeset | 183 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 184 | standard_input = stdin_actor() | 
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 185 | val stdout = raw_output_actor(false) | 
| 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 186 | val stderr = raw_output_actor(true) | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 187 | command_input = input_actor(command_stream) | 
| 39572 | 188 | val message = message_actor(message_stream) | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 189 | |
| 39587 
f84b70e3bb9c
more reactive handling of Isabelle_Process startup errors;
 wenzelm parents: 
39585diff
changeset | 190 | val rc = process_result.join | 
| 39618 | 191 |       system_result("process terminated")
 | 
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 192 | for ((thread, _) <- List(standard_input, stdout, stderr, command_input, message)) | 
| 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 193 | thread.join | 
| 39572 | 194 |       system_result("process_manager terminated")
 | 
| 45666 | 195 | put_result(Isabelle_Markup.EXIT, "Return code: " + rc.toString) | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 196 | } | 
| 45027 
f459e93a038e
more abstract wrapping of fifos as System_Channel;
 wenzelm parents: 
44775diff
changeset | 197 | system_channel.accepted() | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 198 | } | 
| 27973 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 199 | |
| 39572 | 200 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 201 | /* management methods */ | 
| 27973 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 202 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 203 |   def join() { process_manager.join() }
 | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 204 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 205 | def terminate() | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 206 |   {
 | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 207 | close() | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 208 |     system_result("Terminating Isabelle process")
 | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 209 | terminate_process() | 
| 27973 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 210 | } | 
| 
18d02c0b90b6
moved class Result into static object, removed dynamic tree method;
 wenzelm parents: 
27963diff
changeset | 211 | |
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 212 | |
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 213 | |
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 214 | /** stream actors **/ | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 215 | |
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 216 | /* raw stdin */ | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 217 | |
| 39572 | 218 | private def stdin_actor(): (Thread, Actor) = | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 219 |   {
 | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 220 | val name = "standard_input" | 
| 38636 
b7647ca7de5a
module for simplified thread operations (Scala version);
 wenzelm parents: 
38573diff
changeset | 221 |     Simple_Thread.actor(name) {
 | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 222 | var finished = false | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 223 |       while (!finished) {
 | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 224 |         try {
 | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 225 |           //{{{
 | 
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 226 |           receive {
 | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 227 | case Input_Text(text) => | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 228 | process.stdin.write(text) | 
| 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 229 | process.stdin.flush | 
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 230 | case Close => | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 231 | process.stdin.close | 
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 232 | finished = true | 
| 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 233 | case bad => System.err.println(name + ": ignoring bad message " + bad) | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 234 | } | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 235 | //}}} | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 236 | } | 
| 39525 | 237 |         catch { case e: IOException => system_result(name + ": " + e.getMessage) }
 | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 238 | } | 
| 39525 | 239 | system_result(name + " terminated") | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 240 | } | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 241 | } | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 242 | |
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 243 | |
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 244 | /* raw output */ | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 245 | |
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 246 | private def raw_output_actor(err: Boolean): (Thread, Actor) = | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 247 |   {
 | 
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 248 | val (name, reader, markup) = | 
| 45666 | 249 |       if (err) ("standard_error", process.stderr, Isabelle_Markup.STDERR)
 | 
| 250 |       else ("standard_output", process.stdout, Isabelle_Markup.STDOUT)
 | |
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 251 | |
| 38636 
b7647ca7de5a
module for simplified thread operations (Scala version);
 wenzelm parents: 
38573diff
changeset | 252 |     Simple_Thread.actor(name) {
 | 
| 28045 | 253 | var result = new StringBuilder(100) | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 254 | |
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 255 | var finished = false | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 256 |       while (!finished) {
 | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 257 |         try {
 | 
| 28045 | 258 |           //{{{
 | 
| 259 | var c = -1 | |
| 260 | var done = false | |
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 261 |           while (!done && (result.length == 0 || reader.ready)) {
 | 
| 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 262 | c = reader.read | 
| 28045 | 263 | if (c >= 0) result.append(c.asInstanceOf[Char]) | 
| 264 | else done = true | |
| 265 | } | |
| 266 |           if (result.length > 0) {
 | |
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 267 | put_result(markup, result.toString) | 
| 28045 | 268 | result.length = 0 | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 269 | } | 
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 270 |           else {
 | 
| 45633 
2cb7e34f6096
retain stderr and include it in syslog, which is buffered and thus increases the chance that users see remains from crashes etc.;
 wenzelm parents: 
45158diff
changeset | 271 | reader.close | 
| 28045 | 272 | finished = true | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 273 | } | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 274 | //}}} | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 275 | } | 
| 39525 | 276 |         catch { case e: IOException => system_result(name + ": " + e.getMessage) }
 | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 277 | } | 
| 39525 | 278 | system_result(name + " terminated") | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 279 | } | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 280 | } | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 281 | |
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 282 | |
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 283 | /* command input */ | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 284 | |
| 39572 | 285 | private def input_actor(raw_stream: OutputStream): (Thread, Actor) = | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 286 |   {
 | 
| 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 287 | val name = "command_input" | 
| 38636 
b7647ca7de5a
module for simplified thread operations (Scala version);
 wenzelm parents: 
38573diff
changeset | 288 |     Simple_Thread.actor(name) {
 | 
| 39530 
16adc476348f
Isabelle_Process: more robust rendezvous, even without proper blocking on open (Cygwin);
 wenzelm parents: 
39528diff
changeset | 289 | val stream = new BufferedOutputStream(raw_stream) | 
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 290 | var finished = false | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 291 |       while (!finished) {
 | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 292 |         try {
 | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 293 |           //{{{
 | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 294 |           receive {
 | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 295 | case Input_Chunks(chunks) => | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 296 | stream.write(Standard_System.string_bytes( | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 297 |                   chunks.map(_.length).mkString("", ",", "\n")));
 | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 298 | chunks.foreach(stream.write(_)); | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 299 | stream.flush | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 300 | case Close => | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 301 | stream.close | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 302 | finished = true | 
| 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 303 | case bad => System.err.println(name + ": ignoring bad message " + bad) | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 304 | } | 
| 28045 | 305 | //}}} | 
| 27963 | 306 | } | 
| 39525 | 307 |         catch { case e: IOException => system_result(name + ": " + e.getMessage) }
 | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 308 | } | 
| 39525 | 309 | system_result(name + " terminated") | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 310 | } | 
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 311 | } | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 312 | |
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 313 | |
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 314 | /* message output */ | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 315 | |
| 39572 | 316 | private def message_actor(stream: InputStream): (Thread, Actor) = | 
| 39526 | 317 |   {
 | 
| 318 | class EOF extends Exception | |
| 319 | class Protocol_Error(msg: String) extends Exception(msg) | |
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 320 | |
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 321 | val name = "message_output" | 
| 38636 
b7647ca7de5a
module for simplified thread operations (Scala version);
 wenzelm parents: 
38573diff
changeset | 322 |     Simple_Thread.actor(name) {
 | 
| 34100 | 323 | val default_buffer = new Array[Byte](65536) | 
| 324 | var c = -1 | |
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 325 | |
| 43746 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 326 | def read_chunk(decode: Boolean): XML.Body = | 
| 34100 | 327 |       {
 | 
| 328 |         //{{{
 | |
| 329 | // chunk size | |
| 330 | var n = 0 | |
| 331 | c = stream.read | |
| 39526 | 332 | if (c == -1) throw new EOF | 
| 34100 | 333 |         while (48 <= c && c <= 57) {
 | 
| 334 | n = 10 * n + (c - 48) | |
| 335 | c = stream.read | |
| 336 | } | |
| 337 |         if (c != 10) throw new Protocol_Error("bad message chunk header")
 | |
| 338 | ||
| 339 | // chunk content | |
| 340 | val buf = | |
| 341 | if (n <= default_buffer.size) default_buffer | |
| 342 | else new Array[Byte](n) | |
| 343 | ||
| 344 | var i = 0 | |
| 345 | var m = 0 | |
| 346 |         do {
 | |
| 347 | m = stream.read(buf, i, n - i) | |
| 39731 
5cb0d7b0d601
bulk read: observe EOF protocol more carefully -- 0 counts as successful read;
 wenzelm parents: 
39632diff
changeset | 348 | if (m != -1) i += m | 
| 
5cb0d7b0d601
bulk read: observe EOF protocol more carefully -- 0 counts as successful read;
 wenzelm parents: 
39632diff
changeset | 349 | } while (m != -1 && n > i) | 
| 34100 | 350 | |
| 351 |         if (i != n) throw new Protocol_Error("bad message chunk content")
 | |
| 352 | ||
| 43746 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 353 | if (decode) | 
| 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 354 | YXML.parse_body_failsafe(Standard_System.decode_chars(Symbol.decode, buf, 0, n)) | 
| 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 355 | else List(XML.Text(Standard_System.decode_chars(s => s, buf, 0, n).toString)) | 
| 34100 | 356 | //}}} | 
| 357 | } | |
| 358 | ||
| 359 |       do {
 | |
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 360 |         try {
 | 
| 43746 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 361 | val header = read_chunk(true) | 
| 38445 
ba9ea6b9b75c
simplified internal message format: dropped special Symbol.STX header;
 wenzelm parents: 
38372diff
changeset | 362 |           header match {
 | 
| 
ba9ea6b9b75c
simplified internal message format: dropped special Symbol.STX header;
 wenzelm parents: 
38372diff
changeset | 363 | case List(XML.Elem(Markup(name, props), Nil)) | 
| 39525 | 364 | if name.size == 1 && Kind.message_markup.isDefinedAt(name(0)) => | 
| 43746 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 365 | val kind = Kind.message_markup(name(0)) | 
| 45666 | 366 | val body = read_chunk(kind != Isabelle_Markup.RAW) | 
| 43746 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 367 | put_result(kind, props, body) | 
| 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 368 | case _ => | 
| 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 369 | read_chunk(false) | 
| 
a41f618c641d
some support for raw messages, which bypass standard Symbol/YXML decoding;
 wenzelm parents: 
43745diff
changeset | 370 |               throw new Protocol_Error("bad header: " + header.toString)
 | 
| 28063 | 371 | } | 
| 27963 | 372 | } | 
| 28063 | 373 |         catch {
 | 
| 39525 | 374 |           case e: IOException => system_result("Cannot read message:\n" + e.getMessage)
 | 
| 375 |           case e: Protocol_Error => system_result("Malformed message:\n" + e.getMessage)
 | |
| 39526 | 376 | case _: EOF => | 
| 28063 | 377 | } | 
| 34100 | 378 | } while (c != -1) | 
| 379 | stream.close | |
| 380 | ||
| 39525 | 381 | system_result(name + " terminated") | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 382 | } | 
| 39526 | 383 | } | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 384 | |
| 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 385 | |
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 386 | /** main methods **/ | 
| 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 387 | |
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 388 | def input_raw(text: String): Unit = standard_input._2 ! Input_Text(text) | 
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 389 | |
| 38372 
e753f71b6b34
added Isabelle_Process.input_bytes, which avoids the somewhat slow Standard_System.string_bytes (just in case someone wants to stream raw data at 250MB/s);
 wenzelm parents: 
38270diff
changeset | 390 | def input_bytes(name: String, args: Array[Byte]*): Unit = | 
| 39585 
00be8711082f
main Isabelle_Process via Isabelle_System.Managed_Process;
 wenzelm parents: 
39575diff
changeset | 391 | command_input._2 ! Input_Chunks(Standard_System.string_bytes(name) :: args.toList) | 
| 38372 
e753f71b6b34
added Isabelle_Process.input_bytes, which avoids the somewhat slow Standard_System.string_bytes (just in case someone wants to stream raw data at 250MB/s);
 wenzelm parents: 
38270diff
changeset | 392 | |
| 38270 
71bb3c273dd1
native Isabelle_Process commands, based on efficient byte channel protocol for string lists;
 wenzelm parents: 
38262diff
changeset | 393 | def input(name: String, args: String*): Unit = | 
| 43721 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 394 |   {
 | 
| 44732 | 395 | receiver(new Input(name, args.toList)) | 
| 38372 
e753f71b6b34
added Isabelle_Process.input_bytes, which avoids the somewhat slow Standard_System.string_bytes (just in case someone wants to stream raw data at 250MB/s);
 wenzelm parents: 
38270diff
changeset | 396 | input_bytes(name, args.map(Standard_System.string_bytes): _*) | 
| 43721 
fad8634cee62
echo prover input via raw_messages, for improved protocol tracing;
 wenzelm parents: 
43695diff
changeset | 397 | } | 
| 38259 
2b61c5e27399
distinguish proper Isabelle_Process INPUT vs. raw STDIN, tuned corresponding method names;
 wenzelm parents: 
38253diff
changeset | 398 | |
| 39528 
c01d89d18ff0
refined Isabelle_Process startup: emit \002 before rendezvous on fifos, more robust treatment of startup failure with timeout, do not quit() after main loop;
 wenzelm parents: 
39526diff
changeset | 399 |   def close(): Unit = { close(command_input); close(standard_input) }
 | 
| 27949 
6eb0327c0b9b
Isabelle process management -- always reactive due to multi-threaded I/O.
 wenzelm parents: diff
changeset | 400 | } |