Reviewed By: jberdine Differential Revision: D4271772 fbshipit-source-id: 3f9681dmaster
parent
4b98543d35
commit
ff3b6a10db
@ -0,0 +1,72 @@
|
||||
(*
|
||||
* 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.
|
||||
*)
|
||||
|
||||
(** tree of (trace, access path) associations organized by structure of access paths *)
|
||||
module type S = sig
|
||||
module TraceDomain : AbstractDomain.S
|
||||
module AccessMap = AccessPath.AccessMap
|
||||
module BaseMap = AccessPath.BaseMap
|
||||
|
||||
type node = TraceDomain.astate * tree
|
||||
and tree =
|
||||
| Subtree of node AccessMap.t (** map from access -> nodes. a leaf is encoded as an empty map *)
|
||||
| Star (** special leaf for starred access paths *)
|
||||
|
||||
(** map from base var -> access subtree. Here's how to represent a few different kinds of
|
||||
trace * access path associations:
|
||||
(x, T) := { x |-> (T, Subtree {}) }
|
||||
(x.f, T) := { x |-> (empty, Subtree { f |-> (T, Subtree {}) }) }
|
||||
(x*, T) := { x |-> (T, Star) }
|
||||
(x.f*, T) := { x |-> (empty, Subtree { f |-> (T, Star) }) }
|
||||
(x, T1), (y, T2) := { x |-> (T1, Subtree {}), y |-> (T2, Subtree {}) }
|
||||
(x.f, T1), (x.g, T2) := { x |-> (empty, Subtree { f |-> (T1, Subtree {}),
|
||||
g |-> (T2, Subtree {}) }) }
|
||||
*)
|
||||
type t = node BaseMap.t
|
||||
|
||||
include AbstractDomain.S with type astate = t
|
||||
|
||||
val empty : t
|
||||
|
||||
val empty_node : node
|
||||
|
||||
val make_node : TraceDomain.astate -> node AccessMap.t -> node
|
||||
|
||||
(** for testing only *)
|
||||
val make_access_node : TraceDomain.astate -> AccessPath.access -> TraceDomain.astate -> node
|
||||
|
||||
(** create a leaf node with no successors *)
|
||||
val make_normal_leaf : TraceDomain.astate -> node
|
||||
|
||||
(** create a leaf node with a wildcard successor *)
|
||||
val make_starred_leaf : TraceDomain.astate -> node
|
||||
|
||||
(** retrieve the node associated with the given access path *)
|
||||
val get_node : AccessPath.t -> t -> node option
|
||||
|
||||
(** retrieve the trace associated with the given access path *)
|
||||
val get_trace : AccessPath.t -> t -> TraceDomain.astate option
|
||||
|
||||
(** add the given access path to the tree and associate its last access with with the given node.
|
||||
if any of the accesses in the path are not already present in the tree, they will be added
|
||||
with with empty traces associated with each of the inner nodes. *)
|
||||
val add_node : AccessPath.t -> node -> t -> t
|
||||
|
||||
(** add the given access path to the tree and associate its last access with with the given trace.
|
||||
if any of the accesses in the path are not already present in the tree, they will be added
|
||||
with with empty traces associated with each of the inner nodes. *)
|
||||
val add_trace : AccessPath.t -> TraceDomain.astate -> t -> t
|
||||
|
||||
(** apply a function to each (access path, node) pair in the tree. *)
|
||||
val fold : ('a -> AccessPath.t -> TraceDomain.astate -> 'a) -> t -> 'a -> 'a
|
||||
|
||||
val pp_node : Format.formatter -> node -> unit
|
||||
end
|
||||
|
||||
module Make (TraceDomain : AbstractDomain.S) : S with module TraceDomain = TraceDomain
|
Loading…
Reference in new issue