| author | blanchet | 
| Mon, 04 Nov 2013 16:53:43 +0100 | |
| changeset 54246 | 8fdb4dc08ed1 | 
| parent 53976 | da610b507799 | 
| child 54519 | 5fed81762406 | 
| child 54637 | db3d3d99c69d | 
| permissions | -rw-r--r-- | 
| 47336 | 1 | (* Title: Pure/PIDE/command.ML | 
| 2 | Author: Makarius | |
| 3 | ||
| 52534 | 4 | Prover command execution: read -- eval -- print. | 
| 47336 | 5 | *) | 
| 6 | ||
| 7 | signature COMMAND = | |
| 8 | sig | |
| 52600 | 9 | val read: (unit -> theory) -> Token.T list -> Toplevel.transition | 
| 10 | type eval | |
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 11 | val eval_eq: eval * eval -> bool | 
| 52784 
4ba2e8b9972f
de-assign execs that were not registered as running yet -- observe change of perspective more thoroughly;
 wenzelm parents: 
52775diff
changeset | 12 | val eval_running: eval -> bool | 
| 52772 | 13 | val eval_finished: eval -> bool | 
| 52536 | 14 | val eval_result_state: eval -> Toplevel.state | 
| 52535 
b7badd371e4d
tuned signature -- eliminated pointless type synonym;
 wenzelm parents: 
52534diff
changeset | 15 | val eval: (unit -> theory) -> Token.T list -> eval -> eval | 
| 52600 | 16 | type print | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 17 | val print: bool -> (string * string list) list -> string -> | 
| 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 18 | eval -> print list -> print list option | 
| 52526 | 19 | type print_fn = Toplevel.transition -> Toplevel.state -> unit | 
| 52953 
2c927b7501c5
explicit "strict" flag for print functions (flipped internal meaning);
 wenzelm parents: 
52853diff
changeset | 20 | type print_function = | 
| 
2c927b7501c5
explicit "strict" flag for print functions (flipped internal meaning);
 wenzelm parents: 
52853diff
changeset | 21 |     {command_name: string, args: string list} ->
 | 
| 
2c927b7501c5
explicit "strict" flag for print functions (flipped internal meaning);
 wenzelm parents: 
52853diff
changeset | 22 |       {delay: Time.time option, pri: int, persistent: bool, strict: bool, print_fn: print_fn} option
 | 
| 
2c927b7501c5
explicit "strict" flag for print functions (flipped internal meaning);
 wenzelm parents: 
52853diff
changeset | 23 | val print_function: string -> print_function -> unit | 
| 52571 | 24 | val no_print_function: string -> unit | 
| 52600 | 25 | type exec = eval * print list | 
| 26 | val no_exec: exec | |
| 27 | val exec_ids: exec option -> Document_ID.exec list | |
| 52606 | 28 | val exec: Document_ID.execution -> exec -> unit | 
| 47336 | 29 | end; | 
| 30 | ||
| 31 | structure Command: COMMAND = | |
| 32 | struct | |
| 33 | ||
| 52775 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 34 | (** memo results **) | 
| 47341 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 35 | |
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 36 | datatype 'a expr = | 
| 52596 
40298d383463
global management of command execution fragments;
 wenzelm parents: 
52586diff
changeset | 37 | Expr of Document_ID.exec * (unit -> 'a) | | 
| 47341 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 38 | Result of 'a Exn.result; | 
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 39 | |
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 40 | abstype 'a memo = Memo of 'a expr Synchronized.var | 
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 41 | with | 
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 42 | |
| 52596 
40298d383463
global management of command execution fragments;
 wenzelm parents: 
52586diff
changeset | 43 | fun memo exec_id e = Memo (Synchronized.var "Command.memo" (Expr (exec_id, e))); | 
| 47341 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 44 | fun memo_value a = Memo (Synchronized.var "Command.memo" (Result (Exn.Res a))); | 
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 45 | |
| 47342 
7828c7b3c143
more explicit memo_eval vs. memo_result, to enforce bottom-up execution;
 wenzelm parents: 
47341diff
changeset | 46 | fun memo_result (Memo v) = | 
| 
7828c7b3c143
more explicit memo_eval vs. memo_result, to enforce bottom-up execution;
 wenzelm parents: 
47341diff
changeset | 47 | (case Synchronized.value v of | 
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 48 |     Expr (exec_id, _) => error ("Unfinished execution result: " ^ Document_ID.print exec_id)
 | 
| 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 49 | | Result res => Exn.release res); | 
| 47341 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 50 | |
| 52656 
9437f440ef3f
keep persistent prints only if actually finished;
 wenzelm parents: 
52651diff
changeset | 51 | fun memo_finished (Memo v) = | 
| 52785 | 52 | (case Synchronized.value v of Expr _ => false | Result _ => true); | 
| 52656 
9437f440ef3f
keep persistent prints only if actually finished;
 wenzelm parents: 
52651diff
changeset | 53 | |
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 54 | fun memo_exec execution_id (Memo v) = | 
| 52774 
627fb639a2d9
maintain explicit execution frontier: avoid conflict with former task via static dependency;
 wenzelm parents: 
52772diff
changeset | 55 | Synchronized.timed_access v (K (SOME Time.zeroTime)) | 
| 52609 
c8f8c29193de
clarified memo_exec: plain synchronized access without any special tricks;
 wenzelm parents: 
52607diff
changeset | 56 | (fn expr => | 
| 
c8f8c29193de
clarified memo_exec: plain synchronized access without any special tricks;
 wenzelm parents: 
52607diff
changeset | 57 | (case expr of | 
| 52775 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 58 | Expr (exec_id, body) => | 
| 52609 
c8f8c29193de
clarified memo_exec: plain synchronized access without any special tricks;
 wenzelm parents: 
52607diff
changeset | 59 | uninterruptible (fn restore_attributes => fn () => | 
| 53375 
78693e46a237
Execution.fork formally requires registered Execution.running;
 wenzelm parents: 
53192diff
changeset | 60 | if Execution.running execution_id exec_id [Future.the_worker_group ()] then | 
| 52609 
c8f8c29193de
clarified memo_exec: plain synchronized access without any special tricks;
 wenzelm parents: 
52607diff
changeset | 61 | let | 
| 52775 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 62 | val res = | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 63 | (body | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 64 | |> restore_attributes | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 65 | |> Future.worker_nest "Command.memo_exec" | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 66 | |> Exn.interruptible_capture) (); | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 67 | in SOME ((), Result res) end | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 68 | else SOME ((), expr)) () | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 69 | | Result _ => SOME ((), expr))) | 
| 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 70 | |> (fn NONE => error "Conflicting command execution" | _ => ()); | 
| 52596 
40298d383463
global management of command execution fragments;
 wenzelm parents: 
52586diff
changeset | 71 | |
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 72 | fun memo_fork params execution_id (Memo v) = | 
| 52596 
40298d383463
global management of command execution fragments;
 wenzelm parents: 
52586diff
changeset | 73 | (case Synchronized.value v of | 
| 
40298d383463
global management of command execution fragments;
 wenzelm parents: 
52586diff
changeset | 74 | Result _ => () | 
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 75 | | _ => ignore ((singleton o Future.forks) params (fn () => memo_exec execution_id (Memo v)))); | 
| 52526 | 76 | |
| 47341 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 77 | end; | 
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 78 | |
| 
00f6279bb67a
Command.memo including physical interrupts (unlike Lazy.lazy);
 wenzelm parents: 
47336diff
changeset | 79 | |
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 80 | |
| 52536 | 81 | (** main phases of execution **) | 
| 82 | ||
| 52510 | 83 | (* read *) | 
| 52509 | 84 | |
| 52534 | 85 | fun read init span = | 
| 52510 | 86 | let | 
| 87 | val outer_syntax = #2 (Outer_Syntax.get_syntax ()); | |
| 88 | val command_reports = Outer_Syntax.command_reports outer_syntax; | |
| 52509 | 89 | |
| 52534 | 90 | val proper_range = | 
| 91 | Position.set_range (Token.position_range_of (#1 (take_suffix Token.is_improper span))); | |
| 52510 | 92 | val pos = | 
| 93 | (case find_first Token.is_command span of | |
| 94 | SOME tok => Token.position_of tok | |
| 95 | | NONE => proper_range); | |
| 52509 | 96 | |
| 52510 | 97 | val (is_malformed, token_reports) = Thy_Syntax.reports_of_tokens span; | 
| 98 | val _ = Position.reports_text (token_reports @ maps command_reports span); | |
| 99 | in | |
| 100 | if is_malformed then Toplevel.malformed pos "Malformed command syntax" | |
| 101 | else | |
| 102 | (case Outer_Syntax.read_spans outer_syntax span of | |
| 103 | [tr] => | |
| 104 | if Keyword.is_control (Toplevel.name_of tr) then | |
| 105 | Toplevel.malformed pos "Illegal control command" | |
| 52534 | 106 | else Toplevel.modify_init init tr | 
| 107 | | [] => Toplevel.ignored (Position.set_range (Token.position_range_of span)) | |
| 52510 | 108 | | _ => Toplevel.malformed proper_range "Exactly one command expected") | 
| 109 | handle ERROR msg => Toplevel.malformed proper_range msg | |
| 110 | end; | |
| 52509 | 111 | |
| 112 | ||
| 113 | (* eval *) | |
| 47336 | 114 | |
| 52600 | 115 | type eval_state = | 
| 116 |   {failed: bool, malformed: bool, command: Toplevel.transition, state: Toplevel.state};
 | |
| 117 | val init_eval_state = | |
| 118 |   {failed = false, malformed = false, command = Toplevel.empty, state = Toplevel.toplevel};
 | |
| 119 | ||
| 120 | datatype eval = Eval of {exec_id: Document_ID.exec, eval_process: eval_state memo};
 | |
| 121 | ||
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 122 | fun eval_eq (Eval {exec_id, ...}, Eval {exec_id = exec_id', ...}) = exec_id = exec_id';
 | 
| 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 123 | |
| 52784 
4ba2e8b9972f
de-assign execs that were not registered as running yet -- observe change of perspective more thoroughly;
 wenzelm parents: 
52775diff
changeset | 124 | fun eval_running (Eval {exec_id, ...}) = Execution.is_running_exec exec_id;
 | 
| 52772 | 125 | fun eval_finished (Eval {eval_process, ...}) = memo_finished eval_process;
 | 
| 126 | ||
| 52600 | 127 | fun eval_result (Eval {eval_process, ...}) = memo_result eval_process;
 | 
| 128 | val eval_result_state = #state o eval_result; | |
| 129 | ||
| 47336 | 130 | local | 
| 131 | ||
| 132 | fun run int tr st = | |
| 53189 
ee8b8dafef0e
discontinued parallel_subproofs_saturation and related internal counters (superseded by parallel_subproofs_threshold and timing information);
 wenzelm parents: 
52999diff
changeset | 133 | if Goal.future_enabled 1 andalso Keyword.is_diag (Toplevel.name_of tr) then | 
| 53192 | 134 |     (Execution.fork {name = "Toplevel.diag", pos = Toplevel.pos_of tr, pri = ~1}
 | 
| 51605 
eca8acb42e4a
more explicit Goal.fork_params -- avoid implicit arguments via thread data;
 wenzelm parents: 
51603diff
changeset | 135 | (fn () => Toplevel.command_exception int tr st); ([], SOME st)) | 
| 51284 
59a03019f3bf
fork diagnostic commands (theory loader and PIDE interaction);
 wenzelm parents: 
51266diff
changeset | 136 | else Toplevel.command_errors int tr st; | 
| 47336 | 137 | |
| 52510 | 138 | fun check_cmts span tr st' = | 
| 139 | Toplevel.setmp_thread_position tr | |
| 140 | (fn () => | |
| 141 | Outer_Syntax.side_comments span |> maps (fn cmt => | |
| 142 | (Thy_Output.check_text (Token.source_position_of cmt) st'; []) | |
| 52619 
70d5f2f7d27f
reraise interrupts outside command regular transactions -- relevant for memo_stable;
 wenzelm parents: 
52609diff
changeset | 143 | handle exn => | 
| 
70d5f2f7d27f
reraise interrupts outside command regular transactions -- relevant for memo_stable;
 wenzelm parents: 
52609diff
changeset | 144 | if Exn.is_interrupt exn then reraise exn | 
| 
70d5f2f7d27f
reraise interrupts outside command regular transactions -- relevant for memo_stable;
 wenzelm parents: 
52609diff
changeset | 145 | else ML_Compiler.exn_messages_ids exn)) (); | 
| 52510 | 146 | |
| 47336 | 147 | fun proof_status tr st = | 
| 148 | (case try Toplevel.proof_of st of | |
| 149 | SOME prf => Toplevel.status tr (Proof.status_markup prf) | |
| 150 | | NONE => ()); | |
| 151 | ||
| 52534 | 152 | fun eval_state span tr ({malformed, state = st, ...}: eval_state) =
 | 
| 52509 | 153 | if malformed then | 
| 52526 | 154 |     {failed = true, malformed = malformed, command = tr, state = Toplevel.toplevel}
 | 
| 48772 | 155 | else | 
| 156 | let | |
| 157 | val malformed' = Toplevel.is_malformed tr; | |
| 158 | val is_init = Toplevel.is_init tr; | |
| 159 | val is_proof = Keyword.is_proof (Toplevel.name_of tr); | |
| 47336 | 160 | |
| 48772 | 161 | val _ = Multithreading.interrupted (); | 
| 50201 
c26369c9eda6
Isabelle-specific implementation of quasi-abstract markup elements -- back to module arrangement before d83797ef0d2d;
 wenzelm parents: 
49866diff
changeset | 162 | val _ = Toplevel.status tr Markup.running; | 
| 48918 
6e5fd4585512
check side-comments of command spans (normally filtered out in Outer_Syntax.toplevel_source);
 wenzelm parents: 
48772diff
changeset | 163 | val (errs1, result) = run (is_init orelse is_proof) (Toplevel.set_print false tr) st; | 
| 52509 | 164 | val errs2 = (case result of NONE => [] | SOME st' => check_cmts span tr st'); | 
| 48918 
6e5fd4585512
check side-comments of command spans (normally filtered out in Outer_Syntax.toplevel_source);
 wenzelm parents: 
48772diff
changeset | 165 | val errs = errs1 @ errs2; | 
| 50201 
c26369c9eda6
Isabelle-specific implementation of quasi-abstract markup elements -- back to module arrangement before d83797ef0d2d;
 wenzelm parents: 
49866diff
changeset | 166 | val _ = Toplevel.status tr Markup.finished; | 
| 50914 
fe4714886d92
identify future results more carefully, to avoid odd duplication of error messages, notably from forked goals;
 wenzelm parents: 
50911diff
changeset | 167 | val _ = List.app (Future.error_msg (Toplevel.pos_of tr)) errs; | 
| 48772 | 168 | in | 
| 169 | (case result of | |
| 170 | NONE => | |
| 171 | let | |
| 172 | val _ = if null errs then Exn.interrupt () else (); | |
| 50201 
c26369c9eda6
Isabelle-specific implementation of quasi-abstract markup elements -- back to module arrangement before d83797ef0d2d;
 wenzelm parents: 
49866diff
changeset | 173 | val _ = Toplevel.status tr Markup.failed; | 
| 52526 | 174 |           in {failed = true, malformed = malformed', command = tr, state = st} end
 | 
| 48772 | 175 | | SOME st' => | 
| 176 | let | |
| 177 | val _ = proof_status tr st'; | |
| 52526 | 178 |           in {failed = false, malformed = malformed', command = tr, state = st'} end)
 | 
| 48772 | 179 | end; | 
| 47336 | 180 | |
| 52534 | 181 | in | 
| 182 | ||
| 183 | fun eval init span eval0 = | |
| 184 | let | |
| 185 | val exec_id = Document_ID.make (); | |
| 186 | fun process () = | |
| 187 | let | |
| 188 | val tr = | |
| 189 | Position.setmp_thread_data (Position.id_only (Document_ID.print exec_id)) | |
| 52536 | 190 | (fn () => read init span |> Toplevel.exec_id exec_id) (); | 
| 52534 | 191 | in eval_state span tr (eval_result eval0) end; | 
| 52600 | 192 |   in Eval {exec_id = exec_id, eval_process = memo exec_id process} end;
 | 
| 52534 | 193 | |
| 47336 | 194 | end; | 
| 195 | ||
| 52509 | 196 | |
| 197 | (* print *) | |
| 198 | ||
| 52600 | 199 | datatype print = Print of | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 200 |  {name: string, args: string list, delay: Time.time option, pri: int, persistent: bool,
 | 
| 52600 | 201 | exec_id: Document_ID.exec, print_process: unit memo}; | 
| 202 | ||
| 52526 | 203 | type print_fn = Toplevel.transition -> Toplevel.state -> unit; | 
| 52515 | 204 | |
| 52647 
45ce95b8bf69
determine print function parameters dynamically, e.g. depending on runtime options;
 wenzelm parents: 
52619diff
changeset | 205 | type print_function = | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 206 |   {command_name: string, args: string list} ->
 | 
| 52953 
2c927b7501c5
explicit "strict" flag for print functions (flipped internal meaning);
 wenzelm parents: 
52853diff
changeset | 207 |     {delay: Time.time option, pri: int, persistent: bool, strict: bool, print_fn: print_fn} option;
 | 
| 52647 
45ce95b8bf69
determine print function parameters dynamically, e.g. depending on runtime options;
 wenzelm parents: 
52619diff
changeset | 208 | |
| 52511 | 209 | local | 
| 210 | ||
| 52647 
45ce95b8bf69
determine print function parameters dynamically, e.g. depending on runtime options;
 wenzelm parents: 
52619diff
changeset | 211 | val print_functions = | 
| 
45ce95b8bf69
determine print function parameters dynamically, e.g. depending on runtime options;
 wenzelm parents: 
52619diff
changeset | 212 | Synchronized.var "Command.print_functions" ([]: (string * print_function) list); | 
| 52511 | 213 | |
| 52570 | 214 | fun print_error tr e = | 
| 53709 
84522727f9d3
improved printing of exception trace in Poly/ML 5.5.1;
 wenzelm parents: 
53404diff
changeset | 215 | (Toplevel.setmp_thread_position tr o Toplevel.controlled_execution) e () | 
| 52619 
70d5f2f7d27f
reraise interrupts outside command regular transactions -- relevant for memo_stable;
 wenzelm parents: 
52609diff
changeset | 216 | handle exn => | 
| 
70d5f2f7d27f
reraise interrupts outside command regular transactions -- relevant for memo_stable;
 wenzelm parents: 
52609diff
changeset | 217 | if Exn.is_interrupt exn then reraise exn | 
| 
70d5f2f7d27f
reraise interrupts outside command regular transactions -- relevant for memo_stable;
 wenzelm parents: 
52609diff
changeset | 218 | else List.app (Future.error_msg (Toplevel.pos_of tr)) (ML_Compiler.exn_messages_ids exn); | 
| 52516 
b5b3c888df9f
more exception handling -- make print functions total;
 wenzelm parents: 
52515diff
changeset | 219 | |
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 220 | fun print_eq (Print {exec_id, ...}, Print {exec_id = exec_id', ...}) = exec_id = exec_id';
 | 
| 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 221 | |
| 52775 
e0169f13bd37
keep memo_exec execution running, which is important to cancel goal forks eventually;
 wenzelm parents: 
52774diff
changeset | 222 | fun print_finished (Print {print_process, ...}) = memo_finished print_process;
 | 
| 52656 
9437f440ef3f
keep persistent prints only if actually finished;
 wenzelm parents: 
52651diff
changeset | 223 | |
| 52600 | 224 | fun print_persistent (Print {persistent, ...}) = persistent;
 | 
| 52596 
40298d383463
global management of command execution fragments;
 wenzelm parents: 
52586diff
changeset | 225 | |
| 52853 
4ab66773a41f
prefer canonical order, to avoid potential fluctuation due to front-end edits;
 wenzelm parents: 
52850diff
changeset | 226 | val overlay_ord = prod_ord string_ord (list_ord string_ord); | 
| 
4ab66773a41f
prefer canonical order, to avoid potential fluctuation due to front-end edits;
 wenzelm parents: 
52850diff
changeset | 227 | |
| 52511 | 228 | in | 
| 52509 | 229 | |
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 230 | fun print command_visible command_overlays command_name eval old_prints = | 
| 52570 | 231 | let | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 232 | val print_functions = Synchronized.value print_functions; | 
| 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 233 | |
| 52999 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 234 |     fun make_print name args {delay, pri, persistent, strict, print_fn} =
 | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 235 | let | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 236 | val exec_id = Document_ID.make (); | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 237 | fun process () = | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 238 | let | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 239 |             val {failed, command, state = st', ...} = eval_result eval;
 | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 240 | val tr = Toplevel.exec_id exec_id command; | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 241 | in | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 242 | if failed andalso strict then () | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 243 | else print_error tr (fn () => print_fn tr st') | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 244 | end; | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 245 | in | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 246 |         Print {
 | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 247 | name = name, args = args, delay = delay, pri = pri, persistent = persistent, | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 248 | exec_id = exec_id, print_process = memo exec_id process} | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 249 | end; | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 250 | |
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 251 | fun bad_print name args exn = | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 252 |       make_print name args {delay = NONE, pri = 0, persistent = false,
 | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 253 | strict = false, print_fn = fn _ => fn _ => reraise exn}; | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 254 | |
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 255 | fun new_print name args get_pr = | 
| 52570 | 256 | let | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 257 |         val params = {command_name = command_name, args = args};
 | 
| 52570 | 258 | in | 
| 53709 
84522727f9d3
improved printing of exception trace in Poly/ML 5.5.1;
 wenzelm parents: 
53404diff
changeset | 259 | (case Exn.capture (Toplevel.controlled_execution get_pr) params of | 
| 52570 | 260 | Exn.Res NONE => NONE | 
| 52999 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 261 | | Exn.Res (SOME pr) => SOME (make_print name args pr) | 
| 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 262 | | Exn.Exn exn => SOME (bad_print name args exn)) | 
| 52570 | 263 | end; | 
| 264 | ||
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 265 | fun get_print (a, b) = | 
| 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 266 |       (case find_first (fn Print {name, args, ...} => name = a andalso args = b) old_prints of
 | 
| 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 267 | NONE => | 
| 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 268 | (case AList.lookup (op =) print_functions a of | 
| 52999 
1f09c98a3232
more explicit error, e.g. for sledgehammer query operation applied in non-HOL session;
 wenzelm parents: 
52953diff
changeset | 269 |             NONE => SOME (bad_print a b (ERROR ("Missing print function " ^ quote a)))
 | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 270 | | SOME get_pr => new_print a b get_pr) | 
| 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 271 | | some => some); | 
| 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 272 | |
| 52570 | 273 | val new_prints = | 
| 274 | if command_visible then | |
| 52853 
4ab66773a41f
prefer canonical order, to avoid potential fluctuation due to front-end edits;
 wenzelm parents: 
52850diff
changeset | 275 | fold (fn (a, _) => cons (a, [])) print_functions command_overlays | 
| 
4ab66773a41f
prefer canonical order, to avoid potential fluctuation due to front-end edits;
 wenzelm parents: 
52850diff
changeset | 276 | |> sort_distinct overlay_ord | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 277 | |> map_filter get_print | 
| 52656 
9437f440ef3f
keep persistent prints only if actually finished;
 wenzelm parents: 
52651diff
changeset | 278 | else filter (fn print => print_finished print andalso print_persistent print) old_prints; | 
| 52570 | 279 | in | 
| 52600 | 280 | if eq_list print_eq (old_prints, new_prints) then NONE else SOME new_prints | 
| 52570 | 281 | end; | 
| 52511 | 282 | |
| 52647 
45ce95b8bf69
determine print function parameters dynamically, e.g. depending on runtime options;
 wenzelm parents: 
52619diff
changeset | 283 | fun print_function name f = | 
| 52511 | 284 | Synchronized.change print_functions (fn funs => | 
| 285 | (if not (AList.defined (op =) funs name) then () | |
| 286 |     else warning ("Redefining command print function: " ^ quote name);
 | |
| 52647 
45ce95b8bf69
determine print function parameters dynamically, e.g. depending on runtime options;
 wenzelm parents: 
52619diff
changeset | 287 | AList.update (op =) (name, f) funs)); | 
| 52511 | 288 | |
| 52571 | 289 | fun no_print_function name = | 
| 290 | Synchronized.change print_functions (filter_out (equal name o #1)); | |
| 291 | ||
| 52511 | 292 | end; | 
| 293 | ||
| 52526 | 294 | val _ = | 
| 52647 
45ce95b8bf69
determine print function parameters dynamically, e.g. depending on runtime options;
 wenzelm parents: 
52619diff
changeset | 295 | print_function "print_state" | 
| 52850 
9fff9f78240a
support print functions with explicit arguments, as provided by overlays;
 wenzelm parents: 
52785diff
changeset | 296 |     (fn {command_name, ...} =>
 | 
| 53404 
d598b6231ff7
non-persistent print_state: trade-off between JVM space vs. ML time;
 wenzelm parents: 
53375diff
changeset | 297 |       SOME {delay = NONE, pri = 1, persistent = false, strict = true,
 | 
| 52651 | 298 | print_fn = fn tr => fn st' => | 
| 299 | let | |
| 300 | val is_init = Keyword.is_theory_begin command_name; | |
| 301 | val is_proof = Keyword.is_proof command_name; | |
| 302 | val do_print = | |
| 303 | not is_init andalso | |
| 304 | (Toplevel.print_of tr orelse (is_proof andalso Toplevel.is_proof st')); | |
| 305 | in if do_print then Toplevel.print_state false st' else () end}); | |
| 52509 | 306 | |
| 52532 | 307 | |
| 52600 | 308 | (* combined execution *) | 
| 309 | ||
| 310 | type exec = eval * print list; | |
| 311 | val no_exec: exec = | |
| 312 |   (Eval {exec_id = Document_ID.none, eval_process = memo_value init_eval_state}, []);
 | |
| 52532 | 313 | |
| 52600 | 314 | fun exec_ids NONE = [] | 
| 315 |   | exec_ids (SOME (Eval {exec_id, ...}, prints)) =
 | |
| 316 |       exec_id :: map (fn Print {exec_id, ...} => exec_id) prints;
 | |
| 317 | ||
| 318 | local | |
| 319 | ||
| 52651 | 320 | fun run_print execution_id (Print {name, delay, pri, print_process, ...}) =
 | 
| 53976 
da610b507799
low-priority print task is always asynchronous -- relevant for single-core machine and automatically tried tools;
 wenzelm parents: 
53709diff
changeset | 321 | if Multithreading.enabled () orelse pri <= 0 then | 
| 52651 | 322 | let | 
| 323 | val group = Future.worker_subgroup (); | |
| 324 | fun fork () = | |
| 325 |         memo_fork {name = name, group = SOME group, deps = [], pri = pri, interrupts = true}
 | |
| 326 | execution_id print_process; | |
| 327 | in | |
| 52762 | 328 | (case delay of | 
| 329 | NONE => fork () | |
| 330 | | SOME d => ignore (Event_Timer.request (Time.+ (Time.now (), d)) fork)) | |
| 52651 | 331 | end | 
| 332 | else memo_exec execution_id print_process; | |
| 52559 
ddaf277e0d8c
more direct interleaving of eval/print and update/execution -- refrain from crude manipulation of max_threads;
 wenzelm parents: 
52536diff
changeset | 333 | |
| 52600 | 334 | in | 
| 335 | ||
| 52607 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 336 | fun exec execution_id (Eval {eval_process, ...}, prints) =
 | 
| 
33a133d50616
clarified execution: maintain running execs only, check "stable" separately via memo (again);
 wenzelm parents: 
52606diff
changeset | 337 | (memo_exec execution_id eval_process; List.app (run_print execution_id) prints); | 
| 52532 | 338 | |
| 47336 | 339 | end; | 
| 340 | ||
| 52600 | 341 | end; | 
| 342 |