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.
646 lines
23 KiB
646 lines
23 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.
|
|
*)
|
|
|
|
(** Generate unit tests automatically from specs *)
|
|
|
|
open Utils
|
|
module L = Logging
|
|
module F = Format
|
|
|
|
let (++) = Sil.Int.add
|
|
let (--) = Sil.Int.sub
|
|
|
|
module IdMap = Map.Make (Ident) (** maps from identifiers *)
|
|
|
|
(** Constraint solving module *)
|
|
module Constraint : sig
|
|
(** Collect constraints on [vars] from [pi], and return a satisfying instantiation *)
|
|
val solve_from_pure : Sil.atom list -> Ident.t list -> Sil.Int.t IdMap.t
|
|
end = struct
|
|
(** flag for debug mode of the module *)
|
|
let debug = false
|
|
|
|
(** denote a range of values [bottom] <= val <= [top], except [excluded] *)
|
|
type range =
|
|
{ mutable bottom: Sil.Int.t option; (** lower bound *)
|
|
mutable excluded: Sil.Int.t list; (** individual values not in range *)
|
|
mutable top: Sil.Int.t option; (** upper bound *) }
|
|
|
|
type eval = range IdMap.t (** evaluation for variables *)
|
|
|
|
(** create a new range *)
|
|
let new_range () =
|
|
{ bottom = None; excluded = []; top = None }
|
|
|
|
(** pretty print a range *)
|
|
let pp_range id fmt rng =
|
|
let pp_opt fmt = function
|
|
| None -> F.fprintf fmt "_"
|
|
| Some n -> Sil.Int.pp fmt n in
|
|
F.fprintf fmt "%a <= %a <= %a [%a]" pp_opt rng.bottom (Ident.pp pe_text) id pp_opt rng.top (pp_comma_seq Sil.Int.pp) rng.excluded
|
|
|
|
(** pretty print an evaluation *)
|
|
let pp_eval fmt ev =
|
|
let do_id id rng =
|
|
F.fprintf fmt "%a@." (pp_range id) rng in
|
|
IdMap.iter do_id ev
|
|
|
|
(** create a new evaluation *)
|
|
let new_eval vars =
|
|
let ev = ref IdMap.empty in
|
|
let add_var id =
|
|
ev := IdMap.add id (new_range ()) !ev in
|
|
list_iter add_var vars;
|
|
!ev
|
|
|
|
let gt_bottom i r =
|
|
match r.bottom with
|
|
| None -> true
|
|
| Some j -> Sil.Int.gt i j
|
|
|
|
let geq_bottom i r =
|
|
match r.bottom with
|
|
| None -> true
|
|
| Some j -> Sil.Int.geq i j
|
|
|
|
let lt_top i r =
|
|
match r.top with
|
|
| None -> true
|
|
| Some j -> Sil.Int.lt i j
|
|
|
|
let leq_top i r =
|
|
match r.top with
|
|
| None -> true
|
|
| Some j -> Sil.Int.leq i j
|
|
|
|
(** normalize [r]: the excluded elements must be strictly between bottom and top *)
|
|
let normalize r =
|
|
r.excluded <- list_filter (fun i -> geq_bottom i r && leq_top i r) r.excluded;
|
|
let rec normalize_bottom () = match r.bottom with
|
|
| None -> ()
|
|
| Some i ->
|
|
if list_mem Sil.Int.eq i r.excluded then begin
|
|
r.excluded <- list_filter (Sil.Int.neq i) r.excluded;
|
|
r.bottom <- Some (i ++ Sil.Int.one);
|
|
normalize_bottom ()
|
|
end in
|
|
let rec normalize_top () = match r.top with
|
|
| None -> ()
|
|
| Some i ->
|
|
if list_mem Sil.Int.eq i r.excluded then begin
|
|
r.excluded <- list_filter (Sil.Int.neq i) r.excluded;
|
|
r.top <- Some (i -- Sil.Int.one);
|
|
normalize_top ()
|
|
end in
|
|
normalize_bottom ();
|
|
normalize_top ()
|
|
|
|
(** global variable set to true every time a range is changed *)
|
|
let changed = ref false
|
|
|
|
let mark_changed id r =
|
|
changed := true;
|
|
if debug then F.fprintf F.std_formatter "changed: %a@." (pp_range id) r
|
|
|
|
(** exclude one element from the range *)
|
|
let add_excluded r id i =
|
|
if geq_bottom i r && leq_top i r && not (list_mem Sil.Int.eq i r.excluded)
|
|
then begin
|
|
r.excluded <- i :: r.excluded;
|
|
normalize r;
|
|
mark_changed id r;
|
|
end
|
|
|
|
(** make the bottom of the range >= [i] *)
|
|
let add_bottom r id i =
|
|
if gt_bottom i r then
|
|
begin
|
|
r.bottom <- Some i;
|
|
normalize r;
|
|
mark_changed id r
|
|
end
|
|
|
|
(** make the top of the range <= [i] *)
|
|
let add_top r id i =
|
|
if lt_top i r then
|
|
begin
|
|
r.top <- Some i;
|
|
normalize r;
|
|
mark_changed id r
|
|
end
|
|
|
|
(** choose an element in the range *)
|
|
let choose id rng =
|
|
if debug then F.fprintf F.std_formatter "choosing %a@." (pp_range id) rng;
|
|
let found = ref None in
|
|
let num_iter = list_length rng.excluded in
|
|
let try_candidate candidate =
|
|
if geq_bottom candidate rng && leq_top candidate rng && not (list_mem Sil.Int.eq candidate rng.excluded)
|
|
then (found := Some candidate; rng.bottom <- Some candidate; rng.top <- Some candidate; rng.excluded <- []) in
|
|
let search_up () =
|
|
let base = match rng.bottom with None -> Sil.Int.zero | Some n -> n in
|
|
for i = 0 to num_iter do
|
|
if !found = None then
|
|
let candidate = Sil.Int.add base (Sil.Int.of_int i) in
|
|
try_candidate candidate
|
|
done in
|
|
let search_down () =
|
|
let base = match rng.top with None -> Sil.Int.zero | Some n -> n in
|
|
for i = 0 to num_iter do
|
|
if !found = None then
|
|
let candidate = Sil.Int.sub base (Sil.Int.of_int i) in
|
|
try_candidate candidate
|
|
done in
|
|
search_up ();
|
|
if !found = None then search_down ();
|
|
if !found = None then
|
|
(L.err "Constraint Error: empty range %a@." (pp_range id) rng;
|
|
rng.top <- Some Sil.Int.zero;
|
|
rng.bottom <- Some Sil.Int.zero;
|
|
rng.excluded <- [])
|
|
|
|
(** return the solution if the id is solved (has unique solution) *)
|
|
let solved ev id =
|
|
let rng = IdMap.find id ev in
|
|
match rng.bottom, rng.top with
|
|
| Some n1, Some n2 when Sil.Int.eq n1 n2 -> Some n1
|
|
| _ -> None
|
|
|
|
let rec pi_iter do_le do_lt do_neq pi =
|
|
let do_atom a = match a with
|
|
| Sil.Aeq (Sil.BinOp (Sil.Le, e1, e2), Sil.Const (Sil.Cint i)) when Sil.Int.isone i ->
|
|
do_le e1 e2
|
|
| Sil.Aeq (Sil.BinOp (Sil.Lt, e1, e2), Sil.Const (Sil.Cint i)) when Sil.Int.isone i ->
|
|
do_lt e1 e2
|
|
| Sil.Aeq _ -> ()
|
|
| Sil.Aneq (e1, e2) ->
|
|
do_neq e1 e2 in
|
|
changed := false;
|
|
list_iter do_atom pi;
|
|
if !changed then pi_iter do_le do_lt do_neq pi
|
|
|
|
(** Collect constraints on [vars] from [pi], and return a satisfying instantiation *)
|
|
let solve_from_pure pi vars =
|
|
if debug then F.fprintf F.std_formatter "solve_from_pure pi: %a@." (Prop.pp_pi pe_text) pi;
|
|
let vars_fav = Sil.fav_from_list vars in
|
|
let atom_is_relevant a =
|
|
let fav = Sil.atom_fav a in
|
|
Sil.fav_for_all fav (fun id -> Sil.fav_mem vars_fav id) in
|
|
let pi_relevant = list_filter atom_is_relevant pi in
|
|
let ev = new_eval vars in
|
|
let update_top rng id n_op = match rng.top, n_op with
|
|
| Some _, Some n -> add_top rng id n
|
|
| _ -> () in
|
|
let update_bottom rng id n_op = match rng.bottom, n_op with
|
|
| Some _, Some n -> add_bottom rng id n
|
|
| _ -> () in
|
|
let (+++) n1_op n2_op = match n1_op, n2_op with
|
|
| Some n1, Some n2 -> Some (Sil.Int.add n1 n2)
|
|
| _ -> None in
|
|
let (---) n1_op n2_op = match n1_op, n2_op with
|
|
| Some n1, Some n2 -> Some (Sil.Int.sub n1 n2)
|
|
| _ -> None in
|
|
let do_le e1 e2 = match e1, e2 with
|
|
| Sil.Var id, Sil.Const (Sil.Cint n) ->
|
|
let rng = IdMap.find id ev in
|
|
add_top rng id n
|
|
| Sil.BinOp (Sil.MinusA, Sil.Var id1, Sil.Var id2), Sil.Const (Sil.Cint n) ->
|
|
let rng1 = IdMap.find id1 ev in
|
|
let rng2 = IdMap.find id2 ev in
|
|
update_top rng1 id1 (rng2.top +++ (Some n));
|
|
update_bottom rng2 id2 (rng1.bottom --- (Some n))
|
|
| Sil.BinOp (Sil.PlusA, Sil.Var id1, Sil.Var id2), Sil.Const (Sil.Cint n) ->
|
|
let rng1 = IdMap.find id1 ev in
|
|
let rng2 = IdMap.find id2 ev in
|
|
update_top rng1 id1 (Some n --- rng2.bottom);
|
|
update_top rng2 id2 (Some n --- rng1.bottom)
|
|
| _ -> if debug then assert false in
|
|
let do_lt e1 e2 = match e1, e2 with
|
|
| Sil.Const (Sil.Cint n), Sil.Var id ->
|
|
let rng = IdMap.find id ev in
|
|
add_bottom rng id (n ++ Sil.Int.one)
|
|
| Sil.Const (Sil.Cint n), Sil.BinOp (Sil.PlusA, Sil.Var id1, Sil.Var id2) ->
|
|
let rng1 = IdMap.find id1 ev in
|
|
let rng2 = IdMap.find id2 ev in
|
|
update_bottom rng1 id1 (Some (n ++ Sil.Int.one) --- rng2.top);
|
|
update_bottom rng2 id2 (Some (n ++ Sil.Int.one) --- rng1.top)
|
|
| _ -> if debug then assert false in
|
|
let rec do_neq e1 e2 = match e1, e2 with
|
|
| Sil.Var id, Sil.Const (Sil.Cint n)
|
|
| Sil.Const(Sil.Cint n), Sil.Var id ->
|
|
let rng = IdMap.find id ev in
|
|
add_excluded rng id n
|
|
| Sil.Var id1, Sil.Var id2 ->
|
|
(match solved ev id1, solved ev id2 with
|
|
| None, None -> ()
|
|
| Some _, Some _ -> ()
|
|
| Some n1, None ->
|
|
do_neq (Sil.exp_int n1) e2
|
|
| None, Some n2 ->
|
|
do_neq e1 (Sil.exp_int n2))
|
|
| Sil.Var id1, Sil.BinOp(Sil.PlusA, Sil.Var id2, Sil.Const (Sil.Cint n)) ->
|
|
(match solved ev id1, solved ev id2 with
|
|
| None, None -> ()
|
|
| Some _, Some _ -> ()
|
|
| Some n1, None ->
|
|
do_neq (Sil.exp_int (n1 -- n)) (Sil.Var id2)
|
|
| None, Some n2 ->
|
|
do_neq (Sil.Var id1) (Sil.exp_int (n2 ++ n)))
|
|
| _ -> if debug then assert false in
|
|
let do_ident id =
|
|
if debug then F.fprintf F.std_formatter "constraints before doing %a:@.%a@." (Ident.pp pe_text) id pp_eval ev;
|
|
let rng = IdMap.find id ev in
|
|
pi_iter do_le do_lt do_neq pi_relevant;
|
|
choose id rng in
|
|
list_iter do_ident vars;
|
|
if debug then F.fprintf F.std_formatter "solution to pure constraints:@.%a@." pp_eval ev;
|
|
let solution = IdMap.map (function { bottom = Some n } -> n | _ -> assert false) ev in
|
|
solution
|
|
end
|
|
|
|
type varinfo =
|
|
{ typ: Sil.typ; (* type of the variable *)
|
|
alloc: bool (* whether the variable needs allocation (on lhs of |->, lists) *)
|
|
}
|
|
|
|
type idmap = varinfo IdMap.t (* map from identifier to varinfo *)
|
|
|
|
let extend_idmap id vinfo idmap =
|
|
let vinfo' =
|
|
try
|
|
let old_vinfo = IdMap.find id !idmap in
|
|
{ old_vinfo with alloc = old_vinfo.alloc || vinfo.alloc }
|
|
with Not_found -> vinfo in
|
|
idmap := IdMap.add id vinfo' !idmap
|
|
|
|
let pe = pe_text
|
|
|
|
(** Given a sigma, create an idmap *)
|
|
let create_idmap sigma : idmap =
|
|
let idmap = ref IdMap.empty in
|
|
let rec do_exp e typ = match e, typ with
|
|
| Sil.Const _, _ -> ()
|
|
| Sil.Var id, _ ->
|
|
extend_idmap id { typ = typ; alloc = false } idmap
|
|
| Sil.BinOp (Sil.PlusA, e1, e2), _ ->
|
|
do_exp e1 typ;
|
|
do_exp e2 typ
|
|
| Sil.BinOp (Sil.PlusPI, e1, e2), _ ->
|
|
do_exp e1 typ;
|
|
do_exp e2 (Sil.Tint Sil.IULong)
|
|
| Sil.Lfield (e1, f, t), _ ->
|
|
do_exp e1 typ
|
|
| Sil.Sizeof _, _ -> ()
|
|
| _ ->
|
|
L.err "Unmatched exp: %a : %a@." (Sil.pp_exp pe) e (Sil.pp_typ_full pe) typ;
|
|
assert false in
|
|
let rec do_se se typ = match se, typ with
|
|
| Sil.Eexp (e, inst), _ ->
|
|
do_exp e typ
|
|
| Sil.Estruct (fsel, _), Sil.Tstruct (ftal, sftal, _, _, _, _, _) ->
|
|
do_struct fsel ftal
|
|
| Sil.Earray (size, esel, _), Sil.Tarray (typ, size') ->
|
|
do_se (Sil.Eexp (size, Sil.inst_none)) (Sil.Tint Sil.IULong);
|
|
do_array esel typ
|
|
| _ ->
|
|
L.err "Unmatched sexp: %a : %a@." (Sil.pp_sexp pe) se (Sil.pp_typ_full pe) typ;
|
|
assert false
|
|
and do_struct fsel ftal = match fsel, ftal with
|
|
| [], _ -> ()
|
|
| (f1, se) :: fsel', (f2, typ, a2) :: ftl' when Ident.fieldname_equal f1 f2 ->
|
|
do_se se typ;
|
|
do_struct fsel' ftl'
|
|
| (f1, se) :: fsel', (f2, typ, a2) :: ftal' ->
|
|
do_struct fsel ftal'
|
|
| _:: _, [] -> assert false
|
|
and do_array esel typ = match esel with
|
|
| (e, se):: esel' ->
|
|
do_se (Sil.Eexp (e, Sil.inst_none)) (Sil.Tint Sil.IULong);
|
|
do_se se typ;
|
|
do_array esel' typ
|
|
| [] -> () in
|
|
let do_lhs_e e t = match e with
|
|
| Sil.Var id ->
|
|
extend_idmap id { typ = t; alloc = true } idmap
|
|
| _ -> () in
|
|
let do_hpred = function
|
|
| Sil.Hpointsto (e, se, Sil.Sizeof (typ, _)) ->
|
|
do_lhs_e e (Sil.Tptr (typ, Sil.Pk_pointer));
|
|
do_se se typ
|
|
| Sil.Hlseg (k, hpar, e, f, el) ->
|
|
do_lhs_e e (Sil.Tptr (Sil.Tvoid, Sil.Pk_pointer));
|
|
do_se (Sil.Eexp (f, Sil.inst_none)) (Sil.Tptr (Sil.Tvoid, Sil.Pk_pointer));
|
|
list_iter (fun e -> do_se (Sil.Eexp (e, Sil.inst_none)) Sil.Tvoid) el
|
|
| hpred ->
|
|
L.err "do_hpred not implemented %a@." (Sil.pp_hpred pe) hpred in
|
|
list_iter do_hpred sigma;
|
|
!idmap
|
|
|
|
module Code : sig
|
|
type t
|
|
val add_from_pp : t -> (Format.formatter -> unit -> unit) -> unit
|
|
val add_line : t -> string -> unit
|
|
val append : t -> t -> unit
|
|
val empty : unit -> t
|
|
val pp : F.formatter -> t -> unit
|
|
val set_indent : string -> unit
|
|
val to_list : t -> string list
|
|
end = struct
|
|
type t = string list ref
|
|
let indent = ref ""
|
|
let to_list code =
|
|
list_rev !code
|
|
let pp fmt code =
|
|
let doit line = F.fprintf fmt "%s@\n" line in
|
|
list_iter doit (to_list code);
|
|
F.fprintf fmt "@."
|
|
let empty () = ref []
|
|
let add_line code l =
|
|
code := (!indent ^ l) :: !code
|
|
let add_from_pp code pp =
|
|
add_line code (pp_to_string pp ())
|
|
let append code1 code2 =
|
|
code1 := !code2 @ !code1
|
|
let set_indent s =
|
|
indent := s
|
|
end
|
|
|
|
type code = Code.t
|
|
|
|
(** pretty print generated code *)
|
|
let pp_code = Code.pp
|
|
|
|
(** pretty print an ident in C *)
|
|
let pp_id_c pe fmt id =
|
|
let name = Ident.get_name id in
|
|
let stamp = Ident.get_stamp id in
|
|
let varname = Ident.name_to_string name in
|
|
F.fprintf fmt "%s%d" varname stamp
|
|
|
|
(** pretty print an expression in C *)
|
|
let rec pp_exp_c pe fmt = function
|
|
| Sil.Lfield (e, f, t) ->
|
|
F.fprintf fmt "&(%a->%a)" (pp_exp_c pe) e Ident.pp_fieldname f
|
|
| Sil.Var id ->
|
|
pp_id_c pe fmt id
|
|
| e ->
|
|
Sil.pp_exp pe fmt e
|
|
|
|
(** pretty print a type in C *)
|
|
let pp_typ_c pe typ =
|
|
let pp_nil fmt () = () in
|
|
Sil.pp_type_decl pe pp_nil pp_exp_c typ
|
|
|
|
(** Convert a pvar to a string by just extracting the name *)
|
|
let pvar_to_string pvar =
|
|
Mangled.to_string (Sil.pvar_get_name pvar)
|
|
|
|
(** pretty print an expression list in C *)
|
|
let pp_exp_list_c pe f expl =
|
|
(pp_seq (pp_exp_c pe)) f expl
|
|
|
|
(** Name of the recusive function for a specific list para *)
|
|
let mk_lseg_name id proc_name spec_num =
|
|
"mk_lseg_" ^ string_of_int id ^ Procname.to_string proc_name ^ "_spec" ^ string_of_int spec_num
|
|
|
|
let mk_size_name id =
|
|
"_size_" ^ string_of_int id
|
|
|
|
let pp_texp_for_malloc fmt =
|
|
let rec handle_arr_size typ = match typ with
|
|
| Sil.Tvar _ | Sil.Tint _ | Sil.Tfloat _ | Sil.Tvoid | Sil.Tfun _ | Sil.Tenum _ ->
|
|
typ
|
|
| Sil.Tptr (t, pk) ->
|
|
Sil.Tptr (handle_arr_size t, pk)
|
|
| Sil.Tstruct (ftal, sftal, csu, nameo, supers, def_mthds, iann) ->
|
|
Sil.Tstruct (list_map (fun (f, t, a) -> (f, handle_arr_size t, a)) ftal, sftal, csu, nameo, supers, def_mthds, iann)
|
|
| Sil.Tarray (t, e) ->
|
|
Sil.Tarray (handle_arr_size t, e) in
|
|
function
|
|
| Sil.Sizeof (typ, _) ->
|
|
let typ' = handle_arr_size typ in
|
|
F.fprintf fmt "sizeof(%a)" (pp_typ_c pe) typ'
|
|
| e -> pp_exp_c pe fmt e
|
|
|
|
(* generate code for sigma *)
|
|
let gen_sigma code proc_name spec_num env idmap sigma =
|
|
let post_code = Code.empty () in
|
|
let rec do_strexp code' base need_deref = function
|
|
| Sil.Eexp (e, inst) ->
|
|
let lhs = if need_deref then "(*"^base^")" else base in
|
|
let pp f () = F.fprintf f "%s = %a;" lhs (pp_exp_c pe) e in
|
|
Code.add_from_pp code' pp
|
|
| Sil.Estruct (fsel, _) ->
|
|
let accessor = if need_deref then "->" else "." in
|
|
list_iter (fun (f, se) -> do_strexp code' (base ^ accessor ^ Ident.fieldname_to_string f) false se) fsel
|
|
| Sil.Earray (size, esel, _) ->
|
|
list_iter (fun (e, se) ->
|
|
let pp f () = F.fprintf f "%a" (pp_exp_c pe) e in
|
|
let index = pp_to_string pp () in
|
|
do_strexp code' (base ^ "[" ^ index ^ "]") false se) esel in
|
|
|
|
let gen_hpred = function
|
|
| Sil.Hpointsto (Sil.Lvar pvar, se, _) ->
|
|
let base = pvar_to_string pvar in
|
|
do_strexp post_code base false se
|
|
| Sil.Hpointsto (Sil.Var id, se, te) ->
|
|
let pp1 f () =
|
|
F.fprintf f "%a = malloc(%a);" (pp_id_c pe) id pp_texp_for_malloc te in
|
|
let pp2 f () =
|
|
F.fprintf f "if(%a == NULL) exit(12);" (pp_id_c pe) id in
|
|
Code.add_from_pp code pp1;
|
|
Code.add_from_pp code pp2;
|
|
let pp3 f () = F.fprintf f "%a" (pp_id_c pe) id in
|
|
let base = pp_to_string pp3 () in
|
|
do_strexp post_code base true se
|
|
| Sil.Hlseg (k, hpar, Sil.Var id, f, el) ->
|
|
let hpara_id = Sil.Predicates.get_hpara_id env hpar in
|
|
let size_var = mk_size_name hpara_id in
|
|
let mk_name = mk_lseg_name hpara_id proc_name spec_num in
|
|
let pp_el fmt el =
|
|
if el != [] then pp_exp_list_c pe fmt el in
|
|
let pp1 fmt () =
|
|
F.fprintf fmt "int %s = 42;" size_var in
|
|
let pp2 fmt () =
|
|
F.fprintf fmt "%a = %s(%s, %a%a);" (pp_id_c pe) id mk_name size_var (pp_exp_c pe) f pp_el el in
|
|
Code.add_from_pp code pp1;
|
|
Code.add_from_pp code pp2
|
|
| hpred ->
|
|
L.err "gen_hpred not implemented: %a@." (Sil.pp_hpred pe) hpred in
|
|
list_iter gen_hpred sigma;
|
|
Code.append code post_code
|
|
|
|
(* generate code corresponding to equalities in the pure part *)
|
|
let gen_init_equalities code pure =
|
|
let do_atom = function
|
|
| Sil.Aeq (Sil.Var id, e) ->
|
|
let pp f () = F.fprintf f "%a = %a;" (pp_id_c pe) id (pp_exp_c pe) e in
|
|
Code.add_from_pp code pp
|
|
| _ -> () in
|
|
list_iter do_atom pure
|
|
|
|
(** generate variable declarations *)
|
|
let gen_var_decl code idmap parameters =
|
|
let do_parameter (name, typ) =
|
|
let pp_name f () = Format.fprintf f "%s" name in
|
|
let pp f () = F.fprintf f "%a;" (Sil.pp_type_decl pe pp_name pp_exp_c) typ in
|
|
Code.add_from_pp code pp in
|
|
let do_vinfo id { typ = typ; alloc = alloc } =
|
|
let pp_var f () = pp_id_c pe f id in
|
|
let pp f () = F.fprintf f "%a;" (Sil.pp_type_decl pe pp_var pp_exp_c) typ in
|
|
Code.add_from_pp code pp in
|
|
list_iter do_parameter parameters;
|
|
IdMap.iter do_vinfo idmap
|
|
|
|
(** initialize variables not requiring allocation *)
|
|
let gen_init_vars code solutions idmap =
|
|
let get_const id c =
|
|
try Sil.Cint (IdMap.find id solutions)
|
|
with Not_found -> c in
|
|
let do_vinfo id { typ = typ; alloc = alloc } =
|
|
if not alloc then
|
|
let const = match typ with
|
|
| Sil.Tint _ | Sil.Tvoid ->
|
|
get_const id (Sil.Cint Sil.Int.zero)
|
|
| Sil.Tfloat _ ->
|
|
Sil.Cfloat 0.0
|
|
| Sil.Tptr _ ->
|
|
get_const id (Sil.Cint Sil.Int.zero)
|
|
| Sil.Tfun _ ->
|
|
Sil.Cint Sil.Int.zero
|
|
| typ ->
|
|
L.err "do_vinfo type undefined: %a@." (Sil.pp_typ_full pe) typ;
|
|
assert false in
|
|
let pp fmt () =
|
|
F.fprintf fmt "%a = (%a) %a;" (pp_id_c pe) id (Sil.pp_typ_full pe) typ (Sil.pp_exp pe) (Sil.Const const) in
|
|
Code.add_from_pp code pp in
|
|
IdMap.iter do_vinfo idmap
|
|
|
|
(** apply a filter to the identifiers of an idmap *)
|
|
let filter_idmap filter idmap =
|
|
let idmap' = ref IdMap.empty in
|
|
IdMap.iter (fun id x -> if filter id then idmap' := IdMap.add id x !idmap') idmap;
|
|
!idmap'
|
|
|
|
let pp_svars fmt svars =
|
|
if svars != [] then F.fprintf fmt "%a" (pp_comma_seq (pp_id_c pe)) svars
|
|
|
|
|
|
let gen_hpara code proc_name spec_num env id hpara =
|
|
let mk_name = mk_lseg_name id proc_name spec_num in
|
|
let size_name = mk_size_name id in
|
|
let pp1 f () =
|
|
F.fprintf f "void* %s(int %s, void* %a%a) {" mk_name size_name (pp_id_c pe) hpara.Sil.next pp_svars hpara.Sil.svars in
|
|
let pp2 f () =
|
|
F.fprintf f "%a= %s(%s -1 , %a%a);" (pp_id_c pe) hpara.Sil.next mk_name size_name (pp_id_c pe) hpara.Sil.next pp_svars hpara.Sil.svars in
|
|
let line1 = pp_to_string pp1 () in
|
|
let idmap = create_idmap hpara.Sil.body in
|
|
let idmap_ex =
|
|
let filter i =
|
|
list_exists (Ident.equal i) hpara.Sil.evars in
|
|
filter_idmap filter idmap in
|
|
let idmap_no_next =
|
|
let filter i =
|
|
not (Ident.equal i hpara.Sil.next) in
|
|
filter_idmap filter idmap in
|
|
let line11 = "if ("^size_name^" == 0) {" in
|
|
let line12 = "return " ^ (pp_to_string (pp_id_c pe) hpara.Sil.next) ^ ";" in
|
|
let line13 ="} else {" in
|
|
let line14 = pp_to_string pp2 () in
|
|
let line2 = "return " ^ (pp_to_string (pp_id_c pe) hpara.Sil.root) ^ ";" in
|
|
let line3 = "}" in
|
|
Code.add_line code line1;
|
|
Code.set_indent " ";
|
|
gen_var_decl code idmap_no_next [];
|
|
Code.add_line code line11;
|
|
Code.set_indent " ";
|
|
Code.add_line code line12;
|
|
Code.set_indent " ";
|
|
Code.add_line code line13;
|
|
Code.set_indent " ";
|
|
Code.add_line code line14;
|
|
gen_init_vars code IdMap.empty idmap_ex;
|
|
gen_sigma code proc_name spec_num env idmap hpara.Sil.body;
|
|
Code.add_line code line2;
|
|
Code.set_indent " ";
|
|
Code.add_line code line3;
|
|
Code.set_indent "";
|
|
Code.add_line code line3;
|
|
Code.add_line code ""
|
|
|
|
let gen_hpara_dll code proc_name spec_num env id hpara_dll = assert false
|
|
|
|
(** Generate epilog for the test case *)
|
|
let gen_epilog code proc_name parameters =
|
|
let pp_parameter fmt (name, typ) =
|
|
F.fprintf fmt "%s" name in
|
|
let pp f () = F.fprintf f "%a(%a);" Procname.pp proc_name (pp_comma_seq pp_parameter) parameters in
|
|
let line1 = pp_to_string pp () in
|
|
let line2 = "}" in
|
|
Code.add_line code line1;
|
|
Code.set_indent "";
|
|
Code.add_line code line2
|
|
|
|
let test_function_name proc_name spec_num =
|
|
"unit_test_" ^ Procname.to_string proc_name ^ "_spec" ^ string_of_int spec_num
|
|
|
|
(** Generate prolog for test case *)
|
|
let gen_prolog code fname proc_name spec_num =
|
|
let pp f () =
|
|
let fun_name = test_function_name proc_name spec_num in
|
|
F.fprintf f "/**@\n *@\n * Procedure: %a()@\n * File: %s@\n" Procname.pp proc_name fname;
|
|
F.fprintf f " * Unit Test: %s()@\n * Created: %a@\n *@\n */@\n@\n" fun_name pp_current_time ();
|
|
F.fprintf f "void %s() {" fun_name in
|
|
Code.add_from_pp code pp;
|
|
Code.set_indent " "
|
|
|
|
let solve_constraints pure idmap =
|
|
let vars = ref [] in
|
|
let do_vinfo id { typ = typ; alloc = alloc } =
|
|
if not alloc then vars := !vars @ [id] in
|
|
IdMap.iter do_vinfo idmap;
|
|
Constraint.solve_from_pure pure !vars
|
|
|
|
(** generate a unit test form a spec *)
|
|
let genunit fname proc_name spec_num parameters spec =
|
|
let pre = Specs.Jprop.to_prop spec.Specs.pre in
|
|
let pure = Prop.get_pure pre in
|
|
let sigma = Prop.get_sigma pre in
|
|
let env = Prop.prop_pred_env pre in
|
|
let idmap = create_idmap sigma in
|
|
let code = Code.empty () in
|
|
Sil.Predicates.iter env
|
|
(gen_hpara code proc_name spec_num env)
|
|
(gen_hpara_dll code proc_name spec_num env);
|
|
gen_prolog code fname proc_name spec_num;
|
|
gen_var_decl code idmap parameters;
|
|
gen_init_vars code (solve_constraints pure idmap) idmap;
|
|
gen_init_equalities code pure;
|
|
gen_sigma code proc_name spec_num env idmap sigma;
|
|
gen_epilog code proc_name parameters;
|
|
code
|
|
|
|
(** generate code for a main calling all the unit test functions passed as argument *)
|
|
let genmain proc_numspecs_list =
|
|
let code = Code.empty () in
|
|
let do_one_proc (proc_name, num_specs) =
|
|
for i = 1 to num_specs do
|
|
let test_fun_name = test_function_name proc_name i in
|
|
let line = test_fun_name ^ "();" in
|
|
Code.add_line code line done in
|
|
Code.add_line code "int main() {";
|
|
Code.set_indent " ";
|
|
list_iter do_one_proc proc_numspecs_list;
|
|
Code.add_line code "printf(\"unit test terminated\\n\");";
|
|
Code.add_line code "return 0;";
|
|
Code.set_indent "";
|
|
Code.add_line code "}";
|
|
code
|