From 742cd36d3ec23ba91e8bc7a3e73dfb7f7bddbdfc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Basile=20Cl=C3=A9ment?= Date: Tue, 25 Jun 2024 11:31:42 +0200 Subject: [PATCH 1/5] feat(BV): Add support for bv2nat/int2bv normal forms Currently, Alt-Ergo knows the value of [bv2nat] and [int2bv] on constants. We also perform some simplifications during semantic value creation; however, these simplifications are not integrated into the rest of the solver. For instance, we know that [bv2nat(x @ y)] is equal to [bv2nat(x) * 2^n + bv2nat(y)] (where [n] is the bit-width of [y]), but we don't know that [z] is equal to the same under the hypothesis that [z = x @ y]. This patch moves these simplifications to the [Bitv_rel] module, making them an integral part of the solving process. Since we can't compute [bv2nat] (or [int2bv]) of an arbitrary semantic value, we generate fresh variables to represent them when needed. --- src/lib/reasoners/arith.ml | 53 +- src/lib/reasoners/bitv.ml | 91 +-- src/lib/reasoners/bitv_rel.ml | 710 ++++++++++++++++++++---- tests/bitv/testfile-bv2nat-001.expected | 2 + tests/bitv/testfile-bv2nat-001.smt2 | 4 + tests/bitv/testfile-bv2nat-002.expected | 2 + tests/bitv/testfile-bv2nat-002.smt2 | 5 + tests/bitv/testfile-bv2nat-003.expected | 2 + tests/bitv/testfile-bv2nat-003.smt2 | 4 + tests/bitv/testfile-bv2nat-004.expected | 2 + tests/bitv/testfile-bv2nat-004.smt2 | 5 + tests/bitv/testfile-bv2nat-005.expected | 2 + tests/bitv/testfile-bv2nat-005.smt2 | 4 + tests/bitv/testfile-bv2nat-006.expected | 2 + tests/bitv/testfile-bv2nat-006.smt2 | 6 + tests/bitv/testfile-bv2nat-007.expected | 2 + tests/bitv/testfile-bv2nat-007.smt2 | 6 + tests/bitv/testfile-bv2nat-009.expected | 2 + tests/bitv/testfile-bv2nat-009.smt2 | 4 + 19 files changed, 652 insertions(+), 256 deletions(-) create mode 100644 tests/bitv/testfile-bv2nat-001.expected create mode 100644 tests/bitv/testfile-bv2nat-001.smt2 create mode 100644 tests/bitv/testfile-bv2nat-002.expected create mode 100644 tests/bitv/testfile-bv2nat-002.smt2 create mode 100644 tests/bitv/testfile-bv2nat-003.expected create mode 100644 tests/bitv/testfile-bv2nat-003.smt2 create mode 100644 tests/bitv/testfile-bv2nat-004.expected create mode 100644 tests/bitv/testfile-bv2nat-004.smt2 create mode 100644 tests/bitv/testfile-bv2nat-005.expected create mode 100644 tests/bitv/testfile-bv2nat-005.smt2 create mode 100644 tests/bitv/testfile-bv2nat-006.expected create mode 100644 tests/bitv/testfile-bv2nat-006.smt2 create mode 100644 tests/bitv/testfile-bv2nat-007.expected create mode 100644 tests/bitv/testfile-bv2nat-007.smt2 create mode 100644 tests/bitv/testfile-bv2nat-009.expected create mode 100644 tests/bitv/testfile-bv2nat-009.smt2 diff --git a/src/lib/reasoners/arith.ml b/src/lib/reasoners/arith.ml index 01ab0206e..c95bdb1a5 100644 --- a/src/lib/reasoners/arith.ml +++ b/src/lib/reasoners/arith.ml @@ -365,56 +365,11 @@ module Shostak | _ -> P.add p (P.create [coef, a] Q.zero ty), ctx let make t = - let { E.f; xs; _ } = E.term_view t in - match f, xs with - | Op Int2BV n, [x] -> - (* When we have an Int2BV expression, we try our best to convert it to - something that is usable by the bitv theory. - - More precisely: - - - If we have (int2bv c) where [c] is a constant, we convert the - constant to a bitvector constant of the appropriate size and - create the corresponding [Bitv] term. The call to [X.make] will be - dispatched to the bitvector theory. - - - If we have (int2bv [1 * (bv2nat e) + 0]) -- that is, int2bv of a - single alien that is equal to a [bv2nat] expression -- we convert - [e] to the appropriate bitvector size using [extract] or - [zero_extend]. This is done by a roundtrip through E.BV.int2bv. - - - There are some other expressions where we could convert, for - instance, disjoint sums of bv2nat terms multiplied by powers of - two, but we only handle the simple cases for now. - - - In all other cases, Int2BV becomes uninterpreted. *) - let p, ctx = mke Q.one (empty_polynome Tint) x [] in - begin match P.to_list p with - | [], c -> - let c = Q.to_z c in - let c = ZA.(erem c @@ ~$1 lsl n) in - let r, ctx' = E.mk_term (Sy.Bitv (n, c)) [] (Tbitv n) |> X.make in - r, List.rev_append ctx' ctx - | [ coef, x ], const when Q.is_zero const && Q.is_one coef -> - begin match X.term_extract x with - | Some tx, _ -> - begin match (E.term_view tx).f with - | Op BV2Nat -> - (* int2bv will simplify BV2Nat: we must [X.make] again *) - let r, ctx' = E.BV.int2bv n tx |> X.make in - r, List.rev_append ctx' ctx - | _ -> - (* Otherwise we must become uninterpreted *) - E.BV.int2bv n tx |> X.term_embed, ctx - end - | None, _ -> - X.term_embed t, [] - end - | _ -> - X.term_embed t, [] - end + Options.tool_req 4 "TR-Arith-Make"; + match E.term_view t with + | { f = Op (Int2BV _); _ } -> + X.term_embed t, [] | _ -> - Options.tool_req 4 "TR-Arith-Make"; let ty = E.type_info t in let p, ctx = mke Q.one (empty_polynome ty) t [] in is_mine p, ctx diff --git a/src/lib/reasoners/bitv.ml b/src/lib/reasoners/bitv.ml index 1c8fe5fb2..d8117a483 100644 --- a/src/lib/reasoners/bitv.ml +++ b/src/lib/reasoners/bitv.ml @@ -1302,95 +1302,10 @@ module Shostak(X : ALIEN) = struct let print = Debug.print_C_ast - (* This is used to extract terms from non-bitv semantic values. - - We assume that non-bitv semantic values of a bitvector type are - necessarily uninterpreted terms, because that should be the case at the - time this code is written. - - If this ever ceases to be the case, we should either preserve the original - term along with the semantic value, or fail more gracefully here. *) - let term_extract r = - match X.term_extract r with - | Some t, _ -> t - | None, _ -> - Util.internal_error "Non-BV semantic value: %a" X.print r - - (* This is a helper function that converts a [simple_term] to an integer - expression. *) - let simple_term_to_nat acc st = - match st.bv with - | Cte n -> E.Ints.(acc * ~$$Z.(~$1 lsl st.sz) + ~$$n) - | Other r -> - let t = term_extract r.value in - let t = if r.negated then E.BV.bvnot t else t in - E.Ints.(acc * ~$$Z.(~$1 lsl st.sz) + E.BV.bv2nat t) - | Ext (o, _, i, j) -> - assert (st.sz = j - i + 1); - let t = term_extract o.value in - let t = if o.negated then E.BV.bvnot t else t in - E.Ints.( - acc * ~$$Z.(~$1 lsl st.sz) + - (E.BV.bv2nat t / ~$$Z.(~$1 lsl i)) mod ~$$Z.(~$1 lsl st.sz)) - - let abstract_to_nat r = - List.fold_left simple_term_to_nat (E.Ints.of_int 0) r - - (* Ideally, we would want to just call [abstract_to_nat r |> X.make]. But if - we do so, we may end up in a loop where we repeatedly call [X.make] on a - [BV2Nat] term -- so instead if we are a single [Other] term, we become - uninterpreted. *) - let bv2nat ot bv = - match bv with - | [{ bv = Other { value = r; negated }; sz }] -> - let t = term_extract r in - let maybe_negate t = - if negated then E.Ints.(~$$Z.(~$1 lsl sz - ~$1) - t) else t - in - let t', ctx = - begin match E.term_view t with - | { f = Op Int2BV _; _ } -> - (* bv2nat will simplify: we must call [X.make] again *) - E.BV.bv2nat t |> maybe_negate, [] - | { ty = Tbitv n; _ } -> - assert (n = sz); - if negated then - (* if we are negated, we will simplify *) - E.BV.bv2nat t |> maybe_negate, [] - else - (* bv2nat will *not* simplify: become uninterpreted with interval - information *) - let t = E.BV.bv2nat t |> maybe_negate in - t, [ E.Ints.(~$0 <= t) ; E.Ints.(t < ~$$Z.(~$1 lsl n)) ] - | { ty; _ } -> - Util.internal_error "expected bitv, got %a" Ty.print ty - end - in - X.term_embed ot, E.Core.eq ot t' :: ctx - | _ -> - (* Note: we can't just call [X.make] on the result of [abstract_to_nat] - because [X.make] should only be called on subterms. If we do it, it - causes crashes when `IntervalCalculus.add` assumes that the arguments - of division operators have been added to the `Uf` prior to the - division itself. *) - let t' = abstract_to_nat bv in - X.term_embed ot, [ E.Core.eq ot t' ] - let make t = - let { E.f; xs; _ } = E.term_view t in - match f, xs with - | Op BV2Nat, [x] -> - (* When we have a BV2Nat expression, we try our best to convert it to - something that is usable by the arithmetic theory. - - More precisely, after simplification of the argument, we get a - composition of constants and aliens or alien extractions, to which we - apply [bv2nat] recursively. If the alien or alien extraction are - [int2bv] terms, we convert the composition [(bv2nat ((_ int2bv n) x))] - into [(mod x (pow 2 n))]. *) - let r, ctx = Canon.make x in - let r, ctx' = bv2nat t r in - r, List.rev_append ctx' ctx + match E.term_view t with + | { f = Op BV2Nat; _ } -> + X.term_embed t, [] | _ -> let r, ctx = Canon.make t in is_mine r, ctx diff --git a/src/lib/reasoners/bitv_rel.ml b/src/lib/reasoners/bitv_rel.ml index 7f506da80..a827cc61a 100644 --- a/src/lib/reasoners/bitv_rel.ml +++ b/src/lib/reasoners/bitv_rel.ml @@ -27,51 +27,558 @@ module L = Xliteral let timer = Timers.M_Bitv -(* Currently we only compute, but in the future we may want to perform the same - simplifications as in [Bitv.make]. We currently don't, because we don't - really have a way to share code that uses polynome between the theory and the - relations without touching the Shostak [module rec]. - - Note that if we *do* want to compute here, the check for [X.is_constant] in - [Rel_utils.update] needs to be removed, which may have (small) performance - implications. *) -let bv2nat _op bv = - match Bitv.to_Z_opt bv with - | Some n -> Some (Shostak.Polynome.create [] (Q.of_bigint n) Tint) - | None -> None - -(* [int2bv] is in the bitvector theory rather than the arithmetic theory because - we treat the arithmetic as more "primitive" than bit-vectors. *) -let int2bv op p = - match op, Shostak.Polynome.is_const p with - | Symbols.Int2BV n, Some q -> - assert (Z.equal (Q.den q) Z.one); - let m = Q.to_bigint q in - Some (Bitv.int2bv_const n m) - | Int2BV _, None -> None - | _ -> assert false - -let delay1 = Rel_utils.delay1 - -let dispatch = function - | Symbols.BV2Nat -> - Some (delay1 Shostak.Bitv.embed Shostak.Arith.is_mine bv2nat) - | Int2BV _ -> - Some (delay1 Shostak.Arith.embed Shostak.Bitv.is_mine int2bv) - | _ -> None - let is_bv_ty = function | Ty.Tbitv _ -> true | _ -> false -let is_bv_r r = is_bv_ty @@ X.type_info r - let bitwidth r = match X.type_info r with Tbitv n -> n | _ -> assert false + let const sz n = Shostak.Bitv.is_mine [ { bv = Cte (Z.extract n 0 sz); sz } ] +module BitvNormalForm = struct + (** Normal form for bit-vector values. + + We decompose non-constant bit-vector compositions as a variable part, + where all constant bits are set to [0] and all high constant bits are + chopped off, and an offset with all the constant bits. We consider the + variable part atomic if it is a single non-negated variable. + + Assuming [x] and [y] are bit-vectors of width 2, the normal form of: + - [101 @ x] is [x + 10100] ; + - [10 @ x @ 01] is [(x @ 00) + 100001] ; + - [10 @ y<0, 0> @ y<1, 1>] is [(y<0, 0> @ y<1, 1>) + 1000] ; + - [10 @ x @ 11 @ y] is [(x @ 00 @ y) + 10001100] + + ([x] denotes the extraction from bit [i] to [j]) *) + + type constant = Z.t + + module Atom = Rel_utils.XComparable + + let type_info = X.type_info + + module Composite = Rel_utils.XComparable + + let fold_composite f r acc = + List.fold_left (fun acc { Bitv.bv ; _ } -> + match bv with + | Bitv.Cte _ -> acc + | Other { value ; _ } -> f value acc + | Ext ({ value ; _ }, _, _, _) -> f value acc + ) acc (Shostak.Bitv.embed r) + + type t = + | Constant of constant + | Atom of Atom.t * constant + | Composite of Composite.t * constant + + let normal_form r = + let rec loop cte rev_acc = function + | [] -> ( + match rev_acc with + | [] -> + Constant cte + | [ { Bitv.bv = Bitv.Other { value ; negated = false }; _ } ] -> + Atom (value, cte) + | _ -> + Composite (Shostak.Bitv.is_mine (List.rev rev_acc), cte) + ) + | { Bitv.bv = Bitv.Cte n ; sz } :: bv' -> + let cte = Z.(cte lsl sz lor n) in + let acc = + match rev_acc with + | [] -> [] + | _ -> { Bitv.bv = Bitv.Cte Z.zero ; sz } :: rev_acc + in + loop cte acc bv' + | x :: bv' -> + let cte = Z.(cte lsl x.sz) in + loop cte (x :: rev_acc) bv' + in loop Z.zero [] (Shostak.Bitv.embed r) +end + +module BV2Nat = struct + (* Domain for bv2nat and int2bv conversions + + We only keep track of bit-vector variables and their extractions, and the + associated integer polynomial (i.e. for a variable [x] of length [4], we + will keep track of a polynomial for [x] and all of the extractions of [x] + that are encountered in the problem). We also maintain a reverse map from + the polynomials to the bit-vectors so that whenever the representative on + either side (bit-vector or integers) changes, we propagate the + corresponding equality to the other side. + + In order to minimize the number of polynomial variables, we compute the + integer version of an extraction of [x] from bit [i] to [j] (inclusive) as + [0 <= bv2nat(x asr i) - bv2nat(x asr j) * 2^(j-i+1) < 2^(j-i+1)]. + + Note: We currently have no way of defining the bv2nat and int2bv of an + arbitrary semantic value structurally, so we resort to dynamically creating + variables to represent them. *) + module P = Shostak.Polynome + + type extraction = { ofs : int ; len : int } + + module Extraction = struct + type t = extraction + + let compare e1 e2 = + let c = Int.compare e1.ofs e2.ofs in + if c <> 0 then c else Int.compare e1.len e2.len + + let shift_right ~size:sz ofs = { ofs ; len = sz - ofs } + + let full = shift_right 0 + + let of_ae i j = + assert (0 <= i && i <= j); + { ofs = i ; len = j - i + 1 } + + module Map = Map.Make(struct + type nonrec t = t + + let compare = compare + end) + end + + (* Helpers for readability *) + module T = struct + module Ints = struct + module Set = Set.Make(P) + + module Map = Map.Make(P) + + let of_repr r = Shostak.Arith.embed r + + let to_repr p = Shostak.Arith.is_mine p + + let of_bigint n = P.create [] (Q.of_bigint n) Tint + + let (~$$) = of_bigint + + let zero = ~$$Z.zero + + let (~-) = P.mult_const Q.minus_one + + let ( + ) = P.add + + let ( - ) = P.sub + + let ( +$$ ) p n = P.add_const (Q.of_bigint n) p + + let ( *$$ ) p n = P.mult_const (Q.of_bigint n) p + + let mkv_eq p p' = Uf.LX.mkv_eq (to_repr p) (to_repr p') + + let mkv_le p p' = Uf.LX.mkv_builtin true LE [to_repr p; to_repr p'] + end + + module BV = struct + let extract bv { ofs ; len } = + assert (0 <= ofs + len - 1 && ofs + len - 1 < bitwidth bv); + Shostak.Bitv.is_mine @@ + Bitv.extract (bitwidth bv) ofs (ofs + len - 1) @@ + Shostak.Bitv.embed bv + + let zero_extend sz r = + let r_sz = bitwidth r in + if sz < r_sz then invalid_arg "zero_extend"; + if sz = r_sz then r + else + Shostak.Bitv.is_mine @@ + { Bitv.bv = Bitv.Cte Z.zero ; sz = sz - r_sz } :: + Shostak.Bitv.embed r + + let mkv_eq (r, ext) (r', ext') = + let sz = max ext.len ext'.len in + let r = extract r ext and r' = extract r' ext' in + Uf.LX.mkv_eq (zero_extend sz r) (zero_extend sz r') + end + end + + + (* Adds to [eqs] the inequalities [0 <= p < 2^sz]. *) + let add_width_ineqs ~ex sz p eqs = + T.Ints.(mkv_le zero p, ex) :: + T.Ints.(mkv_le p (of_bigint @@ Z.extract Z.minus_one 0 sz), ex) :: + eqs + + type t = + { bv2nat : (P.t * Ex.t) Extraction.Map.t MX.t + (** Map from bit-vector extractions to their integer representation. + + If [(bv, ext)] is associated with [(p, ex)] in this map, it means + that [p = bv2nat(bv)] is justified by [ex]. + + We maintain the invariant that if [(bv, ext)] is associated with + [(p, ex)] in this map, then [(bv, ext, ex)] is associated with [p] + in [nat2bv]. *) + ; nat2bv : (X.r * Extraction.t * Ex.t) T.Ints.Map.t + (** Map from integer polynomials to their bit-vector representation. + + If [p] is associated with [(bv, ext, ex)] in this map, it means that + [p = bv2nat(bv)] is justified by [ex]. + + Note that this is the inverse of the [bv2nat] mapping, not an + implementation of the [int2bv] function (although both coincide for + entries in the map). *) + ; use : T.Ints.Set.t MX.t + (** Map from integer variables to the polynomials that use them. + + This is used for integer substitutions because of our encoding of + extractions in [bv2nat]: we encode [bv2nat(bv)] in + terms of [bv2nat(bv<0, i>)] and [bv2nat(bv<0, j>)] which we create + dynamically. The polynomials associated with these expressions are + thus not tracked by the UnionFind module, and are not given to + [subst]. *) + ; eqs : (X.r Xliteral.view * Ex.t) list + } + + let pp_bv2nat_ext ppf ((x, ext), (p, _ex)) = + Fmt.pf ppf "@[<2>bv2nat(%a<%d, %d>) =@ %a@]" + X.print x ext.ofs ext.len P.print p + + let iter_ext f t = + MX.iter + (fun x m -> Extraction.Map.iter (fun ext -> f (x, ext)) m) + t.bv2nat + + let pp ppf t = + Fmt.pf ppf "@;<2>@[%a@]" + (Fmt.iter_bindings iter_ext pp_bv2nat_ext) t + + let empty = + { bv2nat = MX.empty + ; nat2bv = T.Ints.Map.empty + ; use = MX.empty + ; eqs = [] } + + let fold_ext x f t acc = + assert (X.is_a_leaf x); + match MX.find x t.bv2nat with + | exception Not_found -> acc + | m -> Extraction.Map.fold (f x) m acc + + let fold_use_p x f t acc = + assert (X.is_a_leaf x); + match MX.find x t.use with + | exception Not_found -> acc + | nps -> + T.Ints.Set.fold (fun p acc -> + let v = + try T.Ints.Map.find p t.nat2bv with Not_found -> assert false + in + f p v acc + ) nps acc + + let add_use_p p use = + List.fold_left (fun use r -> + MX.update r (function + | None -> Some (T.Ints.Set.singleton p) + | Some sp -> Some (T.Ints.Set.add p sp) + ) use + ) use (P.leaves p) + + let remove_use_p p use = + List.fold_left (fun use r -> + MX.update r (function + | None -> assert false + | Some ps -> + let ps = T.Ints.Set.remove p ps in + if T.Ints.Set.is_empty ps then None else Some ps + ) use + ) use (P.leaves p) + + let find_ext bv ext bv2nat = + Extraction.Map.find ext @@ MX.find bv @@ bv2nat + + let add_ext ~ex bv ext p bv2nat = + MX.update bv (function + | None -> Some (Extraction.Map.singleton ext (p, ex)) + | Some m -> Some (Extraction.Map.add ext (p, ex) m) + ) bv2nat + + let remove_aux bv ext p t = + let use = remove_use_p p t.use in + let nat2bv = T.Ints.Map.remove p t.nat2bv in + let bv2nat = + MX.update bv (function + | None -> None + | Some m -> + let m = Extraction.Map.remove ext m in + if Extraction.Map.is_empty m then None else Some m + ) t.bv2nat + in + { use ; nat2bv ; bv2nat ; eqs = t.eqs } + + (* Returns the polynomial associated with [bv2nat(bv) asr ofs], or raises + [Not_found] if there is none. *) + let find_asr bv ofs t = + find_ext bv (Extraction.shift_right ~size:(bitwidth bv) ofs) t.bv2nat + + (* Returns the polynomial associated with [bv2nat(bv) asr ofs], creating a + fresh variable for it if it does not exist. *) + let find_or_create_asr bv ofs t = + let ext = Extraction.shift_right ~size:(bitwidth bv) ofs in + try find_ext bv ext t.bv2nat, t + with Not_found -> + if ofs >= bitwidth bv then + (T.Ints.zero, Ex.empty), t + else + let k = T.Ints.of_repr @@ X.term_embed @@ E.fresh_name Tint in + let use = add_use_p k t.use in + let bv2nat = add_ext ~ex:Ex.empty bv ext k t.bv2nat in + let nat2bv = T.Ints.Map.add k (bv, ext, Ex.empty) t.nat2bv in + (k, Ex.empty), { use ; bv2nat ; nat2bv ; eqs = t.eqs } + + (* Add the definining equation for [bv2nat(bv) asr ofs] to [eqs]. + + The defining equation for [bv2nat(bv) asr ofs] is: + + 0 <= bv2nat(bv) - (bv2nat(bv) asr ofs) < 2^ofs + *) + let init_asr bv ofs t = + let (p, ex), t = find_or_create_asr bv 0 t in + if ofs = 0 then (p, ex), t + else + let (p_asr, ex_asr), t = find_or_create_asr bv ofs t in + let delta = T.Ints.(p - p_asr *$$ Z.(one lsl ofs)) in + let eqs = add_width_ineqs ~ex:(Ex.union ex ex_asr) ofs delta t.eqs in + (p_asr, ex_asr), { t with eqs } + + (* Returns the polynomial associated with [bv2nat(bv) asr ofs]. + + If no such polynomial exists, create a fresh variable for it and adds its + defining equation to [t]. *) + let find_or_init_asr bv ofs t = + try find_asr bv ofs t, t with Not_found -> init_asr bv ofs t + + (* Add the defining equation for [bv2nat(bv)] to [t]. + + We define [bv2nat(bv)] in terms of the shifts [bv asr ext.ofs] and + [bv asr ext.len]. *) + let init_ext bv ext t = + let (lo, ex_lo), t = find_or_init_asr bv ext.ofs t in + let (hi, ex_hi), t = find_or_init_asr bv (ext.ofs + ext.len) t in + let delta_p = T.Ints.(lo - hi *$$ Z.(one lsl ext.len)) in + let ex = Ex.union ex_lo ex_hi in + let eqs = add_width_ineqs ~ex ext.len delta_p t.eqs in + (delta_p, ex), { t with eqs } + + let find_or_init_ext bv ext t = + try find_ext bv ext t.bv2nat, t with Not_found -> init_ext bv ext t + + let find_p p t = + T.Ints.Map.find p t.nat2bv + + (* Add the mapping [bv2nat(bv) -> p]. If there is already a polynomial + associated with [bv] or a bit-vector associated with [p], the + appropriate equalities are added to [eqs] beforehand. + + Note that we still insert the new mapping in this situation so that we can + reuse this function for substitutions. *) + let add ~ex bv ext p t = + let t = + match T.Ints.Map.find p t.nat2bv with + | exception Not_found -> t + | (bv', ext', ex') -> + let lit = T.BV.mkv_eq (bv, ext) (bv', ext') in + let eqs = (lit, Ex.union ex ex') :: t.eqs in + remove_aux bv' ext' p { t with eqs } + in + let is_new, t = + match find_ext bv ext t.bv2nat with + | exception Not_found -> + true, t + | (p', ex') -> + let eqs = (T.Ints.mkv_eq p p', Ex.union ex ex') :: t.eqs in + false, remove_aux bv ext p' { t with eqs } + in + match P.is_const p with + | Some cte -> + assert (Z.equal Z.one @@ Q.den cte); + let cte = Q.num cte in + let lit = + Uf.LX.mkv_eq + (T.BV.extract bv ext) + (Shostak.Bitv.is_mine [ + { bv = Cte (Z.extract cte 0 ext.len) ; sz = ext.len } + ]) + in + { t with eqs = (lit, ex) :: t.eqs } + | None -> + let use = add_use_p p t.use in + let bv2nat = add_ext ~ex bv ext p t.bv2nat in + let nat2bv = T.Ints.Map.add p (bv, ext, ex) t.nat2bv in + let t = { use ; bv2nat ; nat2bv ; eqs = t.eqs } in + if is_new then + let (p', ex'), t = init_ext bv ext t in + { t with eqs = (T.Ints.mkv_eq p p', Ex.union ex ex') :: t.eqs } + else + t + + let lognot_p ~size p = + (* ~x = 2^size - 1 - x *) + T.Ints.(-p +$$ Z.extract Z.minus_one 0 size) + + (* Compute [bv2nat(x)] where [x] is an arbitrary bit-vector composition. + + This may require introducing new variables to represent unknown + extractions. *) + let composite bv t = + let rec loop p ex t = function + | [] -> (p, ex), t + | { Bitv.bv = Bitv.Cte n ; sz } :: bv' -> + let p = T.Ints.(p *$$ Z.(one lsl sz) +$$ n) in + loop p ex t bv' + | { bv = Other s ; sz } :: bv' -> + let p = T.Ints. (p *$$ Z.(one lsl sz)) in + let (p_value, ex_value), t = + find_or_init_ext + s.value (Extraction.full ~size:sz) t + in + let p_value = + if s.negated then lognot_p ~size:sz p_value + else p_value + in + loop T.Ints.(p + p_value) (Ex.union ex ex_value) t bv' + | { bv = Ext (s, _s_sz, i, j) ; sz } :: bv' -> + let p = T.Ints.(p *$$ Z.(one lsl sz)) in + let (p_value, ex_value), t = + find_or_init_ext + s.value (Extraction.of_ae i j) t + in + let p_value = + if s.negated then lognot_p ~size:sz p_value + else p_value + in + loop T.Ints.(p + p_value) (Ex.union ex ex_value) t bv' + in + loop T.Ints.zero Ex.empty t (Shostak.Bitv.embed bv) + + (* Add the equality [nat_r = bv2nat(bv_r)]. *) + let add_bv2nat ~ex nat_r bv_r t = + match BitvNormalForm.normal_form bv_r with + | Constant cte -> + (* nat_r = bv2nat(cte) = cte *) + let lit = T.Ints.(mkv_eq (of_repr nat_r) ~$$cte) in + { t with eqs = (lit, ex) :: t.eqs } + | Atom (x, ofs) -> ( + let p = Shostak.Arith.embed nat_r in + let p = T.Ints.(p +$$ Z.neg ofs) in + let eqs = add_width_ineqs ~ex (bitwidth x) p t.eqs in + add ~ex + x (Extraction.full ~size:(bitwidth x)) p { t with eqs } + ) + | Composite (x, ofs) -> + let p = T.Ints.of_repr nat_r in + let p = T.Ints.(p +$$ Z.neg ofs) in + match Shostak.Bitv.embed x with + | [ { bv = Other s ; sz } ] -> + assert (s.negated); (* Otherwise we would be an atom *) + let p = lognot_p ~size:sz p in + add ~ex + s.value (Extraction.full ~size:sz) p t + | [ { bv = Ext (s, _s_sz, i, j) ; sz } ] -> + let p = if s.negated then lognot_p ~size:sz p else p in + add ~ex s.value (Extraction.of_ae i j) p t + | _ -> + let (p', ex'), t = composite x t in + { t with eqs = (T.Ints.mkv_eq p p', Ex.union ex ex') :: t.eqs } + + (* Add the equality [bv_r = int2bv(int_r)]. + + We do not have a table mapping polynomials to their (truncated) bit-vector + representation; instead, we simply add the equation + + bv2nat(int2bv(int_r)) = int_r + 2^|bv_r| * k + + for a fresh variable [k]. + + If we know that [int_r] is an exact bit-vector representation (i.e. [int_r] + is equal to [bv2nat(bv)] for some bit-vector [bv]) we introduce the + appropriate equalities between [bv] and [bv_r] instead. This is required + for completeness. *) + let add_int2bv ~ex bv_r int_r t = + assert (X.is_a_leaf bv_r); + let sz = bitwidth bv_r in + match find_p (T.Ints.of_repr int_r) t with + | (bv', ext', ex') -> + (* int_r is bv2nat(bv'[ext']) -> convert int2bv to extraction *) + let lit = + if sz <= ext'.len then + (* bv_r is extraction of bv' *) + Uf.LX.mkv_eq bv_r (T.BV.extract bv' { ext' with len = sz }) + else + (* bv_r is extension of bv' *) + Uf.LX.mkv_eq + bv_r + (T.BV.zero_extend sz @@ T.BV.extract bv' ext') + in + { t with eqs = (lit, Ex.union ex ex') :: t.eqs } + | exception Not_found -> + (* int_r is not known to be a bv2nat + bv_r = int2bv(int_r) iff + exists k, int_r = bv2nat(bv_r) + 2^|bv_r| * k + *) + let k = T.Ints.of_repr @@ X.term_embed @@ E.fresh_name Tint in + let p_k_2exp = T.Ints.(k *$$ Z.(one lsl sz)) in + add ~ex + bv_r (Extraction.full ~size:sz) + T.Ints.(of_repr int_r + p_k_2exp) t + + let subst_int ~ex rr nrr t = + (* We need to compute our own substitutions because we are tracking + polynomials that are not necessarily associated with representatives in + the union-find, and thus for which [subst] would not get called. *) + if not (X.is_a_leaf rr) then t + else + let nrr_p = Shostak.Arith.embed nrr in + fold_use_p rr (fun p (bv, ext, ex_bv) t -> + let t = remove_aux bv ext p t in + let p' = P.subst rr nrr_p p in + add ~ex:(Ex.union ex ex_bv) bv ext p' t + ) t t + + let subst_bitv ~ex rr nrr t = + match BitvNormalForm.normal_form rr with + | Constant _ -> invalid_arg "subst: cannot substitute constant" + | Composite _ -> t + | Atom (x, ofs) -> + fold_ext x (fun x ext (p, ex_p) t -> + let t = remove_aux x ext p t in + let ofs = + if ext.len < 0 then Z.shift_right ofs ext.ofs + else Z.extract ofs ext.ofs ext.len + in + let p = T.Ints.(p +$$ ofs) in + let nrr = T.BV.extract nrr ext in + add_bv2nat ~ex:(Ex.union ex ex_p) (T.Ints.to_repr p) nrr t + ) t t + + type _ Uf.id += Id : t Uf.id + + let filter_ty = function + | Ty.Tint | Tbitv _ -> true + | _ -> false + + let init _ t = t + + exception Inconsistent of Ex.t + + let subst ~ex rr nrr t = + match X.type_info rr with + | Tint -> subst_int ~ex rr nrr t + | Tbitv _ -> subst_bitv ~ex rr nrr t + | _ -> t + + let flush t = + t.eqs, { t with eqs = [] } +end + module Interval_domain = struct type t = Intervals.Int.t @@ -157,68 +664,6 @@ struct end) end -module BitvNormalForm = struct - (** Normal form for bit-vector values. - - We decompose non-constant bit-vector compositions as a variable part, - where all constant bits are set to [0] and all high constant bits are - chopped off, and an offset with all the constant bits. We consider the - variable part atomic if it is a single non-negated variable. - - Assuming [x] and [y] are bit-vectors of width 2, the normal form of: - - [101 @ x] is [x + 10100] ; - - [10 @ x @ 01] is [(x @ 00) + 100001] ; - - [10 @ y<0, 0> @ y<1, 1>] is [(y<0, 0> @ y<1, 1>) + 1000] ; - - [10 @ x @ 11 @ y] is [(x @ 00 @ y) + 10001100] - - ([x] denotes the extraction from bit [i] to [j]) *) - - type constant = Z.t - - module Atom = Rel_utils.XComparable - - let type_info = X.type_info - - module Composite = Rel_utils.XComparable - - let fold_composite f r acc = - List.fold_left (fun acc { Bitv.bv ; _ } -> - match bv with - | Bitv.Cte _ -> acc - | Other { value ; _ } -> f value acc - | Ext ({ value ; _ }, _, _, _) -> f value acc - ) acc (Shostak.Bitv.embed r) - - type t = - | Constant of constant - | Atom of Atom.t * constant - | Composite of Composite.t * constant - - let normal_form r = - let rec loop cte rev_acc = function - | [] -> ( - match rev_acc with - | [] -> - Constant cte - | [ { Bitv.bv = Bitv.Other { value ; negated = false }; _ } ] -> - Atom (value, cte) - | _ -> - Composite (Shostak.Bitv.is_mine (List.rev rev_acc), cte) - ) - | { Bitv.bv = Bitv.Cte n ; sz } :: bv' -> - let cte = Z.(cte lsl sz lor n) in - let acc = - match rev_acc with - | [] -> [] - | _ -> { Bitv.bv = Bitv.Cte Z.zero ; sz } :: rev_acc - in - loop cte acc bv' - | x :: bv' -> - let cte = Z.(cte lsl x.sz) in - loop cte (x :: rev_acc) bv' - in loop Z.zero [] (Shostak.Bitv.embed r) -end - module Constraint : sig type binop = (* Bitwise operations *) @@ -1528,25 +1973,24 @@ let rec propagate_all uf eqs bdom idom = eqs, bdom, idom type t = - { delayed : Rel_utils.Delayed.t - ; terms : SX.t + { terms : SX.t ; size_splits : Q.t } let empty uf = - { delayed = Rel_utils.Delayed.create ~is_ready:X.is_constant dispatch - ; terms = SX.empty + { terms = SX.empty ; size_splits = Q.one }, + Uf.GlobalDomains.add (module BV2Nat) BV2Nat.empty @@ Uf.GlobalDomains.add (module Bitlist_domains) Bitlist_domains.empty @@ Uf.GlobalDomains.add (module Interval_domains) Interval_domains.empty @@ Uf.domains uf let assume env uf la = let ds = Uf.domains uf in + let bvconv = Uf.GlobalDomains.find (module BV2Nat) ds in let domain = Uf.GlobalDomains.find (module Bitlist_domains) ds in let int_domain = Uf.GlobalDomains.find (module Interval_domains) ds in - let delayed, result = Rel_utils.Delayed.assume env.delayed uf la in let (domain, int_domain, eqs, size_splits) = try let (domain, int_domain, eqs, size_splits) = @@ -1612,10 +2056,16 @@ let assume env uf la = let assume = List.rev_map (fun (lit, ex) -> Literal.LSem lit, ex, Th_util.Other) eqs in + let bv_eqs, bvconv = BV2Nat.flush bvconv in + let bvconv_assume = + List.rev_map (fun (lit, ex) -> Literal.LSem lit, ex, Th_util.Other) bv_eqs + in let result = - { result with assume = List.rev_append assume result.assume } + { Sig_rel.assume = List.rev_append bvconv_assume assume + ; remove = [] } in - { delayed ; size_splits ; terms = env.terms }, + { size_splits ; terms = env.terms }, + Uf.GlobalDomains.add (module BV2Nat) bvconv @@ Uf.GlobalDomains.add (module Bitlist_domains) domain @@ Uf.GlobalDomains.add (module Interval_domains) int_domain ds, result @@ -1681,21 +2131,43 @@ let case_split env uf ~for_model = let add env uf r t = let ds = Uf.domains uf in - let delayed, eqs = Rel_utils.Delayed.add env.delayed uf r t in - let env, ds, eqs = - if is_bv_r r then - let dom = Uf.GlobalDomains.find (module Bitlist_domains) ds in - let idom = Uf.GlobalDomains.find (module Interval_domains) ds in - let terms, dom, idom = extract_constraints env.terms dom idom uf r t in - { env with terms }, - Uf.GlobalDomains.add (module Bitlist_domains) dom @@ - Uf.GlobalDomains.add (module Interval_domains) idom @@ - ds, - eqs - else - env, ds, eqs - in - { env with delayed }, ds, eqs + match X.type_info r with + | Tint -> ( + match E.term_view t with + | { f = Op BV2Nat ; xs = [ x ] ; _ } -> + let bvconv = Uf.GlobalDomains.find (module BV2Nat) ds in + let rx, ex = Uf.find uf x in + let bvconv = BV2Nat.add_bv2nat ~ex r rx bvconv in + let eqs, bvconv = BV2Nat.flush bvconv in + env, + Uf.GlobalDomains.add (module BV2Nat) bvconv ds, + eqs + | _ -> env, ds, [] + ) + | Tbitv _ -> ( + match E.term_view t with + | { f = Op (Int2BV _) ; xs = [ x ]; _ } -> + let bvconv = Uf.GlobalDomains.find (module BV2Nat) ds in + let rx, ex = Uf.find uf x in + let bvconv = BV2Nat.add_int2bv ~ex r rx bvconv in + let eqs, bvconv = BV2Nat.flush bvconv in + env, + Uf.GlobalDomains.add (module BV2Nat) bvconv ds, + eqs + | _ -> + let dom = Uf.GlobalDomains.find (module Bitlist_domains) ds in + let idom = Uf.GlobalDomains.find (module Interval_domains) ds in + let terms, dom, idom = + extract_constraints env.terms dom idom uf r t + in + { env with terms }, + Uf.GlobalDomains.add (module Bitlist_domains) dom @@ + Uf.GlobalDomains.add (module Interval_domains) idom @@ + ds, + [] + ) + | _ -> + env, ds, [] let optimizing_objective _env uf Objective.Function.{ e; is_max; _ } = let ty = E.type_info e in diff --git a/tests/bitv/testfile-bv2nat-001.expected b/tests/bitv/testfile-bv2nat-001.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-001.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-001.smt2 b/tests/bitv/testfile-bv2nat-001.smt2 new file mode 100644 index 000000000..2df02aee9 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-001.smt2 @@ -0,0 +1,4 @@ +(set-logic ALL) +(declare-const x (_ BitVec 2)) +(assert (or (< (bv2nat x) 0) (<= 4 (bv2nat x)))) +(check-sat) diff --git a/tests/bitv/testfile-bv2nat-002.expected b/tests/bitv/testfile-bv2nat-002.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-002.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-002.smt2 b/tests/bitv/testfile-bv2nat-002.smt2 new file mode 100644 index 000000000..e47358de6 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-002.smt2 @@ -0,0 +1,5 @@ +(set-logic ALL) +(declare-const x (_ BitVec 1024)) +(declare-const y (_ BitVec 1024)) +(assert (distinct (bv2nat (concat x y)) (+ (* (bv2nat x) 179769313486231590772930519078902473361797697894230657273430081157732675805500963132708477322407536021120113879871393357658789768814416622492847430639474124377767893424865485276302219601246094119453082952085005768838150682342462881473913110540827237163350510684586298239947245938479716304835356329624224137216) (bv2nat y)))) +(check-sat) diff --git a/tests/bitv/testfile-bv2nat-003.expected b/tests/bitv/testfile-bv2nat-003.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-003.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-003.smt2 b/tests/bitv/testfile-bv2nat-003.smt2 new file mode 100644 index 000000000..4b58a41d0 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-003.smt2 @@ -0,0 +1,4 @@ +(set-logic ALL) +(declare-const x (_ BitVec 2048)) +(assert (distinct (bv2nat x) (+ (* (bv2nat ((_ extract 2047 1024) x)) 179769313486231590772930519078902473361797697894230657273430081157732675805500963132708477322407536021120113879871393357658789768814416622492847430639474124377767893424865485276302219601246094119453082952085005768838150682342462881473913110540827237163350510684586298239947245938479716304835356329624224137216) (bv2nat ((_ extract 1023 0) x))))) +(check-sat) diff --git a/tests/bitv/testfile-bv2nat-004.expected b/tests/bitv/testfile-bv2nat-004.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-004.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-004.smt2 b/tests/bitv/testfile-bv2nat-004.smt2 new file mode 100644 index 000000000..f247bf710 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-004.smt2 @@ -0,0 +1,5 @@ +(set-logic ALL) +(declare-const x (_ BitVec 1024)) + +(assert (distinct ((_ int2bv 2048) (bv2nat x)) (concat (_ bv0 1024) x))) +(check-sat) diff --git a/tests/bitv/testfile-bv2nat-005.expected b/tests/bitv/testfile-bv2nat-005.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-005.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-005.smt2 b/tests/bitv/testfile-bv2nat-005.smt2 new file mode 100644 index 000000000..5f06b4ac4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-005.smt2 @@ -0,0 +1,4 @@ +(set-logic ALL) +(declare-const x (_ BitVec 1024)) +(assert (distinct ((_ int2bv 512) (bv2nat x)) ((_ extract 511 0) x))) +(check-sat) diff --git a/tests/bitv/testfile-bv2nat-006.expected b/tests/bitv/testfile-bv2nat-006.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-006.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-006.smt2 b/tests/bitv/testfile-bv2nat-006.smt2 new file mode 100644 index 000000000..76ade3736 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-006.smt2 @@ -0,0 +1,6 @@ +(set-logic ALL) +(declare-const x (_ BitVec 1024)) +(declare-const y (_ BitVec 1024)) +(assert (= (bv2nat x) (bv2nat y))) +(assert (distinct x y)) +(check-sat) diff --git a/tests/bitv/testfile-bv2nat-007.expected b/tests/bitv/testfile-bv2nat-007.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-007.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-007.smt2 b/tests/bitv/testfile-bv2nat-007.smt2 new file mode 100644 index 000000000..a7d68ebd4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-007.smt2 @@ -0,0 +1,6 @@ +(set-logic ALL) +(declare-const x Int) +(declare-const y Int) +(assert (= ((_ int2bv 1024) x) ((_ int2bv 1024) y))) +(assert (distinct (mod (- x y) 179769313486231590772930519078902473361797697894230657273430081157732675805500963132708477322407536021120113879871393357658789768814416622492847430639474124377767893424865485276302219601246094119453082952085005768838150682342462881473913110540827237163350510684586298239947245938479716304835356329624224137216) 0)) +(check-sat) diff --git a/tests/bitv/testfile-bv2nat-009.expected b/tests/bitv/testfile-bv2nat-009.expected new file mode 100644 index 000000000..6f99ff0f4 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-009.expected @@ -0,0 +1,2 @@ + +unsat diff --git a/tests/bitv/testfile-bv2nat-009.smt2 b/tests/bitv/testfile-bv2nat-009.smt2 new file mode 100644 index 000000000..952f97977 --- /dev/null +++ b/tests/bitv/testfile-bv2nat-009.smt2 @@ -0,0 +1,4 @@ +(set-logic ALL) +(declare-const x (_ BitVec 32)) +(assert (distinct (bv2nat (concat #x00000000 x)) (bv2nat x))) +(check-sat) From 3308a0fe9592c5832ce6c8880a964dd791684987 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Basile=20Cl=C3=A9ment?= Date: Mon, 29 Jul 2024 14:24:42 +0200 Subject: [PATCH 2/5] remove spurious assertion --- src/lib/reasoners/bitv_rel.ml | 1 - 1 file changed, 1 deletion(-) diff --git a/src/lib/reasoners/bitv_rel.ml b/src/lib/reasoners/bitv_rel.ml index a827cc61a..be9b06fdd 100644 --- a/src/lib/reasoners/bitv_rel.ml +++ b/src/lib/reasoners/bitv_rel.ml @@ -257,7 +257,6 @@ module BV2Nat = struct ; eqs = [] } let fold_ext x f t acc = - assert (X.is_a_leaf x); match MX.find x t.bv2nat with | exception Not_found -> acc | m -> Extraction.Map.fold (f x) m acc From d9474129434ea008755de8cc1f9c6a920120ed5b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Basile=20Cl=C3=A9ment?= Date: Tue, 30 Jul 2024 12:29:57 +0200 Subject: [PATCH 3/5] Update CHANGES --- CHANGES.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGES.md b/CHANGES.md index 8888c3461..dc26af6c7 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -50,6 +50,7 @@ #1152) - Add interval domains and arithmetic propagators for the BV theory (#1058, #1083, #1084, #1085) + - Native support for bv2nat of bit-vector normal forms (#1154) - Fix incompleteness issues in the BV solver (#978, #979) - Abstract more arguments of AC symbols to avoid infinite loops when they contain other AC symbols (#990) From 9290d3410db360af9a0e54e32aabe55128191590 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Basile=20Cl=C3=A9ment?= Date: Mon, 19 Aug 2024 10:31:08 +0200 Subject: [PATCH 4/5] Clarifications --- src/lib/reasoners/bitv_rel.ml | 32 ++++++++++++++++++-------------- 1 file changed, 18 insertions(+), 14 deletions(-) diff --git a/src/lib/reasoners/bitv_rel.ml b/src/lib/reasoners/bitv_rel.ml index be9b06fdd..c3566aaf1 100644 --- a/src/lib/reasoners/bitv_rel.ml +++ b/src/lib/reasoners/bitv_rel.ml @@ -213,9 +213,8 @@ module BV2Nat = struct If [(bv, ext)] is associated with [(p, ex)] in this map, it means that [p = bv2nat(bv)] is justified by [ex]. - We maintain the invariant that if [(bv, ext)] is associated with - [(p, ex)] in this map, then [(bv, ext, ex)] is associated with [p] - in [nat2bv]. *) + We maintain the invariant that if [(bv, ext)] maps to [(p, ex)] in + [bv2nat], then [p] maps to [(bv, ext, ex)] in [nat2bv] below. *) ; nat2bv : (X.r * Extraction.t * Ex.t) T.Ints.Map.t (** Map from integer polynomials to their bit-vector representation. @@ -231,7 +230,12 @@ module BV2Nat = struct This is used for integer substitutions because of our encoding of extractions in [bv2nat]: we encode [bv2nat(bv)] in terms of [bv2nat(bv<0, i>)] and [bv2nat(bv<0, j>)] which we create - dynamically. The polynomials associated with these expressions are + dynamically. + + This is done to reduce the number of variables introduced to represent + extractions from quadratic to linear in the bitwidth. + + The polynomials associated with these expressions are thus not tracked by the UnionFind module, and are not given to [subst]. *) ; eqs : (X.r Xliteral.view * Ex.t) list @@ -313,12 +317,12 @@ module BV2Nat = struct in { use ; nat2bv ; bv2nat ; eqs = t.eqs } - (* Returns the polynomial associated with [bv2nat(bv) asr ofs], or raises + (* Returns the polynomial associated with [bv2nat(bv asr ofs)], or raises [Not_found] if there is none. *) let find_asr bv ofs t = find_ext bv (Extraction.shift_right ~size:(bitwidth bv) ofs) t.bv2nat - (* Returns the polynomial associated with [bv2nat(bv) asr ofs], creating a + (* Returns the polynomial associated with [bv2nat(bv asr ofs)], creating a fresh variable for it if it does not exist. *) let find_or_create_asr bv ofs t = let ext = Extraction.shift_right ~size:(bitwidth bv) ofs in @@ -333,11 +337,11 @@ module BV2Nat = struct let nat2bv = T.Ints.Map.add k (bv, ext, Ex.empty) t.nat2bv in (k, Ex.empty), { use ; bv2nat ; nat2bv ; eqs = t.eqs } - (* Add the definining equation for [bv2nat(bv) asr ofs] to [eqs]. + (* Add the definining equation for [bv2nat(bv asr ofs)] to [eqs]. - The defining equation for [bv2nat(bv) asr ofs] is: + The defining equation for [bv2nat(bv asr ofs)] is: - 0 <= bv2nat(bv) - (bv2nat(bv) asr ofs) < 2^ofs + 0 <= bv2nat(bv) - bv2nat(bv asr ofs) < 2^ofs *) let init_asr bv ofs t = let (p, ex), t = find_or_create_asr bv 0 t in @@ -348,7 +352,7 @@ module BV2Nat = struct let eqs = add_width_ineqs ~ex:(Ex.union ex ex_asr) ofs delta t.eqs in (p_asr, ex_asr), { t with eqs } - (* Returns the polynomial associated with [bv2nat(bv) asr ofs]. + (* Returns the polynomial associated with [bv2nat(bv asr ofs)]. If no such polynomial exists, create a fresh variable for it and adds its defining equation to [t]. *) @@ -497,10 +501,10 @@ module BV2Nat = struct for a fresh variable [k]. - If we know that [int_r] is an exact bit-vector representation (i.e. [int_r] - is equal to [bv2nat(bv)] for some bit-vector [bv]) we introduce the - appropriate equalities between [bv] and [bv_r] instead. This is required - for completeness. *) + If we know that [int_r] has an exact bit-vector representation (i.e. + [int_r] is equal to [bv2nat(bv)] for some bit-vector [bv]) we introduce + the appropriate equalities between [bv] and [bv_r] instead. This is + required for completeness. *) let add_int2bv ~ex bv_r int_r t = assert (X.is_a_leaf bv_r); let sz = bitwidth bv_r in From edd544a0104f9242745d074573cf0ae6b8607f68 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Basile=20Cl=C3=A9ment?= Date: Mon, 19 Aug 2024 10:59:58 +0200 Subject: [PATCH 5/5] Move helpers to Polynome module --- src/lib/reasoners/bitv_rel.ml | 51 ++++++++++------------------------ src/lib/reasoners/polynome.ml | 29 +++++++++++++++++++ src/lib/reasoners/polynome.mli | 37 +++++++++++++++++++++++- 3 files changed, 80 insertions(+), 37 deletions(-) diff --git a/src/lib/reasoners/bitv_rel.ml b/src/lib/reasoners/bitv_rel.ml index c3566aaf1..e3ee1c5f1 100644 --- a/src/lib/reasoners/bitv_rel.ml +++ b/src/lib/reasoners/bitv_rel.ml @@ -147,32 +147,11 @@ module BV2Nat = struct (* Helpers for readability *) module T = struct module Ints = struct - module Set = Set.Make(P) - - module Map = Map.Make(P) + include P.Ints let of_repr r = Shostak.Arith.embed r - let to_repr p = Shostak.Arith.is_mine p - - let of_bigint n = P.create [] (Q.of_bigint n) Tint - - let (~$$) = of_bigint - - let zero = ~$$Z.zero - - let (~-) = P.mult_const Q.minus_one - - let ( + ) = P.add - - let ( - ) = P.sub - - let ( +$$ ) p n = P.add_const (Q.of_bigint n) p - - let ( *$$ ) p n = P.mult_const (Q.of_bigint n) p - let mkv_eq p p' = Uf.LX.mkv_eq (to_repr p) (to_repr p') - let mkv_le p p' = Uf.LX.mkv_builtin true LE [to_repr p; to_repr p'] end @@ -215,7 +194,7 @@ module BV2Nat = struct We maintain the invariant that if [(bv, ext)] maps to [(p, ex)] in [bv2nat], then [p] maps to [(bv, ext, ex)] in [nat2bv] below. *) - ; nat2bv : (X.r * Extraction.t * Ex.t) T.Ints.Map.t + ; nat2bv : (X.r * Extraction.t * Ex.t) P.Map.t (** Map from integer polynomials to their bit-vector representation. If [p] is associated with [(bv, ext, ex)] in this map, it means that @@ -224,7 +203,7 @@ module BV2Nat = struct Note that this is the inverse of the [bv2nat] mapping, not an implementation of the [int2bv] function (although both coincide for entries in the map). *) - ; use : T.Ints.Set.t MX.t + ; use : P.Set.t MX.t (** Map from integer variables to the polynomials that use them. This is used for integer substitutions because of our encoding of @@ -256,7 +235,7 @@ module BV2Nat = struct let empty = { bv2nat = MX.empty - ; nat2bv = T.Ints.Map.empty + ; nat2bv = P.Map.empty ; use = MX.empty ; eqs = [] } @@ -270,9 +249,9 @@ module BV2Nat = struct match MX.find x t.use with | exception Not_found -> acc | nps -> - T.Ints.Set.fold (fun p acc -> + P.Set.fold (fun p acc -> let v = - try T.Ints.Map.find p t.nat2bv with Not_found -> assert false + try P.Map.find p t.nat2bv with Not_found -> assert false in f p v acc ) nps acc @@ -280,8 +259,8 @@ module BV2Nat = struct let add_use_p p use = List.fold_left (fun use r -> MX.update r (function - | None -> Some (T.Ints.Set.singleton p) - | Some sp -> Some (T.Ints.Set.add p sp) + | None -> Some (P.Set.singleton p) + | Some sp -> Some (P.Set.add p sp) ) use ) use (P.leaves p) @@ -290,8 +269,8 @@ module BV2Nat = struct MX.update r (function | None -> assert false | Some ps -> - let ps = T.Ints.Set.remove p ps in - if T.Ints.Set.is_empty ps then None else Some ps + let ps = P.Set.remove p ps in + if P.Set.is_empty ps then None else Some ps ) use ) use (P.leaves p) @@ -306,7 +285,7 @@ module BV2Nat = struct let remove_aux bv ext p t = let use = remove_use_p p t.use in - let nat2bv = T.Ints.Map.remove p t.nat2bv in + let nat2bv = P.Map.remove p t.nat2bv in let bv2nat = MX.update bv (function | None -> None @@ -334,7 +313,7 @@ module BV2Nat = struct let k = T.Ints.of_repr @@ X.term_embed @@ E.fresh_name Tint in let use = add_use_p k t.use in let bv2nat = add_ext ~ex:Ex.empty bv ext k t.bv2nat in - let nat2bv = T.Ints.Map.add k (bv, ext, Ex.empty) t.nat2bv in + let nat2bv = P.Map.add k (bv, ext, Ex.empty) t.nat2bv in (k, Ex.empty), { use ; bv2nat ; nat2bv ; eqs = t.eqs } (* Add the definining equation for [bv2nat(bv asr ofs)] to [eqs]. @@ -375,7 +354,7 @@ module BV2Nat = struct try find_ext bv ext t.bv2nat, t with Not_found -> init_ext bv ext t let find_p p t = - T.Ints.Map.find p t.nat2bv + P.Map.find p t.nat2bv (* Add the mapping [bv2nat(bv) -> p]. If there is already a polynomial associated with [bv] or a bit-vector associated with [p], the @@ -385,7 +364,7 @@ module BV2Nat = struct reuse this function for substitutions. *) let add ~ex bv ext p t = let t = - match T.Ints.Map.find p t.nat2bv with + match P.Map.find p t.nat2bv with | exception Not_found -> t | (bv', ext', ex') -> let lit = T.BV.mkv_eq (bv, ext) (bv', ext') in @@ -415,7 +394,7 @@ module BV2Nat = struct | None -> let use = add_use_p p t.use in let bv2nat = add_ext ~ex bv ext p t.bv2nat in - let nat2bv = T.Ints.Map.add p (bv, ext, ex) t.nat2bv in + let nat2bv = P.Map.add p (bv, ext, ex) t.nat2bv in let t = { use ; bv2nat ; nat2bv ; eqs = t.eqs } in if is_new then let (p', ex'), t = init_ext bv ext t in diff --git a/src/lib/reasoners/polynome.ml b/src/lib/reasoners/polynome.ml index a189b0971..e888514dc 100644 --- a/src/lib/reasoners/polynome.ml +++ b/src/lib/reasoners/polynome.ml @@ -73,6 +73,20 @@ module type T = sig val abstract_selectors : t -> (r * r) list -> t * (r * r) list val separate_constant : t -> t * Numbers.Q.t + + module Set : Set.S with type elt = t + module Map : Map.S with type key = t + + module Ints : sig + val of_bigint : Z.t -> t + val zero : t + val (~-) : t -> t + val (+) : t -> t -> t + val (-) : t -> t -> t + val (~$$) : Z.t -> t + val (+$$) : t -> Z.t -> t + val ( *$$ ) : t -> Z.t -> t + end end module type EXTENDED_Polynome = sig @@ -351,4 +365,19 @@ module Make (X : S) = struct let separate_constant t = { t with c = Q.zero}, t.c + (* Operators and helpers for readability *) + + module Set = Set.Make(struct type nonrec t = t let compare = compare end) + module Map = Map.Make(struct type nonrec t = t let compare = compare end) + + module Ints = struct + let of_bigint n = create [] (Q.from_z n) Tint + let (~$$) = of_bigint + let zero = ~$$Z.zero + let (~-) = mult_const Q.m_one + let (+) = add + let (-) = sub + let (+$$) p n = add_const (Q.from_z n) p + let ( *$$ ) p n = mult_const (Q.from_z n) p + end end diff --git a/src/lib/reasoners/polynome.mli b/src/lib/reasoners/polynome.mli index fdcfff5e6..c318b28de 100644 --- a/src/lib/reasoners/polynome.mli +++ b/src/lib/reasoners/polynome.mli @@ -78,6 +78,42 @@ module type T = sig val abstract_selectors : t -> (r * r) list -> t * (r * r) list val separate_constant : t -> t * Numbers.Q.t + + module Set : Set.S with type elt = t + module Map : Map.S with type key = t + + module Ints : sig + (** Helper functions for manipulating polynomials of type [Tint]. *) + + val of_bigint : Z.t -> t + (** Conversion from [Z.t]. *) + + val zero : t + (** The constant [0]. *) + + (** {2 Prefix and infix operators} *) + + val (~-) : t -> t + (** Negation. *) + + val (+) : t -> t -> t + (** Addition [add]. *) + + val (-) : t -> t -> t + (** Subtraction [sub]. *) + + val (~$$) : Z.t -> t + (** Conversion from [Z.t]. *) + + val (+$$) : t -> Z.t -> t + (** Addition with a constant. + + {b Note}: [p +$$ n] is equivalent to [p + ~$$n], but might be more + efficient. *) + + val ( *$$ ) : t -> Z.t -> t + (** Multiplication with a constant. *) + end end module type EXTENDED_Polynome = sig @@ -87,4 +123,3 @@ module type EXTENDED_Polynome = sig end module Make (X : S) : T with type r = X.r -