Skip to content

Commit a333489

Browse files
committed
Use C-type (not index terms) for step in qpred
The previous commit introduce an error for a using array_shift on a void pointer, but the location information was not useful. This was because the array_shift was being deconstructed into a base pointer, and a sizeof term, and the latter term, by virtue of being generated by CN developers than the user, had an 'other' location. When this term was later checked in wellTyped, it would error. Feeling uncomfortable about lying about the origin and made-up nature of the sizeof term, I instead did a long-overdue refactor of QPredicate.t to have its "step" field just be a C-type instead of an arbitrary index term. I believe enough checks and errors about this have been in place about this for long enough such that this is non-breaking change (and I also ran the tests).
1 parent a71492e commit a333489

21 files changed

+60
-123
lines changed

Makefile

+1-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ cn-coq:
5353
$(Q)dune build -p cn-coq
5454

5555
.PHONY: cn-coq-install
56-
cn-coq-install: install_lib cn-coq
56+
cn-coq-install: cn-coq
5757
@echo "[DUNE] install cn-coq"
5858
$(Q)dune install cn-coq
5959

coq/Cn/Request.v

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ Module QPredicate.
2929
pointer : IndexTerms.t;
3030
q : Symbol.t * BaseTypes.t;
3131
q_loc : Locations.t;
32-
step : IndexTerms.t;
32+
step : SCtypes.ctype;
3333
permission : IndexTerms.t;
3434
iargs : list IndexTerms.t
3535
}.

lib/bugExplanation/replicators.ml

+1-10
Original file line numberDiff line numberDiff line change
@@ -479,16 +479,7 @@ let compile_req
479479
let map_sym = Sym.fresh () in
480480
let b_val, s_val, e_val =
481481
aux
482-
(P
483-
{ name;
484-
pointer =
485-
IT.arrayShift_
486-
~base:pointer
487-
~index:(IT.mul_ (q_it, step) (IT.get_loc step))
488-
Sctypes.char_ct
489-
loc;
490-
iargs
491-
})
482+
(P { name; pointer = IT.arrayShift_ ~base:pointer ~index:q_it step loc; iargs })
492483
in
493484
let s2 =
494485
A.

lib/bugExplanation/shapeAnalyzers.ml

+1-10
Original file line numberDiff line numberDiff line change
@@ -198,16 +198,7 @@ let compile_req
198198
let map_sym = Sym.fresh () in
199199
let b_val, s_val, e_val =
200200
aux
201-
(P
202-
{ name;
203-
pointer =
204-
IT.arrayShift_
205-
~base:pointer
206-
~index:(IT.mul_ (q_it, step) (IT.get_loc step))
207-
Sctypes.char_ct
208-
loc;
209-
iargs
210-
})
201+
(P { name; pointer = IT.arrayShift_ ~base:pointer ~index:q_it step loc; iargs })
211202
in
212203
let s2 =
213204
A.

lib/check.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1348,7 +1348,7 @@ let bytes_qpred sym size pointer init : Req.QPredicate.t =
13481348
let bt' = WellTyped.default_quantifier_bt in
13491349
{ q = (sym, bt');
13501350
q_loc = here;
1351-
step = IT.num_lit_ Z.one bt' here;
1351+
step = Sctypes.uchar_ct;
13521352
permission = IT.(lt_ (sym_ (sym, bt', here), size) here);
13531353
name = Owned (Sctypes.uchar_ct, init);
13541354
pointer;

lib/cn_internal_to_ail.ml

+16-10
Original file line numberDiff line numberDiff line change
@@ -2700,7 +2700,13 @@ let cn_to_ail_resource_internal
27002700
Sym.fresh_pretty (Option.get (get_conversion_to_fn_str BT.Integer))
27012701
in *)
27022702
let b2, s2, _e2 = cn_to_ail_expr_internal dts globals q.permission PassBack in
2703-
let b3, s3, _e3 = cn_to_ail_expr_internal dts globals q.step PassBack in
2703+
let b3, s3, _e3 =
2704+
cn_to_ail_expr_internal
2705+
dts
2706+
globals
2707+
(IT.sizeOf_ q.step Cerb_location.unknown)
2708+
PassBack
2709+
in
27042710
(* let conversion_fcall = A.(AilEcall (mk_expr (AilEident convert_to_cn_integer_sym), [e_start])) in *)
27052711
let start_binding = create_binding i_sym cn_integer_ptr_ctype in
27062712
let start_assign = A.(AilSdeclaration [ (i_sym, Some e_start) ]) in
@@ -2713,11 +2719,8 @@ let cn_to_ail_resource_internal
27132719
let return_ctype, _return_bt = calculate_return_type q.name in
27142720
(* Translation of q.pointer *)
27152721
let i_it = IT.IT (IT.(Sym i_sym), i_bt, Cerb_location.unknown) in
2716-
let step_binop =
2717-
IT.IT (IT.(Binop (Mul, i_it, q.step)), i_bt, Cerb_location.unknown)
2718-
in
27192722
let value_it =
2720-
IT.IT (IT.(Binop (Add, q.pointer, step_binop)), BT.(Loc ()), Cerb_location.unknown)
2723+
IT.arrayShift_ ~base:q.pointer ~index:i_it q.step Cerb_location.unknown
27212724
in
27222725
let b4, s4, e4 = cn_to_ail_expr_internal dts globals value_it PassBack in
27232726
let ptr_add_sym = Sym.fresh () in
@@ -3830,7 +3833,13 @@ let cn_to_ail_assume_resource_internal
38303833
Sym.fresh_pretty (Option.get (get_conversion_to_fn_str BT.Integer))
38313834
in *)
38323835
let b2, s2, _e2 = cn_to_ail_expr_internal dts globals q.permission PassBack in
3833-
let b3, s3, _e3 = cn_to_ail_expr_internal dts globals q.step PassBack in
3836+
let b3, s3, _e3 =
3837+
cn_to_ail_expr_internal
3838+
dts
3839+
globals
3840+
(IT.sizeOf_ q.step Cerb_location.unknown)
3841+
PassBack
3842+
in
38343843
(* let conversion_fcall = A.(AilEcall (mk_expr (AilEident convert_to_cn_integer_sym), [e_start])) in *)
38353844
let start_binding = create_binding i_sym cn_integer_ptr_ctype in
38363845
let start_assign = A.(AilSdeclaration [ (i_sym, Some e_start) ]) in
@@ -3843,11 +3852,8 @@ let cn_to_ail_assume_resource_internal
38433852
let return_ctype, _return_bt = calculate_return_type q.name in
38443853
(* Translation of q.pointer *)
38453854
let i_it = IT.IT (IT.(Sym i_sym), i_bt, Cerb_location.unknown) in
3846-
let step_binop =
3847-
IT.IT (IT.(Binop (Mul, i_it, q.step)), i_bt, Cerb_location.unknown)
3848-
in
38493855
let value_it =
3850-
IT.IT (IT.(Binop (Add, q.pointer, step_binop)), BT.(Loc ()), Cerb_location.unknown)
3856+
IT.arrayShift_ ~base:q.pointer ~index:i_it q.step Cerb_location.unknown
38513857
in
38523858
let b4, s4, e4 = cn_to_ail_expr_internal dts globals value_it PassBack in
38533859
let ptr_add_sym = Sym.fresh () in

lib/compile.ml

+1-2
Original file line numberDiff line numberDiff line change
@@ -1142,8 +1142,7 @@ module EffectfulTranslation = struct
11421142
let msg_s = "Iterated predicate pointer must be array_shift<ctype>(ptr, q_var):" in
11431143
match IT.get_term ptr_expr with
11441144
| ArrayShift { base = p; ct; index = x } when Terms.equal_annot SBT.equal x qs ->
1145-
let here = Locations.other __LOC__ in
1146-
return (p, IT.cast_ (SBT.proj bt) (IT.sizeOf_ ct here) here)
1145+
return (p, ct)
11471146
| _ -> fail { loc; msg = Generic (!^msg_s ^^^ IT.pp ptr_expr) [@alert "-deprecated"] }
11481147

11491148

lib/executable_spec_records.ml

+1-2
Original file line numberDiff line numberDiff line change
@@ -58,8 +58,7 @@ let rec add_records_to_map_from_it it =
5858

5959
let add_records_to_map_from_resource = function
6060
| Request.P p -> List.iter add_records_to_map_from_it (p.pointer :: p.iargs)
61-
| Q q ->
62-
List.iter add_records_to_map_from_it (q.pointer :: q.step :: q.permission :: q.iargs)
61+
| Q q -> List.iter add_records_to_map_from_it (q.pointer :: q.permission :: q.iargs)
6362

6463

6564
let add_records_to_map_from_lc = function

lib/explain.ml

+1-3
Original file line numberDiff line numberDiff line change
@@ -244,9 +244,7 @@ let state (ctxt : C.t) log model_with_q extras =
244244
| Some (Q ret) ->
245245
let binder = IT.(Pat (PSym (fst ret.q), snd ret.q, Loc.other __LOC__), None) in
246246
ITSet.union
247-
(ITSet.bigunion_map
248-
(subterms_without_bound_variables [])
249-
[ ret.pointer; ret.step ])
247+
(ITSet.bigunion_map (subterms_without_bound_variables []) [ ret.pointer ])
250248
(ITSet.bigunion_map
251249
(subterms_without_bound_variables [ binder ])
252250
(ret.permission :: ret.iargs))

lib/pack.ml

+2-10
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ let unfolded_array loc init (ict, olength) pointer =
2929
pointer;
3030
q = (q_s, Memory.uintptr_bt);
3131
q_loc = loc;
32-
step = IT.uintptr_int_ (Memory.size_of_ctype ict) loc;
32+
step = ict;
3333
iargs = [];
3434
permission =
3535
IT.(
@@ -166,15 +166,7 @@ let extractable_one (* global *) prove_or_model (predicate_name, index) (ret, O
166166
let at_index =
167167
( P
168168
{ name = ret.name;
169-
pointer =
170-
IT.(
171-
pointer_offset_
172-
( ret.pointer,
173-
mul_
174-
( cast_ Memory.uintptr_bt ret.step loc,
175-
cast_ Memory.uintptr_bt index loc )
176-
loc )
177-
loc);
169+
pointer = IT.(arrayShift_ ~base:ret.pointer ~index ret.step loc);
178170
iargs = List.map (IT.subst su) ret.iargs
179171
},
180172
O (IT.map_get_ o index loc) )

lib/pp_mucore_coq.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1251,7 +1251,7 @@ and pp_request_qpredicate qpred =
12511251
("QPredicate.pointer", pp_index_term qpred.pointer);
12521252
("QPredicate.q", pp_pair pp_symbol (pp_basetype pp_unit) qpred.q);
12531253
("QPredicate.q_loc", pp_location qpred.q_loc);
1254-
("QPredicate.step", pp_index_term qpred.step);
1254+
("QPredicate.step", pp_sctype qpred.step);
12551255
("QPredicate.permission", pp_index_term qpred.permission);
12561256
("QPredicate.iargs", pp_list pp_index_term qpred.iargs)
12571257
]

lib/request.ml

+6-15
Original file line numberDiff line numberDiff line change
@@ -76,23 +76,16 @@ module QPredicate = struct
7676
pointer : IT.t; (* I *)
7777
q : Sym.t * BaseTypes.t;
7878
q_loc : Locations.t; [@equal fun _ _ -> true] [@compare fun _ _ -> 0]
79-
step : IT.t;
79+
step : Sctypes.ctype;
8080
permission : IT.t; (* I, function of q *)
8181
iargs : IT.t list (* I, function of q *)
8282
}
8383
[@@deriving eq, ord]
8484

8585
let pp_aux (p : t) oargs =
8686
let open Pp in
87-
(* ISD: this is `p + i * step` but that's "wrong" in a couple of ways:
88-
- we are not using the correct precedences for `p` and `step`
89-
- in C pointer arithmetic takes account of the types, but here
90-
we seem to be doing it at the byte level. Would `step` ever
91-
differ from the size of elements that `p` points to?
92-
- perhaps print as `&p[i]` or `&p[j + i]`
93-
*)
9487
let pointer =
95-
IT.pp p.pointer ^^^ plus ^^^ Sym.pp (fst p.q) ^^^ star ^^^ IT.pp p.step
88+
IT.pp p.pointer ^^^ plus ^^^ Sym.pp (fst p.q) ^^^ at ^^^ Sctypes.pp p.step
9689
in
9790
let args = pointer :: List.map IT.pp p.iargs in
9891
!^"each"
@@ -126,7 +119,7 @@ module QPredicate = struct
126119
pointer = IT.subst substitution qp.pointer;
127120
q = qp.q;
128121
q_loc = qp.q_loc;
129-
step = IT.subst substitution qp.step;
122+
step = qp.step;
130123
permission = IT.subst substitution qp.permission;
131124
iargs = List.map (IT.subst substitution) qp.iargs
132125
}
@@ -137,7 +130,7 @@ module QPredicate = struct
137130
Dnode
138131
( pp_ctor "qpred",
139132
Dleaf (Pp.parens (Pp.typ (Sym.pp (fst qpred.q)) (BaseTypes.pp (snd qpred.q))))
140-
:: IT.dtree qpred.step
133+
:: Dleaf (Sctypes.pp qpred.step)
141134
:: IT.dtree qpred.permission
142135
:: dtree_of_name qpred.name
143136
:: IT.dtree qpred.pointer
@@ -186,15 +179,15 @@ let free_vars_bts = function
186179
(fun _ bt1 bt2 ->
187180
assert (BaseTypes.equal bt1 bt2);
188181
Some bt1)
189-
(IT.free_vars_bts_list [ p.pointer; p.step ])
182+
(IT.free_vars_bts_list [ p.pointer ])
190183
(Sym.Map.remove (fst p.q) (IT.free_vars_bts_list (p.permission :: p.iargs)))
191184

192185

193186
let free_vars = function
194187
| P p -> IT.free_vars_list (p.pointer :: p.iargs)
195188
| Q p ->
196189
Sym.Set.union
197-
(Sym.Set.union (IT.free_vars p.pointer) (IT.free_vars p.step))
190+
(IT.free_vars p.pointer)
198191
(Sym.Set.remove (fst p.q) (IT.free_vars_list (p.permission :: p.iargs)))
199192

200193

@@ -207,8 +200,6 @@ let alpha_equivalent r1 r2 =
207200
| _ -> false
208201

209202

210-
let steps_constant = function Q qp -> Option.is_some (IT.is_const qp.step) | _ -> true
211-
212203
let dtree = function P pred -> Predicate.dtree pred | Q qpred -> QPredicate.dtree qpred
213204

214205
let get_pointer = function P pred -> pred.pointer | Q qpred -> qpred.pointer

lib/request.mli

+1-3
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ module QPredicate : sig
3737
pointer : IndexTerms.t;
3838
q : Sym.t * BaseTypes.t;
3939
q_loc : Locations.t;
40-
step : IndexTerms.t;
40+
step : Sctypes.ctype;
4141
permission : IndexTerms.t;
4242
iargs : IndexTerms.t list
4343
}
@@ -83,8 +83,6 @@ val free_vars : t -> Sym.Set.t
8383

8484
val alpha_equivalent : t -> t -> bool
8585

86-
val steps_constant : t -> bool
87-
8886
val dtree : t -> Cerb_frontend.Pp_ast.doc_tree
8987

9088
val get_pointer : t -> IndexTerms.t

lib/resourceInference.ml

+3-24
Original file line numberDiff line numberDiff line change
@@ -251,33 +251,19 @@ module General = struct
251251
let@ provable = provable loc in
252252
let@ simp_ctxt = simp_ctxt () in
253253
let needed = requested.permission in
254-
let step = Simplify.IndexTerms.simp simp_ctxt requested.step in
255-
let@ () =
256-
if Option.is_some (IT.is_const step) then
257-
return ()
258-
else (
259-
let doc =
260-
Pp.(
261-
!^"cannot simplify iter-step to constant:"
262-
^^^ IT.pp requested.step
263-
^^ colon
264-
^^^ IT.pp step)
265-
in
266-
fail (fun _ -> { loc; msg = TypeErrors.Generic doc [@alert "-deprecated"] }))
267-
in
254+
let step = requested.step in
268255
let@ (needed, oarg), rw_time =
269256
map_and_fold_resources
270257
loc
271258
(fun re (needed, oarg) ->
272259
let continue = (Unchanged, (needed, oarg)) in
273-
assert (Req.steps_constant (fst re));
274260
if IT.is_false needed then
275261
continue
276262
else (
277263
match re with
278264
| Q p', O p'_oarg
279265
when Req.subsumed requested.name p'.name
280-
&& IT.equal step p'.step
266+
&& Sctypes.equal step p'.step
281267
&& BaseTypes.equal (snd requested.q) (snd p'.q) ->
282268
let p' = Req.QPredicate.alpha_rename_ (fst requested.q) p' in
283269
let here = Locations.other __LOC__ in
@@ -341,14 +327,7 @@ module General = struct
341327
| `True ->
342328
let@ o_re_index =
343329
let pointer =
344-
IT.(
345-
pointer_offset_
346-
( requested.pointer,
347-
mul_
348-
( cast_ Memory.uintptr_bt requested.step here,
349-
cast_ Memory.uintptr_bt index here )
350-
here ))
351-
here
330+
IT.(arrayShift_ ~base:requested.pointer ~index requested.step here)
352331
in
353332
predicate_request
354333
loc

lib/simplify.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -651,7 +651,7 @@ module Request = struct
651651
pointer = IndexTerms.simp simp_ctxt qp.pointer;
652652
q = qp.q;
653653
q_loc = qp.q_loc;
654-
step = IndexTerms.simp simp_ctxt qp.step;
654+
step = qp.step;
655655
permission = and_ permission (IT.get_loc qp.permission);
656656
iargs = List.map (IndexTerms.simp simp_ctxt) qp.iargs
657657
}

lib/testGeneration/genCompile.ml

+2-2
Original file line numberDiff line numberDiff line change
@@ -183,7 +183,7 @@ let rec compile_it_lat
183183
let gt_body =
184184
let sym_val = Sym.fresh () in
185185
let it_q = IT.sym_ (q_sym, k_bt, q_loc) in
186-
let it_p = IT.add_ (pointer, IT.mul_ (it_q, step) (IT.get_loc step)) loc in
186+
let it_p = IT.arrayShift_ ~base:pointer ~index:it_q step loc in
187187
let gt_asgn =
188188
GT.asgn_
189189
( (it_p, ct),
@@ -218,7 +218,7 @@ let rec compile_it_lat
218218
let pred = List.assoc Sym.equal fsym preds in
219219
let arg_syms = pred.pointer :: fst (List.split pred.iargs) in
220220
let it_q = IT.sym_ (q_sym, q_bt, q_loc) in
221-
let it_p = IT.add_ (pointer, IT.mul_ (it_q, step) (IT.get_loc step)) loc in
221+
let it_p = IT.arrayShift_ ~base:pointer ~index:it_q step loc in
222222
let arg_its = it_p :: iargs in
223223
let args = List.combine arg_syms arg_its in
224224
(* Build [GT.t] *)

lib/wellTyped.ml

+12-20
Original file line numberDiff line numberDiff line change
@@ -1111,25 +1111,10 @@ module WReq = struct
11111111
(*normalisation does not change bit types. If this assertion fails, we have to
11121112
adjust the later code to use qbt.*)
11131113
warn_when_not_quantifier_bt "each" loc qbt (Sym.pp (fst p.q));
1114-
let@ step = WIT.check loc (snd p.q) p.step in
1115-
let@ step =
1116-
match step with
1117-
| IT (Const (Bits (_bits_bt, z)), _, _) ->
1118-
if Z.lt Z.zero z then
1119-
return step
1120-
else
1121-
fail
1122-
{ loc;
1123-
msg =
1124-
Generic (!^"Iteration step" ^^^ IT.pp p.step ^^^ !^"must be positive")
1125-
[@alert "-deprecated"]
1126-
}
1127-
| IT (SizeOf _, _, _) -> return step
1128-
| IT (Cast (_, IT (SizeOf _, _, _)), _, _) -> return step
1129-
| _ ->
1130-
let hint = "Only constant iteration steps are allowed." in
1131-
fail { loc; msg = NIA { it = p.step; hint } }
1132-
in
1114+
(* The location is not quite right here (should point to the
1115+
array_shift<>() itself rather than the pointer) but it's good enough
1116+
for now. *)
1117+
let@ () = err_if_ct_void (IT.get_loc p.pointer) `Array_shift p.step in
11331118
let@ permission, iargs =
11341119
pure
11351120
(let@ () = add_l (fst p.q) (snd p.q) (loc, lazy (Pp.string "forall-var")) in
@@ -1153,7 +1138,14 @@ module WReq = struct
11531138
in
11541139
return
11551140
(Req.Q
1156-
{ name = p.name; pointer; q = p.q; q_loc = p.q_loc; step; permission; iargs })
1141+
{ name = p.name;
1142+
pointer;
1143+
q = p.q;
1144+
q_loc = p.q_loc;
1145+
step = p.step;
1146+
permission;
1147+
iargs
1148+
})
11571149
end
11581150

11591151
module WRS = struct

0 commit comments

Comments
 (0)