ocaml/typing/env.ml

1486 lines
47 KiB
OCaml

(***********************************************************************)
(* *)
(* OCaml *)
(* *)
(* Xavier Leroy, projet Cristal, INRIA Rocquencourt *)
(* *)
(* Copyright 1996 Institut National de Recherche en Informatique et *)
(* en Automatique. All rights reserved. This file is distributed *)
(* under the terms of the Q Public License version 1.0. *)
(* *)
(***********************************************************************)
(* Environment handling *)
open Cmi_format
open Config
open Misc
open Asttypes
open Longident
open Path
open Types
open Btype
let add_delayed_check_forward = ref (fun _ -> assert false)
let value_declarations : ((string * Location.t), (unit -> unit)) Hashtbl.t =
Hashtbl.create 16
(* This table is used to usage of value declarations. A declaration is
identified with its name and location. The callback attached to a
declaration is called whenever the value is used explicitly
(lookup_value) or implicitly (inclusion test between signatures,
cf Includemod.value_descriptions). *)
let type_declarations = Hashtbl.create 16
type constructor_usage = Positive | Pattern | Privatize
type constructor_usages =
{
mutable cu_positive: bool;
mutable cu_pattern: bool;
mutable cu_privatize: bool;
}
let add_constructor_usage cu = function
| Positive -> cu.cu_positive <- true
| Pattern -> cu.cu_pattern <- true
| Privatize -> cu.cu_privatize <- true
let constructor_usages () =
{cu_positive = false; cu_pattern = false; cu_privatize = false}
let used_constructors :
(string * Location.t * string, (constructor_usage -> unit)) Hashtbl.t
= Hashtbl.create 16
let prefixed_sg = Hashtbl.create 113
type error =
| Illegal_renaming of string * string
| Inconsistent_import of string * string * string
| Need_recursive_types of string * string
exception Error of error
module EnvLazy : sig
type ('a,'b) t
val force : ('a -> 'b) -> ('a,'b) t -> 'b
val create : 'a -> ('a,'b) t
end = struct
type ('a,'b) t = ('a,'b) eval ref
and ('a,'b) eval =
Done of 'b
| Raise of exn
| Thunk of 'a
let force f x =
match !x with
Done x -> x
| Raise e -> raise e
| Thunk e ->
try
let y = f e in
x := Done y;
y
with e ->
x := Raise e;
raise e
let create x =
let x = ref (Thunk x) in
x
end
type summary =
Env_empty
| Env_value of summary * Ident.t * value_description
| Env_type of summary * Ident.t * type_declaration
| Env_exception of summary * Ident.t * exception_declaration
| Env_module of summary * Ident.t * module_type
| Env_modtype of summary * Ident.t * modtype_declaration
| Env_class of summary * Ident.t * class_declaration
| Env_cltype of summary * Ident.t * class_type_declaration
| Env_open of summary * Path.t
module EnvTbl =
struct
(* A table indexed by identifier, with an extra slot to record usage. *)
type 'a t = ('a * bool ref) Ident.tbl
let empty = Ident.empty
let dummy_slot = ref true
let current_slot = ref dummy_slot
let add id x tbl =
Ident.add id (x, !current_slot) tbl
let add_dont_track id x tbl =
Ident.add id (x, dummy_slot) tbl
let find_same_not_using id tbl =
fst (Ident.find_same id tbl)
let find_same id tbl =
let (x, slot) = Ident.find_same id tbl in
slot := true;
x
let find_name s tbl =
let (x, slot) = Ident.find_name s tbl in
slot := true;
x
let find_all s tbl =
let xs = Ident.find_all s tbl in
List.map (fun (x, slot) -> (x, (fun () -> slot := true))) xs
let with_slot slot f x =
let old_slot = !current_slot in
current_slot := slot;
try_finally
(fun () -> f x)
(fun () -> current_slot := old_slot)
let fold_name f = Ident.fold_name (fun k (d,_) -> f k d)
let keys tbl = Ident.fold_all (fun k _ accu -> k::accu) tbl []
end
type type_descriptions =
constructor_description list * label_description list
type t = {
values: (Path.t * value_description) EnvTbl.t;
constrs: constructor_description EnvTbl.t;
labels: label_description EnvTbl.t;
types: (Path.t * (type_declaration * type_descriptions)) EnvTbl.t;
modules: (Path.t * module_type) EnvTbl.t;
modtypes: (Path.t * modtype_declaration) EnvTbl.t;
components: (Path.t * module_components) EnvTbl.t;
classes: (Path.t * class_declaration) EnvTbl.t;
cltypes: (Path.t * class_type_declaration) EnvTbl.t;
summary: summary;
local_constraints: bool;
gadt_instances: (int * TypeSet.t ref) list;
in_signature: bool;
}
and module_components =
(t * Subst.t * Path.t * Types.module_type, module_components_repr) EnvLazy.t
and module_components_repr =
Structure_comps of structure_components
| Functor_comps of functor_components
and structure_components = {
mutable comp_values: (string, (value_description * int)) Tbl.t;
mutable comp_constrs: (string, (constructor_description * int) list) Tbl.t;
mutable comp_labels: (string, (label_description * int) list) Tbl.t;
mutable comp_types:
(string, ((type_declaration * type_descriptions) * int)) Tbl.t;
mutable comp_modules:
(string, ((Subst.t * Types.module_type,module_type) EnvLazy.t * int)) Tbl.t;
mutable comp_modtypes: (string, (modtype_declaration * int)) Tbl.t;
mutable comp_components: (string, (module_components * int)) Tbl.t;
mutable comp_classes: (string, (class_declaration * int)) Tbl.t;
mutable comp_cltypes: (string, (class_type_declaration * int)) Tbl.t
}
and functor_components = {
fcomp_param: Ident.t; (* Formal parameter *)
fcomp_arg: module_type; (* Argument signature *)
fcomp_res: module_type; (* Result signature *)
fcomp_env: t; (* Environment in which the result signature makes sense *)
fcomp_subst: Subst.t; (* Prefixing substitution for the result signature *)
fcomp_cache: (Path.t, module_components) Hashtbl.t (* For memoization *)
}
let subst_modtype_maker (subst, mty) = Subst.modtype subst mty
let empty = {
values = EnvTbl.empty; constrs = EnvTbl.empty;
labels = EnvTbl.empty; types = EnvTbl.empty;
modules = EnvTbl.empty; modtypes = EnvTbl.empty;
components = EnvTbl.empty; classes = EnvTbl.empty;
cltypes = EnvTbl.empty;
summary = Env_empty; local_constraints = false; gadt_instances = [];
in_signature = false;
}
let in_signature env = {env with in_signature = true}
let diff_keys is_local tbl1 tbl2 =
let keys2 = EnvTbl.keys tbl2 in
List.filter
(fun id ->
is_local (EnvTbl.find_same_not_using id tbl2) &&
try ignore (EnvTbl.find_same_not_using id tbl1); false
with Not_found -> true)
keys2
let is_ident = function
Pident _ -> true
| Pdot _ | Papply _ -> false
let is_local (p, _) = is_ident p
let is_local_exn = function
| {cstr_tag = Cstr_exception (p, _)} -> is_ident p
| _ -> false
let diff env1 env2 =
diff_keys is_local env1.values env2.values @
diff_keys is_local_exn env1.constrs env2.constrs @
diff_keys is_local env1.modules env2.modules @
diff_keys is_local env1.classes env2.classes
(* Forward declarations *)
let components_of_module' =
ref ((fun env sub path mty -> assert false) :
t -> Subst.t -> Path.t -> module_type -> module_components)
let components_of_module_maker' =
ref ((fun (env, sub, path, mty) -> assert false) :
t * Subst.t * Path.t * module_type -> module_components_repr)
let components_of_functor_appl' =
ref ((fun f p1 p2 -> assert false) :
functor_components -> Path.t -> Path.t -> module_components)
let check_modtype_inclusion =
(* to be filled with Includemod.check_modtype_inclusion *)
ref ((fun env mty1 path1 mty2 -> assert false) :
t -> module_type -> Path.t -> module_type -> unit)
(* The name of the compilation unit currently compiled.
"" if outside a compilation unit. *)
let current_unit = ref ""
(* Persistent structure descriptions *)
type pers_struct =
{ ps_name: string;
ps_sig: signature;
ps_comps: module_components;
ps_crcs: (string * Digest.t) list;
ps_filename: string;
ps_flags: pers_flags list }
let persistent_structures =
(Hashtbl.create 17 : (string, pers_struct option) Hashtbl.t)
(* Consistency between persistent structures *)
let crc_units = Consistbl.create()
let check_consistency filename crcs =
try
List.iter
(fun (name, crc) -> Consistbl.check crc_units name crc filename)
crcs
with Consistbl.Inconsistency(name, source, auth) ->
raise(Error(Inconsistent_import(name, auth, source)))
(* Reading persistent structures from .cmi files *)
let read_pers_struct modname filename =
let cmi = read_cmi filename in
let name = cmi.cmi_name in
let sign = cmi.cmi_sign in
let crcs = cmi.cmi_crcs in
let flags = cmi.cmi_flags in
let comps =
!components_of_module' empty Subst.identity
(Pident(Ident.create_persistent name))
(Mty_signature sign) in
let ps = { ps_name = name;
ps_sig = sign;
ps_comps = comps;
ps_crcs = crcs;
ps_filename = filename;
ps_flags = flags } in
if ps.ps_name <> modname then
raise(Error(Illegal_renaming(ps.ps_name, filename)));
check_consistency filename ps.ps_crcs;
List.iter
(function Rectypes ->
if not !Clflags.recursive_types then
raise(Error(Need_recursive_types(ps.ps_name, !current_unit))))
ps.ps_flags;
Hashtbl.add persistent_structures modname (Some ps);
ps
let find_pers_struct name =
if name = "*predef*" then raise Not_found;
let r =
try Some (Hashtbl.find persistent_structures name)
with Not_found -> None
in
match r with
| Some None -> raise Not_found
| Some (Some sg) -> sg
| None ->
let filename =
try find_in_path_uncap !load_path (name ^ ".cmi")
with Not_found ->
Hashtbl.add persistent_structures name None;
raise Not_found
in
read_pers_struct name filename
let reset_cache () =
current_unit := "";
Hashtbl.clear persistent_structures;
Consistbl.clear crc_units;
Hashtbl.clear value_declarations;
Hashtbl.clear type_declarations;
Hashtbl.clear used_constructors;
Hashtbl.clear prefixed_sg
let reset_cache_toplevel () =
(* Delete 'missing cmi' entries from the cache. *)
let l =
Hashtbl.fold
(fun name r acc -> if r = None then name :: acc else acc)
persistent_structures []
in
List.iter (Hashtbl.remove persistent_structures) l;
Hashtbl.clear value_declarations;
Hashtbl.clear type_declarations;
Hashtbl.clear used_constructors;
Hashtbl.clear prefixed_sg
let set_unit_name name =
current_unit := name
(* Lookup by identifier *)
let rec find_module_descr path env =
match path with
Pident id ->
begin try
let (p, desc) = EnvTbl.find_same id env.components
in desc
with Not_found ->
if Ident.persistent id
then (find_pers_struct (Ident.name id)).ps_comps
else raise Not_found
end
| Pdot(p, s, pos) ->
begin match
EnvLazy.force !components_of_module_maker' (find_module_descr p env)
with
Structure_comps c ->
let (descr, pos) = Tbl.find s c.comp_components in
descr
| Functor_comps f ->
raise Not_found
end
| Papply(p1, p2) ->
begin match
EnvLazy.force !components_of_module_maker' (find_module_descr p1 env)
with
Functor_comps f ->
!components_of_functor_appl' f p1 p2
| Structure_comps c ->
raise Not_found
end
let find proj1 proj2 path env =
match path with
Pident id ->
let (p, data) = EnvTbl.find_same id (proj1 env)
in data
| Pdot(p, s, pos) ->
begin match
EnvLazy.force !components_of_module_maker' (find_module_descr p env)
with
Structure_comps c ->
let (data, pos) = Tbl.find s (proj2 c) in data
| Functor_comps f ->
raise Not_found
end
| Papply(p1, p2) ->
raise Not_found
let find_value =
find (fun env -> env.values) (fun sc -> sc.comp_values)
and find_type_full =
find (fun env -> env.types) (fun sc -> sc.comp_types)
and find_modtype =
find (fun env -> env.modtypes) (fun sc -> sc.comp_modtypes)
and find_class =
find (fun env -> env.classes) (fun sc -> sc.comp_classes)
and find_cltype =
find (fun env -> env.cltypes) (fun sc -> sc.comp_cltypes)
let find_type p env =
fst (find_type_full p env)
let find_type_descrs p env =
snd (find_type_full p env)
(* Find the manifest type associated to a type when appropriate:
- the type should be public or should have a private row,
- the type should have an associated manifest type. *)
let find_type_expansion ?level path env =
let decl = find_type path env in
match decl.type_manifest with
| Some body when decl.type_private = Public
|| decl.type_kind <> Type_abstract
|| Btype.has_constr_row body ->
(decl.type_params, body, may_map snd decl.type_newtype_level)
(* The manifest type of Private abstract data types without
private row are still considered unknown to the type system.
Hence, this case is caught by the following clause that also handles
purely abstract data types without manifest type definition. *)
| _ -> raise Not_found
(* Find the manifest type information associated to a type, i.e.
the necessary information for the compiler's type-based optimisations.
In particular, the manifest type associated to a private abstract type
is revealed for the sake of compiler's type-based optimisations. *)
let find_type_expansion_opt path env =
let decl = find_type path env in
match decl.type_manifest with
(* The manifest type of Private abstract data types can still get
an approximation using their manifest type. *)
| Some body -> (decl.type_params, body, may_map snd decl.type_newtype_level)
| _ -> raise Not_found
let find_modtype_expansion path env =
match find_modtype path env with
Modtype_abstract -> raise Not_found
| Modtype_manifest mty -> mty
let find_module path env =
match path with
Pident id ->
begin try
let (p, data) = EnvTbl.find_same id env.modules
in data
with Not_found ->
if Ident.persistent id then
let ps = find_pers_struct (Ident.name id) in
Mty_signature(ps.ps_sig)
else raise Not_found
end
| Pdot(p, s, pos) ->
begin match
EnvLazy.force !components_of_module_maker' (find_module_descr p env)
with
Structure_comps c ->
let (data, pos) = Tbl.find s c.comp_modules in
EnvLazy.force subst_modtype_maker data
| Functor_comps f ->
raise Not_found
end
| Papply(p1, p2) ->
raise Not_found (* not right *)
(* Lookup by name *)
let rec lookup_module_descr lid env =
match lid with
Lident s ->
begin try
EnvTbl.find_name s env.components
with Not_found ->
if s = !current_unit then raise Not_found;
let ps = find_pers_struct s in
(Pident(Ident.create_persistent s), ps.ps_comps)
end
| Ldot(l, s) ->
let (p, descr) = lookup_module_descr l env in
begin match EnvLazy.force !components_of_module_maker' descr with
Structure_comps c ->
let (descr, pos) = Tbl.find s c.comp_components in
(Pdot(p, s, pos), descr)
| Functor_comps f ->
raise Not_found
end
| Lapply(l1, l2) ->
let (p1, desc1) = lookup_module_descr l1 env in
let (p2, mty2) = lookup_module l2 env in
begin match EnvLazy.force !components_of_module_maker' desc1 with
Functor_comps f ->
!check_modtype_inclusion env mty2 p2 f.fcomp_arg;
(Papply(p1, p2), !components_of_functor_appl' f p1 p2)
| Structure_comps c ->
raise Not_found
end
and lookup_module lid env =
match lid with
Lident s ->
begin try
EnvTbl.find_name s env.modules
with Not_found ->
if s = !current_unit then raise Not_found;
let ps = find_pers_struct s in
(Pident(Ident.create_persistent s), Mty_signature ps.ps_sig)
end
| Ldot(l, s) ->
let (p, descr) = lookup_module_descr l env in
begin match EnvLazy.force !components_of_module_maker' descr with
Structure_comps c ->
let (data, pos) = Tbl.find s c.comp_modules in
(Pdot(p, s, pos), EnvLazy.force subst_modtype_maker data)
| Functor_comps f ->
raise Not_found
end
| Lapply(l1, l2) ->
let (p1, desc1) = lookup_module_descr l1 env in
let (p2, mty2) = lookup_module l2 env in
let p = Papply(p1, p2) in
begin match EnvLazy.force !components_of_module_maker' desc1 with
Functor_comps f ->
!check_modtype_inclusion env mty2 p2 f.fcomp_arg;
(p, Subst.modtype (Subst.add_module f.fcomp_param p2 f.fcomp_subst)
f.fcomp_res)
| Structure_comps c ->
raise Not_found
end
let lookup proj1 proj2 lid env =
match lid with
Lident s ->
EnvTbl.find_name s (proj1 env)
| Ldot(l, s) ->
let (p, desc) = lookup_module_descr l env in
begin match EnvLazy.force !components_of_module_maker' desc with
Structure_comps c ->
let (data, pos) = Tbl.find s (proj2 c) in
(Pdot(p, s, pos), data)
| Functor_comps f ->
raise Not_found
end
| Lapply(l1, l2) ->
raise Not_found
let lookup_simple proj1 proj2 lid env =
match lid with
Lident s ->
EnvTbl.find_name s (proj1 env)
| Ldot(l, s) ->
let (p, desc) = lookup_module_descr l env in
begin match EnvLazy.force !components_of_module_maker' desc with
Structure_comps c ->
let (data, pos) = Tbl.find s (proj2 c) in
data
| Functor_comps f ->
raise Not_found
end
| Lapply(l1, l2) ->
raise Not_found
let lookup_all_simple proj1 proj2 shadow lid env =
match lid with
Lident s ->
let xl = EnvTbl.find_all s (proj1 env) in
let rec do_shadow =
function
| [] -> []
| ((x, f) :: xs) ->
(x, f) ::
(do_shadow (List.filter (fun (y, g) -> not (shadow x y)) xs))
in
do_shadow xl
| Ldot(l, s) ->
let (p, desc) = lookup_module_descr l env in
begin match EnvLazy.force !components_of_module_maker' desc with
Structure_comps c ->
let comps =
try Tbl.find s (proj2 c) with Not_found -> []
in
List.map
(fun (data, pos) -> (data, (fun () -> ())))
comps
| Functor_comps f ->
raise Not_found
end
| Lapply(l1, l2) ->
raise Not_found
let has_local_constraints env = env.local_constraints
let cstr_shadow cstr1 cstr2 =
match cstr1.cstr_tag, cstr2.cstr_tag with
Cstr_exception _, Cstr_exception _ -> true
| _ -> false
let lbl_shadow lbl1 lbl2 = false
let lookup_value =
lookup (fun env -> env.values) (fun sc -> sc.comp_values)
and lookup_all_constructors =
lookup_all_simple (fun env -> env.constrs) (fun sc -> sc.comp_constrs)
cstr_shadow
and lookup_all_labels =
lookup_all_simple (fun env -> env.labels) (fun sc -> sc.comp_labels)
lbl_shadow
and lookup_type =
lookup (fun env -> env.types) (fun sc -> sc.comp_types)
and lookup_modtype =
lookup (fun env -> env.modtypes) (fun sc -> sc.comp_modtypes)
and lookup_class =
lookup (fun env -> env.classes) (fun sc -> sc.comp_classes)
and lookup_cltype =
lookup (fun env -> env.cltypes) (fun sc -> sc.comp_cltypes)
let mark_value_used name vd =
try Hashtbl.find value_declarations (name, vd.val_loc) ()
with Not_found -> ()
let mark_type_used name vd =
try Hashtbl.find type_declarations (name, vd.type_loc) ()
with Not_found -> ()
let mark_constructor_used usage name vd constr =
try Hashtbl.find used_constructors (name, vd.type_loc, constr) usage
with Not_found -> ()
let mark_exception_used usage ed constr =
try Hashtbl.find used_constructors ("exn", ed.exn_loc, constr) usage
with Not_found -> ()
let set_value_used_callback name vd callback =
let key = (name, vd.val_loc) in
try
let old = Hashtbl.find value_declarations key in
Hashtbl.replace value_declarations key (fun () -> old (); callback ())
(* this is to support cases like:
let x = let x = 1 in x in x
where the two declarations have the same location
(e.g. resulting from Camlp4 expansion of grammar entries) *)
with Not_found ->
Hashtbl.add value_declarations key callback
let set_type_used_callback name td callback =
let loc = td.type_loc in
if loc.Location.loc_ghost then ()
else let key = (name, loc) in
let old =
try Hashtbl.find type_declarations key
with Not_found -> assert false
in
Hashtbl.replace type_declarations key (fun () -> callback old)
let lookup_value lid env =
let (_, desc) as r = lookup_value lid env in
mark_value_used (Longident.last lid) desc;
r
let lookup_type lid env =
let (path, (decl, _)) = lookup_type lid env in
mark_type_used (Longident.last lid) decl;
(path, decl)
(* [path] must be the path to a type, not to a module ! *)
let path_subst_last path id =
match path with
Pident _ -> Pident id
| Pdot (p, name, pos) -> Pdot(p, Ident.name id, pos)
| Papply (p1, p2) -> assert false
let mark_type_path env path =
try
let decl = find_type path env in
mark_type_used (Path.last path) decl
with Not_found -> ()
let ty_path t =
match repr t with
| {desc=Tconstr(path, _, _)} -> path
| _ -> assert false
let lookup_constructor lid env =
match lookup_all_constructors lid env with
[] -> raise Not_found
| (desc, use) :: _ ->
mark_type_path env (ty_path desc.cstr_res);
use ();
desc
let is_lident = function
Lident _ -> true
| _ -> false
let lookup_all_constructors lid env =
try
let cstrs = lookup_all_constructors lid env in
let wrap_use desc use () =
mark_type_path env (ty_path desc.cstr_res);
use ()
in
List.map (fun (cstr, use) -> (cstr, wrap_use cstr use)) cstrs
with
Not_found when is_lident lid -> []
let mark_constructor usage env name desc =
match desc.cstr_tag with
| Cstr_exception (_, loc) ->
begin
try Hashtbl.find used_constructors ("exn", loc, name) usage
with Not_found -> ()
end
| _ ->
let ty_path = ty_path desc.cstr_res in
let ty_decl = try find_type ty_path env with Not_found -> assert false in
let ty_name = Path.last ty_path in
mark_constructor_used usage ty_name ty_decl name
let lookup_label lid env =
match lookup_all_labels lid env with
[] -> raise Not_found
| (desc, use) :: _ ->
mark_type_path env (ty_path desc.lbl_res);
use ();
desc
let lookup_all_labels lid env =
try
let lbls = lookup_all_labels lid env in
let wrap_use desc use () =
mark_type_path env (ty_path desc.lbl_res);
use ()
in
List.map (fun (lbl, use) -> (lbl, wrap_use lbl use)) lbls
with
Not_found when is_lident lid -> []
let lookup_class lid env =
let (_, desc) as r = lookup_class lid env in
(* special support for Typeclass.unbound_class *)
if Path.name desc.cty_path = "" then ignore (lookup_type lid env)
else mark_type_path env desc.cty_path;
r
let lookup_cltype lid env =
let (_, desc) as r = lookup_cltype lid env in
if Path.name desc.clty_path = "" then ignore (lookup_type lid env)
else mark_type_path env desc.clty_path;
mark_type_path env desc.clty_path;
r
(* GADT instance tracking *)
let add_gadt_instance_level lv env =
{env with
gadt_instances = (lv, ref TypeSet.empty) :: env.gadt_instances}
let is_Tlink = function {desc = Tlink _} -> true | _ -> false
let gadt_instance_level env t =
let rec find_instance = function
[] -> None
| (lv, r) :: rem ->
if TypeSet.exists is_Tlink !r then
(* Should we use set_typeset ? *)
r := TypeSet.fold (fun ty -> TypeSet.add (repr ty)) !r TypeSet.empty;
if TypeSet.mem t !r then Some lv else find_instance rem
in find_instance env.gadt_instances
let add_gadt_instances env lv tl =
let r =
try List.assoc lv env.gadt_instances with Not_found -> assert false in
(* Format.eprintf "Added";
List.iter (fun ty -> Format.eprintf "@ %a" !Btype.print_raw ty) tl;
Format.eprintf "@."; *)
set_typeset r (List.fold_right TypeSet.add tl !r)
(* Only use this after expand_head! *)
let add_gadt_instance_chain env lv t =
let r =
try List.assoc lv env.gadt_instances with Not_found -> assert false in
let rec add_instance t =
let t = repr t in
if not (TypeSet.mem t !r) then begin
(* Format.eprintf "@ %a" !Btype.print_raw t; *)
set_typeset r (TypeSet.add t !r);
match t.desc with
Tconstr (p, _, memo) ->
may add_instance (find_expans Private p !memo)
| _ -> ()
end
in
(* Format.eprintf "Added chain"; *)
add_instance t
(* Format.eprintf "@." *)
(* Expand manifest module type names at the top of the given module type *)
let rec scrape_modtype mty env =
match mty with
Mty_ident path ->
begin try
scrape_modtype (find_modtype_expansion path env) env
with Not_found ->
mty
end
| _ -> mty
(* Compute constructor descriptions *)
let constructors_of_type ty_path decl =
let handle_variants cstrs =
Datarepr.constructor_descrs
(newgenty (Tconstr(ty_path, decl.type_params, ref Mnil)))
cstrs decl.type_private
in
match decl.type_kind with
| Type_variant cstrs -> handle_variants cstrs
| Type_record _ | Type_abstract -> []
(* Compute label descriptions *)
let labels_of_type ty_path decl =
match decl.type_kind with
Type_record(labels, rep) ->
Datarepr.label_descrs
(newgenty (Tconstr(ty_path, decl.type_params, ref Mnil)))
labels rep decl.type_private
| Type_variant _ | Type_abstract -> []
(* Given a signature and a root path, prefix all idents in the signature
by the root path and build the corresponding substitution. *)
let rec prefix_idents root pos sub = function
[] -> ([], sub)
| Sig_value(id, decl) :: rem ->
let p = Pdot(root, Ident.name id, pos) in
let nextpos = match decl.val_kind with Val_prim _ -> pos | _ -> pos+1 in
let (pl, final_sub) = prefix_idents root nextpos sub rem in
(p::pl, final_sub)
| Sig_type(id, decl, _) :: rem ->
let p = Pdot(root, Ident.name id, nopos) in
let (pl, final_sub) =
prefix_idents root pos (Subst.add_type id p sub) rem in
(p::pl, final_sub)
| Sig_exception(id, decl) :: rem ->
let p = Pdot(root, Ident.name id, pos) in
let (pl, final_sub) = prefix_idents root (pos+1) sub rem in
(p::pl, final_sub)
| Sig_module(id, mty, _) :: rem ->
let p = Pdot(root, Ident.name id, pos) in
let (pl, final_sub) =
prefix_idents root (pos+1) (Subst.add_module id p sub) rem in
(p::pl, final_sub)
| Sig_modtype(id, decl) :: rem ->
let p = Pdot(root, Ident.name id, nopos) in
let (pl, final_sub) =
prefix_idents root pos
(Subst.add_modtype id (Mty_ident p) sub) rem in
(p::pl, final_sub)
| Sig_class(id, decl, _) :: rem ->
let p = Pdot(root, Ident.name id, pos) in
let (pl, final_sub) = prefix_idents root (pos + 1) sub rem in
(p::pl, final_sub)
| Sig_class_type(id, decl, _) :: rem ->
let p = Pdot(root, Ident.name id, nopos) in
let (pl, final_sub) = prefix_idents root pos sub rem in
(p::pl, final_sub)
let subst_signature sub sg =
List.map
(fun item ->
match item with
| Sig_value(id, decl) ->
Sig_value (id, Subst.value_description sub decl)
| Sig_type(id, decl, x) ->
Sig_type(id, Subst.type_declaration sub decl, x)
| Sig_exception(id, decl) ->
Sig_exception (id, Subst.exception_declaration sub decl)
| Sig_module(id, mty, x) ->
Sig_module(id, Subst.modtype sub mty,x)
| Sig_modtype(id, decl) ->
Sig_modtype(id, Subst.modtype_declaration sub decl)
| Sig_class(id, decl, x) ->
Sig_class(id, Subst.class_declaration sub decl, x)
| Sig_class_type(id, decl, x) ->
Sig_class_type(id, Subst.cltype_declaration sub decl, x)
)
sg
let prefix_idents_and_subst root sub sg =
let (pl, sub) = prefix_idents root 0 sub sg in
pl, sub, lazy (subst_signature sub sg)
let prefix_idents_and_subst root sub sg =
if sub = Subst.identity then
let sgs =
try
Hashtbl.find prefixed_sg root
with Not_found ->
let sgs = ref [] in
Hashtbl.add prefixed_sg root sgs;
sgs
in
try
List.assq sg !sgs
with Not_found ->
let r = prefix_idents_and_subst root sub sg in
sgs := (sg, r) :: !sgs;
r
else
prefix_idents_and_subst root sub sg
(* Compute structure descriptions *)
let add_to_tbl id decl tbl =
let decls =
try Tbl.find id tbl with Not_found -> [] in
Tbl.add id (decl :: decls) tbl
let rec components_of_module env sub path mty =
EnvLazy.create (env, sub, path, mty)
and components_of_module_maker (env, sub, path, mty) =
(match scrape_modtype mty env with
Mty_signature sg ->
let c =
{ comp_values = Tbl.empty;
comp_constrs = Tbl.empty;
comp_labels = Tbl.empty; comp_types = Tbl.empty;
comp_modules = Tbl.empty; comp_modtypes = Tbl.empty;
comp_components = Tbl.empty; comp_classes = Tbl.empty;
comp_cltypes = Tbl.empty } in
let pl, sub, _ = prefix_idents_and_subst path sub sg in
let env = ref env in
let pos = ref 0 in
List.iter2 (fun item path ->
match item with
Sig_value(id, decl) ->
let decl' = Subst.value_description sub decl in
c.comp_values <-
Tbl.add (Ident.name id) (decl', !pos) c.comp_values;
begin match decl.val_kind with
Val_prim _ -> () | _ -> incr pos
end
| Sig_type(id, decl, _) ->
let decl' = Subst.type_declaration sub decl in
let constructors = List.map snd (constructors_of_type path decl') in
let labels = List.map snd (labels_of_type path decl') in
c.comp_types <-
Tbl.add (Ident.name id)
((decl', (constructors, labels)), nopos)
c.comp_types;
List.iter
(fun descr ->
c.comp_constrs <-
add_to_tbl descr.cstr_name (descr, nopos) c.comp_constrs)
constructors;
List.iter
(fun descr ->
c.comp_labels <-
add_to_tbl descr.lbl_name (descr, nopos) c.comp_labels)
labels;
env := store_type_infos id path decl !env
| Sig_exception(id, decl) ->
let decl' = Subst.exception_declaration sub decl in
let cstr = Datarepr.exception_descr path decl' in
let s = Ident.name id in
c.comp_constrs <-
add_to_tbl s (cstr, !pos) c.comp_constrs;
incr pos
| Sig_module(id, mty, _) ->
let mty' = EnvLazy.create (sub, mty) in
c.comp_modules <-
Tbl.add (Ident.name id) (mty', !pos) c.comp_modules;
let comps = components_of_module !env sub path mty in
c.comp_components <-
Tbl.add (Ident.name id) (comps, !pos) c.comp_components;
env := store_module id path mty !env;
incr pos
| Sig_modtype(id, decl) ->
let decl' = Subst.modtype_declaration sub decl in
c.comp_modtypes <-
Tbl.add (Ident.name id) (decl', nopos) c.comp_modtypes;
env := store_modtype id path decl !env
| Sig_class(id, decl, _) ->
let decl' = Subst.class_declaration sub decl in
c.comp_classes <-
Tbl.add (Ident.name id) (decl', !pos) c.comp_classes;
incr pos
| Sig_class_type(id, decl, _) ->
let decl' = Subst.cltype_declaration sub decl in
c.comp_cltypes <-
Tbl.add (Ident.name id) (decl', !pos) c.comp_cltypes)
sg pl;
Structure_comps c
| Mty_functor(param, ty_arg, ty_res) ->
Functor_comps {
fcomp_param = param;
(* fcomp_arg must be prefixed eagerly, because it is interpreted
in the outer environment, not in env *)
fcomp_arg = Subst.modtype sub ty_arg;
(* fcomp_res is prefixed lazily, because it is interpreted in env *)
fcomp_res = ty_res;
fcomp_env = env;
fcomp_subst = sub;
fcomp_cache = Hashtbl.create 17 }
| Mty_ident p ->
Structure_comps {
comp_values = Tbl.empty;
comp_constrs = Tbl.empty;
comp_labels = Tbl.empty;
comp_types = Tbl.empty;
comp_modules = Tbl.empty; comp_modtypes = Tbl.empty;
comp_components = Tbl.empty; comp_classes = Tbl.empty;
comp_cltypes = Tbl.empty })
(* Insertion of bindings by identifier + path *)
and check_usage loc id warn tbl =
if not loc.Location.loc_ghost && Warnings.is_active (warn "") then begin
let name = Ident.name id in
let key = (name, loc) in
if Hashtbl.mem tbl key then ()
else let used = ref false in
Hashtbl.add tbl key (fun () -> used := true);
if not (name = "" || name.[0] = '_' || name.[0] = '#')
then
!add_delayed_check_forward
(fun () -> if not !used then Location.prerr_warning loc (warn name))
end;
and store_value ?check id path decl env =
may (fun f -> check_usage decl.val_loc id f value_declarations) check;
{ env with
values = EnvTbl.add id (path, decl) env.values;
summary = Env_value(env.summary, id, decl) }
and store_type id path info env =
let loc = info.type_loc in
check_usage loc id (fun s -> Warnings.Unused_type_declaration s)
type_declarations;
let constructors = constructors_of_type path info in
let labels = labels_of_type path info in
let descrs = (List.map snd constructors, List.map snd labels) in
if not loc.Location.loc_ghost &&
Warnings.is_active (Warnings.Unused_constructor ("", false, false))
then begin
let ty = Ident.name id in
List.iter
begin fun (_, {cstr_name = c; _}) ->
let k = (ty, loc, c) in
if not (Hashtbl.mem used_constructors k) then
let used = constructor_usages () in
Hashtbl.add used_constructors k (add_constructor_usage used);
if not (ty = "" || ty.[0] = '_')
then !add_delayed_check_forward
(fun () ->
if not env.in_signature && not used.cu_positive then
Location.prerr_warning loc
(Warnings.Unused_constructor
(c, used.cu_pattern, used.cu_privatize)))
end
constructors
end;
{ env with
constrs =
List.fold_right
(fun (id, descr) constrs -> EnvTbl.add id descr constrs)
constructors
env.constrs;
labels =
List.fold_right
(fun (id, descr) labels -> EnvTbl.add id descr labels)
labels
env.labels;
types = EnvTbl.add id (path, (info, descrs)) env.types;
summary = Env_type(env.summary, id, info) }
and store_type_infos id path info env =
(* Simplified version of store_type that doesn't compute and store
constructor and label infos, but simply record the arity and
manifest-ness of the type. Used in components_of_module to
keep track of type abbreviations (e.g. type t = float) in the
computation of label representations. *)
{ env with
types = EnvTbl.add id (path, (info,([],[]))) env.types;
summary = Env_type(env.summary, id, info) }
and store_exception id path decl env =
let loc = decl.exn_loc in
if not loc.Location.loc_ghost &&
Warnings.is_active (Warnings.Unused_exception ("", false))
then begin
let ty = "exn" in
let c = Ident.name id in
let k = (ty, loc, c) in
if not (Hashtbl.mem used_constructors k) then begin
let used = constructor_usages () in
Hashtbl.add used_constructors k (add_constructor_usage used);
!add_delayed_check_forward
(fun () ->
if not env.in_signature && not used.cu_positive then
Location.prerr_warning loc
(Warnings.Unused_exception
(c, used.cu_pattern)
)
)
end;
end;
{ env with
constrs = EnvTbl.add id (Datarepr.exception_descr path decl) env.constrs;
summary = Env_exception(env.summary, id, decl) }
and store_module id path mty env =
{ env with
modules = EnvTbl.add id (path, mty) env.modules;
components =
EnvTbl.add id (path, components_of_module env Subst.identity path mty)
env.components;
summary = Env_module(env.summary, id, mty) }
and store_modtype id path info env =
{ env with
modtypes = EnvTbl.add id (path, info) env.modtypes;
summary = Env_modtype(env.summary, id, info) }
and store_class id path desc env =
{ env with
classes = EnvTbl.add id (path, desc) env.classes;
summary = Env_class(env.summary, id, desc) }
and store_cltype id path desc env =
{ env with
cltypes = EnvTbl.add id (path, desc) env.cltypes;
summary = Env_cltype(env.summary, id, desc) }
(* Compute the components of a functor application in a path. *)
let components_of_functor_appl f p1 p2 =
try
Hashtbl.find f.fcomp_cache p2
with Not_found ->
let p = Papply(p1, p2) in
let mty =
Subst.modtype (Subst.add_module f.fcomp_param p2 Subst.identity)
f.fcomp_res in
let comps = components_of_module f.fcomp_env f.fcomp_subst p mty in
Hashtbl.add f.fcomp_cache p2 comps;
comps
(* Define forward functions *)
let _ =
components_of_module' := components_of_module;
components_of_functor_appl' := components_of_functor_appl;
components_of_module_maker' := components_of_module_maker
(* Insertion of bindings by identifier *)
let add_value ?check id desc env =
store_value ?check id (Pident id) desc env
let add_type id info env =
store_type id (Pident id) info env
and add_exception id decl env =
store_exception id (Pident id) decl env
and add_module id mty env =
store_module id (Pident id) mty env
and add_modtype id info env =
store_modtype id (Pident id) info env
and add_class id ty env =
store_class id (Pident id) ty env
and add_cltype id ty env =
store_cltype id (Pident id) ty env
let add_local_constraint id info elv env =
match info with
{type_manifest = Some ty; type_newtype_level = Some (lv, _)} ->
(* elv is the expansion level, lv is the definition level *)
let env =
add_type id {info with type_newtype_level = Some (lv, elv)} env in
{ env with local_constraints = true }
| _ -> assert false
(* Insertion of bindings by name *)
let enter store_fun name data env =
let id = Ident.create name in (id, store_fun id (Pident id) data env)
let enter_value ?check = enter (store_value ?check)
and enter_type = enter store_type
and enter_exception = enter store_exception
and enter_module = enter store_module
and enter_modtype = enter store_modtype
and enter_class = enter store_class
and enter_cltype = enter store_cltype
(* Insertion of all components of a signature *)
let add_item comp env =
match comp with
Sig_value(id, decl) -> add_value id decl env
| Sig_type(id, decl, _) -> add_type id decl env
| Sig_exception(id, decl) -> add_exception id decl env
| Sig_module(id, mty, _) -> add_module id mty env
| Sig_modtype(id, decl) -> add_modtype id decl env
| Sig_class(id, decl, _) -> add_class id decl env
| Sig_class_type(id, decl, _) -> add_cltype id decl env
let rec add_signature sg env =
match sg with
[] -> env
| comp :: rem -> add_signature rem (add_item comp env)
(* Open a signature path *)
let open_signature root sg env =
(* First build the paths and substitution *)
let (pl, sub, sg) = prefix_idents_and_subst root Subst.identity sg in
let sg = Lazy.force sg in
(* Then enter the components in the environment after substitution *)
let newenv =
List.fold_left2
(fun env item p ->
match item with
Sig_value(id, decl) ->
store_value (Ident.hide id) p decl env
| Sig_type(id, decl, _) ->
store_type (Ident.hide id) p decl env
| Sig_exception(id, decl) ->
store_exception (Ident.hide id) p decl env
| Sig_module(id, mty, _) ->
store_module (Ident.hide id) p mty env
| Sig_modtype(id, decl) ->
store_modtype (Ident.hide id) p decl env
| Sig_class(id, decl, _) ->
store_class (Ident.hide id) p decl env
| Sig_class_type(id, decl, _) ->
store_cltype (Ident.hide id) p decl env
)
env sg pl in
{ newenv with summary = Env_open(env.summary, root) }
(* Open a signature from a file *)
let open_pers_signature name env =
let ps = find_pers_struct name in
open_signature (Pident(Ident.create_persistent name)) ps.ps_sig env
let open_signature ?(loc = Location.none) ?(toplevel = false) root sg env =
if not toplevel && not loc.Location.loc_ghost && Warnings.is_active (Warnings.Unused_open "")
then begin
let used = ref false in
!add_delayed_check_forward
(fun () ->
if not !used then
Location.prerr_warning loc (Warnings.Unused_open (Path.name root))
);
EnvTbl.with_slot used (open_signature root sg) env
end
else open_signature root sg env
(* Read a signature from a file *)
let read_signature modname filename =
let ps = read_pers_struct modname filename in ps.ps_sig
(* Return the CRC of the interface of the given compilation unit *)
let crc_of_unit name =
let ps = find_pers_struct name in
try
List.assoc name ps.ps_crcs
with Not_found ->
assert false
(* Return the list of imported interfaces with their CRCs *)
let imported_units() =
Consistbl.extract crc_units
(* Save a signature to a file *)
let save_signature_with_imports sg modname filename imports =
Btype.cleanup_abbrev ();
Subst.reset_for_saving ();
let sg = Subst.signature (Subst.for_saving Subst.identity) sg in
let oc = open_out_bin filename in
try
let cmi = {
cmi_name = modname;
cmi_sign = sg;
cmi_crcs = imports;
cmi_flags = if !Clflags.recursive_types then [Rectypes] else [];
} in
let crc = output_cmi filename oc cmi in
close_out oc;
(* Enter signature in persistent table so that imported_unit()
will also return its crc *)
let comps =
components_of_module empty Subst.identity
(Pident(Ident.create_persistent modname)) (Mty_signature sg) in
let ps =
{ ps_name = modname;
ps_sig = sg;
ps_comps = comps;
ps_crcs = (cmi.cmi_name, crc) :: imports;
ps_filename = filename;
ps_flags = cmi.cmi_flags } in
Hashtbl.add persistent_structures modname (Some ps);
Consistbl.set crc_units modname crc filename;
sg
with exn ->
close_out oc;
remove_file filename;
raise exn
let save_signature sg modname filename =
save_signature_with_imports sg modname filename (imported_units())
(* Folding on environments *)
let find_all proj1 proj2 f lid env acc =
match lid with
| None ->
EnvTbl.fold_name
(fun id (p, data) acc -> f (Ident.name id) p data acc)
(proj1 env) acc
| Some l ->
let p, desc = lookup_module_descr l env in
begin match EnvLazy.force components_of_module_maker desc with
Structure_comps c ->
Tbl.fold
(fun s (data, pos) acc -> f s (Pdot (p, s, pos)) data acc)
(proj2 c) acc
| Functor_comps _ ->
acc
end
let find_all_simple_list proj1 proj2 f lid env acc =
match lid with
| None ->
EnvTbl.fold_name
(fun id data acc -> f data acc)
(proj1 env) acc
| Some l ->
let p, desc = lookup_module_descr l env in
begin match EnvLazy.force components_of_module_maker desc with
Structure_comps c ->
Tbl.fold
(fun s comps acc ->
match comps with
[] -> acc
| (data, pos) :: _ ->
f data acc)
(proj2 c) acc
| Functor_comps _ ->
acc
end
let fold_modules f lid env acc =
match lid with
| None ->
let acc =
EnvTbl.fold_name
(fun id (p, data) acc -> f (Ident.name id) p data acc)
env.modules
acc
in
Hashtbl.fold
(fun name ps acc ->
match ps with
None -> acc
| Some ps ->
f name (Pident(Ident.create_persistent name))
(Mty_signature ps.ps_sig) acc)
persistent_structures
acc
| Some l ->
let p, desc = lookup_module_descr l env in
begin match EnvLazy.force components_of_module_maker desc with
Structure_comps c ->
Tbl.fold
(fun s (data, pos) acc ->
f s (Pdot (p, s, pos))
(EnvLazy.force subst_modtype_maker data) acc)
c.comp_modules
acc
| Functor_comps _ ->
acc
end
let fold_values f =
find_all (fun env -> env.values) (fun sc -> sc.comp_values) f
and fold_constructors f =
find_all_simple_list (fun env -> env.constrs) (fun sc -> sc.comp_constrs) f
and fold_labels f =
find_all_simple_list (fun env -> env.labels) (fun sc -> sc.comp_labels) f
and fold_types f =
find_all (fun env -> env.types) (fun sc -> sc.comp_types) f
and fold_modtypes f =
find_all (fun env -> env.modtypes) (fun sc -> sc.comp_modtypes) f
and fold_classs f =
find_all (fun env -> env.classes) (fun sc -> sc.comp_classes) f
and fold_cltypes f =
find_all (fun env -> env.cltypes) (fun sc -> sc.comp_cltypes) f
(* Make the initial environment *)
let initial = Predef.build_initial_env add_type add_exception empty
(* Return the environment summary *)
let summary env = env.summary
let last_env = ref empty
let last_reduced_env = ref empty
let keep_only_summary env =
if !last_env == env then !last_reduced_env
else begin
let new_env =
{
empty with
summary = env.summary;
local_constraints = env.local_constraints;
in_signature = env.in_signature;
}
in
last_env := env;
last_reduced_env := new_env;
new_env
end
let env_of_only_summary env_from_summary env =
let new_env = env_from_summary env.summary Subst.identity in
{ new_env with
local_constraints = env.local_constraints;
in_signature = env.in_signature;
}
(* Error report *)
open Format
let report_error ppf = function
| Illegal_renaming(modname, filename) -> fprintf ppf
"Wrong file naming: %a@ contains the compiled interface for@ %s"
Location.print_filename filename modname
| Inconsistent_import(name, source1, source2) -> fprintf ppf
"@[<hov>The files %a@ and %a@ \
make inconsistent assumptions@ over interface %s@]"
Location.print_filename source1 Location.print_filename source2 name
| Need_recursive_types(import, export) ->
fprintf ppf
"@[<hov>Unit %s imports from %s, which uses recursive types.@ %s@]"
export import "The compilation flag -rectypes is required"