|
|
|
(*
|
|
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the MIT license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree.
|
|
|
|
*)
|
|
|
|
|
|
|
|
(** Abstract domain *)
|
|
|
|
|
|
|
|
module X = Llair_to_Fol
|
|
|
|
open Fol
|
|
|
|
|
|
|
|
type t = Sh.t [@@deriving compare, equal, sexp]
|
|
|
|
|
|
|
|
let pp fs q = Format.fprintf fs "@[{ %a@ }@]" Sh.pp q
|
|
|
|
let report_fmt_thunk = Fun.flip pp
|
|
|
|
|
|
|
|
(* set by cli *)
|
|
|
|
let simplify_states = ref true
|
|
|
|
let simplify q = if !simplify_states then Sh.simplify q else q
|
|
|
|
|
|
|
|
let init globals =
|
|
|
|
IArray.fold globals Sh.emp ~f:(fun global q ->
|
|
|
|
match (global : Llair.GlobalDefn.t) with
|
|
|
|
| {name; init= Some seq} ->
|
|
|
|
let loc = X.global name in
|
|
|
|
let siz =
|
|
|
|
match Llair.Global.typ name with
|
|
|
|
| Pointer {elt} -> Llair.Typ.size_of elt
|
|
|
|
| _ -> violates Llair.GlobalDefn.invariant global
|
|
|
|
in
|
|
|
|
let len = Term.integer (Z.of_int siz) in
|
|
|
|
let cnt = X.term seq in
|
|
|
|
Sh.star q (Sh.seg {loc; bas= loc; len; siz= len; cnt})
|
|
|
|
| _ -> q )
|
|
|
|
|
|
|
|
let join p q =
|
|
|
|
[%Trace.call fun {pf} -> pf "@ %a@ %a" pp p pp q]
|
|
|
|
;
|
|
|
|
Some (Sh.or_ p q) |> Option.map ~f:simplify
|
|
|
|
|>
|
|
|
|
[%Trace.retn fun {pf} -> pf "%a" (Option.pp "%a" pp)]
|
|
|
|
|
|
|
|
let dnf = Sh.dnf
|
|
|
|
|
|
|
|
let exec_assume q b =
|
|
|
|
Exec.assume q (X.formula b)
|
|
|
|
|> simplify
|
|
|
|
|> fun q -> if Sh.is_unsat q then None else Some q
|
|
|
|
|
|
|
|
let exec_kill r q = Exec.kill q (X.reg r) |> simplify
|
|
|
|
|
|
|
|
let exec_move res q =
|
|
|
|
Exec.move q (IArray.map res ~f:(fun (r, e) -> (X.reg r, X.term e)))
|
|
|
|
|> simplify
|
|
|
|
|
|
|
|
let exec_inst inst pre =
|
|
|
|
( match (inst : Llair.inst) with
|
|
|
|
| Move {reg_exps; _} ->
|
|
|
|
Some
|
|
|
|
(Exec.move pre
|
|
|
|
(IArray.map reg_exps ~f:(fun (r, e) -> (X.reg r, X.term e))))
|
|
|
|
| Load {reg; ptr; len; _} ->
|
|
|
|
Exec.load pre ~reg:(X.reg reg) ~ptr:(X.term ptr) ~len:(X.term len)
|
|
|
|
| Store {ptr; exp; len; _} ->
|
|
|
|
Exec.store pre ~ptr:(X.term ptr) ~exp:(X.term exp) ~len:(X.term len)
|
|
|
|
| Alloc {reg; num; len; _} ->
|
|
|
|
Exec.alloc pre ~reg:(X.reg reg) ~num:(X.term num) ~len
|
|
|
|
| Free {ptr; _} -> Exec.free pre ~ptr:(X.term ptr)
|
|
|
|
| Nondet {reg; _} -> Some (Exec.nondet pre (Option.map ~f:X.reg reg))
|
|
|
|
| Abort _ -> Exec.abort pre
|
|
|
|
| Intrinsic {reg; name; args; _} ->
|
|
|
|
let areturn = Option.map ~f:X.reg reg in
|
|
|
|
let actuals = IArray.map ~f:X.term args in
|
|
|
|
Exec.intrinsic pre areturn name actuals )
|
|
|
|
|> Option.map ~f:simplify
|
|
|
|
|
|
|
|
let value_determined_by ctx us a =
|
|
|
|
List.exists (Context.class_of ctx a) ~f:(fun b ->
|
|
|
|
Term.Set.subset (Term.Set.of_iter (Term.atoms b)) ~of_:us )
|
|
|
|
|
|
|
|
let garbage_collect (q : Sh.t) ~wrt =
|
|
|
|
[%Trace.call fun {pf} -> pf "@ %a" pp q]
|
|
|
|
;
|
|
|
|
(* only support DNF for now *)
|
|
|
|
assert (List.is_empty q.djns) ;
|
|
|
|
let rec all_reachable_vars previous current (q : t) =
|
|
|
|
if Term.Set.equal previous current then current
|
|
|
|
else
|
|
|
|
let new_set =
|
|
|
|
List.fold q.heap current ~f:(fun seg current ->
|
|
|
|
if value_determined_by q.ctx current seg.loc then
|
|
|
|
List.fold (Context.class_of q.ctx seg.cnt) current
|
|
|
|
~f:(fun e c ->
|
|
|
|
Term.Set.union c (Term.Set.of_iter (Term.atoms e)) )
|
|
|
|
else current )
|
|
|
|
in
|
|
|
|
all_reachable_vars current new_set q
|
|
|
|
in
|
|
|
|
let r_vars = all_reachable_vars Term.Set.empty wrt q in
|
|
|
|
Sh.filter_heap q ~f:(fun seg -> value_determined_by q.ctx r_vars seg.loc)
|
|
|
|
|>
|
|
|
|
[%Trace.retn fun {pf} -> pf "%a" pp]
|
|
|
|
|
|
|
|
let and_eqs sub formals actuals q =
|
|
|
|
let and_eq formal actual q =
|
|
|
|
let actual' = Term.rename sub actual in
|
|
|
|
Sh.and_ (Formula.eq (Term.var formal) actual') q
|
|
|
|
in
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
IArray.fold2_exn ~f:and_eq formals actuals q
|
|
|
|
|
|
|
|
let localize_entry globals actuals formals freturn locals shadow pre entry =
|
|
|
|
(* Add the formals here to do garbage collection and then get rid of them *)
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
let formals_set = Var.Set.of_iter (IArray.to_iter formals) in
|
|
|
|
let freturn_locals = X.regs (Llair.Reg.Set.add_option freturn locals) in
|
|
|
|
let wrt =
|
|
|
|
Term.Set.of_iter
|
|
|
|
(Iter.append
|
|
|
|
(Iter.map ~f:X.global (Llair.Global.Set.to_iter globals))
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
(Iter.map ~f:Term.var (IArray.to_iter formals)))
|
|
|
|
in
|
|
|
|
let function_summary_pre = garbage_collect entry ~wrt in
|
|
|
|
[%Trace.info "function summary pre %a" pp function_summary_pre] ;
|
|
|
|
let foot = Sh.exists formals_set function_summary_pre in
|
|
|
|
let xs, foot = Sh.bind_exists ~wrt:pre.Sh.us foot in
|
|
|
|
let frame =
|
|
|
|
try Option.get_exn (Solver.infer_frame pre xs foot)
|
|
|
|
with _ ->
|
|
|
|
fail "Solver couldn't infer frame of a garbage-collected pre" ()
|
|
|
|
in
|
|
|
|
let q'' =
|
|
|
|
Sh.extend_us freturn_locals (and_eqs shadow formals actuals foot)
|
|
|
|
in
|
|
|
|
(q'', frame)
|
|
|
|
|
|
|
|
type from_call = {areturn: Var.t option; unshadow: Var.Subst.t; frame: Sh.t}
|
|
|
|
[@@deriving compare, equal, sexp]
|
|
|
|
|
|
|
|
(** Express formula in terms of formals instead of actuals, and enter scope
|
|
|
|
of locals: rename formals to fresh vars in formula and actuals, add
|
|
|
|
equations between each formal and actual, and quantify fresh vars. *)
|
|
|
|
let call ~summaries ~globals ~actuals ~areturn ~formals ~freturn ~locals q =
|
|
|
|
[%Trace.call fun {pf} ->
|
|
|
|
pf "@ @[<hv>locals: {@[%a@]}@ globals: {@[%a@]}@ q: %a@]"
|
|
|
|
Llair.Reg.Set.pp locals Llair.Global.Set.pp globals pp q ;
|
|
|
|
assert (
|
|
|
|
(* modifs do not appear in actuals (otherwise incomplete) *)
|
|
|
|
let fv_actuals =
|
|
|
|
actuals
|
|
|
|
|> IArray.to_iter
|
|
|
|
|> Iter.map ~f:X.term
|
|
|
|
|> Iter.flat_map ~f:Term.vars
|
|
|
|
in
|
|
|
|
not
|
|
|
|
(Option.exists areturn ~f:(fun modif ->
|
|
|
|
Iter.exists ~f:(Var.equal (X.reg modif)) fv_actuals )) )]
|
|
|
|
;
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
let actuals = IArray.map ~f:X.term actuals in
|
|
|
|
let areturn = Option.map ~f:X.reg areturn in
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
let formals = IArray.map ~f:X.reg formals in
|
|
|
|
let freturn_locals = X.regs (Llair.Reg.Set.add_option freturn locals) in
|
|
|
|
let modifs = Var.Set.of_option areturn in
|
|
|
|
(* quantify modifs, their current values will be overwritten and so should
|
|
|
|
not be saved and restored on return *)
|
|
|
|
let q = Sh.exists modifs q in
|
|
|
|
(* save current values of shadowed formals and locals with a renaming *)
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
let formals_freturn_locals =
|
|
|
|
Iter.fold ~f:Var.Set.add (IArray.to_iter formals) freturn_locals
|
|
|
|
in
|
|
|
|
let q, shadow = Sh.freshen q ~wrt:formals_freturn_locals in
|
|
|
|
let unshadow = Var.Subst.invert shadow in
|
|
|
|
assert (Var.Set.disjoint modifs (Var.Subst.domain shadow)) ;
|
|
|
|
(* pass arguments by conjoining equations between formals and actuals *)
|
|
|
|
let entry = and_eqs shadow formals actuals q in
|
|
|
|
(* note: locals and formals are in scope *)
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
assert (Var.Set.subset formals_freturn_locals ~of_:entry.us) ;
|
|
|
|
(* simplify *)
|
|
|
|
let entry = simplify entry in
|
|
|
|
( if not summaries then (entry, {areturn; unshadow; frame= Sh.emp})
|
|
|
|
else
|
|
|
|
let q, frame =
|
|
|
|
localize_entry globals actuals formals freturn locals shadow q entry
|
|
|
|
in
|
|
|
|
(q, {areturn; unshadow; frame}) )
|
|
|
|
|>
|
|
|
|
[%Trace.retn fun {pf} (entry, {unshadow; frame}) ->
|
|
|
|
pf "@[<v>unshadow: %a@ frame: %a@ entry: %a@]" Var.Subst.pp unshadow pp
|
|
|
|
frame pp entry]
|
|
|
|
|
|
|
|
(** Leave scope of locals: existentially quantify locals. *)
|
|
|
|
let post locals _ q =
|
|
|
|
[%Trace.call fun {pf} ->
|
|
|
|
pf "@ @[<hv>locals: {@[%a@]}@ q: %a@]" Llair.Reg.Set.pp locals Sh.pp q]
|
|
|
|
;
|
|
|
|
Sh.exists (X.regs locals) q |> simplify
|
|
|
|
|>
|
|
|
|
[%Trace.retn fun {pf} -> pf "%a" Sh.pp]
|
|
|
|
|
|
|
|
(** Express in terms of actuals instead of formals: existentially quantify
|
|
|
|
formals, and apply inverse of fresh variables for formals renaming to
|
|
|
|
restore the shadowed variables. *)
|
|
|
|
let retn formals freturn {areturn; unshadow; frame} q =
|
|
|
|
[%Trace.call fun {pf} ->
|
|
|
|
pf "@ @[<v>formals: {@[%a@]}%a%a@ unshadow: %a@ q: %a@ frame: %a@]"
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
(IArray.pp ", " Llair.Reg.pp)
|
|
|
|
formals
|
|
|
|
(Option.pp "@ freturn: %a" Llair.Reg.pp)
|
|
|
|
freturn
|
|
|
|
(Option.pp "@ areturn: %a" Var.pp)
|
|
|
|
areturn Var.Subst.pp unshadow pp q pp frame]
|
|
|
|
;
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
let formals =
|
|
|
|
Var.Set.of_iter (Iter.map ~f:X.reg (IArray.to_iter formals))
|
|
|
|
in
|
|
|
|
let freturn = Option.map ~f:X.reg freturn in
|
|
|
|
let q =
|
|
|
|
match areturn with
|
|
|
|
| Some areturn -> (
|
|
|
|
(* reenter scope of areturn just before exiting scope of formals *)
|
|
|
|
let q = Sh.extend_us (Var.Set.of_ areturn) q in
|
|
|
|
(* pass return value *)
|
|
|
|
match freturn with
|
|
|
|
| Some freturn ->
|
|
|
|
Exec.move q (IArray.of_ (areturn, Term.var freturn))
|
|
|
|
| None -> Exec.kill q areturn )
|
|
|
|
| None -> q
|
|
|
|
in
|
|
|
|
(* exit scope of formals, except for areturn, which move/kill handled *)
|
|
|
|
let outscoped =
|
|
|
|
Var.Set.diff
|
|
|
|
(Var.Set.union formals (Var.Set.of_option freturn))
|
|
|
|
(Var.Set.of_option areturn)
|
|
|
|
in
|
|
|
|
let q = Sh.exists outscoped q in
|
|
|
|
(* reinstate shadowed values of locals *)
|
|
|
|
let q = Sh.rename unshadow q in
|
|
|
|
(* reconjoin frame *)
|
|
|
|
Sh.star frame q
|
|
|
|
(* simplify *)
|
|
|
|
|> simplify
|
|
|
|
|>
|
|
|
|
[%Trace.retn fun {pf} -> pf "%a" pp]
|
|
|
|
|
|
|
|
let resolve_callee lookup ptr (q : Sh.t) =
|
|
|
|
Context.class_of q.ctx (X.term ptr)
|
|
|
|
|> List.find_map ~f:(X.lookup_func lookup)
|
|
|
|
|> Option.to_list
|
|
|
|
|
|
|
|
let recursion_beyond_bound = `prune
|
|
|
|
|
|
|
|
type summary = {xs: Var.Set.t; foot: t; post: t}
|
|
|
|
|
|
|
|
let pp_summary fs {xs; foot; post} =
|
|
|
|
Format.fprintf fs "@[<v>xs: @[%a@]@ foot: %a@ post: %a @]" Var.Set.pp xs
|
|
|
|
pp foot pp post
|
|
|
|
|
|
|
|
let create_summary ~locals ~formals ~entry ~current:(post : Sh.t) =
|
|
|
|
[%Trace.call fun {pf} ->
|
|
|
|
pf "@ formals %a@ entry: %a@ current: %a"
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
(IArray.pp ",@ " Llair.Reg.pp)
|
|
|
|
formals pp entry pp post]
|
|
|
|
;
|
|
|
|
let formals =
|
[sledge] Represent function formal parameters and actual arguments in order
Summary:
Previously, when LLAIR was in SSA form, blocks took parameters just
like functions, and it was sometimes necessary to partially apply a
block to some of the parameters. For example, blocks to which function
calls return would need to accept the return value as an argument, and
sometimes immediately jump to another block passing the rest of the
arguments as well. These "trampoline" blocks were partial applications
of the eventual block to all but the final, return value,
argument.
This partial application mechanism meant that function parameters and
arguments were represented as a stack, with the first argument at the
bottom, that is, in reverse order.
Now that LLAIR is free of SSA, this confusion is no longer needed, and
this diff changes the representation of function formal parameters and
actual arguments to be in the natural order. This also brings Call
instructions in line with Intrinsic instructions, which will make
changing the handling of intrinsics from Calls to Intrinsic less
error-prone.
Reviewed By: jvillard
Differential Revision: D25146163
fbshipit-source-id: d3ed07a45
4 years ago
|
|
|
Var.Set.of_iter (Iter.map ~f:X.reg (IArray.to_iter formals))
|
|
|
|
in
|
|
|
|
let locals = X.regs locals in
|
|
|
|
let foot = Sh.exists locals entry in
|
|
|
|
let foot, subst = Sh.freshen ~wrt:(Var.Set.union foot.us post.us) foot in
|
|
|
|
let restore_formals q =
|
|
|
|
Var.Set.fold formals q ~f:(fun var q ->
|
|
|
|
let var = Term.var var in
|
|
|
|
let renamed_var = Term.rename subst var in
|
|
|
|
Sh.and_ (Formula.eq renamed_var var) q )
|
|
|
|
in
|
|
|
|
(* Add back the original formals name *)
|
|
|
|
let post = Sh.rename subst post in
|
|
|
|
let foot = restore_formals foot in
|
|
|
|
let post = restore_formals post in
|
|
|
|
[%Trace.info "subst: %a" Var.Subst.pp subst] ;
|
|
|
|
let xs = Var.Set.inter (Sh.fv foot) (Sh.fv post) in
|
|
|
|
let xs = Var.Set.diff xs formals in
|
|
|
|
let xs_and_formals = Var.Set.union xs formals in
|
|
|
|
let foot = Sh.exists (Var.Set.diff foot.us xs_and_formals) foot in
|
|
|
|
let post = Sh.exists (Var.Set.diff post.us xs_and_formals) post in
|
|
|
|
let current = Sh.extend_us xs post in
|
|
|
|
({xs; foot; post}, current)
|
|
|
|
|>
|
|
|
|
[%Trace.retn fun {pf} (fs, _) -> pf "@,%a" pp_summary fs]
|
|
|
|
|
|
|
|
let apply_summary q ({xs; foot; post} as fs) =
|
|
|
|
[%Trace.call fun {pf} -> pf "@ fs: %a@ q: %a" pp_summary fs pp q]
|
|
|
|
;
|
|
|
|
let xs_in_q = Var.Set.inter xs q.Sh.us in
|
|
|
|
let xs_in_fv_q = Var.Set.inter xs (Sh.fv q) in
|
|
|
|
(* Between creation of a summary and its use, the vocabulary of q (q.us)
|
|
|
|
might have been extended. That means infer_frame would fail, because q
|
|
|
|
and foot have different vocabulary. This might indicate that the
|
|
|
|
summary cannot be applied to q, however in the case where
|
|
|
|
free-variables of q and foot match it is benign. In the case where free
|
|
|
|
variables match, we temporarily reduce the vocabulary of q to match the
|
|
|
|
vocabulary of foot. *)
|
|
|
|
[%Trace.info "xs inter q.us: %a" Var.Set.pp xs_in_q] ;
|
|
|
|
[%Trace.info "xs inter fv.q %a" Var.Set.pp xs_in_fv_q] ;
|
|
|
|
let q, add_back =
|
|
|
|
if Var.Set.is_empty xs_in_fv_q then (Sh.exists xs_in_q q, xs_in_q)
|
|
|
|
else (q, Var.Set.empty)
|
|
|
|
in
|
|
|
|
let frame =
|
|
|
|
if Var.Set.is_empty xs_in_fv_q then Solver.infer_frame q xs foot
|
|
|
|
else None
|
|
|
|
in
|
|
|
|
[%Trace.info "frame %a" (Option.pp "%a" pp) frame] ;
|
|
|
|
Option.map ~f:(Sh.extend_us add_back) (Option.map ~f:(Sh.star post) frame)
|
|
|
|
|>
|
|
|
|
[%Trace.retn fun {pf} r ->
|
|
|
|
match r with None -> pf "None" | Some q -> pf "@,%a" pp q]
|
|
|
|
|
|
|
|
let%test_module _ =
|
|
|
|
( module struct
|
|
|
|
let () = Trace.init ~margin:68 ()
|
|
|
|
let pp = Format.printf "@.%a@." Sh.pp
|
|
|
|
let wrt = Var.Set.empty
|
|
|
|
let main_, wrt = Var.fresh "main" ~wrt
|
|
|
|
let a_, wrt = Var.fresh "a" ~wrt
|
|
|
|
let n_, wrt = Var.fresh "n" ~wrt
|
|
|
|
let b_, wrt = Var.fresh "b" ~wrt
|
|
|
|
let end_, _ = Var.fresh "end" ~wrt
|
|
|
|
let a = Term.var a_
|
|
|
|
let main = Term.var main_
|
|
|
|
let b = Term.var b_
|
|
|
|
let n = Term.var n_
|
|
|
|
let endV = Term.var end_
|
|
|
|
let seg_main = Sh.seg {loc= main; bas= b; len= n; siz= n; cnt= a}
|
|
|
|
let seg_a = Sh.seg {loc= a; bas= b; len= n; siz= n; cnt= endV}
|
|
|
|
let seg_cycle = Sh.seg {loc= a; bas= b; len= n; siz= n; cnt= main}
|
|
|
|
|
|
|
|
let%expect_test _ =
|
|
|
|
pp (garbage_collect seg_main ~wrt:(Term.Set.of_list [])) ;
|
|
|
|
[%expect {| emp |}]
|
|
|
|
|
|
|
|
let%expect_test _ =
|
|
|
|
pp
|
|
|
|
(garbage_collect (Sh.star seg_a seg_main)
|
|
|
|
~wrt:(Term.Set.of_list [a])) ;
|
|
|
|
[%expect {| %a_2 -[ %b_4, %n_3 )-> ⟨%n_3,%end_5⟩ |}]
|
|
|
|
|
|
|
|
let%expect_test _ =
|
|
|
|
pp
|
|
|
|
(garbage_collect (Sh.star seg_a seg_main)
|
|
|
|
~wrt:(Term.Set.of_list [main])) ;
|
|
|
|
[%expect
|
|
|
|
{|
|
|
|
|
%main_1 -[ %b_4, %n_3 )-> ⟨%n_3,%a_2⟩
|
|
|
|
* %a_2 -[ %b_4, %n_3 )-> ⟨%n_3,%end_5⟩ |}]
|
|
|
|
|
|
|
|
let%expect_test _ =
|
|
|
|
pp
|
|
|
|
(garbage_collect
|
|
|
|
(Sh.star seg_cycle seg_main)
|
|
|
|
~wrt:(Term.Set.of_list [a])) ;
|
|
|
|
[%expect
|
|
|
|
{|
|
|
|
|
%main_1 -[ %b_4, %n_3 )-> ⟨%n_3,%a_2⟩
|
|
|
|
* %a_2 -[ %b_4, %n_3 )-> ⟨%n_3,%main_1⟩ |}]
|
|
|
|
end )
|