Skip to content

Commit daf0c49

Browse files
committed
feat(BV, CP): Propagators for addition and multiplication
This patch adds constraints to represent bit-vector addition and multiplication, with propagators in the interval domain and a simple propagator for multiplication that only considers the factors-of-two multiples in the bit-vector domain (this information cannot be captured by the interval domain due to imprecise handling of overflow). No propagator for addition in the bit-vector domain is provided yet, although the plan is to add a full-adder propagator to provide a limited form of *local* bit-blasting, as suggested in S. Bardin, P. Herrmann, and F. Perroud. “An Alternative to SAT-Based Approaches for Bit-Vectors”. In: TACAS. 2010.
1 parent 5c14a39 commit daf0c49

12 files changed

+408
-45
lines changed

src/lib/reasoners/bitlist.ml

+37-1
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@ let explanation { ex; _ } = ex
4949

5050
let exact width value ex =
5151
{ width
52-
; bits_set = value
52+
; bits_set = Z.extract value 0 width
5353
; bits_clr = Z.extract (Z.lognot value) 0 width
5454
; ex }
5555

@@ -264,3 +264,39 @@ let fold_domain f b acc =
264264
(ofs + 1) { b with bits_set = Z.logor b.bits_set mask } acc
265265
in
266266
fold_domain_aux 0 b acc
267+
268+
(* simple propagator: only compute known low bits *)
269+
let mul a b =
270+
let sz = width a in
271+
assert (width b = sz);
272+
273+
let ex = Ex.union (explanation a) (explanation b) in
274+
275+
(* (a * 2^n) * (b * 2^m) = (a * b) * 2^(n + m) *)
276+
let zeroes_a = Z.trailing_zeros @@ Z.lognot a.bits_clr in
277+
let zeroes_b = Z.trailing_zeros @@ Z.lognot b.bits_clr in
278+
if zeroes_a + zeroes_b >= sz then
279+
exact sz Z.zero ex
280+
else
281+
let low_bits =
282+
if zeroes_a + zeroes_b = 0 then empty
283+
else exact (zeroes_a + zeroes_b) Z.zero ex
284+
in
285+
let a = extract a zeroes_a (zeroes_a + sz - width low_bits - 1) in
286+
assert (width a + width low_bits = sz);
287+
let b = extract b zeroes_b (zeroes_b + sz - width low_bits - 1) in
288+
assert (width b + width low_bits = sz);
289+
(* ((ah * 2^n) + al) * ((bh * 2^m) + bl) =
290+
al * bl (mod 2^(min n m)) *)
291+
let low_a_known = Z.trailing_zeros @@ Z.lognot @@ bits_known a in
292+
let low_b_known = Z.trailing_zeros @@ Z.lognot @@ bits_known b in
293+
let low_known = min low_a_known low_b_known in
294+
let mid_bits =
295+
if low_known = 0 then empty
296+
else exact
297+
low_known
298+
Z.(extract (value a) 0 low_known * extract (value b) 0 low_known)
299+
ex
300+
in
301+
concat (unknown (sz - width mid_bits - width low_bits) Ex.empty) @@
302+
concat mid_bits low_bits

src/lib/reasoners/bitlist.mli

+3
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,9 @@ val logor : t -> t -> t
118118
val logxor : t -> t -> t
119119
(** Bitwise xor. *)
120120

121+
val mul : t -> t -> t
122+
(** Multiplication. *)
123+
121124
val concat : t -> t -> t
122125
(** Bit-vector concatenation. *)
123126

src/lib/reasoners/bitv.ml

+8-2
Original file line numberDiff line numberDiff line change
@@ -347,7 +347,10 @@ module Shostak(X : ALIEN) = struct
347347
let is_mine_symb sy =
348348
match sy with
349349
| Sy.Bitv _
350-
| Op (Concat | Extract _ | BV2Nat | BVnot | BVand | BVor | BVxor)
350+
| Op (
351+
Concat | Extract _ | BV2Nat
352+
| BVnot | BVand | BVor | BVxor
353+
| BVadd | BVsub | BVmul)
351354
-> true
352355
| _ -> false
353356

@@ -403,7 +406,10 @@ module Shostak(X : ALIEN) = struct
403406
let other ~neg t _sz ctx =
404407
let r, ctx' =
405408
match E.term_view t with
406-
| { f = Op (BVand | BVor | BVxor); _ } ->
409+
| { f = Op (
410+
BVand | BVor | BVxor
411+
| BVadd | BVsub | BVmul
412+
); _ } ->
407413
X.term_embed t, []
408414
| _ -> X.make t
409415
in

0 commit comments

Comments
 (0)