forked from cubicle-model-checker/cubicle
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpre.ml
341 lines (301 loc) · 10.2 KB
/
pre.ml
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
338
339
340
341
(**************************************************************************)
(* *)
(* Cubicle *)
(* *)
(* Copyright (C) 2011-2014 *)
(* *)
(* Sylvain Conchon and Alain Mebsout *)
(* Universite Paris-Sud 11 *)
(* *)
(* *)
(* This file is distributed under the terms of the Apache Software *)
(* License version 2.0 *)
(* *)
(**************************************************************************)
open Options
open Format
open Ast
open Util
open Types
type reset_memo = unit -> unit
module H = Hstring
module Debug = struct
let fixpoint =
if not debug then fun _ -> () else
fun ls ->
eprintf "\nAfter simplification, subsumption and fixpoint check : @.";
match ls with
| [] -> eprintf "No new branches@."
| _ ->
List.iter (eprintf "@.New branch : %a@." Node.print) ls
let unsafe =
if not debug then fun _ -> () else
fun s -> eprintf " %a@." Node.print s
let invariant =
fun s -> eprintf "Invariant ?@. %a@." Cube.print s
let pre =
if not debug then fun _ _ -> () else
fun tr p ->
eprintf "\nResult of the pre for transition %s (%a):@.%a@."
(H.view tr.tr_name)
Variable.print_vars tr.tr_args
SAtom.print p
let pre_cubes =
if not debug then fun _ _ -> () else
fun p args ->
eprintf "Cubes (%a) :%a@." Variable.print_vars args SAtom.print p
end
(***********************************************************************)
(* Pre-image of an atom w.r.t a transition, simply represented here by *)
(* a function tau *)
(***********************************************************************)
let rec pre_atom tau a =
match a with
| Atom.True | Atom.False -> a
| Atom.Comp (x, op, y) -> tau x op y
| Atom.Ite (sa, a1, a2) ->
let pre_sa =
SAtom.fold (fun a -> SAtom.add (pre_atom tau a)) sa SAtom.empty
in
let pre_a1 = pre_atom tau a1 in
let pre_a2 = pre_atom tau a2 in
Atom.Ite(pre_sa, pre_a1, pre_a2)
(****************************************)
(* Convert a transition into a function *)
(****************************************)
type assign = Single of term | Branch of swts
let fresh_nondet =
let cpt = ref 0 in
fun (args, ret) ->
incr cpt;
let s = H.make ("!"^(string_of_int !cpt)) in
Smt.Symbol.declare s args ret;
s
let rec find_update a li = function
| [] -> raise Not_found
| { up_loc = loc; up_arr = a'; up_arg = lj; up_swts = ls} :: _ when a=a' ->
let ls =
List.map
(fun (ci, ti) ->
let sigma = List.combine lj li in
SAtom.subst sigma ci,
Term.subst sigma ti) ls in
Branch ls
| _ :: l -> find_update a li l
exception Remove_lit_var of Hstring.t
let rec find_assign memo tr = function
| Elem (x, sx) ->
let gu =
if H.list_mem x tr.tr_nondets then
(*raise (Remove_lit_var x)*)
try List.assoc (Hstring.view x) !memo
with Not_found ->
let nv =
UTerm (Elem (fresh_nondet (Smt.Symbol.type_of x), sx)) in
memo := (Hstring.view x,nv) :: !memo;
nv
else
try H.list_assoc x tr.tr_assigns
with Not_found -> UTerm (Elem (x, sx))
in
begin
match gu with
| UTerm t -> Single t
| UCase swts -> Branch swts
end
| Const i as a -> Single a
| Arith (x, cs1) ->
begin
let t = find_assign memo tr x in
match t with
| Single (Const cs2) ->
let c =
Const (add_constants cs1 cs2)
in
Single c
| Single (Arith (y, cs2)) ->
Single (Arith (y, add_constants cs1 cs2))
| Single y -> Single (Arith (y, cs1))
| Branch up_swts ->
Branch (List.map (fun (sa, y) -> (sa, (Arith (y, cs1))))
up_swts)
end
| Access (a, li) ->
let nli = li in
(* List.map (fun i -> *)
(* if H.list_mem i tr.tr_nondets then *)
(* (assert false; *)
(* fresh_nondet (Smt.Symbol.type_of i)) *)
(* else i *)
(* try (match H.list_assoc i tr.tr_assigns with *)
(* | Elem (ni, _) -> ni *)
(* | Const _ | Arith _ | Access _ -> assert false) *)
(* with Not_found -> i *)
(* ) li in *)
try find_update a nli tr.tr_upds
with Not_found -> Single (Access (a, nli))
(* let na = *)
(* try (match H.list_assoc a tr.tr_assigns with *)
(* | Elem (na, _) -> na *)
(* | Const _ | Arith _ | Access _ -> assert false) *)
(* with Not_found -> a *)
(* in *)
(* Single (Access (na, nli)) *)
let make_tau tr =
let memo = ref [] in
(fun x op y ->
try
match find_assign memo tr x, find_assign memo tr y with
| Single tx, Single ty -> Atom.Comp (tx, op, ty)
| Single tx, Branch ls ->
List.fold_right
(fun (ci, ti) f -> Atom.Ite(ci, Atom.Comp(tx, op, ti), f))
ls Atom.True
| Branch ls, Single tx ->
List.fold_right
(fun (ci, ti) f -> Atom.Ite(ci, Atom.Comp(ti, op, tx), f))
ls Atom.True
| Branch ls1, Branch ls2 ->
List.fold_right
(fun (ci, ti) f ->
List.fold_right
(fun (cj, tj) f ->
Atom.Ite(SAtom.union ci cj, Atom.Comp(ti, op, tj), f)) ls2 f)
ls1 Atom.True
with Remove_lit_var _ -> Atom.True),
(fun () -> memo := [])
(**********************)
(* Postponed Formulas *)
(**********************)
let postpone args p np =
let sa1 = SAtom.filter (Atom.has_vars args) p in
let sa2 = SAtom.filter (Atom.has_vars args) np in
SAtom.equal sa2 sa1
let uguard sigma args tr_args = function
| [] -> [SAtom.empty]
| [j, dnf] ->
let uargs = List.filter (fun a -> not (H.list_mem a tr_args)) args in
List.fold_left
(fun lureq z ->
let m = List.map (SAtom.subst ((j, z)::sigma)) dnf in
List.fold_left
(fun acc sa ->
(List.map (fun zy-> SAtom.union zy sa) m) @ acc ) [] lureq
)
[SAtom.empty]
uargs
| _ -> assert false
let add_list n l =
if List.exists (fun n' -> Node.subset n' n) l then l
else
let l =
if true || delete then
List.filter (fun n' -> not (Node.subset n n')) l
else l
in
n :: l
let add_list a l = a :: l
let add_array_to_list n l =
if List.exists (fun n' -> ArrayAtom.subset n' n) l then l
else
let l =
if true || delete then
List.filter (fun n' -> not (ArrayAtom.subset n n')) l
else l
in
n :: l
let make_cubes (ls, post) rargs s tr cnp =
let { cube = { Cube.vars = uargs; litterals = p}; tag = nb } = s in
let nb_uargs = List.length uargs in
let args = cnp.Cube.vars in
let cube acc sigma =
let tr_args = List.map (Variable.subst sigma) tr.tr_args in
let lnp = Cube.elim_ite_simplify (Cube.subst sigma cnp) in
(* cubes are in normal form *)
List.fold_left
(fun (ls, post) cnp ->
let np, nargs = cnp.Cube.litterals, cnp.Cube.vars in
let lureq = uguard sigma nargs tr_args tr.tr_ureq in
List.fold_left
(fun (ls, post) ureq ->
try
let ureq = Cube.simplify_atoms_base np ureq in
let np = SAtom.union ureq np in
if debug && verbose > 0 then Debug.pre_cubes np nargs;
if Cube.inconsistent_set np then
begin
if debug && verbose > 0 then eprintf "(inconsistent)@.";
(ls, post)
end
else
let new_cube = Cube.create nargs np in
let new_s = Node.create ~from:(Some (tr, tr_args, s)) new_cube in
match post_strategy with
| 0 -> add_list new_s ls, post
| 1 ->
if List.length nargs > nb_uargs then
ls, add_list new_s post
else add_list new_s ls, post
| 2 ->
if not (SAtom.is_empty ureq) || postpone args p np
then ls, add_list new_s post
else add_list new_s ls, post
| _ -> assert false
with Exit -> ls, post
) (ls, post) lureq ) acc lnp
in
if List.length tr.tr_args > List.length rargs then
begin
if !size_proc = 0 then assert false;
(ls, post)
end
else
(* let d_old = Variable.all_permutations tr.tr_args rargs in *)
(* TODO: Benchmark this *)
let d = Variable.permutations_missing tr.tr_args args in
(* assert (List.length d_old >= List.length d); *)
List.fold_left cube (ls, post) d
(* The following version computes the pre-image once for each transition and
instantiates afterwards.
FIXME: This is supposed to be faster but this implementation is not.
(check order, ...)
*)
let make_cubes_new (ls, post) rargs s tr cnp =
failwith "To implement"
(*****************************************************)
(* Pre-image of an unsafe formula w.r.t a transition *)
(*****************************************************)
let pre { tr_info = tri; tr_tau = tau; tr_reset = reset } unsafe =
(* let tau = tr.tr_tau in *)
let pre_unsafe =
SAtom.union tri.tr_reqs
(SAtom.fold (fun a -> SAtom.add (pre_atom tau a)) unsafe SAtom.empty)
in
let pre_u = Cube.create_normal pre_unsafe in
if debug && verbose > 0 then Debug.pre tri pre_unsafe;
reset();
let args = pre_u.Cube.vars in
if tri.tr_args = [] then tri, pre_u, args
else
let nargs = Variable.append_extra_procs args tri.tr_args in
if !size_proc <> 0 && List.length nargs > !size_proc then
tri, pre_u, args
else tri, pre_u, nargs
(*********************************************************************)
(* Pre-image of a system s : computing the cubes gives a list of new *)
(* systems *)
(*********************************************************************)
let pre_image trs s =
TimePre.start ();
Debug.unsafe s;
let u = Node.litterals s in
let ls, post =
List.fold_left
(fun acc tr ->
let trinfo, pre_u, info_args = pre tr u in
make_cubes acc info_args s trinfo pre_u)
([], [])
trs
in
TimePre.pause ();
List.rev ls, List.rev post