diff --git a/src/plugins/wp/Cint.ml b/src/plugins/wp/Cint.ml index dfea2eac9518ed9d743b7d65243e34a6bceb875b..650f7f466b48d0c1d9fb11e29327bf6648137aa1 100644 --- a/src/plugins/wp/Cint.ml +++ b/src/plugins/wp/Cint.ml @@ -1287,7 +1287,7 @@ let mask_simplifier = let find t dom = Tmap.find t dom - let get t dom = + let get dom t = let r = match F.repr t with | Kint set -> { set; unset=Integer.lognot set } @@ -1318,28 +1318,32 @@ let mask_simplifier = else Some v) t v ctx let neutral_land = {set=Integer.minus_one ; unset=Integer.zero } + let eval_land eval es = + List.fold_left (fun {set;unset} x -> + let v = eval x in + { set=Integer.logand v.set set ; + unset=Integer.logor v.unset unset}) neutral_land es + let neutral_lor = {set=Integer.zero ; unset=Integer.minus_one } + let eval_lor eval es = + List.fold_left (fun {set;unset} x -> + let v = eval x in + { set=Integer.logor v.set set ; + unset=Integer.logand v.unset unset}) neutral_lor es + let eval ctx e = (* non recursive evaluation *) let r = try match F.repr e with | Kint set -> { set; unset=Integer.lognot set } - | Fun(f,es) when f == f_land -> - List.fold_left (fun {set;unset} x -> - let v = get x ctx in - { set=Integer.logand v.set set ; - unset=Integer.logor v.unset unset}) neutral_land es - | Fun(f,es) when f == f_lor -> - List.fold_left (fun {set;unset} x -> - let v = get x ctx in - { set=Integer.logor v.set set ; - unset=Integer.logand v.unset unset}) neutral_lor es + | Fun(f,es) when f == f_land -> eval_land (get ctx) es + | Fun(f,es) when f == f_lor -> eval_lor (get ctx) es | Fun(f,[x]) when f == f_lnot -> - let v = get x ctx in + let v = get ctx x in { unset=v.set ; set=v.unset } | Fun(f,[x]) -> let iota = to_cint f in - let v = get x ctx in + let v = get ctx x in if not (Ctypes.signed iota) then (* The uppest bits are unset *) let mask = snd (Ctypes.bounds iota) in @@ -1434,13 +1438,13 @@ let mask_simplifier = | Eq(a,b) when is_int a && is_int b -> (* b may give a better constraint because it could be a constant *) let ctx = reduce ctx a (eval ctx b) in - reduce ctx b (get a ctx) + reduce ctx b (get ctx a) | _ -> ctx with Not_found -> ctx let eval ctx h = let ({set;unset} as v) = eval ctx h in - let r = try masks_narrow ~unset ~set (get h ctx) with Not_found -> v in + let r = try masks_narrow ~unset ~set (get ctx h) with Not_found -> v in Wp_parameters.debug ~dkey "Value %a@." pretty (h,r); r @@ -1463,22 +1467,75 @@ let mask_simplifier = x) es xs in if !modified then f xs else e in - let rewrite_cst ctx e = (* [r = rewrite ctx e] such that [ctx |- e = r] *) + let rewrite_cst ~highest ctx e = (* [r = rewrite ctx e] such that [ctx |- e = r] *) match F.repr e with | Kint _ -> e + | Fun(f,[x;k]) when highest && + f == f_bit_positive -> (* rewrites [bit_test(x,k)] *) + (try let k = match_integer k in (* may raise Not_found *) + let v = Masks.eval ctx x in + let m = two_power_k k in (* may raise Not_found *) + if Masks.is_bottom v then + (* Does not rewrite [e] because the polarity is unknown *) + e + else if not (Integer.is_zero (Integer.logand v.Masks.set m)) then + (* [ctx] gives that the bit [k] of [x] is set *) + e_true + else if not (Integer.is_zero (Integer.logand v.Masks.unset m)) then + (* [ctx] gives that the bit [k] of [x] is unset *) + e_false + else e + with _ -> e) + | Eq (a, b) when highest -> + (try + let b = match_integer b in (* may raise Not_found *) + match F.repr a with + | Fun(f,es) when f == f_land -> + (* [k & t == b] specifies some bits of [t] *) + let k,es = match_list_head match_integer es in (* may raise Not_found *) + if not (Integer.is_zero (Integer.logand b (Integer.lognot k))) then + (* [b] and [k] are such that the equality is false *) + e_false + else + let set = b (* the bits of [t] that have to be set *) + and unset = (* the bits of [t] that have to be unset *) + Integer.logand k (Integer.lognot b) + and v = (* the current bits of [t] *) + try Masks.find (F.e_fun f_land es) ctx + with Not_found -> + Masks.eval_land (fun i -> i) (List.map (Masks.eval ctx) es) + in + if Masks.is_bottom v then + (* Does not rewrite [e] because the polarity is unknown *) + e + else if not (Integer.is_zero (Integer.logand set v.Masks.unset)) then + (* Some bits of [t] that has to be set is unset *) + e_false + else if not (Integer.is_zero (Integer.logand unset v.Masks.set)) then + (* Some bits of [t] that has to be unset is set *) + e_false + else if (Integer.equal (Integer.logand set v.Masks.set) set) && + (Integer.equal (Integer.logand unset v.Masks.unset) unset) + then (* The bits of [t] that have to be set are set && + those that have to be unset are unset *) + e_true + else e + | _ -> e + with _ -> e) | _ when is_int e -> let open Masks in let v = eval ctx e in if Integer.equal v.set (Integer.lognot v.unset) then + (* [ctx] gives a value (0 or else 1) for all bits of [e] *) F.e_zint v.set else e | _ -> e in - let rec rewrite ctx e = (* [r = rewrite ctx e] such that [ctx |- e = r] *) + let rec rewrite ~highest ctx e = (* [r = rewrite ctx e] such that [ctx |- e = r] *) let e = match F.repr e with | Fun(f,xs) when f == f_land -> - let es = List.map (rewrite ctx) xs in + let es = List.map (rewrite ~highest ctx) xs in Wp_parameters.debug ~dkey "Rewrite AND %a@." Lang.F.pp_term e; let reduce unset x = match F.repr x with | Kint v -> F.e_zint (Integer.logand (Integer.lognot unset) v) @@ -1494,13 +1551,19 @@ let mask_simplifier = else if Integer.equal unset_mask Integer.minus_one then e_zero else nary_op e xs (F.e_fun f_land) (reduce unset_mask) es | Fun(f,xs) -> (try - nary_op e xs (F.e_fun ~result:(F.typeof e) f) (rewrite ctx) xs + nary_op e xs (F.e_fun ~result:(F.typeof e) f) (rewrite ~highest ctx) xs with _ -> e) - | Not a -> unary_op e e_not (rewrite ctx) a - | Eq (a, b) -> bin_op e e_eq (rewrite ctx) a b - | Leq (a, b) -> bin_op e e_leq (rewrite ctx) a b + | Not a -> unary_op e e_not (rewrite ~highest ctx) a + | And es -> nary_op e es e_and (rewrite ~highest ctx) es + | Or es -> nary_op e es e_or (rewrite ~highest ctx) es + | Imply(xs, b) -> + let l = b::xs in + nary_op e l (function b::xs -> e_imply xs b | _ -> assert false) (rewrite ~highest ctx) l + | Eq (a, b) -> bin_op e e_eq (rewrite ~highest ctx) a b + | Leq (a, b) -> bin_op e e_leq (rewrite ~highest ctx) a b + | Lt (a, b) -> bin_op e e_lt (rewrite ~highest ctx) a b | _ -> e - in rewrite_cst ctx e + in rewrite_cst ~highest ctx e in object @@ -1522,23 +1585,25 @@ let mask_simplifier = method equivalent_exp e = if Tmap.is_empty masks then e else (Wp_parameters.debug ~dkey "Rewrite Exp: %a@." Lang.F.pp_term e; - rewrite masks e) + rewrite ~highest:true masks e) method weaker_hyp p = if Tmap.is_empty masks then p else (Wp_parameters.debug ~dkey "Rewrite Hyp: %a@." Lang.F.pp_pred p; - F.p_bool (rewrite masks (F.e_prop p))) + F.p_bool (rewrite ~highest:false masks (F.e_prop p))) method equivalent_branch p = if Tmap.is_empty masks then p else (Wp_parameters.debug ~dkey "Rewrite Branch: %a@." Lang.F.pp_pred p; - F.p_bool (rewrite masks (F.e_prop p)) + (* Does not rewrite [hyp] as much as possible. + Any way, contradiction may be found later when [hyp] will be assumed *) + F.p_bool (rewrite ~highest:true masks (F.e_prop p)) ) method stronger_goal p = if Tmap.is_empty masks then p else (Wp_parameters.debug ~dkey "Rewrite Goal: %a@." Lang.F.pp_pred p; - F.p_bool (rewrite masks (F.e_prop p))) + F.p_bool (rewrite ~highest:true masks (F.e_prop p))) end diff --git a/src/plugins/wp/tests/wp_typed/oracle/user_bitwise.0.res.oracle b/src/plugins/wp/tests/wp_typed/oracle/user_bitwise.0.res.oracle index 6f871cd6fc4d62ac24e925302c564e5ac6df812a..c5466b451ca3497f5848c48e6ceedf92e864cbd2 100644 --- a/src/plugins/wp/tests/wp_typed/oracle/user_bitwise.0.res.oracle +++ b/src/plugins/wp/tests/wp_typed/oracle/user_bitwise.0.res.oracle @@ -93,8 +93,7 @@ Prove: (land(lor(to_uint64(lsl(x, n)), x_2), lsl(1, x_1)) != 0) <-> ------------------------------------------------------------ Goal Post-condition 'b0' in 'rr1': -Assume { Type: is_uint32(x) /\ is_uint32(lsr(x, 1)). } -Prove: (bit_test(x, 0) \/ bit_test(x, 32)) <-> bit_test(x, 0). +Prove: true. ------------------------------------------------------------ diff --git a/src/plugins/wp/tests/wp_typed/oracle_qualif/user_bitwise.0.res.oracle b/src/plugins/wp/tests/wp_typed/oracle_qualif/user_bitwise.0.res.oracle index adb01c5d3ab87c27ccdf34f5b7717cdf85f016b9..552cfc2b33328a43cb354572a3ffae67dcb5df63 100644 --- a/src/plugins/wp/tests/wp_typed/oracle_qualif/user_bitwise.0.res.oracle +++ b/src/plugins/wp/tests/wp_typed/oracle_qualif/user_bitwise.0.res.oracle @@ -5,7 +5,7 @@ [wp] 12 goals scheduled [wp] [Qed] Goal typed_ref_rl1_ensures_b0 : Valid [wp] [Alt-Ergo] Goal typed_ref_rl1_ensures_bk : Valid -[wp] [Alt-Ergo] Goal typed_ref_rr1_ensures_b0 : Valid +[wp] [Qed] Goal typed_ref_rr1_ensures_b0 : Valid [wp] [Alt-Ergo] Goal typed_ref_rr1_ensures_bk : Valid [wp] [Alt-Ergo] Goal typed_ref_rln32_ensures_b1 : Valid [wp] [Alt-Ergo] Goal typed_ref_rln32_ensures_b2 : Valid @@ -16,12 +16,12 @@ [wp] [Alt-Ergo] Goal typed_ref_rrn64_ensures_b1 : Valid [wp] [Alt-Ergo] Goal typed_ref_rrn64_ensures_b2 : Valid [wp] Proved goals: 12 / 12 - Qed: 1 - Alt-Ergo: 11 + Qed: 2 + Alt-Ergo: 10 ------------------------------------------------------------ Functions WP Alt-Ergo Total Success rl1 1 1 2 100% - rr1 - 2 2 100% + rr1 1 1 2 100% rln32 - 2 2 100% rrn32 - 2 2 100% rln64 - 2 2 100%