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.
259 lines
6.5 KiB
259 lines
6.5 KiB
(*
|
|
* Copyright (c) 2009 - 2013 Monoidics ltd.
|
|
* 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.
|
|
*)
|
|
|
|
(** Module for Names and Identifiers *)
|
|
|
|
open! IStd
|
|
module Hashtbl = Caml.Hashtbl
|
|
module L = Logging
|
|
module F = Format
|
|
|
|
module Name = struct
|
|
type t = Primed | Normal | Footprint | Spec | FromString of string [@@deriving compare]
|
|
|
|
let primed = "t"
|
|
|
|
let normal = "n"
|
|
|
|
let footprint = "f"
|
|
|
|
let spec = "val"
|
|
|
|
let from_string s = FromString s
|
|
|
|
let to_string = function
|
|
| Primed ->
|
|
primed
|
|
| Normal ->
|
|
normal
|
|
| Footprint ->
|
|
footprint
|
|
| Spec ->
|
|
spec
|
|
| FromString s ->
|
|
s
|
|
end
|
|
|
|
type name = Name.t [@@deriving compare]
|
|
|
|
let name_spec = Name.Spec
|
|
|
|
let equal_name = [%compare.equal : name]
|
|
|
|
type kind =
|
|
| KNone
|
|
(** special kind of "null ident" (basically, a more compact way of implementing an ident option).
|
|
useful for situations when an instruction requires an id, but no one should read the result. *)
|
|
| KFootprint
|
|
| KNormal
|
|
| KPrimed
|
|
[@@deriving compare]
|
|
|
|
let kfootprint = KFootprint
|
|
|
|
let knormal = KNormal
|
|
|
|
let kprimed = KPrimed
|
|
|
|
let equal_kind = [%compare.equal : kind]
|
|
|
|
(* timestamp for a path identifier *)
|
|
let path_ident_stamp = -3
|
|
|
|
type t = {kind: kind; name: Name.t; stamp: int} [@@deriving compare]
|
|
|
|
(* most unlikely first *)
|
|
let equal i1 i2 =
|
|
Int.equal i1.stamp i2.stamp && equal_kind i1.kind i2.kind && equal_name i1.name i2.name
|
|
|
|
|
|
(** {2 Set for identifiers} *)
|
|
module Set = Caml.Set.Make (struct
|
|
type nonrec t = t
|
|
|
|
let compare = compare
|
|
end)
|
|
|
|
module Map = Caml.Map.Make (struct
|
|
type nonrec t = t
|
|
|
|
let compare = compare
|
|
end)
|
|
|
|
module Hash = Hashtbl.Make (struct
|
|
type nonrec t = t
|
|
|
|
let equal = equal
|
|
|
|
let hash (id: t) = Hashtbl.hash id
|
|
end)
|
|
|
|
let idlist_to_idset ids = List.fold ~f:(fun set id -> Set.add id set) ~init:Set.empty ids
|
|
|
|
(** {2 Conversion between Names and Strings} *)
|
|
module NameHash = Hashtbl.Make (struct
|
|
type t = name
|
|
|
|
let equal = equal_name
|
|
|
|
let hash = Hashtbl.hash
|
|
end)
|
|
|
|
(** Convert a string to a name *)
|
|
let string_to_name = Name.from_string
|
|
|
|
(** Convert a name to a string. *)
|
|
let name_to_string = Name.to_string
|
|
|
|
(** {2 Functions and Hash Tables for Managing Stamps} *)
|
|
|
|
(** Set the stamp of the identifier *)
|
|
let set_stamp i stamp = {i with stamp}
|
|
|
|
(** Get the stamp of the identifier *)
|
|
let get_stamp i = i.stamp
|
|
|
|
module NameGenerator = struct
|
|
type t = int NameHash.t
|
|
|
|
let create () : t = NameHash.create 17
|
|
|
|
(** Map from names to stamps. *)
|
|
let name_map = ref (create ())
|
|
|
|
let get_current () = !name_map
|
|
|
|
let set_current map = name_map := map
|
|
|
|
(** Reset the name generator *)
|
|
let reset () = name_map := create ()
|
|
|
|
(** Create a fresh identifier with the given kind and name. *)
|
|
let create_fresh_ident kind name =
|
|
let stamp =
|
|
try
|
|
let stamp = NameHash.find !name_map name in
|
|
NameHash.replace !name_map name (stamp + 1) ;
|
|
stamp + 1
|
|
with Not_found ->
|
|
NameHash.add !name_map name 0 ;
|
|
0
|
|
in
|
|
{kind; name; stamp}
|
|
|
|
|
|
(** Make sure that fresh ids after whis one will be with different stamps *)
|
|
let update_name_hash name stamp =
|
|
try
|
|
let curr_stamp = NameHash.find !name_map name in
|
|
let new_stamp = max curr_stamp stamp in
|
|
NameHash.replace !name_map name new_stamp
|
|
with Not_found -> NameHash.add !name_map name stamp
|
|
end
|
|
|
|
(** Name used for the return variable *)
|
|
let name_return = Mangled.from_string "return"
|
|
|
|
(** Return the standard name for the given kind *)
|
|
let standard_name kind =
|
|
if equal_kind kind KNormal || equal_kind kind KNone then Name.Normal
|
|
else if equal_kind kind KFootprint then Name.Footprint
|
|
else Name.Primed
|
|
|
|
|
|
(** Every identifier with a given stamp should unltimately be created using this function *)
|
|
let create_with_stamp kind name stamp =
|
|
NameGenerator.update_name_hash name stamp ;
|
|
{kind; name; stamp}
|
|
|
|
|
|
(** Create an identifier with default name for the given kind *)
|
|
let create kind stamp = create_with_stamp kind (standard_name kind) stamp
|
|
|
|
(** Generate a normal identifier with the given name and stamp *)
|
|
let create_normal name stamp = create_with_stamp KNormal name stamp
|
|
|
|
(** Create a fresh identifier with default name for the given kind. *)
|
|
let create_fresh kind = NameGenerator.create_fresh_ident kind (standard_name kind)
|
|
|
|
let create_none () = create_fresh KNone
|
|
|
|
(** Generate a footprint identifier with the given name and stamp *)
|
|
let create_footprint name stamp = create_with_stamp KFootprint name stamp
|
|
|
|
(** {2 Functions for Identifiers} *)
|
|
|
|
(** Get a name of an identifier *)
|
|
let get_name id = id.name
|
|
|
|
let has_kind id kind = equal_kind id.kind kind
|
|
|
|
let is_primed (id: t) = has_kind id KPrimed
|
|
|
|
let is_normal (id: t) = has_kind id KNormal || has_kind id KNone
|
|
|
|
let is_footprint (id: t) = has_kind id KFootprint
|
|
|
|
let is_none (id: t) = has_kind id KNone
|
|
|
|
let is_path (id: t) = has_kind id KNormal && Int.equal id.stamp path_ident_stamp
|
|
|
|
(** Update the name generator so that the given id's are not generated again *)
|
|
let update_name_generator ids =
|
|
let upd id = ignore (create_with_stamp id.kind id.name id.stamp) in
|
|
List.iter ~f:upd ids
|
|
|
|
|
|
(** Generate a normal identifier whose name encodes a path given as a string. *)
|
|
let create_path pathstring =
|
|
create_normal (string_to_name ("%path%" ^ pathstring)) path_ident_stamp
|
|
|
|
|
|
(** {2 Pretty Printing} *)
|
|
|
|
(** Convert an identifier to a string. *)
|
|
let to_string id =
|
|
if has_kind id KNone then "_"
|
|
else
|
|
let base_name = name_to_string id.name in
|
|
let prefix = if has_kind id KFootprint then "@" else if has_kind id KNormal then "" else "_" in
|
|
let suffix = "$" ^ string_of_int id.stamp in
|
|
prefix ^ base_name ^ suffix
|
|
|
|
|
|
(** Pretty print a name. *)
|
|
let pp_name f name = F.fprintf f "%s" (name_to_string name)
|
|
|
|
(** Pretty print an identifier. *)
|
|
let pp f id = F.fprintf f "%s" (to_string id)
|
|
|
|
(** pretty printer for lists of identifiers *)
|
|
let pp_list = Pp.comma_seq pp
|
|
|
|
module HashQueue = Hash_queue.Make (struct
|
|
type nonrec t = t
|
|
|
|
let compare = compare
|
|
|
|
let hash = Hashtbl.hash
|
|
|
|
let sexp_of_t id = Sexp.of_string (to_string id)
|
|
end)
|
|
|
|
let hashqueue_of_sequence ?init s =
|
|
let q = match init with None -> HashQueue.create () | Some q0 -> q0 in
|
|
Sequence.iter s ~f:(fun id ->
|
|
let _ : [`Key_already_present | `Ok] = HashQueue.enqueue q id () in
|
|
() ) ;
|
|
q
|
|
|
|
|
|
let set_of_sequence ?(init= Set.empty) s = Sequence.fold s ~init ~f:(fun ids id -> Set.add id ids)
|