(* * 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 Utils module L = Logging (** Module to register and invoke callbacks *) (** Inline a synthetic (access or bridge) method. *) let inline_synthetic_method ret_ids etl proc_desc proc_name loc_call : Sil.instr option = let modified = ref None in let debug = false in let found instr instr' = modified := Some instr'; if debug then begin L.stderr "XX inline_synthetic_method found instr: %a@." (Sil.pp_instr pe_text) instr; L.stderr "XX inline_synthetic_method instr': %a@." (Sil.pp_instr pe_text) instr' end in let do_instr node instr = match instr, ret_ids, etl with | Sil.Letderef (id1, Sil.Lfield (Sil.Var id2, fn, ft), bt, loc), [ret_id], [(e1, t1)] -> (* getter for fields *) let instr' = Sil.Letderef (ret_id, Sil.Lfield (e1, fn, ft), bt, loc_call) in found instr instr' | Sil.Letderef (id1, Sil.Lfield (Sil.Lvar pvar, fn, ft), bt, loc), [ret_id], [] when Sil.pvar_is_global pvar -> (* getter for static fields *) let instr' = Sil.Letderef (ret_id, Sil.Lfield (Sil.Lvar pvar, fn, ft), bt, loc_call) in found instr instr' | Sil.Set (Sil.Lfield (ex1, fn, ft), bt , ex2, loc), _, [(e1, t1); (e2, t2)] -> (* setter for fields *) let instr' = Sil.Set (Sil.Lfield (e1, fn, ft), bt , e2, loc_call) in found instr instr' | Sil.Set (Sil.Lfield (Sil.Lvar pvar, fn, ft), bt , ex2, loc), _, [(e1, t1)] when Sil.pvar_is_global pvar -> (* setter for static fields *) let instr' = Sil.Set (Sil.Lfield (Sil.Lvar pvar, fn, ft), bt , e1, loc_call) in found instr instr' | Sil.Call (ret_ids', Sil.Const (Sil.Cfun pn), etl', loc', cf), _, _ when list_length ret_ids = list_length ret_ids' && list_length etl' = list_length etl -> let instr' = Sil.Call (ret_ids, Sil.Const (Sil.Cfun pn), etl, loc_call, cf) in found instr instr' | Sil.Call (ret_ids', Sil.Const (Sil.Cfun pn), etl', loc', cf), _, _ when list_length ret_ids = list_length ret_ids' && list_length etl' + 1 = list_length etl -> let etl1 = match list_rev etl with (* remove last element *) | _ :: l -> list_rev l | [] -> assert false in let instr' = Sil.Call (ret_ids, Sil.Const (Sil.Cfun pn), etl1, loc_call, cf) in found instr instr' | _ -> () in Cfg.Procdesc.iter_instrs do_instr proc_desc; !modified (** Find synthetic (access or bridge) methods in the procedure and inline them in the cfg. *) let proc_inline_synthetic_methods cfg proc_desc : unit = let instr_inline_synthetic_method = function | Sil.Call (ret_ids, Sil.Const (Sil.Cfun pn), etl, loc, _) -> (match Cfg.Procdesc.find_from_name cfg pn with | Some pd -> let is_access = Procname.java_is_access_method pn in let attributes = Cfg.Procdesc.get_attributes pd in let is_synthetic = attributes.ProcAttributes.is_synthetic_method in let is_bridge = attributes.ProcAttributes.is_bridge_method in if is_access || is_bridge || is_synthetic then inline_synthetic_method ret_ids etl pd pn loc else None | None -> None) | _ -> None in let node_inline_synthetic_methods node = let modified = ref false in let do_instr instr = match instr_inline_synthetic_method instr with | None -> instr | Some instr' -> modified := true; instr' in let instrs = Cfg.Node.get_instrs node in let instrs' = list_map do_instr instrs in if !modified then Cfg.Node.replace_instrs node instrs' in Cfg.Procdesc.iter_nodes node_inline_synthetic_methods proc_desc type proc_callback_t = Procname.t list -> (Procname.t -> Cfg.Procdesc.t option) -> Idenv.t -> Sil.tenv -> Procname.t -> Cfg.Procdesc.t -> unit type cluster_callback_t = Procname.t list -> (Procname.t -> Cfg.Procdesc.t option) -> (Idenv.t * Sil.tenv * Procname.t * Cfg.Procdesc.t) list -> unit let procedure_callbacks = ref [] let cluster_callbacks = ref [] let register_procedure_callback language_opt (callback: proc_callback_t) = procedure_callbacks := (language_opt, callback):: !procedure_callbacks let register_cluster_callback language_opt (callback: cluster_callback_t) = cluster_callbacks := (language_opt, callback):: !cluster_callbacks let unregister_all_callbacks () = procedure_callbacks := []; cluster_callbacks := [] (** Collect what we need to know about a procedure for the analysis. *) let get_procedure_definition exe_env proc_name = let cfg = Exe_env.get_cfg exe_env proc_name in let tenv = Exe_env.get_tenv exe_env proc_name in Option.map (fun proc_desc -> proc_inline_synthetic_methods cfg proc_desc; let idenv = Idenv.create cfg proc_desc and language = (Cfg.Procdesc.get_attributes proc_desc).ProcAttributes.language in (idenv, tenv, proc_name, proc_desc, language)) (Cfg.Procdesc.find_from_name cfg proc_name) let get_language proc_name = if Procname.is_java proc_name then Config.Java else Config.C_CPP (** Invoke all registered procedure callbacks on a set of procedures. *) let iterate_procedure_callbacks all_procs exe_env proc_name = let procedure_language = get_language proc_name in Config.curr_language := procedure_language; let cfg = Exe_env.get_cfg exe_env proc_name in let get_procdesc proc_name = let cfg = try Exe_env.get_cfg exe_env proc_name with Not_found -> cfg in Cfg.Procdesc.find_from_name cfg proc_name in let update_time proc_name elapsed = let prev_summary = Specs.get_summary_unsafe proc_name in let stats_time = prev_summary.Specs.stats.Specs.stats_time +. elapsed in let stats = { prev_summary.Specs.stats with Specs.stats_time = stats_time } in let summary = { prev_summary with Specs.stats = stats } in Specs.add_summary proc_name summary in Option.may (fun (idenv, tenv, proc_name, proc_desc, language) -> list_iter (fun (language_opt, proc_callback) -> let language_matches = match language_opt with | Some language -> language = procedure_language | None -> true in if language_matches then begin let init_time = Unix.gettimeofday () in proc_callback all_procs get_procdesc idenv tenv proc_name proc_desc; let elapsed = Unix.gettimeofday () -. init_time in update_time proc_name elapsed end) !procedure_callbacks) (get_procedure_definition exe_env proc_name) (** Invoke all registered cluster callbacks on a cluster of procedures. *) let iterate_cluster_callbacks all_procs exe_env proc_names = let get_procdesc proc_name = try let cfg = Exe_env.get_cfg exe_env proc_name in Cfg.Procdesc.find_from_name cfg proc_name with Not_found -> None in let procedure_definitions = list_map (get_procedure_definition exe_env) proc_names |> list_flatten_options in let environment = list_map (fun (idenv, tenv, proc_name, proc_desc, _) -> (idenv, tenv, proc_name, proc_desc)) procedure_definitions in (** Procedures matching the given language or all if no language is specified. *) let relevant_procedures language_opt = Option.map_default (fun l -> list_filter (fun p -> l = get_language p) proc_names) proc_names language_opt in list_iter (fun (language_opt, cluster_callback) -> let proc_names = relevant_procedures language_opt in if list_length proc_names > 0 then cluster_callback all_procs get_procdesc environment) !cluster_callbacks (** Invoke all procedure and cluster callbacks on a given environment. *) let iterate_callbacks store_summary call_graph exe_env = let proc_names = Cg.get_defined_nodes call_graph in let saved_language = !Config.curr_language in let cluster_id proc_name = match get_language proc_name with | Config.Java -> Procname.java_get_class proc_name | _ -> "unknown" in let cluster proc_names = let cluster_map = list_fold_left (fun map proc_name -> let proc_cluster = cluster_id proc_name in let bucket = try StringMap.find proc_cluster map with Not_found -> [] in StringMap.add proc_cluster (proc_name:: bucket) map) StringMap.empty proc_names in (* Return all values of the map *) list_map snd (StringMap.bindings cluster_map) in let reset_summary proc_name = let cfg_opt = try Some (Exe_env.get_cfg exe_env proc_name) with | Not_found -> None in let procdesc_opt = match cfg_opt with | Some cfg -> Cfg.Procdesc.find_from_name cfg proc_name | None -> None in let loc = match procdesc_opt with | Some proc_desc -> Cfg.Procdesc.get_loc proc_desc | None -> Location.dummy in Specs.reset_summary call_graph proc_name loc in (* Make sure summaries exists. *) list_iter reset_summary proc_names; (* Invoke callbacks. *) list_iter (iterate_procedure_callbacks proc_names exe_env) proc_names; list_iter (iterate_cluster_callbacks proc_names exe_env) (cluster proc_names); list_iter store_summary proc_names; Config.curr_language := saved_language