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.

148 lines
3.6 KiB

(*
* 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
let all_formals_untainted pdesc =
let make_untainted (name, typ) = (name, typ, None) in
List.map ~f:make_untainted (Procdesc.get_formals pdesc)
module type Kind = sig
include TraceElem.Kind
val unknown : t
val get : Typ.Procname.t -> Tenv.t -> (t * int option) option
val get_tainted_formals : Procdesc.t -> Tenv.t -> (Mangled.t * Typ.t * t option) list
end
module type S = sig
include TraceElem.S
type spec = {source: t; index: int option}
val is_footprint : t -> bool
val make_footprint : AccessPath.t -> Procdesc.t -> t
val get_footprint_access_path : t -> AccessPath.t option
val get : CallSite.t -> Tenv.t -> spec option
val get_tainted_formals : Procdesc.t -> Tenv.t -> (Mangled.t * Typ.t * t option) list
end
module Make (Kind : Kind) = struct
module Kind = Kind
type kind =
| Normal of Kind.t (** known source returned directly or transitively from a callee *)
| Footprint of AccessPath.t (** unknown source read from the environment *)
[@@deriving compare]
let pp_kind fmt = function
| Normal kind
-> Kind.pp fmt kind
| Footprint ap
-> F.fprintf fmt "Footprint(%a)" AccessPath.pp ap
type t = {kind: kind; site: CallSite.t} [@@deriving compare]
type spec = {source: t; index: int option}
let is_footprint t = match t.kind with Footprint _ -> true | _ -> false
let get_footprint_access_path t = match t.kind with Footprint ap -> Some ap | _ -> None
let call_site t = t.site
let kind t = match t.kind with Normal kind -> kind | Footprint _ -> Kind.unknown
let make ?indexes:_ kind site = {site; kind= Normal kind}
let make_footprint ap pdesc =
let kind = Footprint ap in
let site = CallSite.make (Procdesc.get_proc_name pdesc) (Procdesc.get_loc pdesc) in
{site; kind}
let get site tenv =
match Kind.get (CallSite.pname site) tenv with
| Some (kind, index)
-> let source = make kind site in
Some {source; index}
| None
-> None
let get_tainted_formals pdesc tenv =
let site = CallSite.make (Procdesc.get_proc_name pdesc) (Procdesc.get_loc pdesc) in
List.map
~f:(fun (name, typ, kind_opt) ->
(name, typ, Option.map kind_opt ~f:(fun kind -> make kind site)))
(Kind.get_tainted_formals pdesc tenv)
let pp fmt s = F.fprintf fmt "%a(%a)" pp_kind s.kind CallSite.pp s.site
let with_callsite t callee_site =
if is_footprint t then failwithf "Can't change the call site of footprint source %a" pp t ;
{t with site= callee_site}
module Set = PrettyPrintable.MakePPSet (struct
type nonrec t = t
let compare = compare
let pp = pp
end)
end
module Dummy = struct
type t = unit [@@deriving compare]
type spec = {source: t; index: int option}
let call_site _ = CallSite.dummy
let kind t = t
let make ?indexes:_ kind _ = kind
let pp _ () = ()
let is_footprint _ = false
let make_footprint _ _ = assert false
let get_footprint_access_path _ = assert false
let get _ _ = None
let get_tainted_formals pdesc _ =
List.map ~f:(fun (name, typ) -> (name, typ, None)) (Procdesc.get_formals pdesc)
module Kind = struct
type nonrec t = t
let compare = compare
let pp = pp
end
module Set = PrettyPrintable.MakePPSet (struct
type nonrec t = t
let compare = compare
let pp = pp
end)
let with_callsite t _ = t
end