(* * Copyright (c) 2016 - 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 module F = Format module L = Logging module type S = sig include Trace.S (** a path from some procedure via the given passthroughs to the given call stack, with passthroughs for each callee *) type sink_path = Passthroughs.t * (Sink.t * Passthroughs.t) list (** get a path for each of the reportable flows to a sink in this trace *) val get_reportable_sink_paths : t -> trace_of_pname:(Procname.t -> t) -> sink_path list (** update sink with the given call site *) val with_callsite : t -> CallSite.t -> t val to_sink_loc_trace : ?desc_of_sink:(Sink.t -> string) -> ?sink_should_nest:(Sink.t -> bool) -> sink_path -> Errlog.loc_trace_elem list end module MakeSink(TraceElem : TraceElem.S) = struct include TraceElem type parameter = { sink : t; index : int; report_reachable : bool; } let get _ _ = [] end module Make (TraceElem : TraceElem.S) = struct include Trace.Make(struct module Source = Source.Dummy module Sink = MakeSink(TraceElem) let should_report _ _ = true end) type sink_path = Passthroughs.t * (Sink.t * Passthroughs.t) list let initial = let dummy_source = () in of_source dummy_source let get_reportable_sink_paths t ~trace_of_pname = IList.map (fun (passthroughs, _, sinks) -> passthroughs, sinks) (get_reportable_paths t ~trace_of_pname) let to_sink_loc_trace ?desc_of_sink ?sink_should_nest (passthroughs, sinks) = to_loc_trace ?desc_of_sink ?sink_should_nest (passthroughs, [], sinks) let with_callsite t call_site = IList.fold_left (fun t_acc sink -> let callee_sink = Sink.with_callsite sink call_site in add_sink callee_sink t_acc) initial (Sinks.elements (sinks t)) let pp fmt t = let pp_passthroughs_if_not_empty fmt p = if not (Passthroughs.is_empty p) then F.fprintf fmt " via %a" Passthroughs.pp p in F.fprintf fmt "%a%a" Sinks.pp (sinks t) pp_passthroughs_if_not_empty (passthroughs t) end