author | paulson |
Fri, 12 Jan 2001 16:11:55 +0100 | |
changeset 10881 | 03f06372230b |
parent 9857 | 2f994c499bef |
child 11628 | e57a6e51715e |
permissions | -rw-r--r-- |
923 | 1 |
(* Title: HOL/thy_syntax.ML |
2 |
ID: $Id$ |
|
3 |
Author: Markus Wenzel and Lawrence C Paulson and Carsten Clasohm |
|
4 |
||
5 |
Additional theory file sections for HOL. |
|
6 |
*) |
|
7 |
||
3622 | 8 |
local |
923 | 9 |
|
10 |
open ThyParse; |
|
11 |
||
12 |
||
1475 | 13 |
(** typedef **) |
923 | 14 |
|
1475 | 15 |
fun mk_typedef_decl (((((opt_name, vs), t), mx), rhs), wt) = |
923 | 16 |
let |
17 |
val name' = if_none opt_name t; |
|
6642 | 18 |
val name = unenclose name'; |
923 | 19 |
in |
20 |
(cat_lines [name', mk_triple (t, mk_list vs, mx), rhs, wt], |
|
21 |
[name ^ "_def", "Rep_" ^ name, "Rep_" ^ name ^ "_inverse", |
|
22 |
"Abs_" ^ name ^ "_inverse"]) |
|
23 |
end; |
|
24 |
||
1475 | 25 |
val typedef_decl = |
923 | 26 |
optional ("(" $$-- name --$$ ")" >> Some) None -- |
27 |
type_args -- name -- opt_infix --$$ "=" -- string -- opt_witness |
|
1475 | 28 |
>> mk_typedef_decl; |
923 | 29 |
|
30 |
||
31 |
||
3980 | 32 |
(** record **) |
33 |
||
34 |
val record_decl = |
|
4225 | 35 |
(type_args -- name >> (mk_pair o apfst mk_list)) --$$ "=" |
4226 | 36 |
-- optional (typ --$$ "+" >> (parens o cat "Some")) "None" |
4225 | 37 |
-- repeat1 ((name --$$ "::" -- typ) >> mk_pair) |
4001 | 38 |
>> (fn ((x, y), zs) => cat_lines [x, y, mk_big_list zs]); |
3980 | 39 |
|
40 |
||
6477 | 41 |
|
923 | 42 |
(** (co)inductive **) |
43 |
||
8813
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
44 |
(*Skipping initial blanks, find the first identifier*) (* FIXME slightly broken! *) |
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
45 |
fun scan_to_id s = |
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
46 |
s |> Symbol.explode |
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
47 |
|> Scan.error (Scan.finite Symbol.stopper |
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
48 |
(Scan.!! (fn _ => "Expected to find an identifier in " ^ s) |
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
49 |
(Scan.any Symbol.is_blank |-- Syntax.scan_id))) |
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
50 |
|> #1; |
abc0f3722aed
added scan_to_id (used to be in Pure/section_utils.ML);
wenzelm
parents:
8657
diff
changeset
|
51 |
|
5097 | 52 |
fun inductive_decl coind = |
923 | 53 |
let |
6426 | 54 |
val no_atts = map (mk_pair o rpair "[]"); |
923 | 55 |
fun mk_intr_name (s, _) = (*the "op" cancels any infix status*) |
56 |
if Syntax.is_identifier s then "op " ^ s else "_"; |
|
57 |
fun mk_params (((recs, ipairs), monos), con_defs) = |
|
6642 | 58 |
let val big_rec_name = space_implode "_" (map (scan_to_id o unenclose) recs) |
923 | 59 |
and srec_tms = mk_list recs |
6426 | 60 |
and sintrs = mk_big_list (no_atts (map (mk_pair o apfst quote) ipairs)) |
923 | 61 |
in |
5097 | 62 |
";\n\n\ |
63 |
\local\n\ |
|
64 |
\val (thy, {defs, mono, unfold, intrs, elims, mk_cases, induct, ...}) =\n\ |
|
65 |
\ InductivePackage.add_inductive true " ^ |
|
6523 | 66 |
(if coind then "true " else "false ") ^ srec_tms ^ " [] " ^ |
6426 | 67 |
sintrs ^ " " ^ mk_list (no_atts monos) ^ " " ^ mk_list (no_atts con_defs) ^ " thy;\n\ |
5097 | 68 |
\in\n\ |
69 |
\structure " ^ big_rec_name ^ " =\n\ |
|
70 |
\struct\n\ |
|
71 |
\ val defs = defs;\n\ |
|
72 |
\ val mono = mono;\n\ |
|
73 |
\ val unfold = unfold;\n\ |
|
74 |
\ val intrs = intrs;\n\ |
|
75 |
\ val elims = elims;\n\ |
|
76 |
\ val elim = hd elims;\n\ |
|
77 |
\ val " ^ (if coind then "co" else "") ^ "induct = induct;\n\ |
|
78 |
\ val mk_cases = mk_cases;\n\ |
|
79 |
\ val " ^ mk_list (map mk_intr_name ipairs) ^ " = intrs;\n\ |
|
80 |
\end;\n\ |
|
81 |
\val thy = thy;\nend;\n\ |
|
82 |
\val thy = thy\n" |
|
923 | 83 |
end |
84 |
val ipairs = "intrs" $$-- repeat1 (ident -- !! string) |
|
5716
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
85 |
fun optlist s = optional (s $$-- list1 name) [] |
923 | 86 |
in |
5716
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
87 |
repeat1 name -- ipairs -- optlist "monos" -- optlist "con_defs" |
923 | 88 |
>> mk_params |
89 |
end; |
|
90 |
||
91 |
||
6477 | 92 |
|
923 | 93 |
(** datatype **) |
94 |
||
95 |
local |
|
5183 | 96 |
(*** generate string for calling add_datatype ***) |
97 |
(*** and bindig theorems to ML identifiers ***) |
|
923 | 98 |
|
5183 | 99 |
fun mk_bind_thms_string names = |
100 |
(case find_first (not o Syntax.is_identifier) names of |
|
101 |
Some id => (warning (id ^ " is not a valid identifier"); "") |
|
102 |
| None => |
|
103 |
let |
|
104 |
fun mk_dt_struct (s, (id, i)) = |
|
105 |
s ^ "structure " ^ id ^ " =\n\ |
|
106 |
\struct\n\ |
|
107 |
\ val distinct = nth_elem (" ^ i ^ ", distinct);\n\ |
|
108 |
\ val inject = nth_elem (" ^ i ^ ", inject);\n\ |
|
109 |
\ val exhaust = nth_elem (" ^ i ^ ", exhaustion);\n\ |
|
110 |
\ val cases = nth_elem (" ^ i ^ ", case_thms);\n\ |
|
111 |
\ val (split, split_asm) = nth_elem (" ^ i ^ ", split_thms);\n" ^ |
|
112 |
(if length names = 1 then |
|
113 |
" val induct = induction;\n\ |
|
114 |
\ val recs = rec_thms;\n\ |
|
115 |
\ val simps = simps;\n\ |
|
116 |
\ val size = size;\n" |
|
117 |
else "") ^ |
|
118 |
"end;\n"; |
|
119 |
||
120 |
val structs = foldl mk_dt_struct |
|
121 |
("", (names ~~ (map string_of_int (0 upto length names - 1)))); |
|
122 |
||
123 |
in |
|
124 |
(if length names > 1 then |
|
125 |
"structure " ^ (space_implode "_" names) ^ " =\n\ |
|
126 |
\struct\n\ |
|
5658 | 127 |
\ val induct = induction;\n\ |
128 |
\ val recs = rec_thms;\n\ |
|
129 |
\ val simps = simps;\n\ |
|
130 |
\ val size = size;\nend;\n" |
|
131 |
else "") ^ structs |
|
5183 | 132 |
end); |
133 |
||
134 |
fun mk_dt_string dts = |
|
923 | 135 |
let |
5183 | 136 |
val names = map (fn ((((alt_name, _), name), _), _) => |
6642 | 137 |
unenclose (if_none alt_name name)) dts; |
5183 | 138 |
|
139 |
val add_datatype_args = brackets (commas (map quote names)) ^ " " ^ |
|
140 |
brackets (commas (map (fn ((((_, vs), name), mx), cs) => |
|
141 |
parens (brackets (commas vs) ^ ", " ^ name ^ ", " ^ mx ^ ", " ^ |
|
142 |
brackets (commas cs))) dts)); |
|
143 |
||
923 | 144 |
in |
5183 | 145 |
";\nlocal\n\ |
146 |
\val (thy, {distinct, inject, exhaustion, rec_thms,\n\ |
|
147 |
\ case_thms, split_thms, induction, size, simps}) =\n\ |
|
6103 | 148 |
\ DatatypePackage.add_datatype false " ^ add_datatype_args ^ " thy;\n\ |
5183 | 149 |
\in\n" ^ mk_bind_thms_string names ^ |
150 |
"val thy = thy;\nend;\nval thy = thy\n" |
|
923 | 151 |
end; |
152 |
||
6381 | 153 |
fun mk_thmss namess = mk_list (map (mk_list o map (mk_pair o rpair "[]")) namess); |
154 |
fun mk_thm name = mk_pair (name, "[]"); |
|
6103 | 155 |
|
5183 | 156 |
fun mk_rep_dt_string (((names, distinct), inject), induct) = |
157 |
";\nlocal\n\ |
|
158 |
\val (thy, {distinct, inject, exhaustion, rec_thms,\n\ |
|
159 |
\ case_thms, split_thms, induction, size, simps}) =\n\ |
|
6103 | 160 |
\ DatatypePackage.rep_datatype " ^ |
5183 | 161 |
(case names of |
6103 | 162 |
Some names => "(Some [" ^ commas_quote names ^ "])\n " ^ |
163 |
mk_thmss distinct ^ "\n " ^ mk_thmss inject ^ |
|
164 |
"\n " ^ mk_thm induct ^ " thy;\nin\n" ^ mk_bind_thms_string names |
|
165 |
| None => "None\n " ^ mk_thmss distinct ^ "\n " ^ mk_thmss inject ^ |
|
166 |
"\n " ^ mk_thm induct ^ " thy;\nin\n") ^ |
|
5183 | 167 |
"val thy = thy;\nend;\nval thy = thy\n"; |
923 | 168 |
|
5183 | 169 |
(*** parsers ***) |
4106 | 170 |
|
6642 | 171 |
val simple_typ = ident || (type_var >> unenclose); |
1316 | 172 |
|
1251
81fc4d8e3eda
added nested types on right hand side of datatype definitions
clasohm
parents:
977
diff
changeset
|
173 |
fun complex_typ toks = |
1316 | 174 |
let val typ = simple_typ || "(" $$-- complex_typ --$$ ")"; |
175 |
val typ2 = complex_typ || "(" $$-- complex_typ --$$ ")"; |
|
176 |
in |
|
5183 | 177 |
(typ ^^ (repeat ident >> (cat "" o space_implode " ")) || |
178 |
"(" $$-- !! (list1 typ2 >> (parens o commas)) --$$ ")" ^^ !! |
|
179 |
(repeat1 ident >> (cat "" o space_implode " "))) toks |
|
1316 | 180 |
end; |
181 |
||
6642 | 182 |
val opt_typs = repeat ((string >> unenclose) || |
5183 | 183 |
simple_typ || ("(" $$-- complex_typ --$$ ")")); |
184 |
val constructor = name -- opt_typs -- opt_mixfix >> (fn ((n, Ts), mx) => |
|
5716
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
185 |
parens (n ^ ", " ^ brackets (commas_quote Ts) ^ ", " ^ mx)); |
5183 | 186 |
val opt_name = optional ("(" $$-- name --$$ ")" >> Some) None |
187 |
||
5716
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
188 |
fun optlistlist s = optional (s $$-- enum "and" (list name)) [[]] |
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
189 |
|
923 | 190 |
in |
191 |
val datatype_decl = |
|
5183 | 192 |
enum1 "and" (opt_name -- type_args -- name -- opt_infix --$$ "=" -- |
193 |
enum1 "|" constructor) >> mk_dt_string; |
|
194 |
val rep_datatype_decl = |
|
6642 | 195 |
((optional ((repeat1 (name >> unenclose)) >> Some) None) -- |
5716
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
196 |
optlistlist "distinct" -- optlistlist "inject" -- |
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
197 |
("induct" $$-- name)) >> mk_rep_dt_string; |
923 | 198 |
end; |
199 |
||
200 |
||
6477 | 201 |
|
923 | 202 |
(** primrec **) |
203 |
||
8657 | 204 |
fun mk_patterns eqns = mk_list (map (fn (s, _) => if s = "" then "_" else s) eqns); |
205 |
fun mk_eqns eqns = mk_list (map (fn (x, y) => mk_pair (mk_pair (quote x, y), "[]")) eqns); |
|
206 |
||
5716
a3d2a6b6693e
Changed syntax of rep_datatype and inductive: Theorems
berghofe
parents:
5658
diff
changeset
|
207 |
fun mk_primrec_decl (alt_name, eqns) = |
8657 | 208 |
";\nval (thy, " ^ mk_patterns eqns ^ ") = PrimrecPackage.add_primrec " ^ alt_name ^ " " |
209 |
^ mk_eqns eqns ^ " " ^ " thy;\n\ |
|
210 |
\val thy = thy\n" |
|
923 | 211 |
|
5183 | 212 |
(* either names and axioms or just axioms *) |
8657 | 213 |
val primrec_decl = optional ("(" $$-- name --$$ ")") "\"\"" -- |
214 |
repeat1 (ident -- string || (string >> pair "")) >> mk_primrec_decl; |
|
923 | 215 |
|
2922 | 216 |
|
6497 | 217 |
(*** recdef: interface to Slind's TFL ***) |
2922 | 218 |
|
6497 | 219 |
(** TFL with explicit WF relations **) |
2922 | 220 |
|
3194 | 221 |
(*fname: name of function being defined; rel: well-founded relation*) |
8657 | 222 |
fun mk_recdef_decl ((((fname, rel), congs), ss), eqns) = |
9857 | 223 |
let val fid = unenclose fname in |
6477 | 224 |
";\n\n\ |
225 |
\local\n\ |
|
8623
5668aaf41c36
mod in recdef allows to access the correct simpset via simpset().
nipkow
parents:
6642
diff
changeset
|
226 |
\fun simpset() = Simplifier.simpset_of thy;\n\ |
9857 | 227 |
\val (thy, result) = thy |> RecdefPackage.add_recdef_old " ^ quote fid ^ " " ^ rel ^ "\n" ^ |
228 |
mk_eqns eqns ^ "\n(" ^ ss ^ ",\n " ^ mk_list congs ^ ");\n\ |
|
6477 | 229 |
\in\n\ |
230 |
\structure " ^ fid ^ " =\n\ |
|
231 |
\struct\n\ |
|
8657 | 232 |
\ val {simps, rules = " ^ mk_patterns eqns ^ ", induct, tcs} = result;\n\ |
6477 | 233 |
\end;\n\ |
234 |
\val thy = thy;\n\ |
|
235 |
\end;\n\ |
|
236 |
\val thy = thy\n" |
|
2922 | 237 |
end; |
238 |
||
6555 | 239 |
val recdef_decl = |
8657 | 240 |
name -- string -- |
9857 | 241 |
optional ("congs" $$-- list1 ident) [] -- |
6642 | 242 |
optional ("simpset" $$-- string >> unenclose) "simpset()" -- |
8657 | 243 |
repeat1 (ident -- string || (string >> pair "")) >> mk_recdef_decl; |
6555 | 244 |
|
6497 | 245 |
|
246 |
(** TFL with no WF relation supplied **) |
|
247 |
||
248 |
(*fname: name of function being defined; rel: well-founded relation*) |
|
6555 | 249 |
fun mk_defer_recdef_decl ((fname, congs), axms) = |
250 |
let |
|
6642 | 251 |
val fid = unenclose fname; |
6555 | 252 |
val congs_text = mk_list (map (fn c => mk_pair (c, "[]")) congs); |
253 |
val axms_text = mk_big_list axms; |
|
6497 | 254 |
in |
6555 | 255 |
";\n\n\ |
256 |
\local\n\ |
|
257 |
\val (thy, result) = thy |> RecdefPackage.defer_recdef " ^ quote fid ^ "\n" ^ |
|
258 |
axms_text ^ "\n" ^ congs_text ^ ";\n\ |
|
259 |
\in\n\ |
|
260 |
\structure " ^ fid ^ " =\n\ |
|
261 |
\struct\n\ |
|
262 |
\ val {induct_rules} = result;\n\ |
|
263 |
\end;\n\ |
|
264 |
\val thy = thy;\n\ |
|
265 |
\end;\n\ |
|
266 |
\val thy = thy\n" |
|
6497 | 267 |
end; |
268 |
||
6555 | 269 |
val defer_recdef_decl = |
270 |
(name -- |
|
271 |
optional ("congs" $$-- list1 name) [] -- |
|
272 |
repeat1 string >> mk_defer_recdef_decl); |
|
2922 | 273 |
|
274 |
||
275 |
||
3622 | 276 |
(** augment thy syntax **) |
923 | 277 |
|
3622 | 278 |
in |
923 | 279 |
|
3622 | 280 |
val _ = ThySyn.add_syntax |
5183 | 281 |
["intrs", "monos", "con_defs", "congs", "simpset", "|", |
6555 | 282 |
"and", "distinct", "inject", "induct"] |
4873 | 283 |
[axm_section "typedef" "|> TypedefPackage.add_typedef" typedef_decl, |
6523 | 284 |
section "record" "|> (#1 oooo RecordPackage.add_record)" record_decl, |
6497 | 285 |
section "inductive" "" (inductive_decl false), |
286 |
section "coinductive" "" (inductive_decl true), |
|
287 |
section "datatype" "" datatype_decl, |
|
5183 | 288 |
section "rep_datatype" "" rep_datatype_decl, |
6497 | 289 |
section "primrec" "" primrec_decl, |
6555 | 290 |
section "recdef" "" recdef_decl, |
291 |
section "defer_recdef" "" defer_recdef_decl]; |
|
923 | 292 |
|
293 |
end; |