[sledge] Strengthen quantifier witnessing

Summary:
Strengthen existential quantifier witnessing to enable witnessing an
existential with a term containing another existential if no universal
witness is available. Additionally, strengthen existential witnessing
to enable terms of interpreted theories to witness existential
variables.

Also strengthen and simplify the representation invariant checking for
existential witnessing code.

Reviewed By: jvillard

Differential Revision: D20120271

fbshipit-source-id: 4c44fe9ef
master
Josh Berdine 5 years ago committed by Facebook Github Bot
parent ae8cd953f8
commit 0b88d99c79

@ -42,6 +42,7 @@ module Subst : sig
val is_empty : t -> bool val is_empty : t -> bool
val length : t -> int val length : t -> int
val mem : t -> Term.t -> bool val mem : t -> Term.t -> bool
val find : t -> Term.t -> Term.t option
val fold : t -> init:'a -> f:(key:Term.t -> data:Term.t -> 'a -> 'a) -> 'a val fold : t -> init:'a -> f:(key:Term.t -> data:Term.t -> 'a -> 'a) -> 'a
val iteri : t -> f:(key:Term.t -> data:Term.t -> unit) -> unit val iteri : t -> f:(key:Term.t -> data:Term.t -> unit) -> unit
val for_alli : t -> f:(key:Term.t -> data:Term.t -> bool) -> bool val for_alli : t -> f:(key:Term.t -> data:Term.t -> bool) -> bool
@ -65,6 +66,7 @@ end = struct
let is_empty = Map.is_empty let is_empty = Map.is_empty
let length = Map.length let length = Map.length
let mem = Map.mem let mem = Map.mem
let find = Map.find
let fold = Map.fold let fold = Map.fold
let iteri = Map.iteri let iteri = Map.iteri
let for_alli = Map.for_alli let for_alli = Map.for_alli
@ -636,6 +638,20 @@ let ppx_classes_diff x fs (r, s) =
(** Existential Witnessing and Elimination *) (** Existential Witnessing and Elimination *)
let subst_invariant us s0 s =
assert (s0 == s || not (Subst.equal s0 s)) ;
assert (
Subst.iteri s ~f:(fun ~key ~data ->
(* dom of new entries not ito us *)
assert (
Option.for_all ~f:(Term.equal data) (Subst.find s0 key)
|| not (Set.is_subset (Term.fv key) ~of_:us) ) ;
(* rep not ito us implies trm not ito us *)
assert (
Set.is_subset (Term.fv data) ~of_:us
|| not (Set.is_subset (Term.fv key) ~of_:us) ) ) ;
true )
type 'a zom = Zero | One of 'a | Many type 'a zom = Zero | One of 'a | Many
(* try to solve [p = q] such that [fv (p - q) ⊆ us xs] and [p - q] has at (* try to solve [p = q] such that [fv (p - q) ⊆ us xs] and [p - q] has at
@ -680,12 +696,7 @@ let solve_interp_eq us e' (cls, subst) =
|> |>
[%Trace.retn fun {pf} subst' -> [%Trace.retn fun {pf} subst' ->
pf "@[%a@]" Subst.pp_diff (subst, Option.value subst' ~default:subst) ; pf "@[%a@]" Subst.pp_diff (subst, Option.value subst' ~default:subst) ;
Option.iter subst' ~f:(fun subst' -> Option.iter ~f:(subst_invariant us subst) subst']
Subst.iteri subst' ~f:(fun ~key ~data ->
assert (
Subst.mem subst key
|| not (Set.is_subset (Term.fv key) ~of_:us) ) ;
assert (Set.is_subset (Term.fv data) ~of_:us) ) )]
(* move equations from [cls] to [subst] which are between [Interpreted] (* move equations from [cls] to [subst] which are between [Interpreted]
terms and can be expressed, after normalizing with [subst], as [x u] terms and can be expressed, after normalizing with [subst], as [x u]
@ -713,50 +724,88 @@ let rec solve_interp_eqs us (cls, subst) =
pf "cls: @[%a@]@ subst: @[%a@]" pp_diff_cls (cls, cls') Subst.pp_diff pf "cls: @[%a@]@ subst: @[%a@]" pp_diff_cls (cls, cls') Subst.pp_diff
(subst, subst')] (subst, subst')]
(* move equations from [cls] (which is assumed to be normalized by [subst]) type cls_solve_state =
to [subst] which are between non-[Interpreted] terms and can be expressed { rep_us: Term.t option (** rep, that is ito us, for class *)
as [x u] where [us xs fv x us fv u] *) ; cls_us: Term.t list (** cls that is ito us, or interpreted *)
; rep_xs: Term.t option (** rep, that is *not* ito us, for class *)
; cls_xs: Term.t list (** cls that is *not* ito us *) }
let dom_trm e =
match (e : Term.t) with
| Ap2 (Memory, _, (Var _ as v)) -> Some v
| _ when non_interpreted e -> Some e
| _ -> None
(** move equations from [cls] (which is assumed to be normalized by [subst])
to [subst] which can be expressed as [x u] where [x] is
non-interpreted [us xs fv x us] and [fv u us] or else
[fv u us xs] *)
let solve_uninterp_eqs us (cls, subst) = let solve_uninterp_eqs us (cls, subst) =
[%Trace.call fun {pf} -> [%Trace.call fun {pf} ->
pf "cls: @[%a@]@ subst: @[%a@]" pp_cls cls Subst.pp subst] pf "cls: @[%a@]@ subst: @[%a@]" pp_cls cls Subst.pp subst]
; ;
let rep_ito_us, cls_not_ito_us, cls_delay = let compare e f =
List.fold cls ~init:(None, [], []) [%compare: kind * Term.t] (classify e, e) (classify f, f)
~f:(fun (rep_ito_us, cls_not_ito_us, cls_delay) trm -> in
if non_interpreted trm then let {rep_us; cls_us; rep_xs; cls_xs} =
if Set.is_subset (Term.fv trm) ~of_:us then List.fold cls ~init:{rep_us= None; cls_us= []; rep_xs= None; cls_xs= []}
match rep_ito_us with ~f:(fun ({rep_us; cls_us; rep_xs; cls_xs} as s) trm ->
| Some rep when Term.compare rep trm <= 0 -> if Set.is_subset (Term.fv trm) ~of_:us then
(rep_ito_us, cls_not_ito_us, trm :: cls_delay) match rep_us with
| Some rep -> (Some trm, cls_not_ito_us, rep :: cls_delay) | Some rep when compare rep trm <= 0 ->
| None -> (Some trm, cls_not_ito_us, cls_delay) {s with cls_us= trm :: cls_us}
else (rep_ito_us, trm :: cls_not_ito_us, cls_delay) | Some rep -> {s with rep_us= Some trm; cls_us= rep :: cls_us}
else (rep_ito_us, cls_not_ito_us, trm :: cls_delay) ) | None -> {s with rep_us= Some trm}
else
match rep_xs with
| Some rep -> (
if compare rep trm <= 0 then
match dom_trm trm with
| Some trm -> {s with cls_xs= trm :: cls_xs}
| None -> {s with cls_us= trm :: cls_us}
else
match dom_trm rep with
| Some rep ->
{s with rep_xs= Some trm; cls_xs= rep :: cls_xs}
| None -> {s with rep_xs= Some trm; cls_us= rep :: cls_us} )
| None -> {s with rep_xs= Some trm} )
in in
( match rep_ito_us with ( match rep_us with
| None -> (cls, subst) | Some rep_us ->
| Some rep_ito_us -> let cls = rep_us :: cls_us in
let cls = let cls, cls_xs =
if List.is_empty cls_delay then [] else rep_ito_us :: cls_delay match rep_xs with
| Some rep -> (
match dom_trm rep with
| Some rep -> (cls, rep :: cls_xs)
| None -> (rep :: cls, cls_xs) )
| None -> (cls, cls_xs)
in in
let subst = let subst =
List.fold cls_not_ito_us ~init:subst ~f:(fun subst trm_not_ito_us -> List.fold cls_xs ~init:subst ~f:(fun subst trm_xs ->
Subst.compose1 ~key:trm_not_ito_us ~data:rep_ito_us subst ) Subst.compose1 ~key:trm_xs ~data:rep_us subst )
in in
(cls, subst) ) (cls, subst)
| None -> (
match rep_xs with
| Some rep_xs ->
let cls = rep_xs :: cls_us in
let subst =
List.fold cls_xs ~init:subst ~f:(fun subst trm_xs ->
Subst.compose1 ~key:trm_xs ~data:rep_xs subst )
in
(cls, subst)
| None -> (cls, subst) ) )
|> |>
[%Trace.retn fun {pf} (cls', subst') -> [%Trace.retn fun {pf} (cls', subst') ->
pf "cls: @[%a@]@ subst: @[%a@]" pp_diff_cls (cls, cls') Subst.pp_diff pf "cls: @[%a@]@ subst: @[%a@]" pp_diff_cls (cls, cls') Subst.pp_diff
(subst, subst') ; (subst, subst') ;
Subst.iteri subst' ~f:(fun ~key ~data -> subst_invariant us subst subst']
assert (
Subst.mem subst key || not (Set.is_subset (Term.fv key) ~of_:us)
) ;
assert (Set.is_subset (Term.fv data) ~of_:us) )]
(* move equations between terms in [rep]'s class [cls] from [classes] to (** move equations between terms in [rep]'s class [cls] from [classes] to
[subst] which can be expressed, after normalizing with [subst], as [x [subst] which can be expressed, after normalizing with [subst], as
u] where [us xs fv x us fv u] *) [x u] where [us xs fv x us] and [fv u us] or else
[fv u us xs] *)
let solve_class us us_xs ~key:rep ~data:cls (classes, subst) = let solve_class us us_xs ~key:rep ~data:cls (classes, subst) =
let classes0 = classes in let classes0 = classes in
[%Trace.call fun {pf} -> [%Trace.call fun {pf} ->
@ -764,9 +813,11 @@ let solve_class us us_xs ~key:rep ~data:cls (classes, subst) =
Subst.pp subst] Subst.pp subst]
; ;
let cls, cls_not_ito_us_xs = let cls, cls_not_ito_us_xs =
List.partition_tf ~f:(fun e -> Set.is_subset (Term.fv e) ~of_:us_xs) cls List.partition_tf
~f:(fun e -> Set.is_subset (Term.fv e) ~of_:us_xs)
(rep :: cls)
in in
let cls, subst = solve_interp_eqs us (rep :: cls, subst) in let cls, subst = solve_interp_eqs us (cls, subst) in
let cls, subst = solve_uninterp_eqs us (cls, subst) in let cls, subst = solve_uninterp_eqs us (cls, subst) in
let cls = List.rev_append cls_not_ito_us_xs cls in let cls = List.rev_append cls_not_ito_us_xs cls in
let cls = let cls =
@ -843,10 +894,12 @@ let solve_for_xs r us xs (classes, subst, us_xs) =
if Subst.mem subst x then (classes, subst, us_xs) if Subst.mem subst x then (classes, subst, us_xs)
else solve_concat_extracts r us x (classes, subst, us_xs) ) else solve_concat_extracts r us x (classes, subst, us_xs) )
(* move equations from [classes] to [subst] which can be expressed, after (** move equations from [classes] to [subst] which can be expressed, after
normalizing with [subst], as [x u] where [us xs fv x us fv u] *) normalizing with [subst], as [x u] where [us xs fv x us]
and [fv u us] or else [fv u us xs]. *)
let solve_classes r (classes, subst, us) xs = let solve_classes r (classes, subst, us) xs =
[%Trace.call fun {pf} -> pf "xs: {@[%a@]}" Var.Set.pp xs] [%Trace.call fun {pf} ->
pf "us: {@[%a@]}@ xs: {@[%a@]}" Var.Set.pp us Var.Set.pp xs]
; ;
let rec solve_classes_ (classes0, subst0, us_xs) = let rec solve_classes_ (classes0, subst0, us_xs) =
let classes, subst = let classes, subst =
@ -867,15 +920,18 @@ let pp_vss fs vss =
(List.pp ";@ " (fun fs vs -> Format.fprintf fs "{@[%a@]}" Var.Set.pp vs)) (List.pp ";@ " (fun fs vs -> Format.fprintf fs "{@[%a@]}" Var.Set.pp vs))
vss vss
(* enumerate variable contexts vᵢ in [v₁;…] and accumulate a solution subst (** enumerate variable contexts vᵢ in [v₁;…] and accumulate a solution
with entries [x u] where [r] entails [x = u] and [ v fv x subst with entries [x u] where [r] entails [x = u] and
¹ v fv u] *) [ v fv x ¹ v] and
[fv u ¹ v] if possible and otherwise
[fv u v] *)
let solve_for_vars vss r = let solve_for_vars vss r =
[%Trace.call fun {pf} -> pf "%a@ @[%a@]" pp_vss vss pp_classes r] [%Trace.call fun {pf} -> pf "%a@ @[%a@]" pp_vss vss pp_classes r]
; ;
List.fold ~f:(solve_classes r) let us, vss =
~init:(classes r, Subst.empty, Var.Set.empty) match vss with us :: vss -> (us, vss) | [] -> (Var.Set.empty, vss)
vss in
List.fold ~f:(solve_classes r) ~init:(classes r, Subst.empty, us) vss
|> snd3 |> snd3
|> |>
[%Trace.retn fun {pf} subst -> [%Trace.retn fun {pf} subst ->
@ -886,11 +942,16 @@ let solve_for_vars vss r =
|| fail "@[%a = %a not entailed by@ %a@]" Term.pp key Term.pp data || fail "@[%a = %a not entailed by@ %a@]" Term.pp key Term.pp data
pp_classes r () ) ; pp_classes r () ) ;
assert ( assert (
List.exists vss ~f:(fun vs -> List.fold_until vss ~init:us
match ~f:(fun us xs ->
( Set.is_subset (Term.fv key) ~of_:vs let us_xs = Set.union us xs in
, Set.is_subset (Term.fv data) ~of_:vs ) let ks = Term.fv key in
with let ds = Term.fv data in
| false, true -> true if
| true, false -> assert false Set.is_subset ks ~of_:us_xs
| _ -> false ) ) )] && Set.is_subset ds ~of_:us_xs
&& ( Set.is_subset ds ~of_:us
|| not (Set.is_subset ks ~of_:us) )
then Stop true
else Continue us_xs )
~finish:(fun _ -> false) ) )]

@ -82,7 +82,8 @@ module Subst : sig
end end
val solve_for_vars : Var.Set.t list -> t -> Subst.t val solve_for_vars : Var.Set.t list -> t -> Subst.t
(** [solve_for_vars \[v₁;…\] r] is a solution substitution that is (** [solve_for_vars vss r] is a solution substitution that is entailed by
entailed by [r] and consists of oriented equalities [x u] such that [r] and consists of oriented equalities [x e] that map terms [x]
[fv x v fv u] where [i] is minimal such that [v] with free variables contained in (the union of) a prefix [uss] of [vss]
distinguishes [fv x] and [fv u], if one exists. *) to terms [e] with free variables contained in as short a prefix of [uss]
as possible. *)

@ -49,7 +49,10 @@ let excise_exists goal =
if Set.is_empty goal.xs then goal if Set.is_empty goal.xs then goal
else else
let solutions_for_xs = let solutions_for_xs =
Equality.solve_for_vars [goal.us; goal.xs] goal.sub.cong let xs =
Set.diff goal.xs (Sh.fv ~ignore_cong:() (Sh.with_pure [] goal.sub))
in
Equality.solve_for_vars [Var.Set.empty; goal.us; xs] goal.sub.cong
in in
if Equality.Subst.is_empty solutions_for_xs then goal if Equality.Subst.is_empty solutions_for_xs then goal
else else

@ -69,7 +69,7 @@ let%test_module _ =
[%expect [%expect
{| {|
( infer_frame: emp \- %m_8, %n_9 . %m_8 = %n_9 emp ( infer_frame: emp \- %m_8, %n_9 . %m_8 = %n_9 emp
) infer_frame: emp |}] ) infer_frame: %m_8 = %n_9 emp |}]
let%expect_test _ = let%expect_test _ =
check_frame check_frame
@ -109,7 +109,8 @@ let%test_module _ =
%l_6 -[ %b_4, 10 )-> 10,%a_1 * %l_7 -[ %b_4, 10 )-> 10,%a_2 %l_6 -[ %b_4, 10 )-> 10,%a_1 * %l_7 -[ %b_4, 10 )-> 10,%a_2
\- %m_8, %n_9 . \- %m_8, %n_9 .
%m_10 . %m_8 = %n_9 %l_7 -[ %b_4, 10 )-> 10,%a_2 %m_10 . %m_8 = %n_9 %l_7 -[ %b_4, 10 )-> 10,%a_2
) infer_frame: %m_10 . %l_6 -[ %b_4, 10 )-> 10,%a_1 |}] ) infer_frame:
%m_10 . %m_8 = %n_9 %l_6 -[ %b_4, 10 )-> 10,%a_1 |}]
let%expect_test _ = let%expect_test _ =
check_frame check_frame
@ -272,4 +273,15 @@ let%test_module _ =
\- %a_1, %m_8 . \- %a_1, %m_8 .
%l_6 -[ %l_6, %m_8 )-> %m_8,%a_1 %l_6 -[ %l_6, %m_8 )-> %m_8,%a_1
) infer_frame: |}] ) infer_frame: |}]
(* Incompleteness: cannot witness existentials to satisfy non-equality
pure constraints *)
let%expect_test _ =
let subtrahend = Sh.and_ (Term.eq m a) (Sh.pure (Term.dq m !0)) in
let minuend = Sh.extend_us (Var.Set.of_ a_) Sh.emp in
infer_frame minuend [m_] subtrahend ;
[%expect
{|
( infer_frame: emp \- %m_8 . %a_1 = %m_8 (%a_1 0) emp
) infer_frame: |}]
end ) end )

Loading…
Cancel
Save