| author | wenzelm | 
| Fri, 17 Jul 2020 14:56:55 +0200 | |
| changeset 72053 | 4ed33ea8d957 | 
| parent 71879 | fe7ee970c425 | 
| child 72103 | 7b318273a4aa | 
| permissions | -rw-r--r-- | 
| 45709 
87017fcbad83
clarified modules (again) -- NB: both Document and Protocol are specific to this particular prover;
 wenzelm parents: 
45672diff
changeset | 1 | (* Title: Pure/PIDE/protocol.ML | 
| 38412 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 2 | Author: Makarius | 
| 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 3 | |
| 45709 
87017fcbad83
clarified modules (again) -- NB: both Document and Protocol are specific to this particular prover;
 wenzelm parents: 
45672diff
changeset | 4 | Protocol message formats for interactive proof documents. | 
| 38412 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 5 | *) | 
| 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 6 | |
| 45709 
87017fcbad83
clarified modules (again) -- NB: both Document and Protocol are specific to this particular prover;
 wenzelm parents: 
45672diff
changeset | 7 | structure Protocol: sig end = | 
| 38412 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 8 | struct | 
| 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 9 | |
| 38418 
9a7af64d71bb
more explicit / functional ML version of document model;
 wenzelm parents: 
38417diff
changeset | 10 | val _ = | 
| 56387 | 11 | Isabelle_Process.protocol_command "Prover.echo" | 
| 52786 | 12 | (fn args => List.app writeln args); | 
| 13 | ||
| 14 | val _ = | |
| 71879 | 15 | Isabelle_Process.protocol_command "Prover.stop" | 
| 16 | (fn [rc] => raise Isabelle_Process.STOP (Value.parse_int rc)); | |
| 17 | ||
| 18 | val _ = | |
| 56387 | 19 | Isabelle_Process.protocol_command "Prover.options" | 
| 52786 | 20 | (fn [options_yxml] => | 
| 65300 | 21 | (Options.set_default (Options.decode (YXML.parse_body options_yxml)); | 
| 22 | Isabelle_Process.init_options_interactive ())); | |
| 52585 | 23 | |
| 24 | val _ = | |
| 71875 | 25 | Isabelle_Process.protocol_command "Prover.init_session" | 
| 70683 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70665diff
changeset | 26 | (fn [session_positions_yxml, session_directories_yxml, doc_names_yxml, global_theories_yxml, | 
| 70712 
a3cfe859d915
find theories via session directories only -- ignore known_theories;
 wenzelm parents: 
70683diff
changeset | 27 | loaded_theories_yxml] => | 
| 65470 | 28 | let | 
| 29 | val decode_table = YXML.parse_body #> let open XML.Decode in list (pair string string) end; | |
| 66712 | 30 | val decode_list = YXML.parse_body #> let open XML.Decode in list string end; | 
| 67493 
c4e9e0c50487
treat sessions as entities with defining position;
 wenzelm parents: 
67471diff
changeset | 31 | val decode_sessions = | 
| 
c4e9e0c50487
treat sessions as entities with defining position;
 wenzelm parents: 
67471diff
changeset | 32 | YXML.parse_body #> let open XML.Decode in list (pair string properties) end; | 
| 65470 | 33 | in | 
| 71875 | 34 | Resources.init_session | 
| 71876 | 35 |           {pide = true,
 | 
| 36 | session_positions = decode_sessions session_positions_yxml, | |
| 70683 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70665diff
changeset | 37 | session_directories = decode_table session_directories_yxml, | 
| 67493 
c4e9e0c50487
treat sessions as entities with defining position;
 wenzelm parents: 
67471diff
changeset | 38 | docs = decode_list doc_names_yxml, | 
| 67219 | 39 | global_theories = decode_table global_theories_yxml, | 
| 70712 
a3cfe859d915
find theories via session directories only -- ignore known_theories;
 wenzelm parents: 
70683diff
changeset | 40 | loaded_theories = decode_list loaded_theories_yxml} | 
| 65470 | 41 | end); | 
| 42 | ||
| 43 | val _ = | |
| 54519 
5fed81762406
maintain blobs within document state: digest + text in ML, digest-only in Scala;
 wenzelm parents: 
53192diff
changeset | 44 | Isabelle_Process.protocol_command "Document.define_blob" | 
| 
5fed81762406
maintain blobs within document state: digest + text in ML, digest-only in Scala;
 wenzelm parents: 
53192diff
changeset | 45 | (fn [digest, content] => Document.change_state (Document.define_blob digest content)); | 
| 
5fed81762406
maintain blobs within document state: digest + text in ML, digest-only in Scala;
 wenzelm parents: 
53192diff
changeset | 46 | |
| 70664 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 47 | fun decode_command id name blobs_xml toks_xml sources : Document.command = | 
| 70663 | 48 | let | 
| 49 | open XML.Decode; | |
| 50 | val (blobs_digests, blobs_index) = | |
| 70664 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 51 | blobs_xml |> | 
| 70663 | 52 | let | 
| 53 | val message = YXML.string_of_body o Protocol_Message.command_positions id; | |
| 54 | in | |
| 55 | pair | |
| 56 | (list (variant | |
| 57 | [fn ([], a) => Exn.Res (pair string (option string) a), | |
| 58 | fn ([], a) => Exn.Exn (ERROR (message a))])) | |
| 59 | int | |
| 60 | end; | |
| 70664 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 61 | val toks = list (pair int int) toks_xml; | 
| 70663 | 62 | in | 
| 63 |    {command_id = Document_ID.parse id,
 | |
| 64 | name = name, | |
| 65 | blobs_digests = blobs_digests, | |
| 66 | blobs_index = blobs_index, | |
| 67 | tokens = toks ~~ sources} | |
| 68 | end; | |
| 69 | ||
| 71024 | 70 | fun commands_accepted ids = | 
| 71 | Output.protocol_message Markup.commands_accepted [XML.Text (space_implode "," ids)]; | |
| 70665 
94442fce40a5
prefer commands_accepted: fewer protocol messages;
 wenzelm parents: 
70664diff
changeset | 72 | |
| 54519 
5fed81762406
maintain blobs within document state: digest + text in ML, digest-only in Scala;
 wenzelm parents: 
53192diff
changeset | 73 | val _ = | 
| 46119 
0d7172a7672c
tuned signature -- emphasize special nature of protocol commands;
 wenzelm parents: 
45709diff
changeset | 74 | Isabelle_Process.protocol_command "Document.define_command" | 
| 70664 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 75 | (fn id :: name :: blobs :: toks :: sources => | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 76 | let | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 77 | val command = decode_command id name (YXML.parse_body blobs) (YXML.parse_body toks) sources; | 
| 70665 
94442fce40a5
prefer commands_accepted: fewer protocol messages;
 wenzelm parents: 
70664diff
changeset | 78 | val _ = Document.change_state (Document.define_command command); | 
| 
94442fce40a5
prefer commands_accepted: fewer protocol messages;
 wenzelm parents: 
70664diff
changeset | 79 | in commands_accepted [id] end); | 
| 70664 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 80 | |
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 81 | val _ = | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 82 | Isabelle_Process.protocol_command "Document.define_commands" | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 83 | (fn args => | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 84 | let | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 85 | fun decode arg = | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 86 | let | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 87 | open XML.Decode; | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 88 | val (id, (name, (blobs_xml, (toks_xml, sources)))) = | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 89 | pair string (pair string (pair I (pair I (list string)))) (YXML.parse_body arg); | 
| 
2bd9e30183b1
prefer define_commands_bulk: fewer protocol messages;
 wenzelm parents: 
70663diff
changeset | 90 | in decode_command id name blobs_xml toks_xml sources end; | 
| 70665 
94442fce40a5
prefer commands_accepted: fewer protocol messages;
 wenzelm parents: 
70664diff
changeset | 91 | |
| 
94442fce40a5
prefer commands_accepted: fewer protocol messages;
 wenzelm parents: 
70664diff
changeset | 92 | val commands = map decode args; | 
| 
94442fce40a5
prefer commands_accepted: fewer protocol messages;
 wenzelm parents: 
70664diff
changeset | 93 | val _ = Document.change_state (fold Document.define_command commands); | 
| 
94442fce40a5
prefer commands_accepted: fewer protocol messages;
 wenzelm parents: 
70664diff
changeset | 94 | in commands_accepted (map (Value.print_int o #command_id) commands) end); | 
| 38412 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 95 | |
| 38418 
9a7af64d71bb
more explicit / functional ML version of document model;
 wenzelm parents: 
38417diff
changeset | 96 | val _ = | 
| 47343 
b8aeab386414
less aggressive discontinue_execution before document update, to avoid unstable execs that need to be re-assigned;
 wenzelm parents: 
46938diff
changeset | 97 | Isabelle_Process.protocol_command "Document.discontinue_execution" | 
| 52606 | 98 | (fn [] => Execution.discontinue ()); | 
| 47343 
b8aeab386414
less aggressive discontinue_execution before document update, to avoid unstable execs that need to be re-assigned;
 wenzelm parents: 
46938diff
changeset | 99 | |
| 
b8aeab386414
less aggressive discontinue_execution before document update, to avoid unstable execs that need to be re-assigned;
 wenzelm parents: 
46938diff
changeset | 100 | val _ = | 
| 52931 
ac6648c0c0fb
cancel_query via direct access to the exec_id of the running query process;
 wenzelm parents: 
52862diff
changeset | 101 | Isabelle_Process.protocol_command "Document.cancel_exec" | 
| 
ac6648c0c0fb
cancel_query via direct access to the exec_id of the running query process;
 wenzelm parents: 
52862diff
changeset | 102 | (fn [exec_id] => Execution.cancel (Document_ID.parse exec_id)); | 
| 
ac6648c0c0fb
cancel_query via direct access to the exec_id of the running query process;
 wenzelm parents: 
52862diff
changeset | 103 | |
| 
ac6648c0c0fb
cancel_query via direct access to the exec_id of the running query process;
 wenzelm parents: 
52862diff
changeset | 104 | val _ = | 
| 46119 
0d7172a7672c
tuned signature -- emphasize special nature of protocol commands;
 wenzelm parents: 
45709diff
changeset | 105 | Isabelle_Process.protocol_command "Document.update" | 
| 59370 
b13ff987c559
refrain from default task_context for all protocol commands, e.g. relevant for "build_theories" to admit Session.shutdown;
 wenzelm parents: 
59366diff
changeset | 106 | (Future.task_context "Document.update" (Future.new_group NONE) | 
| 69849 | 107 | (fn old_id_string :: new_id_string :: consolidate_yxml :: edits_yxml => | 
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 108 | Document.change_state (fn state => | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 109 | let | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 110 | val old_id = Document_ID.parse old_id_string; | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 111 | val new_id = Document_ID.parse new_id_string; | 
| 69849 | 112 | val consolidate = | 
| 113 | YXML.parse_body consolidate_yxml |> | |
| 114 | let open XML.Decode in list string end; | |
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 115 | val edits = | 
| 69849 | 116 | edits_yxml |> map (YXML.parse_body #> | 
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 117 | let open XML.Decode in | 
| 69849 | 118 | pair string | 
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 119 | (variant | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 120 | [fn ([], a) => Document.Edits (list (pair (option int) (option int)) a), | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 121 | fn ([], a) => | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 122 | let | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 123 | val (master, (name, (imports, (keywords, errors)))) = | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 124 | pair string (pair string (pair (list string) | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 125 | (pair (list (pair string | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 126 | (pair (pair string (list string)) (list string)))) | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 127 | (list YXML.string_of_body)))) a; | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 128 | val imports' = map (rpair Position.none) imports; | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 129 | val keywords' = map (fn (x, y) => ((x, Position.none), y)) keywords; | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 130 | val header = Thy_Header.make (name, Position.none) imports' keywords'; | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 131 |                         in Document.Deps {master = master, header = header, errors = errors} end,
 | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 132 | fn (a :: b, c) => | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 133 | Document.Perspective (bool_atom a, map int_atom b, | 
| 69849 | 134 | list (pair int (pair string (list string))) c)]) | 
| 135 | end); | |
| 47404 
e6e5750f1311
simplified Future.cancel/cancel_group (again) -- running threads only;
 wenzelm parents: 
47388diff
changeset | 136 | |
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 137 | val _ = Execution.discontinue (); | 
| 38412 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 138 | |
| 70284 
3e17c3a5fd39
more thorough assignment, e.g. when "purge" removes commands that were not assigned;
 wenzelm parents: 
69849diff
changeset | 139 | val (edited, removed, assign_update, state') = | 
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 140 | Document.update old_id new_id edits consolidate state; | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 141 | val _ = | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 142 | (singleton o Future.forks) | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 143 |                {name = "Document.update/remove", group = NONE,
 | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 144 | deps = Execution.snapshot removed, | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 145 | pri = Task_Queue.urgent_pri + 2, interrupts = false} | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 146 | (fn () => (Execution.purge removed; List.app Isabelle_Process.reset_tracing removed)); | 
| 52601 | 147 | |
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 148 | val _ = | 
| 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 149 | Output.protocol_message Markup.assign_update | 
| 70284 
3e17c3a5fd39
more thorough assignment, e.g. when "purge" removes commands that were not assigned;
 wenzelm parents: 
69849diff
changeset | 150 | ((new_id, edited, assign_update) |> | 
| 69846 | 151 | let | 
| 152 | open XML.Encode; | |
| 153 | fun encode_upd (a, bs) = | |
| 154 | string (space_implode "," (map Value.print_int (a :: bs))); | |
| 70991 
f9f7c34b7dd4
more scalable protocol_message: use XML.body directly (Output.output hook is not required);
 wenzelm parents: 
70712diff
changeset | 155 | in triple int (list string) (list encode_upd) end); | 
| 68336 
09ac56914b29
Document.update includes node consolidation / presentation as regular print operation: avoid user operations on protocol thread;
 wenzelm parents: 
67493diff
changeset | 156 | in Document.start_execution state' end))); | 
| 38412 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 157 | |
| 43748 | 158 | val _ = | 
| 46119 
0d7172a7672c
tuned signature -- emphasize special nature of protocol commands;
 wenzelm parents: 
45709diff
changeset | 159 | Isabelle_Process.protocol_command "Document.remove_versions" | 
| 44673 | 160 | (fn [versions_yxml] => Document.change_state (fn state => | 
| 161 | let | |
| 162 | val versions = | |
| 163 | YXML.parse_body versions_yxml |> | |
| 164 | let open XML.Decode in list int end; | |
| 44676 | 165 | val state1 = Document.remove_versions versions state; | 
| 70991 
f9f7c34b7dd4
more scalable protocol_message: use XML.body directly (Output.output hook is not required);
 wenzelm parents: 
70712diff
changeset | 166 | val _ = Output.protocol_message Markup.removed_versions [XML.Text (versions_yxml)]; | 
| 44676 | 167 | in state1 end)); | 
| 44673 | 168 | |
| 169 | val _ = | |
| 50498 | 170 | Isabelle_Process.protocol_command "Document.dialog_result" | 
| 50500 
c94bba7906d2
identify dialogs via official serial and maintain as result message;
 wenzelm parents: 
50498diff
changeset | 171 | (fn [serial, result] => | 
| 63806 | 172 | Active.dialog_result (Value.parse_int serial) result | 
| 62505 | 173 | handle exn => if Exn.is_interrupt exn then () (*sic!*) else Exn.reraise exn); | 
| 50498 | 174 | |
| 56616 
abc2da18d08d
added protocol command "use_theories", with core functionality of batch build;
 wenzelm parents: 
56458diff
changeset | 175 | val _ = | 
| 62467 | 176 | Isabelle_Process.protocol_command "ML_Heap.share_common_data" | 
| 177 | (fn [] => ML_Heap.share_common_data ()); | |
| 57868 
0b954ac94827
protocol command for heap management, e.g. in Isabelle/jEdit/Scala console: PIDE.session.protocol_command("ML_System.share_common_data");
 wenzelm parents: 
56616diff
changeset | 178 | |
| 38412 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 179 | end; | 
| 
c23f3abbf42d
moved isar_document.ML/scala to Pure/System/ -- side-by-side with isar.ML;
 wenzelm parents: diff
changeset | 180 |