author | wenzelm |
Wed, 25 Mar 2015 14:39:40 +0100 | |
changeset 59812 | 675d0c692c41 |
parent 56467 | 8d7d6f17c6a7 |
child 62712 | 22a17cec2efe |
permissions | -rw-r--r-- |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
1 |
(* Title: Pure/System/options.ML |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
2 |
Author: Makarius |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
3 |
|
51945 | 4 |
System options with external string representation. |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
5 |
*) |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
6 |
|
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
7 |
signature OPTIONS = |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
8 |
sig |
51943 | 9 |
val boolT: string |
10 |
val intT: string |
|
11 |
val realT: string |
|
12 |
val stringT: string |
|
13 |
val unknownT: string |
|
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
14 |
type T |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
15 |
val empty: T |
59812 | 16 |
val names: T -> string list |
56467 | 17 |
val markup: T -> string * Position.T -> Markup.T |
51942 | 18 |
val typ: T -> string -> string |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
19 |
val bool: T -> string -> bool |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
20 |
val int: T -> string -> int |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
21 |
val real: T -> string -> real |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
22 |
val string: T -> string -> string |
51943 | 23 |
val put_bool: string -> bool -> T -> T |
24 |
val put_int: string -> int -> T -> T |
|
25 |
val put_real: string -> real -> T -> T |
|
26 |
val put_string: string -> string -> T -> T |
|
56465 | 27 |
val declare: {pos: Position.T, name: string, typ: string, value: string} -> T -> T |
51943 | 28 |
val update: string -> string -> T -> T |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
29 |
val decode: XML.body -> T |
48698
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
30 |
val default: unit -> T |
56467 | 31 |
val default_markup: string * Position.T -> Markup.T |
51946 | 32 |
val default_typ: string -> string |
51942 | 33 |
val default_bool: string -> bool |
34 |
val default_int: string -> int |
|
35 |
val default_real: string -> real |
|
36 |
val default_string: string -> string |
|
51943 | 37 |
val default_put_bool: string -> bool -> unit |
38 |
val default_put_int: string -> int -> unit |
|
39 |
val default_put_real: string -> real -> unit |
|
40 |
val default_put_string: string -> string -> unit |
|
41 |
val get_default: string -> string |
|
42 |
val put_default: string -> string -> unit |
|
48698
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
43 |
val set_default: T -> unit |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
44 |
val reset_default: unit -> unit |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
45 |
val load_default: unit -> unit |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
46 |
end; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
47 |
|
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
48 |
structure Options: OPTIONS = |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
49 |
struct |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
50 |
|
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
51 |
(* representation *) |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
52 |
|
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
53 |
val boolT = "bool"; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
54 |
val intT = "int"; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
55 |
val realT = "real"; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
56 |
val stringT = "string"; |
51943 | 57 |
val unknownT = "unknown"; |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
58 |
|
56465 | 59 |
datatype T = Options of {pos: Position.T, typ: string, value: string} Symtab.table; |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
60 |
|
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
61 |
val empty = Options Symtab.empty; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
62 |
|
59812 | 63 |
fun names (Options tab) = sort_strings (Symtab.keys tab); |
64 |
||
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
65 |
|
51943 | 66 |
(* check *) |
51942 | 67 |
|
51943 | 68 |
fun check_name (Options tab) name = |
69 |
let val opt = Symtab.lookup tab name in |
|
70 |
if is_some opt andalso #typ (the opt) <> unknownT then the opt |
|
51978 | 71 |
else error ("Unknown system option " ^ quote name) |
51943 | 72 |
end; |
73 |
||
74 |
fun check_type options name typ = |
|
75 |
let val opt = check_name options name in |
|
76 |
if #typ opt = typ then opt |
|
51978 | 77 |
else error ("Ill-typed system option " ^ quote name ^ " : " ^ #typ opt ^ " vs. " ^ typ) |
51943 | 78 |
end; |
51942 | 79 |
|
80 |
||
56467 | 81 |
(* markup *) |
51943 | 82 |
|
56467 | 83 |
fun markup options (name, pos) = |
84 |
let |
|
85 |
val opt = |
|
86 |
check_name options name |
|
87 |
handle ERROR msg => error (msg ^ Position.here pos); |
|
88 |
val props = Position.def_properties_of (#pos opt); |
|
89 |
in Markup.properties props (Markup.entity Markup.system_optionN name) end; |
|
90 |
||
91 |
||
92 |
(* typ *) |
|
93 |
||
51943 | 94 |
fun typ options name = #typ (check_name options name); |
95 |
||
96 |
||
97 |
(* basic operations *) |
|
98 |
||
99 |
fun put T print name x (options as Options tab) = |
|
100 |
let val opt = check_type options name T |
|
56465 | 101 |
in Options (Symtab.update (name, {pos = #pos opt, typ = #typ opt, value = print x}) tab) end; |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
102 |
|
51943 | 103 |
fun get T parse options name = |
104 |
let val opt = check_type options name T in |
|
105 |
(case parse (#value opt) of |
|
106 |
SOME x => x |
|
107 |
| NONE => |
|
51978 | 108 |
error ("Malformed value for system option " ^ quote name ^ |
51943 | 109 |
" : " ^ T ^ " =\n" ^ quote (#value opt))) |
110 |
end; |
|
111 |
||
112 |
||
113 |
(* internal lookup and update *) |
|
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
114 |
|
51951
fab4ab92e812
more standard Isabelle/ML operations -- avoid inaccurate Bool.fromString;
wenzelm
parents:
51946
diff
changeset
|
115 |
val bool = get boolT (try Markup.parse_bool); |
fab4ab92e812
more standard Isabelle/ML operations -- avoid inaccurate Bool.fromString;
wenzelm
parents:
51946
diff
changeset
|
116 |
val int = get intT (try Markup.parse_int); |
51988 | 117 |
val real = get realT (try Markup.parse_real); |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
118 |
val string = get stringT SOME; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
119 |
|
51951
fab4ab92e812
more standard Isabelle/ML operations -- avoid inaccurate Bool.fromString;
wenzelm
parents:
51946
diff
changeset
|
120 |
val put_bool = put boolT Markup.print_bool; |
fab4ab92e812
more standard Isabelle/ML operations -- avoid inaccurate Bool.fromString;
wenzelm
parents:
51946
diff
changeset
|
121 |
val put_int = put intT Markup.print_int; |
51990 | 122 |
val put_real = put realT Markup.print_real; |
51943 | 123 |
val put_string = put stringT I; |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
124 |
|
51943 | 125 |
|
126 |
(* external updates *) |
|
127 |
||
128 |
fun check_value options name = |
|
129 |
let val opt = check_name options name in |
|
130 |
if #typ opt = boolT then ignore (bool options name) |
|
131 |
else if #typ opt = intT then ignore (int options name) |
|
132 |
else if #typ opt = realT then ignore (real options name) |
|
133 |
else if #typ opt = stringT then ignore (string options name) |
|
134 |
else () |
|
135 |
end; |
|
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
136 |
|
56465 | 137 |
fun declare {pos, name, typ, value} (Options tab) = |
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
138 |
let |
56465 | 139 |
val options' = |
140 |
(case Symtab.lookup tab name of |
|
141 |
SOME other => |
|
142 |
error ("Duplicate declaration of system option " ^ quote name ^ Position.here pos ^ |
|
143 |
Position.here (#pos other)) |
|
144 |
| NONE => Options (Symtab.update (name, {pos = pos, typ = typ, value = value}) tab)); |
|
51943 | 145 |
val _ = |
146 |
typ = boolT orelse typ = intT orelse typ = realT orelse typ = stringT orelse |
|
56465 | 147 |
error ("Unknown type for system option " ^ quote name ^ " : " ^ quote typ ^ |
148 |
Position.here pos); |
|
51943 | 149 |
val _ = check_value options' name; |
150 |
in options' end; |
|
151 |
||
152 |
fun update name value (options as Options tab) = |
|
153 |
let |
|
154 |
val opt = check_name options name; |
|
56465 | 155 |
val options' = |
156 |
Options (Symtab.update (name, {pos = #pos opt, typ = #typ opt, value = value}) tab); |
|
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
157 |
val _ = check_value options' name; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
158 |
in options' end; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
159 |
|
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
160 |
|
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
161 |
(* decode *) |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
162 |
|
56465 | 163 |
fun decode_opt body = |
164 |
let open XML.Decode |
|
165 |
in list (pair properties (pair string (pair string string))) end body |
|
166 |
|> map (fn (props, (name, (typ, value))) => |
|
167 |
{pos = Position.of_properties props, name = name, typ = typ, value = value}); |
|
168 |
||
169 |
fun decode body = fold declare (decode_opt body) empty; |
|
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
170 |
|
48698
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
171 |
|
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
172 |
|
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
173 |
(** global default **) |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
174 |
|
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
175 |
val global_default = Synchronized.var "Options.default" (NONE: T option); |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
176 |
|
51978 | 177 |
fun err_no_default () = error "Missing default for system options within Isabelle process"; |
51943 | 178 |
|
179 |
fun change_default f x y = |
|
180 |
Synchronized.change global_default |
|
181 |
(fn SOME options => SOME (f x y options) |
|
182 |
| NONE => err_no_default ()); |
|
183 |
||
48698
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
184 |
fun default () = |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
185 |
(case Synchronized.value global_default of |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
186 |
SOME options => options |
51943 | 187 |
| NONE => err_no_default ()); |
48698
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
188 |
|
56467 | 189 |
fun default_markup arg = markup (default ()) arg; |
51946 | 190 |
fun default_typ name = typ (default ()) name; |
51942 | 191 |
fun default_bool name = bool (default ()) name; |
192 |
fun default_int name = int (default ()) name; |
|
193 |
fun default_real name = real (default ()) name; |
|
194 |
fun default_string name = string (default ()) name; |
|
195 |
||
51943 | 196 |
val default_put_bool = change_default put_bool; |
197 |
val default_put_int = change_default put_int; |
|
198 |
val default_put_real = change_default put_real; |
|
199 |
val default_put_string = change_default put_string; |
|
200 |
||
201 |
fun get_default name = |
|
202 |
let val options = default () in get (typ options name) SOME options name end; |
|
203 |
val put_default = change_default update; |
|
204 |
||
48698
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
205 |
fun set_default options = Synchronized.change global_default (K (SOME options)); |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
206 |
fun reset_default () = Synchronized.change global_default (K NONE); |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
207 |
|
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
208 |
fun load_default () = |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
209 |
(case getenv "ISABELLE_PROCESS_OPTIONS" of |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
210 |
"" => () |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
211 |
| name => |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
212 |
let val path = Path.explode name in |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
213 |
(case try File.read path of |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
214 |
SOME s => (set_default (decode (YXML.parse_body s)); ignore (try File.rm path)) |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
215 |
| NONE => ()) |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
216 |
end); |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
217 |
|
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
218 |
val _ = load_default (); |
2585042b1a30
pass Isabelle/Scala system options into ML process of Isar tty or build jobs;
wenzelm
parents:
48456
diff
changeset
|
219 |
|
48456
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
220 |
end; |
d8ff14f44a40
added ML version of stand-alone options, with XML.encode/decode operations (unidirectional from Scala to ML);
wenzelm
parents:
diff
changeset
|
221 |