-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathAutoLevity.thy
339 lines (242 loc) · 10.5 KB
/
AutoLevity.thy
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
(*
* Copyright 2014, NICTA
*
* This software may be distributed and modified according to the terms of
* the BSD 2-Clause license. Note that NO WARRANTY is provided.
* See "LICENSE_BSD2.txt" for details.
*
* @TAG(NICTA_BSD)
*)
theory AutoLevity (* FIXME: broken *)
imports "../tools/proofcount/ProofGraph"
begin
ML {* val infoflow_file = "~~/../lib/proof_counting/Noninterference_Refinement_graphs.xml" *}
ML {* val (full_spec,proof_spec_raw,thy_deps) = Proof_Graph.read_graph_spec_from infoflow_file *}
ML {* val proof_spec = Proof_Graph.merge_multi_thms proof_spec_raw *}
(* Extracts "general" lemmas from proof graph and produces lemma buckets from them *)
(* Does theory and lemma dependency management. Might be better to produce a report and
have a bash/perl script consume this, rather than doing the file manipulation in ML *)
(* Cleanup: Not really using name spaces properly *)
(*TODO: Can possibly extract lemmas from the middle of locales *)
ML {*
fun range (beg,fin) l = List.take (List.drop (l,beg),fin - beg)
fun map_range f (beg,fin) l =
let
fun do_clear (h :: l') i = if beg <= i andalso i <= fin then (f h) :: (do_clear l' (i + 1)) else h :: (do_clear l' (i + 1))
| do_clear [] i = []
in
do_clear l 0 end
fun add_thy thy_deps thy thys' =
let
fun reachable i j =
let
val deps = Option.map snd (Symtab.lookup thy_deps i)
in
case deps of SOME deps => member (op =) deps j | NONE => false end
val thys = distinct (op =) thys'
in
if exists (fn ex_thy => reachable ex_thy thy) thys then thys else
thys
|> filter_out (fn ex_thy => reachable thy ex_thy)
|> cons thy
end
fun thy_list thy_deps thys = fold (add_thy thy_deps) thys []
|> sort_wrt I
fun add_thy_consts spec_graph (e : Proof_Graph.proof_entry) =
let
val thys' = map (Int_Graph.get_node spec_graph #> #theory) (#contains e)
in
(union (op =) thys') end
fun sort_lemmas (proof_graph,spec_graph : Spec_Graph.graph_entry Int_Graph.T,thy_deps) spec_theories facts =
let
fun do_sort (nm,e) (thy_names,thy_imports) =
let
val thy_list = thy_list thy_deps (add_thy_consts spec_graph e [])
val i = (space_implode "__" thy_list) ^ "_Bucket"
in
(Symtab.update_new (nm,i) thy_names,
Symtab.update (i,thy_list) thy_imports) end
val (thy_names,thy_imports) = fold do_sort facts (Symtab.empty,Symtab.empty)
fun extra_deps (nm,e) (thm_deps,thytab) =
let
val thy_name = Symtab.lookup thy_names nm |> the
val all_succs = String_Graph.all_succs proof_graph [nm]
fun get_dep nm' b =
let
val thyn = Long_Name.explode nm' |> hd
fun warn _ = if not (Symtab.defined thy_names nm') then (warning (nm ^ " depends on " ^ nm' ^ " which is not marked for levitation and is outside theory import chain. NOT levitating.");true) else b
in if member (op =) spec_theories thyn then (NONE,warn ()) else (SOME thyn,b) end
val (deps,rem) = fold_map get_dep all_succs false
|>> map_filter I
|>> append (map_filter (Symtab.lookup thy_names) all_succs)
|>> remove (op =) thy_name
val imm_succs = String_Graph.immediate_succs proof_graph nm
in
(if rem then (thm_deps,thytab) else
(Symtab.update (nm,(e,imm_succs)) thm_deps,fold (fn dep => Symtab.insert_list (op =) (thy_name,dep)) deps thytab)) end
fun rem_cycles (thy_names,thy_imports) =
let
val thy_graph = thy_imports
|> Symtab.dest
|> map (fn (id,e) => ((id,()),filter (Symtab.defined thy_imports) e))
|> String_Graph.make
fun merge_deps (n :: l) (thy_names,thy_graph) =
let
val thy_graph' =
Proof_Graph.restrict_subgraph (fn (i,_) => not (member (op =) l i)) thy_graph
val thy_names' = Symtab.map (fn _ => fn thyn => if member (op =) l thyn then n else thyn) thy_names
in
(thy_names',thy_graph') end;
fun cycles_of i =
fold (fn j => append (String_Graph.irreducible_paths thy_graph (j,i))) (String_Graph.immediate_succs thy_graph i) []
val cycles = map (flat o cycles_of) (String_Graph.keys thy_graph)
|> filter_out null
|> map (sort_wrt I #> distinct (op =) #> rev)
|> distinct (op =)
val thy_graph = thy_imports
|> Symtab.dest
|> map (fn (id,e) => ((id,()),e))
|> append (map (fn (id,(_,e)) => ((id,()),e)) (Symtab.dest thy_deps))
|> String_Graph.make
val (thy_names',thy_graph') = fold merge_deps cycles (thy_names,thy_graph)
val thy_imports' = map (fn i => (i,String_Graph.all_succs thy_graph' [i] |> remove (op =) i)) (String_Graph.keys thy_graph')
|> filter (fn (i,_) => Symtab.defined thy_imports i)
|> Symtab.make
in
(thy_names',thy_imports') end
val (thm_deps,thy_imports) = fold extra_deps facts (Symtab.empty,thy_imports)
val (thy_names',thy_imports') = rem_cycles (thy_names,thy_imports)
||> Symtab.map (K (thy_list thy_deps))
val order = String_Graph.topological_order proof_graph
|> (fn l => fold_map (fn nm => fn i => ((nm,i),i+1)) l 0)
|> fst
|> Symtab.make
fun do_lookup e tab nm = (Symtab.lookup tab nm |> the) handle Option => error ("No entry " ^ nm ^ " in symtab:\n" ^ e)
val compare_facts = make_ord (fn ((nm,_),(nm',_)) => (do_lookup "order" order nm) < (do_lookup "order" order nm'))
val thys = Symtab.fold (fn (nm,e) => Symtab.insert_list (K false) (do_lookup "thy_names" thy_names' nm,(nm,e))) thm_deps Symtab.empty
|> Symtab.map (K (sort compare_facts))
|> Symtab.map (fn i => fn e => (do_lookup "thy_imports" thy_imports' i,e))
|> Symtab.dest
val base_thy =
let
val all_imports = thy_imports'
|> Symtab.dest
|> map snd
|> flat
|> distinct (op =)
|> filter_out (member (op =) (Symtab.keys thy_imports'))
|> thy_list thy_deps
|> map (fn n => "\"" ^ (Symtab.lookup thy_deps n |> the |> fst) ^ "/" ^ n ^ "\"")
|> distinct (op =)
in
("AutoLevity_Base",(all_imports,[])) end
val toplevel_thy = ("AutoLevity_Top",(thy_list thy_deps (Symtab.keys thy_imports'),[]))
in
(toplevel_thy :: base_thy :: thys) end
(*TODO: Have a more sensible way of generating this list*)
val declare_attribs = ["simp","wp","intro","intro!","elim","elim!"]
fun get_attribs str =
let
val toks = (Outer_Syntax.scan Position.none str
|> filter (Token.is_proper)
) @ ([Token.eof])
val ((nm,attribs),_) =
(Parse.command |-- Parse_Spec.opt_thm_name ":") toks handle _ => error ("Failed to parse lemma attributes" ^ str)
in
map (fst o fst o Args.dest_src) attribs
|> filter (member (op =) declare_attribs) end
fun parse_attribs sorted =
let
fun do_parse (id,(e : Proof_Graph.proof_entry,thm_deps)) =
let
val thy_file = File.read_lines (Path.explode (#file e))
val (begin,fin) = (#lines e)
val text = range (begin -1,fin) thy_file
|> space_implode "\n"
val new_attribs = get_attribs text
in
(id,(e,thm_deps,new_attribs)) end
in
map (fn (file,(deps,lemmas)) => (file,(deps,map do_parse lemmas))) sorted end
fun get_raw_proofs debug (id : string,(e : Proof_Graph.proof_entry,thm_deps,attribs)) =
let
val thy_file = File.read_lines (Path.explode (#file e))
val (begin,fin) = (#lines e)
val text = range (begin -1,fin) thy_file
|> space_implode "\n"
val new_attribs = attribs
|> map (fn s => s ^ " del")
|> space_implode ","
val text' = text ^ (if new_attribs = "" then "" else ("\ndeclare " ^ (#name e) ^ "[" ^ new_attribs ^ "]"))
^ (if debug then "\n(* THM_DEPS: " ^ space_implode " " thm_deps ^ "*)\n" else "")
(* Make sure we don't have any ride-alongs *)
val _ = if (String.isSubstring "method_setup" text) then warning ("Found method setup" ^ (@{make_string} (id,e)) ^ "\n" ^ text) else ()
in
cons (#file e,text') end
fun add_buffer (f,text) (cur_f,buf) =
let
val buf' = buf
|> Buffer.add (if f = cur_f then "" else "\n(* " ^ f ^ " *)\n\n")
|> Buffer.add ("\n" ^ text ^ "\n")
in
(f,buf') end
fun get_thy debug (nm,(imports,lemmas)) =
let
val lemma_text = fold (get_raw_proofs debug) lemmas []
val buf = Buffer.empty
|> Buffer.add ("theory " ^ nm ^ "\nimports ")
|> Buffer.add (space_implode " " imports)
|> Buffer.add ("\nbegin\n")
|> (fn b => fold add_buffer lemma_text ("",b))
|> snd
|> Buffer.add ("\nend")
in
(nm,buf) end
fun write_thys debug sorted base_path =
let
val bufs = map (get_thy debug) sorted
in
(map (fn (nm,buf) => (File.write_buffer (Path.append base_path (Path.explode (nm ^ ".thy"))) buf)) bufs;()) end
fun remove_lemma thy_nm (nm,(e : Proof_Graph.proof_entry,thm_deps,attribs)) (new_deps,cache) =
let
val (beg',fin) = #lines e
val beg = beg' -1
val thy_file = case Symtab.lookup cache (#file e) of SOME s => s | NONE => File.read_lines (Path.explode (#file e)) |> map SOME
val header = "(* " ^ #name e ^ " moved to " ^ thy_nm ^ " *)"
val new_attribs = attribs
|> space_implode ","
val declare = if new_attribs = "" then "" else "\ndeclare " ^ thy_nm ^ "." ^ (#name e) ^ "[" ^ new_attribs ^ "]"
val thy_file' = thy_file
|> map_range (K NONE) (beg,fin-1)
|> map_range (K (SOME (header ^ declare) )) (beg,beg)
in
(Symtab.insert_list (op =) (#file e,thy_nm) new_deps,Symtab.update (#file e,thy_file') cache) end
(*TODO: Auto-generate thy dependencies or assume everyone will import us?*)
fun remove_thy (thy_nm,(deps,lemmas)) cache = fold (remove_lemma thy_nm) lemmas cache
fun clear_thys thy_deps sorted =
let
val (new_deps,cache) = fold remove_thy sorted (Symtab.empty,Symtab.empty)
fun thy_to_file thy_nm = (Symtab.lookup thy_deps thy_nm |> the |> fst) ^ "/" ^ thy_nm ^ ".thy"
val file_deps = map (fn (thy,(f,deps)) => (thy_to_file thy,map thy_to_file deps)) (Symtab.dest thy_deps)
|> Symtab.make
fun is_already_imported file_nm dep =
let
val deps = Symtab.lookup file_deps file_nm |> the
|> remove (op =) file_nm
in
exists (fn d => case (Symtab.lookup new_deps d) of SOME deps' => member (op =) deps' dep | NONE => false) deps end
fun add_deps file_nm file =
let
val deps = Symtab.lookup new_deps file_nm |> the
|> filter_out (is_already_imported file_nm)
val index_line = find_index (fn (SOME s) => String.isPrefix "imports" s | NONE => false) file
val new_dep_line = if null deps then "" else " " ^ (space_implode " " deps)
val file' =
map_range (fn SOME s => SOME (s ^ new_dep_line) | NONE => error ("No import line in file " ^ file_nm)) (index_line,index_line) file
in
file' end
val cache' = Symtab.map add_deps cache
in
(Symtab.map (fn file => fn body => File.write_list (Path.explode file) (map (fn SOME s => s ^ "\n" | NONE => "") body)) cache';()) end;
*}
end