1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
|
(***********************************************************************)
(* v * The Coq Proof Assistant / The Coq Development Team *)
(* <O___,, * INRIA-Rocquencourt & LRI-CNRS-Orsay *)
(* \VV/ *************************************************************)
(* // * This file is distributed under the terms of the *)
(* * GNU Lesser General Public License Version 2.1 *)
(***********************************************************************)
(*i $Id$ i*)
open Pp
open Util
open Names
open Declarations
open Entries
open Libnames
open Libobject
open Lib
open Nametab
(* modules and components *)
(* This type is a functional closure of substitutive lib_objects.
The first part is a partial substitution (which will be later
applied to lib_objects when completed).
The second one is a list of bound identifiers which is nonempty
only if the objects are owned by a fuctor
The third one is the "self" ident of the signature (or structure),
which should be substituted in lib_objects with the real name of
the module.
The fourth one is the segment itself which can contain references
to identifiers in the domain of the substitution or in other two
parts. These references are invalid in the current scope and
therefore must be substitued with valid names before use.
*)
type substitutive_objects =
substitution * mod_bound_id list * mod_self_id * lib_objects
(* For each module, we store the following things:
In modtab_substobjs: substitutive_objects
when we will do Module M:=N, the objects of N will be reloaded
with M after substitution
In modtab_objects: "substituted objects" @ "keep objects"
substituted objects -
roughly the objects above after the substitution - we need to
keep them to call open_object when the module is opened (imported)
keep objects -
The list of non-substitutive objects - as above, for each of
them we will call open_object when the module is opened
(Some) Invariants:
* If the module is a functor, the two latter lists are empty.
* Module objects in substitutive_objects part have empty substituted
objects.
* Modules which where created with Module M:=mexpr or with
Module M:SIG. ... End M. have the keep list empty.
*)
let modtab_substobjs =
ref (MPmap.empty : substitutive_objects MPmap.t)
let modtab_objects =
ref (MPmap.empty : (object_prefix * lib_objects) MPmap.t)
(* currently started interactive module (if any) - its arguments (if it
is a functor) and declared output type *)
let openmod_info =
ref (([],None,None) : mod_bound_id list * module_type_entry option
* module_type_body option)
let _ = Summary.declare_summary "MODULE-INFO"
{ Summary.freeze_function = (fun () ->
!modtab_substobjs,
!modtab_objects,
!openmod_info);
Summary.unfreeze_function = (fun (sobjs,objs,info) ->
modtab_substobjs := sobjs;
modtab_objects := objs;
openmod_info := info);
Summary.init_function = (fun () ->
modtab_substobjs := MPmap.empty;
modtab_objects := MPmap.empty;
openmod_info := ([],None,None));
Summary.survive_module = false;
Summary.survive_section = false }
(* auxiliary functions to transform section_path and kernel_name given
by Lib into module_path and dir_path needed for modules *)
let mp_of_kn kn =
let mp,sec,l = repr_kn kn in
if sec=empty_dirpath then
MPdot (mp,l)
else
anomaly ("Non-empty section in module name!" ^ string_of_kn kn)
let dir_of_sp sp =
let dir,id = repr_path sp in
extend_dirpath dir id
let msid_of_mp = function
MPself msid -> msid
| _ -> anomaly "'Self' module path expected!"
let msid_of_prefix (_,(mp,sec)) =
if sec=empty_dirpath then
msid_of_mp mp
else
anomaly ("Non-empty section in module name!" ^
string_of_mp mp ^ "." ^ string_of_dirpath sec)
(* This function checks if the type calculated for the module [mp] is
a subtype of [sub_mtb]. Uses only the global environment. *)
let check_subtypes mp sub_mtb =
let env = Global.env () in
let mtb = (Environ.lookup_module mp env).mod_type in
let _ = Environ.add_constraints
(Subtyping.check_subtypes env mtb sub_mtb)
in
() (* The constraints are checked and forgot immediately! *)
(* This function registers the visibility of the module and iterates
through its components. It is called by plenty module functions *)
let do_module exists what iter_objects i dir mp substobjs objects =
let prefix = (dir,(mp,empty_dirpath)) in
let dirinfo = DirModule (dir,(mp,empty_dirpath)) in
let vis =
if exists then
if
try Nametab.locate_dir (qualid_of_dirpath dir) = dirinfo
with Not_found -> false
then
Nametab.Exactly i
else
errorlabstrm (what^"_module")
(pr_dirpath dir ++ str " should already exist!")
else
if Nametab.exists_dir dir then
errorlabstrm (what^"_module") (pr_dirpath dir ++ str " already exists")
else
Nametab.Until i
in
Nametab.push_dir vis dir dirinfo;
modtab_substobjs := MPmap.add mp substobjs !modtab_substobjs;
match objects with
Some seg ->
modtab_objects := MPmap.add mp (prefix,seg) !modtab_objects;
iter_objects (i+1) prefix seg
| None -> ()
let conv_names_do_module exists what iter_objects i
(sp,kn) substobjs substituted =
let dir,mp = dir_of_sp sp, mp_of_kn kn in
do_module exists what iter_objects i dir mp substobjs substituted
(* Interactive modules and module types cannot be recached! cache_mod*
functions can be called only once (and "end_mod*" set the flag to
false then)
*)
let cache_module ((sp,kn as oname),(entry,substobjs,substituted)) =
let _ = match entry with
| None ->
anomaly "You must not recache interactive modules!"
| Some (me,sub_mte_o) ->
let sub_mtb_o = match sub_mte_o with
None -> None
| Some mte -> Some (Mod_typing.translate_modtype (Global.env()) mte)
in
let mp = Global.add_module (basename sp) me in
if mp <> mp_of_kn kn then
anomaly "Kernel and Library names do not match";
match sub_mtb_o with
None -> ()
| Some sub_mtb -> check_subtypes mp sub_mtb
in
conv_names_do_module false "cache" load_objects 1 oname substobjs substituted
(* TODO: This check is not essential *)
let check_empty s = function
| None -> ()
| Some _ ->
anomaly ("We should never have full info in " ^ s^"!")
(* When this function is called the module itself is already in the
environment. This function loads its objects only *)
let load_module i (oname,(entry,substobjs,substituted)) =
(* TODO: This check is not essential *)
check_empty "load_module" entry;
conv_names_do_module false "load" load_objects i oname substobjs substituted
let open_module i (oname,(entry,substobjs,substituted)) =
(* TODO: This check is not essential *)
check_empty "open_module" entry;
conv_names_do_module true "open" open_objects i oname substobjs substituted
let subst_substobjs dir mp (subst,mbids,msid,objs) =
match mbids with
| [] ->
let prefix = dir,(mp,empty_dirpath) in
Some (subst_objects prefix (add_msid msid mp subst) objs)
| _ -> None
let subst_module ((sp,kn),subst,(entry,substobjs,_)) =
check_empty "subst_module" entry;
let dir,mp = dir_of_sp sp, mp_of_kn kn in
let (sub,mbids,msid,objs) = substobjs in
let subst' = join sub subst in
(* substitutive_objects get the new substitution *)
let substobjs = (subst',mbids,msid,objs) in
(* if we are not a functor - calculate substitued.
We add "msid |-> mp" to the substitution *)
let substituted = subst_substobjs dir mp substobjs
in
(None,substobjs,substituted)
let classify_module (_,(_,substobjs,_)) =
Substitute (None,substobjs,None)
let (in_module,out_module) =
declare_object {(default_object "MODULE") with
cache_function = cache_module;
load_function = load_module;
open_function = open_module;
subst_function = subst_module;
classify_function = classify_module;
export_function = (fun _ -> anomaly "No modules in sections!") }
let cache_keep _ = anomaly "This module should not be cached!"
let load_keep i ((sp,kn),seg) =
let mp = mp_of_kn kn in
let prefix = dir_of_sp sp, (mp,empty_dirpath) in
begin
try
let prefix',objects = MPmap.find mp !modtab_objects in
if prefix' <> prefix then
anomaly "Two different modules with the same path!";
modtab_objects := MPmap.add mp (prefix,objects@seg) !modtab_objects;
with
Not_found -> anomaly "Keep objects before substitutive"
end;
load_objects i prefix seg
let open_keep i ((sp,kn),seg) =
let dirpath,mp = dir_of_sp sp, mp_of_kn kn in
open_objects i (dirpath,(mp,empty_dirpath)) seg
let (in_modkeep,out_modkeep) =
declare_object {(default_object "MODULE KEEP OBJECTS") with
cache_function = cache_keep;
load_function = load_keep;
open_function = open_keep;
export_function = (fun _ -> anomaly "No modules in sections!") }
(* we remember objects for a module type. In case of a declaration:
Module M:SIG:=...
The module M gets its objects from SIG
*)
let modtypetab =
ref (KNmap.empty : substitutive_objects KNmap.t)
(* currently started interactive module type. We remember its arguments
if it is a functor type *)
let openmodtype_info =
ref ([] : mod_bound_id list)
let _ = Summary.declare_summary "MODTYPE-INFO"
{ Summary.freeze_function = (fun () ->
!modtypetab,!openmodtype_info);
Summary.unfreeze_function = (fun ft ->
modtypetab := fst ft;
openmodtype_info := snd ft);
Summary.init_function = (fun () ->
modtypetab := KNmap.empty;
openmodtype_info := []);
Summary.survive_module = false;
Summary.survive_section = true }
let cache_modtype ((sp,kn),(entry,modtypeobjs)) =
let _ =
match entry with
| None ->
anomaly "You must not recache interactive module types!"
| Some mte ->
let kn' = Global.add_modtype (basename sp) mte in
if kn' <> kn then
anomaly "Kernel and Library names do not match"
in
if Nametab.exists_modtype sp then
errorlabstrm "cache_modtype"
(pr_sp sp ++ str " already exists") ;
Nametab.push_modtype (Nametab.Until 1) sp kn;
modtypetab := KNmap.add kn modtypeobjs !modtypetab
let load_modtype i ((sp,kn),(entry,modtypeobjs)) =
check_empty "load_modtype" entry;
if Nametab.exists_modtype sp then
errorlabstrm "cache_modtype"
(pr_sp sp ++ str " already exists") ;
Nametab.push_modtype (Nametab.Until i) sp kn;
modtypetab := KNmap.add kn modtypeobjs !modtypetab
let open_modtype i ((sp,kn),(entry,_)) =
check_empty "open_modtype" entry;
if
try Nametab.locate_modtype (qualid_of_sp sp) <> kn
with Not_found -> true
then
errorlabstrm ("open_modtype")
(pr_sp sp ++ str " should already exist!");
Nametab.push_modtype (Nametab.Exactly i) sp kn
let subst_modtype (_,subst,(entry,(subs,mbids,msid,objs))) =
check_empty "subst_modtype" entry;
(entry,(join subs subst,mbids,msid,objs))
let classify_modtype (_,(_,substobjs)) =
Substitute (None,substobjs)
let (in_modtype,out_modtype) =
declare_object {(default_object "MODULE TYPE") with
cache_function = cache_modtype;
open_function = open_modtype;
load_function = load_modtype;
subst_function = subst_modtype;
classify_function = classify_modtype;
export_function = in_some }
let replace_module_object id (subst, mbids, msid, lib_stack) modobjs =
if mbids<>[] then
error "Unexpected functor objects"
else
let rec replace_id = function
| [] -> []
| (id',obj)::tail when id = id' ->
if object_tag obj = "MODULE" then
(id, in_module (None,modobjs,None))::tail
else error "MODULE expected!"
| lobj::tail -> lobj::replace_id tail
in
(subst, mbids, msid, replace_id lib_stack)
let abstract_substobjs mbids1 (subst, mbids2, msid, lib_stack) =
(subst, mbids1@mbids2, msid, lib_stack)
let rec get_modtype_substobjs = function
MTEident ln -> KNmap.find ln !modtypetab
| MTEfunsig (mbid,_,mte) ->
let (subst, mbids, msid, objs) = get_modtype_substobjs mte in
(subst, mbid::mbids, msid, objs)
| MTEwith (mty, With_Definition _) -> get_modtype_substobjs mty
| MTEwith (mty, With_Module (id,mp)) ->
let substobjs = get_modtype_substobjs mty in
let modobjs = MPmap.find mp !modtab_substobjs in
replace_module_object id substobjs modobjs
| MTEsig (msid,_) ->
todo "Anonymous module types"; (empty_subst, [], msid, [])
(* push names of bound modules (and their components) to Nametab *)
(* add objects associated to them *)
let process_module_bindings argids args =
let process_arg id (mbid,mty) =
let dir = make_dirpath [id] in
let mp = MPbound mbid in
let substobjs = get_modtype_substobjs mty in
let substituted = subst_substobjs dir mp substobjs in
do_module false "start" load_objects 1 dir mp substobjs substituted
in
List.iter2 process_arg argids args
let replace_module mtb id mb = todo "replace module after with"; mtb
let rec get_some_body mty env = match mty with
MTEident kn -> Environ.lookup_modtype kn env
| MTEfunsig _
| MTEsig _ -> anomaly "anonymous module types not supported"
| MTEwith (mty,With_Definition _) -> get_some_body mty env
| MTEwith (mty,With_Module (id,mp)) ->
replace_module (get_some_body mty env) id (Environ.lookup_module mp env)
let intern_args interp_modtype (env,oldargs) (idl,arg) =
let lib_dir = Lib.library_dp() in
let mbids = List.map (fun (_,id) -> make_mbid lib_dir (string_of_id id)) idl in
let mty = interp_modtype env arg in
let dirs = List.map (fun (_,id) -> make_dirpath [id]) idl in
let mps = List.map (fun mbid -> MPbound mbid) mbids in
let substobjs = get_modtype_substobjs mty in
let substituted's =
List.map2
(fun dir mp -> dir, mp, subst_substobjs dir mp substobjs)
dirs mps
in
List.iter
(fun (dir, mp, substituted) ->
do_module false "interp" load_objects 1 dir mp substobjs substituted)
substituted's;
let body = Modops.module_body_of_type (get_some_body mty env) in
let env =
List.fold_left (fun env mp -> Modops.add_module mp body env) env mps
in
env, List.map (fun mbid -> mbid,mty) mbids :: oldargs
let start_module interp_modtype id args res_o =
let fs = Summary.freeze_summaries () in
let env = Global.env () in
let env,arg_entries_revlist =
List.fold_left (intern_args interp_modtype) (env,[]) args
in
let arg_entries = List.concat (List.rev arg_entries_revlist) in
let res_entry_o, sub_body_o = match res_o with
None -> None, None
| Some (res, true) ->
Some (interp_modtype env res), None
| Some (res, false) ->
(* If the module type is non-restricting, we must translate it
here to catch errors as early as possible. If it is
estricting, the kernel takes care of it. *)
let sub_mte =
List.fold_right
(fun (arg_id,arg_t) mte -> MTEfunsig(arg_id,arg_t,mte))
arg_entries
(interp_modtype env res)
in
let sub_mtb =
Mod_typing.translate_modtype (Global.env()) sub_mte
in
None, Some sub_mtb
in
let mp = Global.start_module id arg_entries res_entry_o in
let mbids = List.map fst arg_entries in
openmod_info:=(mbids,res_entry_o,sub_body_o);
let prefix = Lib.start_module id mp fs in
Nametab.push_dir (Nametab.Until 1) (fst prefix) (DirOpenModule prefix);
Lib.add_frozen_state ()
let end_module id =
let oldoname,oldprefix,fs,lib_stack = Lib.end_module id in
let mp = Global.end_module id in
let mbids, res_o, sub_o = !openmod_info in
begin match sub_o with
None -> ()
| Some sub_mtb -> check_subtypes mp sub_mtb
end;
let substitute, keep, special = Lib.classify_segment lib_stack in
let dir = fst oldprefix in
let msid = msid_of_prefix oldprefix in
let substobjs = try
match res_o with
| None ->
empty_subst, mbids, msid, substitute
| Some (MTEident ln) ->
abstract_substobjs mbids (KNmap.find ln (!modtypetab))
| Some (MTEsig (msid,_)) ->
todo "Anonymous signatures not supported";
empty_subst, mbids, msid, []
| Some (MTEwith _ as mty) ->
abstract_substobjs mbids (get_modtype_substobjs mty)
| Some (MTEfunsig _) ->
anomaly "Funsig cannot be here..."
with
Not_found -> anomaly "Module objects not found..."
in
(* must be called after get_modtype_substobjs, because of possible
dependencies on functor arguments *)
Summary.module_unfreeze_summaries fs;
let substituted = subst_substobjs dir mp substobjs in
let node = in_module (None,substobjs,substituted) in
let objects =
if keep = [] || mbids <> [] then
special@[node] (* no keep objects or we are defining a functor *)
else
special@[node;in_modkeep keep] (* otherwise *)
in
let newoname = Lib.add_leaves id objects in
if (fst newoname) <> (fst oldoname) then
anomaly "Names generated on start_ and end_module do not match";
if mp_of_kn (snd newoname) <> mp then
anomaly "Kernel and Library names do not match";
Lib.add_frozen_state () (* to prevent recaching *)
let module_objects mp =
let prefix,objects = MPmap.find mp !modtab_objects in
segment_of_objects prefix objects
(***********************************************************************)
(* libraries *)
type library_name = dir_path
(* The first two will form substitutive_objects, the last one is keep *)
type library_objects =
mod_self_id * lib_objects * lib_objects
(* The library_cache here is needed to avoid recalculations of
substituted modules object during "reloading" of libraries *)
let library_cache = Hashtbl.create 17
let register_library dir cenv objs digest =
let mp = MPfile dir in
let prefix = dir, (mp, empty_dirpath) in
let substobjs, objects =
try
ignore(Global.lookup_module mp);
(* if it's in the environment, the cached objects should be correct *)
Hashtbl.find library_cache dir
with
Not_found ->
if mp <> Global.import cenv digest then
anomaly "Unexpected disk module name";
let msid,substitute,keep = objs in
let substobjs = empty_subst, [], msid, substitute in
let substituted = subst_substobjs dir mp substobjs in
let objects = option_app (fun seg -> seg@keep) substituted in
let modobjs = substobjs, objects in
Hashtbl.add library_cache dir modobjs;
modobjs
in
do_module false "register_library" load_objects 1 dir mp substobjs objects
let start_library dir =
let mp = Global.start_library dir in
openmod_info:=[],None,None;
Lib.start_compilation dir mp;
Lib.add_frozen_state ()
let end_library dir =
let prefix, lib_stack = Lib.end_compilation dir in
let cenv = Global.export dir in
let msid = msid_of_prefix prefix in
let substitute, keep, _ = Lib.classify_segment lib_stack in
cenv,(msid,substitute,keep)
(* implementation of Export M and Import M *)
let really_import_module mp =
let prefix,objects = MPmap.find mp !modtab_objects in
open_objects 1 prefix objects
let cache_import (_,(_,mp)) =
(* for non-substitutive exports:
let mp = Nametab.locate_module (qualid_of_dirpath dir) in *)
really_import_module mp
let classify_import (_,(export,_ as obj)) =
if export then Substitute obj else Dispose
let subst_import (_,subst,(export,mp as obj)) =
let mp' = subst_mp subst mp in
if mp'==mp then obj else
(export,mp')
let (in_import,out_import) =
declare_object {(default_object "IMPORT MODULE") with
cache_function = cache_import;
open_function = (fun i o -> if i=1 then cache_import o);
subst_function = subst_import;
classify_function = classify_import }
let import_module export mp =
Lib.add_anonymous_leaf (in_import (export,mp))
(************************************************************************)
(* module types *)
let start_modtype interp_modtype id args =
let fs = Summary.freeze_summaries () in
let env = Global.env () in
let env,arg_entries_revlist =
List.fold_left (intern_args interp_modtype) (env,[]) args
in
let arg_entries = List.concat (List.rev arg_entries_revlist) in
let mp = Global.start_modtype id arg_entries in
let mbids = List.map fst arg_entries in
openmodtype_info := mbids;
let prefix = Lib.start_modtype id mp fs in
Nametab.push_dir (Nametab.Until 1) (fst prefix) (DirOpenModtype prefix);
Lib.add_frozen_state ()
let end_modtype id =
let oldoname,prefix,fs,lib_stack = Lib.end_modtype id in
let ln = Global.end_modtype id in
let substitute, _, special = Lib.classify_segment lib_stack in
let msid = msid_of_prefix prefix in
let mbids = !openmodtype_info in
Summary.module_unfreeze_summaries fs;
let modtypeobjs = empty_subst, mbids, msid, substitute in
let oname = Lib.add_leaves id (special@[in_modtype (None, modtypeobjs)]) in
if fst oname <> fst oldoname then
anomaly
"Section paths generated on start_ and end_modtype do not match";
if snd oname <> ln then
anomaly
"Kernel and Library names do not match";
Lib.add_frozen_state ()(* to prevent recaching *)
let declare_modtype interp_modtype id args mty =
let fs = Summary.freeze_summaries () in
let env = Global.env () in
let env,arg_entries_revlist =
List.fold_left (intern_args interp_modtype) (env,[]) args
in
let arg_entries = List.concat (List.rev arg_entries_revlist) in
let base_mty = interp_modtype env mty in
let entry =
List.fold_right
(fun (arg_id,arg_t) mte -> MTEfunsig(arg_id,arg_t,mte))
arg_entries
base_mty
in
let substobjs = get_modtype_substobjs entry in
Summary.unfreeze_summaries fs;
ignore (add_leaf id (in_modtype (Some entry, substobjs)))
let rec get_module_substobjs = function
| MEident mp -> MPmap.find mp !modtab_substobjs
| MEfunctor (mbid,mty,mexpr) ->
let (subst, mbids, msid, objs) =
get_module_substobjs mexpr
in
(subst, mbid::mbids, msid, objs)
| MEstruct (msid,_) ->
(empty_subst, [], msid, [])
| MEapply (mexpr, MEident mp) ->
let (subst, mbids, msid, objs) = get_module_substobjs mexpr in
(match mbids with
| mbid::mbids ->
(add_mbid mbid mp subst, mbids, msid, objs)
| [] -> match mexpr with
| MEident _ | MEstruct _ -> error "Application of a non-functor"
| _ -> error "Application of a functor with too few arguments")
| MEapply (_,mexpr) ->
Modops.error_application_to_not_path mexpr
let declare_module interp_modtype interp_modexpr id args mty_o mexpr_o =
let fs = Summary.freeze_summaries () in
let env = Global.env () in
let env,arg_entries_revlist =
List.fold_left (intern_args interp_modtype) (env,[]) args
in
let arg_entries = List.concat (List.rev arg_entries_revlist) in
let mty_entry_o, mty_sub_o = match mty_o with
None -> None, None
| (Some (mty, true)) ->
Some (List.fold_right
(fun (arg_id,arg_t) mte -> MTEfunsig(arg_id,arg_t,mte))
arg_entries
(interp_modtype env mty)),
None
| (Some (mty, false)) ->
None,
Some (List.fold_right
(fun (arg_id,arg_t) mte -> MTEfunsig(arg_id,arg_t,mte))
arg_entries
(interp_modtype env mty))
in
let mexpr_entry_o = match mexpr_o with
None -> None
| Some mexpr ->
Some (List.fold_right
(fun (mbid,mte) me -> MEfunctor(mbid,mte,me))
arg_entries
(interp_modexpr env mexpr))
in
let entry =
{mod_entry_type = mty_entry_o;
mod_entry_expr = mexpr_entry_o }
in
let substobjs =
match entry with
| {mod_entry_type = Some mte} -> get_modtype_substobjs mte
| {mod_entry_expr = Some mexpr} -> get_module_substobjs mexpr
| _ -> anomaly "declare_module: No type, no body ..."
in
Summary.unfreeze_summaries fs;
let dir,mp = dir_of_sp (Lib.make_path id), mp_of_kn (Lib.make_kn id) in
let substituted = subst_substobjs dir mp substobjs in
ignore (add_leaf
id
(in_module (Some (entry, mty_sub_o), substobjs, substituted)))
(*s Iterators. *)
let fold_all_segments insec f x =
let acc' =
MPmap.fold
(fun _ (prefix,objects) acc ->
let apply_obj acc (id,obj) = f acc (make_oname prefix id) obj in
List.fold_left apply_obj acc objects)
!modtab_objects x
in
let rec apply_node acc = function
| sp, Leaf o -> f acc sp o
| _, ClosedSection (_,_,seg) ->
if insec then List.fold_left apply_node acc seg else acc
| _ -> acc
in
List.fold_left apply_node acc' (Lib.contents_after None)
let iter_all_segments insec f =
let _ =
MPmap.iter
(fun _ (prefix,objects) ->
let apply_obj (id,obj) = f (make_oname prefix id) obj in
List.iter apply_obj objects)
!modtab_objects
in
let rec apply_node = function
| sp, Leaf o -> f sp o
| _, ClosedSection (_,_,seg) -> if insec then List.iter apply_node seg
| _ -> ()
in
List.iter apply_node (Lib.contents_after None)
let debug_print_modtab _ =
let pr_seg = function
| [] -> str "[]"
| l -> str ("[." ^ string_of_int (List.length l) ^ ".]")
in
let pr_modinfo mp (prefix,objects) s =
s ++ str (string_of_mp mp) ++ (spc ())
++ (pr_seg (segment_of_objects prefix objects))
in
let modules = MPmap.fold pr_modinfo !modtab_objects (mt ()) in
hov 0 modules
|