(** Signature for a module describing an identifier. We often need identifiers (for definitions, variables, etc.) and in order to make sure we don't mix them, we use a generative functor (see [IdGen]). *) module type Id = sig type id type generator (** Id generator - simply a counter *) type 'a vector val zero : id val generator_zero : generator val fresh : generator -> id * generator val to_string : id -> string val pp_id : Format.formatter -> id -> unit val show_id : id -> string val pp_generator : Format.formatter -> generator -> unit val show_generator : generator -> string val to_int : id -> int val of_int : id -> int val pp_vector : (Format.formatter -> 'a -> unit) -> Format.formatter -> 'a vector -> unit val show_vector : (Format.formatter -> 'a -> unit) -> 'a vector -> string val empty_vector : 'a vector val vector_to_list : 'a vector -> 'a list val vector_of_list : 'a list -> 'a vector val length : 'a vector -> int val nth : 'a vector -> id -> 'a val nth_opt : 'a vector -> id -> 'a option val update_nth : 'a vector -> id -> 'a -> 'a vector (** Update the nth element of the vector. Raises [Invalid_argument] if the identifier is out of range. *) val iter : ('a -> unit) -> 'a vector -> unit val map : ('a -> 'b) -> 'a vector -> 'b vector val mapi : (id -> 'a -> 'b) -> 'a vector -> 'b vector val mapi_from1 : (id -> 'a -> 'b) -> 'a vector -> 'b vector (** Same as [mapi], but where the indices start with 1 *) val for_all : ('a -> bool) -> 'a vector -> bool val exists : ('a -> bool) -> 'a vector -> bool module Ord : Map.OrderedType with type t = id module Set : Set.S with type elt = id val set_to_string : Set.t -> string module Map : Map.S with type key = id val id_of_json : Yojson.Basic.t -> (id, string) result val vector_of_json : (Yojson.Basic.t -> ('a, string) result) -> Yojson.Basic.t -> ('a vector, string) result end (** Generative functor for identifiers. See [Id]. *) module IdGen () : Id = struct (* TODO: use Z.t *) type id = int [@@deriving show] type generator = id [@@deriving show] type 'a vector = 'a list [@@deriving show] let zero = 0 let generator_zero = 0 let incr x = (* Identifiers should never overflow (because max_int is a really big * value - but we really want to make sure we detect overflows if * they happen *) if x == max_int then raise (Errors.IntegerOverflow ()) else x + 1 let fresh gen = (gen, incr gen) let to_string = string_of_int let to_int x = x let of_int x = x let empty_vector = [] let vector_to_list v = v let vector_of_list v = v let length v = List.length v let nth v id = List.nth v id let nth_opt v id = List.nth_opt v id let rec update_nth vec id v = match (vec, id) with | [], _ -> raise (Invalid_argument "Out of range") | _ :: vec', 0 -> v :: vec' | x :: vec', _ -> x :: update_nth vec' (id - 1) v let iter = List.iter let map = List.map let mapi = List.mapi let mapi_from1 f ls = let rec aux i ls = match ls with [] -> [] | x :: ls' -> f i x :: aux (i + 1) ls' in aux 1 ls let for_all = List.for_all let exists = List.exists module Ord = struct type t = id let compare = compare end module Set = Set.Make (Ord) module Map = Map.Make (Ord) let set_to_string ids = let ids = Set.fold (fun id ids -> to_string id :: ids) ids [] in "{" ^ String.concat ", " (List.rev ids) ^ "}" let id_of_json js = (* TODO: check boundaries ? *) match js with | `Int i -> Ok i | _ -> Error ("id_of_json: failed on " ^ Yojson.Basic.show js) let vector_of_json a_of_json js = OfJsonBasic.combine_error_msgs js "vector_of_json" (OfJsonBasic.list_of_json a_of_json js) end type name = string list [@@deriving show] (** A name such as: `std::collections::vector` (which would be represented as [["std"; "collections"; "vector"]]) *)