You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

909 lines
35 KiB

(*
* Copyright (c) 2009 - 2013 Monoidics ltd.
* Copyright (c) 2013 - present Facebook, Inc.
* All rights reserved.
*
* This source code is licensed under the BSD style license found in the
* LICENSE file in the root directory of this source tree. An additional grant
* of patent rights can be found in the PATENTS file in the same directory.
*)
open! IStd
(** Functions for "Smart" Pattern Matching *)
module L = Logging
module F = Format
let mem_idlist i l = List.exists ~f:(Ident.equal i) l
(** Type for a hpred pattern. flag=false means that the implication
between hpreds is not considered, and flag = true means that it is
considered during pattern matching *)
type hpred_pat = {hpred: Sil.hpred; flag: bool}
(** Checks e1 = e2[sub ++ sub'] for some sub' with dom(sub') subseteq vars.
Returns (sub ++ sub', vars - dom(sub')). *)
let rec exp_match e1 sub vars e2 : (Sil.exp_subst * Ident.t list) option =
let check_equal sub vars e1 e2 =
let e2_inst = Sil.exp_sub (`Exp sub) e2 in
if Exp.equal e1 e2_inst then Some (sub, vars) else None
in
match (e1, e2) with
| _, Exp.Var id2 when Ident.is_primed id2 && mem_idlist id2 vars ->
let vars_new = List.filter ~f:(fun id -> not (Ident.equal id id2)) vars in
let sub_new =
match Sil.extend_sub sub id2 e1 with
| None ->
assert false (* happens when vars contains the same variable twice. *)
| Some sub_new ->
sub_new
in
Some (sub_new, vars_new)
| _, Exp.Var _ ->
check_equal sub vars e1 e2
| Exp.Var _, _ ->
None
| Exp.Const _, _ | _, Exp.Const _ ->
check_equal sub vars e1 e2
| Exp.Sizeof _, _ | _, Exp.Sizeof _ ->
check_equal sub vars e1 e2
| Exp.Cast (_, e1'), Exp.Cast (_, e2') ->
(* we are currently ignoring cast *)
exp_match e1' sub vars e2'
| Exp.Cast _, _ | _, Exp.Cast _ ->
None
| Exp.UnOp (o1, e1', _), Exp.UnOp (o2, e2', _) when Unop.equal o1 o2 ->
exp_match e1' sub vars e2'
| Exp.UnOp _, _ | _, Exp.UnOp _ ->
None (* Naive *)
| Exp.BinOp (b1, e1', e1''), Exp.BinOp (b2, e2', e2'') when Binop.equal b1 b2 -> (
match exp_match e1' sub vars e2' with
| None ->
None
| Some (sub', vars') ->
exp_match e1'' sub' vars' e2'' )
| Exp.BinOp _, _ | _, Exp.BinOp _ ->
None (* Naive *)
| Exp.Exn _, _ | _, Exp.Exn _ ->
check_equal sub vars e1 e2
| Exp.Closure _, _ | _, Exp.Closure _ ->
check_equal sub vars e1 e2
| Exp.Lvar _, _ | _, Exp.Lvar _ ->
check_equal sub vars e1 e2
| Exp.Lfield (e1', fld1, _), Exp.Lfield (e2', fld2, _) when Typ.Fieldname.equal fld1 fld2 ->
exp_match e1' sub vars e2'
| Exp.Lfield _, _ | _, Exp.Lfield _ ->
None
| Exp.Lindex (base1, idx1), Exp.Lindex (base2, idx2) ->
match exp_match base1 sub vars base2 with
| None ->
None
| Some (sub', vars') ->
exp_match idx1 sub' vars' idx2
let exp_list_match es1 sub vars es2 =
let f res_acc (e1, e2) =
match res_acc with
| None ->
None
| Some (sub_acc, vars_leftover) ->
exp_match e1 sub_acc vars_leftover e2
in
Option.find_map
~f:(fun es_combined -> List.fold ~f ~init:(Some (sub, vars)) es_combined)
(List.zip es1 es2)
(** Checks sexp1 = sexp2[sub ++ sub'] for some sub' with
dom(sub') subseteq vars. Returns (sub ++ sub', vars - dom(sub')).
WARNING: This function does not consider the fact that the analyzer
sometimes forgets fields of hpred. It can possibly cause a problem. *)
let rec strexp_match sexp1 sub vars sexp2 : (Sil.exp_subst * Ident.t list) option =
match (sexp1, sexp2) with
| Sil.Eexp (exp1, _), Sil.Eexp (exp2, _) ->
exp_match exp1 sub vars exp2
| Sil.Eexp _, _ | _, Sil.Eexp _ ->
None
| Sil.Estruct (fsel1, _), Sil.Estruct (fsel2, _) ->
fsel_match fsel1 sub vars fsel2
| Sil.Estruct _, _ | _, Sil.Estruct _ ->
None
| Sil.Earray (len1, isel1, _), Sil.Earray (len2, isel2, _) ->
match exp_match len1 sub vars len2 with
| Some (sub', vars') ->
isel_match isel1 sub' vars' isel2
| None ->
None
(** Checks fsel1 = fsel2[sub ++ sub'] for some sub' with
dom(sub') subseteq vars. Returns (sub ++ sub', vars - dom(sub')). *)
and fsel_match fsel1 sub vars fsel2 =
match (fsel1, fsel2) with
| [], [] ->
Some (sub, vars)
| [], _ ->
None
| _, [] ->
if Config.abs_struct <= 0 then None else Some (sub, vars)
(* This can lead to great information loss *)
| (fld1, se1') :: fsel1', (fld2, se2') :: fsel2' ->
let n = Typ.Fieldname.compare fld1 fld2 in
if Int.equal n 0 then
match strexp_match se1' sub vars se2' with
| None ->
None
| Some (sub', vars') ->
fsel_match fsel1' sub' vars' fsel2'
else if n < 0 && Config.abs_struct > 0 then fsel_match fsel1' sub vars fsel2
(* This can lead to great information loss *)
else None
(** Checks isel1 = isel2[sub ++ sub'] for some sub' with
dom(sub') subseteq vars. Returns (sub ++ sub', vars - dom(sub')). *)
and isel_match isel1 sub vars isel2 =
match (isel1, isel2) with
| [], [] ->
Some (sub, vars)
| [], _ | _, [] ->
None
| (idx1, se1') :: isel1', (idx2, se2') :: isel2' ->
let idx2 = Sil.exp_sub (`Exp sub) idx2 in
let sanity_check = not (List.exists ~f:(fun id -> Sil.ident_in_exp id idx2) vars) in
if not sanity_check then
let pe = Pp.text in
L.internal_error "@[.... Sanity Check Failure while Matching Index-Strexps ....@\n" ;
L.internal_error "@[<4> IDX1: %a, STREXP1: %a@\n" (Sil.pp_exp_printenv pe) idx1
(Sil.pp_sexp pe) se1' ;
L.internal_error "@[<4> IDX2: %a, STREXP2: %a@\n@." (Sil.pp_exp_printenv pe) idx2
(Sil.pp_sexp pe) se2' ;
assert false
else if Exp.equal idx1 idx2 then
match strexp_match se1' sub vars se2' with
| None ->
None
| Some (sub', vars') ->
isel_match isel1' sub' vars' isel2'
else None
(* extends substitution sub by creating a new substitution for vars *)
let sub_extend_with_ren (sub: Sil.exp_subst) vars =
let f id = (id, Exp.Var (Ident.create_fresh Ident.kprimed)) in
let renaming_for_vars = Sil.exp_subst_of_list (List.map ~f vars) in
Sil.sub_join sub renaming_for_vars
type sidecondition = Prop.normal Prop.t -> Sil.exp_subst -> bool
let rec execute_with_backtracking = function
| [] ->
None
| [f] ->
f ()
| f :: fs ->
let res_f = f () in
match res_f with None -> execute_with_backtracking fs | Some _ -> res_f
let rec instantiate_to_emp p condition (sub: Sil.exp_subst) vars = function
| [] ->
if condition p sub then Some (sub, p) else None
| hpat :: hpats ->
if not hpat.flag then None
else
match hpat.hpred with
| Sil.Hpointsto _
| Sil.Hlseg (Sil.Lseg_NE, _, _, _, _)
| Sil.Hdllseg (Sil.Lseg_NE, _, _, _, _, _, _) ->
None
| Sil.Hlseg (_, _, e1, e2, _)
-> (
let fully_instantiated =
not (List.exists ~f:(fun id -> Sil.ident_in_exp id e1) vars)
in
if not fully_instantiated then None
else
let e1' = Sil.exp_sub (`Exp sub) e1 in
match exp_match e1' sub vars e2 with
| None ->
None
| Some (sub_new, vars_leftover) ->
instantiate_to_emp p condition sub_new vars_leftover hpats )
| Sil.Hdllseg (_, _, iF, oB, oF, iB, _) ->
let fully_instantiated =
not
(List.exists ~f:(fun id -> Sil.ident_in_exp id iF || Sil.ident_in_exp id oB) vars)
in
if not fully_instantiated then None
else
let iF' = Sil.exp_sub (`Exp sub) iF in
let oB' = Sil.exp_sub (`Exp sub) oB in
match exp_list_match [iF'; oB'] sub vars [oF; iB] with
| None ->
None
| Some (sub_new, vars_leftover) ->
instantiate_to_emp p condition sub_new vars_leftover hpats
(* This function has to be changed in order to
* implement the idea "All lsegs outside are NE, and all lsegs inside
* are PE" *)
let rec iter_match_with_impl tenv iter condition sub vars hpat hpats =
(*
L.out "@[.... iter_match_with_impl ....@.";
L.out "@[<4> sub: %a@\n@." pp_sub sub;
L.out "@[<4> PROP: %a@\n@." pp_prop (Prop.prop_iter_to_prop iter);
L.out "@[<4> hpred: %a@\n@." pp_hpat hpat;
L.out "@[<4> hpred_rest: %a@\n@." pp_hpat_list hpats;
*)
let do_next iter_cur _ =
match Prop.prop_iter_next iter_cur with
| None ->
None
| Some iter_next ->
iter_match_with_impl tenv iter_next condition sub vars hpat hpats
in
let do_empty_hpats iter_cur _ =
let sub_new, vars_leftover =
match Prop.prop_iter_current tenv iter_cur with _, (sub_new, vars_leftover) ->
(sub_new, vars_leftover)
in
let sub_res = sub_extend_with_ren sub_new vars_leftover in
let p_leftover = Prop.prop_iter_remove_curr_then_to_prop tenv iter_cur in
(*
L.out "@[.... iter_match_with_impl (final condtion check) ....@\n@.";
L.out "@[<4> sub_res : %a@\n@." pp_sub sub_res;
L.out "@[<4> p_leftover : %a@\n@." pp_prop p_leftover;
*)
if condition p_leftover sub_res then Some (sub_res, p_leftover) else None
in
let do_nonempty_hpats iter_cur _ =
let sub_new, vars_leftover =
match Prop.prop_iter_current tenv iter_cur with _, (sub_new, vars_leftover) ->
(sub_new, vars_leftover)
in
let hpat_next, hpats_rest =
match hpats with [] -> assert false | hpat_next :: hpats_rest -> (hpat_next, hpats_rest)
in
let p_rest = Prop.prop_iter_remove_curr_then_to_prop tenv iter_cur in
prop_match_with_impl_sub tenv p_rest condition sub_new vars_leftover hpat_next hpats_rest
in
let gen_filter_pointsto lexp2 strexp2 te2 = function
| Sil.Hpointsto (lexp1, strexp1, te1) when Exp.equal te1 te2 -> (
match exp_match lexp1 sub vars lexp2 with
| None ->
None
| Some (sub', vars_leftover) ->
strexp_match strexp1 sub' vars_leftover strexp2 )
| _ ->
None
in
let gen_filter_lseg k2 para2 e_start2 e_end2 es_shared2 = function
| Sil.Hpointsto _ ->
None
| Sil.Hlseg (k1, para1, e_start1, e_end1, es_shared1) ->
let do_kinds_match =
match (k1, k2) with
| Sil.Lseg_NE, Sil.Lseg_NE | Sil.Lseg_NE, Sil.Lseg_PE | Sil.Lseg_PE, Sil.Lseg_PE ->
true
| Sil.Lseg_PE, Sil.Lseg_NE ->
false
in
(* let do_paras_match = hpara_match_with_impl tenv hpat.flag para1 para2 *)
let do_paras_match = hpara_match_with_impl tenv true para1 para2 in
if not (do_kinds_match && do_paras_match) then None
else
let es1 = [e_start1; e_end1] @ es_shared1 in
let es2 = [e_start2; e_end2] @ es_shared2 in
exp_list_match es1 sub vars es2
| Sil.Hdllseg _ ->
None
in
let gen_filter_dllseg k2 para2 iF2 oB2 oF2 iB2 es_shared2 = function
| Sil.Hpointsto _ | Sil.Hlseg _ ->
None
| Sil.Hdllseg (k1, para1, iF1, oB1, oF1, iB1, es_shared1) ->
let do_kinds_match =
match (k1, k2) with
| Sil.Lseg_NE, Sil.Lseg_NE | Sil.Lseg_NE, Sil.Lseg_PE | Sil.Lseg_PE, Sil.Lseg_PE ->
true
| Sil.Lseg_PE, Sil.Lseg_NE ->
false
in
(* let do_paras_match = hpara_dll_match_with_impl tenv hpat.flag para1 para2 *)
let do_paras_match = hpara_dll_match_with_impl tenv true para1 para2 in
if not (do_kinds_match && do_paras_match) then None
else
let es1 = [iF1; oB1; oF1; iB1] @ es_shared1 in
let es2 = [iF2; oB2; oF2; iB2] @ es_shared2 in
exp_list_match es1 sub vars es2
in
match hpat.hpred with
| Sil.Hpointsto (lexp2, strexp2, te2)
-> (
let filter = gen_filter_pointsto lexp2 strexp2 te2 in
match (Prop.prop_iter_find iter filter, hpats) with
| None, _ ->
None
| Some iter_cur, [] ->
do_empty_hpats iter_cur ()
| Some iter_cur, _ ->
execute_with_backtracking [do_nonempty_hpats iter_cur; do_next iter_cur] )
| Sil.Hlseg (k2, para2, e_start2, e_end2, es_shared2)
-> (
let filter = gen_filter_lseg k2 para2 e_start2 e_end2 es_shared2 in
let do_emp_lseg _ =
let fully_instantiated_start2 =
not (List.exists ~f:(fun id -> Sil.ident_in_exp id e_start2) vars)
in
if not fully_instantiated_start2 then None
else
let e_start2' = Sil.exp_sub (`Exp sub) e_start2 in
match (exp_match e_start2' sub vars e_end2, hpats) with
| None, _ ->
(*
L.out "@.... iter_match_with_impl (empty_case, fail) ....@\n@.";
L.out "@[<4> sub: %a@\n@." pp_sub sub;
L.out "@[<4> e_start2': %a@\n@." pp_exp e_start2';
L.out "@[<4> e_end2: %a@\n@." pp_exp e_end2;
*)
None
| Some (sub_new, vars_leftover), [] ->
let sub_res = sub_extend_with_ren sub_new vars_leftover in
let p_leftover = Prop.prop_iter_to_prop tenv iter in
if condition p_leftover sub_res then Some (sub_res, p_leftover) else None
| Some (sub_new, vars_leftover), hpat_next :: hpats_rest ->
let p = Prop.prop_iter_to_prop tenv iter in
prop_match_with_impl_sub tenv p condition sub_new vars_leftover hpat_next hpats_rest
in
let do_para_lseg _ =
let para2_exist_vars, para2_inst =
Sil.hpara_instantiate para2 e_start2 e_end2 es_shared2
in
(* let allow_impl hpred = {hpred=hpred; flag=hpat.flag} in *)
let allow_impl hpred = {hpred; flag= true} in
let para2_hpat, para2_hpats =
match List.map ~f:allow_impl para2_inst with
| [] ->
assert false (* the body of a parameter should contain at least one * conjunct *)
| para2_pat :: para2_pats ->
(para2_pat, para2_pats)
in
let new_vars = para2_exist_vars @ vars in
let new_hpats = para2_hpats @ hpats in
match iter_match_with_impl tenv iter condition sub new_vars para2_hpat new_hpats with
| None ->
None
| Some (sub_res, p_leftover) when condition p_leftover sub_res ->
let not_in_para2_exist_vars id =
not (List.exists ~f:(fun id' -> Ident.equal id id') para2_exist_vars)
in
let sub_res' = Sil.sub_filter not_in_para2_exist_vars sub_res in
Some (sub_res', p_leftover)
| Some _ ->
None
in
match (Prop.prop_iter_find iter filter, hpats) with
| None, _ when not hpat.flag ->
(* L.out "@[.... iter_match_with_impl (lseg not-matched) ....@\n@."; *)
None
| None, _ when Sil.equal_lseg_kind k2 Sil.Lseg_NE ->
(* L.out "@[.... iter_match_with_impl (lseg not-matched) ....@\n@."; *)
do_para_lseg ()
| None, _ ->
(* L.out "@[.... iter_match_with_impl (lseg not-matched) ....@\n@."; *)
execute_with_backtracking [do_emp_lseg; do_para_lseg]
| Some iter_cur, [] ->
(* L.out "@[.... iter_match_with_impl (lseg matched) ....@\n@."; *)
do_empty_hpats iter_cur ()
| Some iter_cur, _ ->
(* L.out "@[.... iter_match_with_impl (lseg matched) ....@\n@."; *)
execute_with_backtracking [do_nonempty_hpats iter_cur; do_next iter_cur] )
| Sil.Hdllseg (k2, para2, iF2, oB2, oF2, iB2, es_shared2) ->
let filter = gen_filter_dllseg k2 para2 iF2 oB2 oF2 iB2 es_shared2 in
let do_emp_dllseg _ =
let fully_instantiated_iFoB2 =
not (List.exists ~f:(fun id -> Sil.ident_in_exp id iF2 || Sil.ident_in_exp id oB2) vars)
in
if not fully_instantiated_iFoB2 then None
else
let iF2' = Sil.exp_sub (`Exp sub) iF2 in
let oB2' = Sil.exp_sub (`Exp sub) oB2 in
match (exp_list_match [iF2'; oB2'] sub vars [oF2; iB2], hpats) with
| None, _ ->
None
| Some (sub_new, vars_leftover), [] ->
let sub_res = sub_extend_with_ren sub_new vars_leftover in
let p_leftover = Prop.prop_iter_to_prop tenv iter in
if condition p_leftover sub_res then Some (sub_res, p_leftover) else None
| Some (sub_new, vars_leftover), hpat_next :: hpats_rest ->
let p = Prop.prop_iter_to_prop tenv iter in
prop_match_with_impl_sub tenv p condition sub_new vars_leftover hpat_next hpats_rest
in
let do_para_dllseg _ =
let fully_instantiated_iF2 =
not (List.exists ~f:(fun id -> Sil.ident_in_exp id iF2) vars)
in
if not fully_instantiated_iF2 then None
else
let iF2' = Sil.exp_sub (`Exp sub) iF2 in
match exp_match iF2' sub vars iB2 with
| None ->
None
| Some (sub_new, vars_leftover) ->
let para2_exist_vars, para2_inst =
Sil.hpara_dll_instantiate para2 iF2 oB2 oF2 es_shared2
in
(* let allow_impl hpred = {hpred=hpred; flag=hpat.flag} in *)
let allow_impl hpred = {hpred; flag= true} in
let para2_hpat, para2_hpats =
match List.map ~f:allow_impl para2_inst with
| [] ->
assert false
(* the body of a parameter should contain at least one * conjunct *)
| para2_pat :: para2_pats ->
(para2_pat, para2_pats)
in
let new_vars = para2_exist_vars @ vars_leftover in
let new_hpats = para2_hpats @ hpats in
match
iter_match_with_impl tenv iter condition sub_new new_vars para2_hpat new_hpats
with
| None ->
None
| Some (sub_res, p_leftover) when condition p_leftover sub_res ->
let not_in_para2_exist_vars id =
not (List.exists ~f:(fun id' -> Ident.equal id id') para2_exist_vars)
in
let sub_res' = Sil.sub_filter not_in_para2_exist_vars sub_res in
Some (sub_res', p_leftover)
| Some _ ->
None
in
match (Prop.prop_iter_find iter filter, hpats) with
| None, _ when not hpat.flag ->
None
| None, _ when Sil.equal_lseg_kind k2 Sil.Lseg_NE ->
do_para_dllseg ()
| None, _ ->
execute_with_backtracking [do_emp_dllseg; do_para_dllseg]
| Some iter_cur, [] ->
do_empty_hpats iter_cur ()
| Some iter_cur, _ ->
execute_with_backtracking [do_nonempty_hpats iter_cur; do_next iter_cur]
and prop_match_with_impl_sub tenv p condition sub vars hpat hpats =
(*
L.out "@[.... prop_match_with_impl_sub ....@.";
L.out "@[<4> sub: %a@\n@." pp_sub sub;
L.out "@[<4> PROP: %a@\n@." pp_prop p;
L.out "@[<4> hpat: %a@\n@." pp_hpat hpat;
L.out "@[<4> hpred_rest: %a@\n@." pp_hpat_list hpats;
*)
match Prop.prop_iter_create p with
| None ->
instantiate_to_emp p condition sub vars (hpat :: hpats)
| Some iter ->
iter_match_with_impl tenv iter condition sub vars hpat hpats
and hpara_common_match_with_impl tenv impl_ok ids1 sigma1 eids2 ids2 sigma2 =
try
let sub_ids =
let ren_ids = List.zip_exn ids2 ids1 in
let f (id2, id1) = (id2, Exp.Var id1) in
List.map ~f ren_ids
in
let sub_eids, eids_fresh =
let f id = (id, Ident.create_fresh Ident.kprimed) in
let ren_eids = List.map ~f eids2 in
let eids_fresh = List.map ~f:snd ren_eids in
let sub_eids = List.map ~f:(fun (id2, id1) -> (id2, Exp.Var id1)) ren_eids in
(sub_eids, eids_fresh)
in
let sub = Sil.exp_subst_of_list (sub_ids @ sub_eids) in
match sigma2 with
| [] ->
if List.is_empty sigma1 then true else false
| hpred2 :: sigma2 ->
let hpat2, hpats2 =
let hpred2_ren, sigma2_ren =
(Sil.hpred_sub (`Exp sub) hpred2, Prop.sigma_sub (`Exp sub) sigma2)
in
let allow_impl hpred = {hpred; flag= impl_ok} in
(allow_impl hpred2_ren, List.map ~f:allow_impl sigma2_ren)
in
let condition _ _ = true in
let p1 = Prop.normalize tenv (Prop.from_sigma sigma1) in
match
prop_match_with_impl_sub tenv p1 condition Sil.exp_sub_empty eids_fresh hpat2 hpats2
with
| None ->
false
| Some (_, p1') when Prop.prop_is_emp p1' ->
true
| _ ->
false
with Invalid_argument _ -> false
and hpara_match_with_impl tenv impl_ok para1 para2 : bool =
(*
L.out "@[.... hpara_match_with_impl_sub ....@.";
L.out "@[<4> HPARA1: %a@\n@." pp_hpara para1;
L.out "@[<4> HPARA2: %a@\n@." pp_hpara para2;
*)
let ids1 = para1.Sil.root :: para1.Sil.next :: para1.Sil.svars in
let ids2 = para2.Sil.root :: para2.Sil.next :: para2.Sil.svars in
let eids2 = para2.Sil.evars in
hpara_common_match_with_impl tenv impl_ok ids1 para1.Sil.body eids2 ids2 para2.Sil.body
and hpara_dll_match_with_impl tenv impl_ok para1 para2 : bool =
(*
L.out "@[.... hpara_dll_match_with_impl_sub ....@.";
L.out "@[<4> HPARA1: %a@\n@." pp_hpara_dll para1;
L.out "@[<4> HPARA2: %a@\n@." pp_hpara_dll para2;
*)
let ids1 = para1.Sil.cell :: para1.Sil.blink :: para1.Sil.flink :: para1.Sil.svars_dll in
let ids2 = para2.Sil.cell :: para2.Sil.blink :: para2.Sil.flink :: para2.Sil.svars_dll in
let eids2 = para2.Sil.evars_dll in
hpara_common_match_with_impl tenv impl_ok ids1 para1.Sil.body_dll eids2 ids2 para2.Sil.body_dll
(** [prop_match_with_impl p condition vars hpat hpats]
returns [(subst, p_leftover)] such that
1) [dom(subst) = vars]
2) [p |- (hpat.hpred * hpats.hpred)[subst] * p_leftover].
Using the flag [field], we can control the strength of |-. *)
let prop_match_with_impl tenv p condition vars hpat hpats =
prop_match_with_impl_sub tenv p condition Sil.exp_sub_empty vars hpat hpats
let sigma_remove_hpred eq sigma e =
let filter = function
| Sil.Hpointsto (root, _, _)
| Sil.Hlseg (_, _, root, _, _)
| Sil.Hdllseg (_, _, root, _, _, _, _) ->
eq root e
in
let sigma_e, sigma_no_e = List.partition_tf ~f:filter sigma in
match sigma_e with
| [] ->
(None, sigma)
| [hpred_e] ->
(Some hpred_e, sigma_no_e)
| _ ->
assert false
(** {2 Routines used when finding disjoint isomorphic sigmas from a single sigma} *)
type iso_mode = Exact | LFieldForget | RFieldForget [@@deriving compare]
let equal_iso_mode = [%compare.equal : iso_mode]
let rec generate_todos_from_strexp mode todos sexp1 sexp2 =
match (sexp1, sexp2) with
| Sil.Eexp (exp1, _), Sil.Eexp (exp2, _) ->
let new_todos = (exp1, exp2) :: todos in
Some new_todos
| Sil.Eexp _, _ ->
None
| Sil.Estruct (fel1, _), Sil.Estruct (fel2, _) ->
(* assume sorted w.r.t. fields *)
if List.length fel1 <> List.length fel2 && equal_iso_mode mode Exact then None
else generate_todos_from_fel mode todos fel1 fel2
| Sil.Estruct _, _ ->
None
| Sil.Earray (len1, iel1, _), Sil.Earray (len2, iel2, _) ->
if not (Exp.equal len1 len2) || List.length iel1 <> List.length iel2 then None
else generate_todos_from_iel mode todos iel1 iel2
| Sil.Earray _, _ ->
None
and generate_todos_from_fel mode todos fel1 fel2 =
match (fel1, fel2) with
| [], [] ->
Some todos
| [], _ ->
if equal_iso_mode mode RFieldForget then Some todos else None
| _, [] ->
if equal_iso_mode mode LFieldForget then Some todos else None
| (fld1, strexp1) :: fel1', (fld2, strexp2) :: fel2' ->
let n = Typ.Fieldname.compare fld1 fld2 in
if Int.equal n 0 then
match generate_todos_from_strexp mode todos strexp1 strexp2 with
| None ->
None
| Some todos' ->
generate_todos_from_fel mode todos' fel1' fel2'
else if n < 0 && equal_iso_mode mode LFieldForget then
generate_todos_from_fel mode todos fel1' fel2
else if n > 0 && equal_iso_mode mode RFieldForget then
generate_todos_from_fel mode todos fel1 fel2'
else None
and generate_todos_from_iel mode todos iel1 iel2 =
match (iel1, iel2) with
| [], [] ->
Some todos
| (idx1, strexp1) :: iel1', (idx2, strexp2) :: iel2' -> (
match generate_todos_from_strexp mode todos strexp1 strexp2 with
| None ->
None
| Some todos' ->
let new_todos = (idx1, idx2) :: todos' in
generate_todos_from_iel mode new_todos iel1' iel2' )
| _ ->
None
(** add (e1,e2) at the front of corres, if necessary. *)
let corres_extend_front e1 e2 corres =
let filter (e1', e2') = Exp.equal e1 e1' || Exp.equal e2 e2' in
let checker e1' e2' = Exp.equal e1 e1' && Exp.equal e2 e2' in
match List.filter ~f:filter corres with
| [] ->
Some ((e1, e2) :: corres)
| [(e1', e2')] when checker e1' e2' ->
Some corres
| _ ->
None
let corres_extensible corres e1 e2 =
let predicate (e1', e2') = Exp.equal e1 e1' || Exp.equal e2 e2' in
not (List.exists ~f:predicate corres) && not (Exp.equal e1 e2)
let corres_related corres e1 e2 =
let filter (e1', e2') = Exp.equal e1 e1' || Exp.equal e2 e2' in
let checker e1' e2' = Exp.equal e1 e1' && Exp.equal e2 e2' in
match List.filter ~f:filter corres with
| [] ->
Exp.equal e1 e2
| [(e1', e2')] when checker e1' e2' ->
true
| _ ->
false
(* TO DO. Perhaps OK. Need to implemenet a better isomorphism check later.*)
let hpara_iso tenv para1 para2 =
hpara_match_with_impl tenv false para1 para2 && hpara_match_with_impl tenv false para2 para1
let hpara_dll_iso tenv para1 para2 =
hpara_dll_match_with_impl tenv false para1 para2
&& hpara_dll_match_with_impl tenv false para2 para1
(** [generic_find_partial_iso] finds isomorphic subsigmas of [sigma_todo].
The function [update] is used to get rid of hpred pairs from [sigma_todo].
[sigma_corres] records the isormophic copies discovered so far. The first
parameter determines how much flexibility we will allow during this partial
isomorphism finding. *)
let rec generic_find_partial_iso tenv mode update corres sigma_corres todos sigma_todo =
match todos with
| [] ->
let sigma1, sigma2 = sigma_corres in
Some (List.rev corres, List.rev sigma1, List.rev sigma2, sigma_todo)
| (e1, e2) :: todos' when corres_related corres e1 e2 -> (
match corres_extend_front e1 e2 corres with
| None ->
assert false
| Some new_corres ->
generic_find_partial_iso tenv mode update new_corres sigma_corres todos' sigma_todo )
| (e1, e2) :: todos' when corres_extensible corres e1 e2
-> (
let hpredo1, hpredo2, new_sigma_todo = update e1 e2 sigma_todo in
match (hpredo1, hpredo2) with
| None, None -> (
match corres_extend_front e1 e2 corres with
| None ->
assert false
| Some new_corres ->
generic_find_partial_iso tenv mode update new_corres sigma_corres todos' sigma_todo )
| None, _ | _, None ->
None
| Some Sil.Hpointsto (_, _, te1), Some Sil.Hpointsto (_, _, te2) when not (Exp.equal te1 te2) ->
None
| Some (Sil.Hpointsto (_, se1, _) as hpred1), Some (Sil.Hpointsto (_, se2, _) as hpred2) -> (
match generate_todos_from_strexp mode [] se1 se2 with
| None ->
None
| Some todos'' ->
let new_corres =
match corres_extend_front e1 e2 corres with
| None ->
assert false
| Some new_corres ->
new_corres
in
let new_sigma_corres =
let sigma1, sigma2 = sigma_corres in
let new_sigma1 = hpred1 :: sigma1 in
let new_sigma2 = hpred2 :: sigma2 in
(new_sigma1, new_sigma2)
in
let new_todos = todos'' @ todos' in
generic_find_partial_iso tenv mode update new_corres new_sigma_corres new_todos
new_sigma_todo )
| ( Some (Sil.Hlseg (k1, para1, root1, next1, shared1) as hpred1)
, Some (Sil.Hlseg (k2, para2, root2, next2, shared2) as hpred2) )
-> (
if k1 <> k2 || not (hpara_iso tenv para1 para2) then None
else
try
let new_corres =
match corres_extend_front e1 e2 corres with
| None ->
assert false
| Some new_corres ->
new_corres
in
let new_sigma_corres =
let sigma1, sigma2 = sigma_corres in
let new_sigma1 = hpred1 :: sigma1 in
let new_sigma2 = hpred2 :: sigma2 in
(new_sigma1, new_sigma2)
in
let new_todos =
let shared12 = List.zip_exn shared1 shared2 in
(root1, root2) :: (next1, next2) :: shared12 @ todos'
in
generic_find_partial_iso tenv mode update new_corres new_sigma_corres new_todos
new_sigma_todo
with Invalid_argument _ -> None )
| ( Some (Sil.Hdllseg (k1, para1, iF1, oB1, oF1, iB1, shared1) as hpred1)
, Some (Sil.Hdllseg (k2, para2, iF2, oB2, oF2, iB2, shared2) as hpred2) )
-> (
if k1 <> k2 || not (hpara_dll_iso tenv para1 para2) then None
else
try
let new_corres =
match corres_extend_front e1 e2 corres with
| None ->
assert false
| Some new_corres ->
new_corres
in
let new_sigma_corres =
let sigma1, sigma2 = sigma_corres in
let new_sigma1 = hpred1 :: sigma1 in
let new_sigma2 = hpred2 :: sigma2 in
(new_sigma1, new_sigma2)
in
let new_todos =
let shared12 = List.zip_exn shared1 shared2 in
(iF1, iF2) :: (oB1, oB2) :: (oF1, oF2) :: (iB1, iB2) :: shared12 @ todos'
in
generic_find_partial_iso tenv mode update new_corres new_sigma_corres new_todos
new_sigma_todo
with Invalid_argument _ -> None )
| _ ->
None )
| _ ->
None
(** [find_partial_iso] finds disjoint isomorphic sub-sigmas inside a given sigma.
The function returns a partial iso and three sigmas. The first sigma is the first
copy of the two isomorphic sigmas, so it uses expressions in the domain of
the returned isomorphism. The second is the second copy of the two isomorphic sigmas,
and it uses expressions in the range of the isomorphism. The third is the unused
part of the input sigma. *)
let find_partial_iso tenv eq corres todos sigma =
let update e1 e2 sigma0 =
let hpredo1, sigma0_no_e1 = sigma_remove_hpred eq sigma0 e1 in
let hpredo2, sigma0_no_e12 = sigma_remove_hpred eq sigma0_no_e1 e2 in
(hpredo1, hpredo2, sigma0_no_e12)
in
let init_sigma_corres = ([], []) in
let init_sigma_todo = sigma in
generic_find_partial_iso tenv Exact update corres init_sigma_corres todos init_sigma_todo
(** Lift the kind of list segment predicates to PE *)
let hpred_lift_to_pe hpred =
match hpred with
| Sil.Hpointsto _ ->
hpred
| Sil.Hlseg (_, para, root, next, shared) ->
Sil.Hlseg (Sil.Lseg_PE, para, root, next, shared)
| Sil.Hdllseg (_, para, iF, oB, oF, iB, shared) ->
Sil.Hdllseg (Sil.Lseg_PE, para, iF, oB, oF, iB, shared)
(** Lift the kind of list segment predicates to PE in a given sigma *)
let sigma_lift_to_pe sigma = List.map ~f:hpred_lift_to_pe sigma
(** [generic_para_create] takes a correspondence, and a sigma
and a list of expressions for the first part of this
correspondence. Then, it creates a renaming of expressions
in the domain of the given correspondence, and applies this
renaming to the given sigma. The result is a tuple of the renaming,
the renamed sigma, ids for existentially quantified expressions,
ids for shared expressions, and shared expressions. *)
let generic_para_create tenv corres sigma1 elist1 =
let corres_ids =
let not_same_consts = function
| Exp.Const c1, Exp.Const c2 ->
not (Const.equal c1 c2)
| _ ->
true
in
let new_corres' = List.filter ~f:not_same_consts corres in
let add_fresh_id pair = (pair, Ident.create_fresh Ident.kprimed) in
List.map ~f:add_fresh_id new_corres'
in
let es_shared, ids_shared, ids_exists =
let not_in_elist1 ((e1, _), _) = not (List.exists ~f:(Exp.equal e1) elist1) in
let corres_ids_no_elist1 = List.filter ~f:not_in_elist1 corres_ids in
let should_be_shared ((e1, e2), _) = Exp.equal e1 e2 in
let shared, exists = List.partition_tf ~f:should_be_shared corres_ids_no_elist1 in
let es_shared = List.map ~f:(fun ((e1, _), _) -> e1) shared in
(es_shared, List.map ~f:snd shared, List.map ~f:snd exists)
in
let renaming = List.map ~f:(fun ((e1, _), id) -> (e1, id)) corres_ids in
let body =
let sigma1' = sigma_lift_to_pe sigma1 in
let renaming_exp = List.map ~f:(fun (e1, id) -> (e1, Exp.Var id)) renaming in
Prop.sigma_replace_exp tenv renaming_exp sigma1'
in
(renaming, body, ids_exists, ids_shared, es_shared)
(** [hpara_create] takes a correspondence, and a sigma, a root
and a next for the first part of this correspondence. Then, it creates a
hpara and discovers a list of shared expressions that are
passed as arguments to hpara. Both of them are returned as a result. *)
let hpara_create tenv corres sigma1 root1 next1 =
let renaming, body, ids_exists, ids_shared, es_shared =
generic_para_create tenv corres sigma1 [root1; next1]
in
let get_id1 e1 =
let is_equal_to_e1 (e1', _) = Exp.equal e1 e1' in
match List.find ~f:is_equal_to_e1 renaming with Some (_, id) -> id | None -> assert false
in
let id_root = get_id1 root1 in
let id_next = get_id1 next1 in
let hpara =
{Sil.root= id_root; Sil.next= id_next; Sil.svars= ids_shared; Sil.evars= ids_exists; Sil.body}
in
(hpara, es_shared)
(** [hpara_dll_create] takes a correspondence, and a sigma, a root,
a blink and a flink for the first part of this correspondence. Then, it creates a
hpara_dll and discovers a list of shared expressions that are
passed as arguments to hpara. Both of them are returned as a result. *)
let hpara_dll_create tenv corres sigma1 root1 blink1 flink1 =
let renaming, body, ids_exists, ids_shared, es_shared =
generic_para_create tenv corres sigma1 [root1; blink1; flink1]
in
let get_id1 e1 =
let is_equal_to_e1 (e1', _) = Exp.equal e1 e1' in
match List.find ~f:is_equal_to_e1 renaming with Some (_, id) -> id | None -> assert false
in
let id_root = get_id1 root1 in
let id_blink = get_id1 blink1 in
let id_flink = get_id1 flink1 in
let hpara_dll =
{ Sil.cell= id_root
; Sil.blink= id_blink
; Sil.flink= id_flink
; Sil.svars_dll= ids_shared
; Sil.evars_dll= ids_exists
; Sil.body_dll= body }
in
(hpara_dll, es_shared)