(* * Copyright (c) 2016 - present * * Programming Research Laboratory (ROPAS) * Seoul National University, Korea * 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 Allocsite = struct include String let pp fmt s = Format.pp_print_string fmt s let make x = x let unknown = "Unknown" end module Loc = struct type t = Var of Var.t | Allocsite of Allocsite.t | Field of t * Typ.Fieldname.t [@@deriving compare] let equal = [%compare.equal : t] let unknown = Allocsite Allocsite.unknown let rec pp fmt = function | Var v -> Var.pp F.str_formatter v ; let s = F.flush_str_formatter () in if Char.equal s.[0] '&' then F.pp_print_string fmt (String.sub s ~pos:1 ~len:(String.length s - 1)) else F.pp_print_string fmt s | Allocsite a -> Allocsite.pp fmt a | Field (l, f) -> F.fprintf fmt "%a.%a" pp l Typ.Fieldname.pp f let is_var = function Var _ -> true | _ -> false let rec contains_allocsite = function | Var _ -> false | Allocsite _ -> true | Field (loc, _) -> contains_allocsite loc let of_var v = Var v let of_allocsite a = Allocsite a let of_pvar pvar = Var (Var.of_pvar pvar) let of_id id = Var (Var.of_id id) let append_field l ~fn = Field (l, fn) let is_return = function | Var (Var.ProgramVar x) -> Mangled.equal (Pvar.get_name x) Ident.name_return | _ -> false let is_field_of ~loc ~field_loc = match field_loc with Field (l, _) -> equal loc l | _ -> false end module PowLoc = struct include AbstractDomain.FiniteSet (Loc) let bot = empty let is_bot = is_empty let unknown = singleton Loc.unknown let append_field ploc ~fn = if is_bot ploc then singleton Loc.unknown else fold (fun l -> add (Loc.append_field l ~fn)) ploc empty let is_singleton x = Int.equal (cardinal x) 1 end