(* * Copyright (c) 2016-present, Facebook, Inc. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. *) open! IStd module F = Format module L = Logging (* forward dependency analysis for computing set of variables that affect the looping behavior of the program 1. perform a control flow dependency analysis by getting all the variables that appear in the guards of the loops. 2. for each control dependency per node, find its respective data dependency 3. remove invariant vars in the loop from control vars *) module CFG = ProcCfg.Normal module LoopHead = Procdesc.Node (* For each CV, track the loop head where it is originating from. This is needed for invariant analysis. *) module CVar = struct type t = {cvar: Var.t; loop_head: LoopHead.t} [@@deriving compare] let pp fmt {cvar; loop_head} = F.fprintf fmt "(cvar : %a; loop_head : %a)" Var.pp cvar LoopHead.pp loop_head end module ControlDepSet = AbstractDomain.FiniteSet (CVar) (** Map control var -> loop head location *) module ControlMap = PrettyPrintable.MakePPMap (Var) module GuardNodes = AbstractDomain.FiniteSet (Procdesc.Node) module LoopHeads = Procdesc.NodeSet (** Map exit node -> loop head set *) module ExitNodeToLoopHeads = Procdesc.NodeMap (** Map loop head -> prune nodes in the loop guard *) module LoopHeadToGuardNodes = Procdesc.NodeMap type loop_control_maps = { exit_map: LoopHeads.t ExitNodeToLoopHeads.t ; loop_head_to_guard_nodes: GuardNodes.t LoopHeadToGuardNodes.t } (* forward transfer function for control dependencies *) module TransferFunctionsControlDeps (CFG : ProcCfg.S) = struct module CFG = CFG module Domain = ControlDepSet type extras = loop_control_maps let collect_vars_in_exp exp loop_head = Var.get_all_vars_in_exp exp |> Sequence.fold ~init:ControlDepSet.empty ~f:(fun acc cvar -> ControlDepSet.add {cvar; loop_head} acc ) let find_vars_in_decl id loop_head _ = function | Sil.Load (lhs_id, exp, _, _) when Ident.equal lhs_id id -> collect_vars_in_exp exp loop_head |> Option.some | Sil.Call ((lhs_id, _), _, arg_list, _, _) when Ident.equal lhs_id id -> List.fold_left arg_list ~init:ControlDepSet.empty ~f:(fun deps (exp, _) -> collect_vars_in_exp exp loop_head |> ControlDepSet.union deps ) |> Option.some | _ -> None let get_vars_in_exp exp prune_node loop_head = let program_control_vars = Exp.program_vars exp |> Sequence.fold ~init:ControlDepSet.empty ~f:(fun acc pvar -> ControlDepSet.add {cvar= Var.of_pvar pvar; loop_head} acc ) in Exp.free_vars exp |> Sequence.fold ~init:program_control_vars ~f:(fun acc id -> match Procdesc.Node.find_in_node_or_preds prune_node ~f:(find_vars_in_decl id loop_head) with | Some deps -> ControlDepSet.union deps acc | None -> L.internal_error "Failed to get the definition of the control variable %a in exp %a \n" Ident.pp id Exp.pp exp ; acc ) (* extract vars from the prune instructions in the node *) let get_control_vars loop_nodes loop_head = GuardNodes.fold (fun prune_node acc -> let instrs = Procdesc.Node.get_instrs prune_node in Instrs.fold ~init:acc ~f:(fun astate instr -> match instr with | Sil.Prune (exp, _, _, _) -> Domain.union (get_vars_in_exp exp prune_node loop_head) astate | _ -> (* prune nodes include other instructions like REMOVE_TEMPS or loads *) astate ) instrs ) loop_nodes Domain.empty (* Each time we pass through - a loop header, add control variables(CVs) of its guard nodes, along with the loop header that CV is originating from - a loop exit node, remove control variables of its guard nodes This is correct because the CVs are only going to be temporaries. *) let exec_instr astate {ProcData.extras= {exit_map; loop_head_to_guard_nodes}} (node : CFG.Node.t) _ = let node = CFG.Node.underlying_node node in let astate' = match LoopHeadToGuardNodes.find_opt node loop_head_to_guard_nodes with | Some loop_nodes -> Domain.union astate (get_control_vars loop_nodes node) | _ -> astate in match Procdesc.Node.get_kind node with | Procdesc.Node.Prune_node _ -> ( match ExitNodeToLoopHeads.find_opt node exit_map with | Some loop_heads -> LoopHeads.fold (fun loop_head astate_acc -> match LoopHeadToGuardNodes.find_opt loop_head loop_head_to_guard_nodes with | Some guard_nodes -> L.(debug Analysis Medium) "@\n>>>Exit node %a, Loop head %a, guard nodes=%a @\n\n" Procdesc.Node.pp node Procdesc.Node.pp loop_head GuardNodes.pp guard_nodes ; get_control_vars guard_nodes loop_head |> Domain.diff astate_acc | _ -> (* Every loop head must have a guard node *) assert false ) loop_heads astate' | _ -> astate' ) | _ -> (* Exit node must be a prune node *) assert (not (ExitNodeToLoopHeads.mem node exit_map)) ; astate' let pp_session_name node fmt = F.fprintf fmt "control dependency analysis %a" CFG.Node.pp_id (CFG.Node.id node) end module ControlDepAnalyzer = AbstractInterpreter.MakeRPO (TransferFunctionsControlDeps (CFG)) type invariant_map = ControlDepAnalyzer.invariant_map let compute_invariant_map pdesc tenv control_maps : invariant_map = let proc_data = ProcData.make pdesc tenv control_maps in let node_cfg = CFG.from_pdesc pdesc in ControlDepAnalyzer.exec_cfg node_cfg proc_data ~initial:ControlDepSet.empty (* Filter CVs which are invariant in the loop where the CV originated from *) let remove_invariant_vars control_vars loop_inv_map = ControlDepSet.fold (fun {cvar; loop_head} acc -> match LoopInvariant.LoopHeadToInvVars.find_opt loop_head loop_inv_map with | Some inv_vars -> if LoopInvariant.InvariantVars.mem cvar inv_vars then acc else ControlMap.add cvar (Procdesc.Node.get_loc loop_head) acc | _ -> (* Each cvar is always attached to a loop head *) assert false ) control_vars ControlMap.empty let compute_control_vars control_invariant_map loop_inv_map node = let node_id = CFG.Node.id node in let deps = ControlMap.empty in ControlDepAnalyzer.extract_post node_id control_invariant_map |> Option.map ~f:(fun control_deps -> (* loop_inv_map: loop head -> variables that are invariant in the loop *) L.(debug Analysis Medium) "@\n>>> Control dependencies of node %a : %a @\n" Procdesc.Node.pp node ControlDepSet.pp control_deps ; remove_invariant_vars control_deps loop_inv_map ) |> Option.value ~default:deps