-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathbtype.ml
771 lines (656 loc) · 24.1 KB
/
btype.ml
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
(**************************************************************************)
(* *)
(* OCaml *)
(* *)
(* Xavier Leroy and Jerome Vouillon, 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 GNU Lesser General Public License version 2.1, with the *)
(* special exception on linking described in the file LICENSE. *)
(* *)
(**************************************************************************)
(* Basic operations on core types *)
open Asttypes
open Types
open Local_store
(**** Sets, maps and hashtables of types ****)
let wrap_repr f ty = f (Transient_expr.repr ty)
let wrap_type_expr f tty = f (Transient_expr.type_expr tty)
module TransientTypeSet = Set.Make(TransientTypeOps)
module TypeSet = struct
include TransientTypeSet
let add = wrap_repr add
let mem = wrap_repr mem
let singleton = wrap_repr singleton
let exists p = TransientTypeSet.exists (wrap_type_expr p)
let elements set =
List.map Transient_expr.type_expr (TransientTypeSet.elements set)
end
module TransientTypeMap = Map.Make(TransientTypeOps)
module TypeMap = struct
include TransientTypeMap
let add ty = wrap_repr add ty
let find ty = wrap_repr find ty
let singleton ty = wrap_repr singleton ty
let fold f = TransientTypeMap.fold (wrap_type_expr f)
end
module TransientTypeHash = Hashtbl.Make(TransientTypeOps)
module TypeHash = struct
include TransientTypeHash
let add hash = wrap_repr (add hash)
let find hash = wrap_repr (find hash)
let iter f = TransientTypeHash.iter (wrap_type_expr f)
end
module TransientTypePairs =
Hashtbl.Make (struct
type t = transient_expr * transient_expr
let equal (t1, t1') (t2, t2') = (t1 == t2) && (t1' == t2')
let hash (t, t') = t.id + 93 * t'.id
end)
module TypePairs = struct
module H = TransientTypePairs
open Transient_expr
type t = {
set : unit H.t;
mutable elems : (transient_expr * transient_expr) list;
(* elems preserves the (reversed) insertion order of elements *)
}
let create n =
{ elems = []; set = H.create n }
let clear t =
t.elems <- [];
H.clear t.set
let repr2 (t1, t2) = (repr t1, repr t2)
let add t p =
let p = repr2 p in
if H.mem t.set p then () else begin
H.add t.set p ();
t.elems <- p :: t.elems
end
let mem t p = H.mem t.set (repr2 p)
let iter f t =
(* iterate in insertion order, not Hashtbl.iter order *)
List.rev t.elems
|> List.iter (fun (t1,t2) ->
f (type_expr t1, type_expr t2))
end
(**** Forward declarations ****)
let print_raw =
ref (fun _ -> assert false : Format.formatter -> type_expr -> unit)
(**** Type level management ****)
let generic_level = Ident.highest_scope
(* Used to mark a type during a traversal. *)
let lowest_level = Ident.lowest_scope
let pivot_level = 2 * lowest_level - 1
(* pivot_level - lowest_level < lowest_level *)
(**** Some type creators ****)
let newgenty desc = newty2 ~level:generic_level desc
let newgenvar ?name () = newgenty (Tvar name)
let newgenstub ~scope = newty3 ~level:generic_level ~scope (Tvar None)
(*
let newmarkedvar level =
incr new_id; { desc = Tvar; level = pivot_level - level; id = !new_id }
let newmarkedgenvar () =
incr new_id;
{ desc = Tvar; level = pivot_level - generic_level; id = !new_id }
*)
(**** Check some types ****)
let is_Tvar ty = match get_desc ty with Tvar _ -> true | _ -> false
let is_Tunivar ty = match get_desc ty with Tunivar _ -> true | _ -> false
let is_Tconstr ty = match get_desc ty with Tconstr _ -> true | _ -> false
let dummy_method = "*dummy method*"
(**** Representative of a type ****)
let merge_fixed_explanation fixed1 fixed2 =
match fixed1, fixed2 with
| Some Univar _ as x, _ | _, (Some Univar _ as x) -> x
| Some Fixed_private as x, _ | _, (Some Fixed_private as x) -> x
| Some Reified _ as x, _ | _, (Some Reified _ as x) -> x
| Some Rigid as x, _ | _, (Some Rigid as x) -> x
| None, None -> None
let fixed_explanation row =
match row_fixed row with
| Some _ as x -> x
| None ->
let ty = row_more row in
match get_desc ty with
| Tvar _ | Tnil -> None
| Tunivar _ -> Some (Univar ty)
| Tconstr (p,_,_) -> Some (Reified p)
| _ -> assert false
let is_fixed row = match row_fixed row with
| None -> false
| Some _ -> true
let has_fixed_explanation row = fixed_explanation row <> None
let static_row row =
row_closed row &&
List.for_all
(fun (_,f) -> match row_field_repr f with Reither _ -> false | _ -> true)
(row_fields row)
let hash_variant s =
let accu = ref 0 in
for i = 0 to String.length s - 1 do
accu := 223 * !accu + Char.code s.[i]
done;
(* reduce to 31 bits *)
accu := !accu land (1 lsl 31 - 1);
(* make it signed for 64 bits architectures *)
if !accu > 0x3FFFFFFF then !accu - (1 lsl 31) else !accu
let proxy ty =
match get_desc ty with
| Tvariant row when not (static_row row) ->
row_more row
| Tobject (ty, _) ->
let rec proxy_obj ty =
match get_desc ty with
Tfield (_, _, _, ty) -> proxy_obj ty
| Tvar _ | Tunivar _ | Tconstr _ -> ty
| Tnil -> ty
| _ -> assert false
in proxy_obj ty
| _ -> ty
(**** Utilities for fixed row private types ****)
let row_of_type t =
match get_desc t with
Tobject(t,_) ->
let rec get_row t =
match get_desc t with
Tfield(_,_,_,t) -> get_row t
| _ -> t
in get_row t
| Tvariant row ->
row_more row
| _ ->
t
let has_constr_row t =
not (is_Tconstr t) && is_Tconstr (row_of_type t)
let is_row_name s =
let l = String.length s in
(* PR#10661: when l=4 and s is "#row", this is not a row name
but the valid #-type name of a class named "row". *)
l > 4 && String.sub s (l-4) 4 = "#row"
let is_constr_row ~allow_ident t =
match get_desc t with
Tconstr (Path.Pident id, _, _) when allow_ident ->
is_row_name (Ident.name id)
| Tconstr (Path.Pdot (_, s), _, _) -> is_row_name s
| _ -> false
(* TODO: where should this really be *)
(* Set row_name in Env, cf. GPR#1204/1329 *)
let set_static_row_name decl path =
match decl.type_manifest with
None -> ()
| Some ty ->
match get_desc ty with
Tvariant row when static_row row ->
let row =
set_row_name row (Some (path, decl.type_params)) in
set_type_desc ty (Tvariant row)
| _ -> ()
(**********************************)
(* Utilities for type traversal *)
(**********************************)
let fold_row f init row =
let result =
List.fold_left
(fun init (_, fi) ->
match row_field_repr fi with
| Rpresent(Some ty) -> f init ty
| Reither(_, tl, _) -> List.fold_left f init tl
| _ -> init)
init
(row_fields row)
in
match get_desc (row_more row) with
| Tvar _ | Tunivar _ | Tsubst _ | Tconstr _ | Tnil ->
begin match
Option.map (fun (_,l) -> List.fold_left f result l) (row_name row)
with
| None -> result
| Some result -> result
end
| _ -> assert false
let iter_row f row =
fold_row (fun () v -> f v) () row
let fold_type_expr f init ty =
match get_desc ty with
Tvar _ -> init
| Tarrow (_, ty1, ty2, _) ->
let result = f init ty1 in
f result ty2
| Ttuple l -> List.fold_left f init l
| Tconstr (_, l, _) -> List.fold_left f init l
| Tobject(ty, {contents = Some (_, p)}) ->
let result = f init ty in
List.fold_left f result p
| Tobject (ty, _) -> f init ty
| Tvariant row ->
let result = fold_row f init row in
f result (row_more row)
| Tfield (_, _, ty1, ty2) ->
let result = f init ty1 in
f result ty2
| Tnil -> init
| Tlink _
| Tsubst _ -> assert false
| Tunivar _ -> init
| Tpoly (ty, tyl) ->
let result = f init ty in
List.fold_left f result tyl
| Tpackage (_, fl) ->
List.fold_left (fun result (_n, ty) -> f result ty) init fl
let iter_type_expr f ty =
fold_type_expr (fun () v -> f v) () ty
let rec iter_abbrev f = function
Mnil -> ()
| Mcons(_, _, ty, ty', rem) -> f ty; f ty'; iter_abbrev f rem
| Mlink rem -> iter_abbrev f !rem
type type_iterators =
{ it_signature: type_iterators -> signature -> unit;
it_signature_item: type_iterators -> signature_item -> unit;
it_value_description: type_iterators -> value_description -> unit;
it_type_declaration: type_iterators -> type_declaration -> unit;
it_extension_constructor: type_iterators -> extension_constructor -> unit;
it_module_declaration: type_iterators -> module_declaration -> unit;
it_modtype_declaration: type_iterators -> modtype_declaration -> unit;
it_class_declaration: type_iterators -> class_declaration -> unit;
it_class_type_declaration: type_iterators -> class_type_declaration -> unit;
it_functor_param: type_iterators -> functor_parameter -> unit;
it_module_type: type_iterators -> module_type -> unit;
it_class_type: type_iterators -> class_type -> unit;
it_type_kind: type_iterators -> type_decl_kind -> unit;
it_do_type_expr: type_iterators -> type_expr -> unit;
it_type_expr: type_iterators -> type_expr -> unit;
it_path: Path.t -> unit; }
let iter_type_expr_cstr_args f = function
| Cstr_tuple tl -> List.iter f tl
| Cstr_record lbls -> List.iter (fun d -> f d.ld_type) lbls
let map_type_expr_cstr_args f = function
| Cstr_tuple tl -> Cstr_tuple (List.map f tl)
| Cstr_record lbls ->
Cstr_record (List.map (fun d -> {d with ld_type=f d.ld_type}) lbls)
let iter_type_expr_kind f = function
| Type_abstract -> ()
| Type_variant (cstrs, _) ->
List.iter
(fun cd ->
iter_type_expr_cstr_args f cd.cd_args;
Option.iter f cd.cd_res
)
cstrs
| Type_record(lbls, _) ->
List.iter (fun d -> f d.ld_type) lbls
| Type_open ->
()
let type_iterators =
let it_signature it =
List.iter (it.it_signature_item it)
and it_signature_item it = function
Sig_value (_, vd, _) -> it.it_value_description it vd
| Sig_type (_, td, _, _) -> it.it_type_declaration it td
| Sig_typext (_, td, _, _) -> it.it_extension_constructor it td
| Sig_module (_, _, md, _, _) -> it.it_module_declaration it md
| Sig_modtype (_, mtd, _) -> it.it_modtype_declaration it mtd
| Sig_class (_, cd, _, _) -> it.it_class_declaration it cd
| Sig_class_type (_, ctd, _, _) -> it.it_class_type_declaration it ctd
and it_value_description it vd =
it.it_type_expr it vd.val_type
and it_type_declaration it td =
List.iter (it.it_type_expr it) td.type_params;
Option.iter (it.it_type_expr it) td.type_manifest;
it.it_type_kind it td.type_kind
and it_extension_constructor it td =
it.it_path td.ext_type_path;
List.iter (it.it_type_expr it) td.ext_type_params;
iter_type_expr_cstr_args (it.it_type_expr it) td.ext_args;
Option.iter (it.it_type_expr it) td.ext_ret_type
and it_module_declaration it md =
it.it_module_type it md.md_type
and it_modtype_declaration it mtd =
Option.iter (it.it_module_type it) mtd.mtd_type
and it_class_declaration it cd =
List.iter (it.it_type_expr it) cd.cty_params;
it.it_class_type it cd.cty_type;
Option.iter (it.it_type_expr it) cd.cty_new;
it.it_path cd.cty_path
and it_class_type_declaration it ctd =
List.iter (it.it_type_expr it) ctd.clty_params;
it.it_class_type it ctd.clty_type;
it.it_path ctd.clty_path
and it_functor_param it = function
| Unit -> ()
| Named (_, mt) -> it.it_module_type it mt
and it_module_type it = function
Mty_ident p
| Mty_alias p -> it.it_path p
| Mty_signature sg -> it.it_signature it sg
| Mty_functor (p, mt) ->
it.it_functor_param it p;
it.it_module_type it mt
and it_class_type it = function
Cty_constr (p, tyl, cty) ->
it.it_path p;
List.iter (it.it_type_expr it) tyl;
it.it_class_type it cty
| Cty_signature cs ->
it.it_type_expr it cs.csig_self;
it.it_type_expr it cs.csig_self_row;
Vars.iter (fun _ (_,_,ty) -> it.it_type_expr it ty) cs.csig_vars;
Meths.iter (fun _ (_,_,ty) -> it.it_type_expr it ty) cs.csig_meths
| Cty_arrow (_, ty, cty) ->
it.it_type_expr it ty;
it.it_class_type it cty
and it_type_kind it kind =
iter_type_expr_kind (it.it_type_expr it) kind
and it_do_type_expr it ty =
iter_type_expr (it.it_type_expr it) ty;
match get_desc ty with
Tconstr (p, _, _)
| Tobject (_, {contents=Some (p, _)})
| Tpackage (p, _) ->
it.it_path p
| Tvariant row ->
Option.iter (fun (p,_) -> it.it_path p) (row_name row)
| _ -> ()
and it_path _p = ()
in
{ it_path; it_type_expr = it_do_type_expr; it_do_type_expr;
it_type_kind; it_class_type; it_functor_param; it_module_type;
it_signature; it_class_type_declaration; it_class_declaration;
it_modtype_declaration; it_module_declaration; it_extension_constructor;
it_type_declaration; it_value_description; it_signature_item; }
let copy_row f fixed row keep more =
let Row {fields = orig_fields; fixed = orig_fixed; closed; name = orig_name} =
row_repr row in
let fields = List.map
(fun (l, fi) -> l,
match row_field_repr fi with
| Rpresent oty -> rf_present (Option.map f oty)
| Reither(c, tl, m) ->
let use_ext_of = if keep then Some fi else None in
let m = if is_fixed row then fixed else m in
let tl = List.map f tl in
rf_either tl ?use_ext_of ~no_arg:c ~matched:m
| Rabsent -> rf_absent)
orig_fields in
let name =
match orig_name with
| None -> None
| Some (path, tl) -> Some (path, List.map f tl) in
let fixed = if fixed then orig_fixed else None in
create_row ~fields ~more ~fixed ~closed ~name
let copy_commu c = if is_commu_ok c then commu_ok else commu_var ()
let rec copy_type_desc ?(keep_names=false) f = function
Tvar _ as ty -> if keep_names then ty else Tvar None
| Tarrow (p, ty1, ty2, c)-> Tarrow (p, f ty1, f ty2, copy_commu c)
| Ttuple l -> Ttuple (List.map f l)
| Tconstr (p, l, _) -> Tconstr (p, List.map f l, ref Mnil)
| Tobject(ty, {contents = Some (p, tl)})
-> Tobject (f ty, ref (Some(p, List.map f tl)))
| Tobject (ty, _) -> Tobject (f ty, ref None)
| Tvariant _ -> assert false (* too ambiguous *)
| Tfield (p, k, ty1, ty2) ->
Tfield (p, field_kind_internal_repr k, f ty1, f ty2)
(* the kind is kept shared, with indirections removed for performance *)
| Tnil -> Tnil
| Tlink ty -> copy_type_desc f (get_desc ty)
| Tsubst _ -> assert false
| Tunivar _ as ty -> ty (* always keep the name *)
| Tpoly (ty, tyl) ->
let tyl = List.map f tyl in
Tpoly (f ty, tyl)
| Tpackage (p, fl) -> Tpackage (p, List.map (fun (n, ty) -> (n, f ty)) fl)
(* Utilities for copying *)
module For_copy : sig
type copy_scope
val redirect_desc: copy_scope -> type_expr -> type_desc -> unit
val with_scope: (copy_scope -> 'a) -> 'a
end = struct
type copy_scope = {
mutable saved_desc : (transient_expr * type_desc) list;
(* Save association of generic nodes with their description. *)
}
let redirect_desc copy_scope ty desc =
let ty = Transient_expr.repr ty in
copy_scope.saved_desc <- (ty, ty.desc) :: copy_scope.saved_desc;
Transient_expr.set_desc ty desc
(* Restore type descriptions. *)
let cleanup { saved_desc; _ } =
List.iter (fun (ty, desc) -> Transient_expr.set_desc ty desc) saved_desc
let with_scope f =
let scope = { saved_desc = [] } in
let res = f scope in
cleanup scope;
res
end
(*******************************************)
(* Memorization of abbreviation expansion *)
(*******************************************)
(* Search whether the expansion has been memorized. *)
let lte_public p1 p2 = (* Private <= Public *)
match p1, p2 with
| Private, _ | _, Public -> true
| Public, Private -> false
let rec find_expans priv p1 = function
Mnil -> None
| Mcons (priv', p2, _ty0, ty, _)
when lte_public priv priv' && Path.same p1 p2 -> Some ty
| Mcons (_, _, _, _, rem) -> find_expans priv p1 rem
| Mlink {contents = rem} -> find_expans priv p1 rem
(* debug: check for cycles in abbreviation. only works with -principal
let rec check_expans visited ty =
let ty = repr ty in
assert (not (List.memq ty visited));
match ty.desc with
Tconstr (path, args, abbrev) ->
begin match find_expans path !abbrev with
Some ty' -> check_expans (ty :: visited) ty'
| None -> ()
end
| _ -> ()
*)
let memo = s_ref []
(* Contains the list of saved abbreviation expansions. *)
let cleanup_abbrev () =
(* Remove all memorized abbreviation expansions. *)
List.iter (fun abbr -> abbr := Mnil) !memo;
memo := []
let memorize_abbrev mem priv path v v' =
(* Memorize the expansion of an abbreviation. *)
mem := Mcons (priv, path, v, v', !mem);
(* check_expans [] v; *)
memo := mem :: !memo
let rec forget_abbrev_rec mem path =
match mem with
Mnil ->
mem
| Mcons (_, path', _, _, rem) when Path.same path path' ->
rem
| Mcons (priv, path', v, v', rem) ->
Mcons (priv, path', v, v', forget_abbrev_rec rem path)
| Mlink mem' ->
mem' := forget_abbrev_rec !mem' path;
raise Exit
let forget_abbrev mem path =
try mem := forget_abbrev_rec !mem path with Exit -> ()
(* debug: check for invalid abbreviations
let rec check_abbrev_rec = function
Mnil -> true
| Mcons (_, ty1, ty2, rem) ->
repr ty1 != repr ty2
| Mlink mem' ->
check_abbrev_rec !mem'
let check_memorized_abbrevs () =
List.for_all (fun mem -> check_abbrev_rec !mem) !memo
*)
(* Re-export backtrack *)
let snapshot = snapshot
let backtrack = backtrack ~cleanup_abbrev
(**********************************)
(* Utilities for labels *)
(**********************************)
let is_optional = function Optional _ -> true | _ -> false
let label_name = function
Nolabel -> ""
| Labelled s
| Optional s -> s
let prefixed_label_name = function
Nolabel -> ""
| Labelled s -> "~" ^ s
| Optional s -> "?" ^ s
let rec extract_label_aux hd l = function
| [] -> None
| (l',t as p) :: ls ->
if label_name l' = l then
Some (l', t, hd <> [], List.rev_append hd ls)
else
extract_label_aux (p::hd) l ls
let extract_label l ls = extract_label_aux [] l ls
(*******************************)
(* Operations on class types *)
(*******************************)
let rec signature_of_class_type =
function
Cty_constr (_, _, cty) -> signature_of_class_type cty
| Cty_signature sign -> sign
| Cty_arrow (_, _, cty) -> signature_of_class_type cty
let rec class_body cty =
match cty with
Cty_constr _ ->
cty (* Only class bodies can be abbreviated *)
| Cty_signature _ ->
cty
| Cty_arrow (_, _, cty) ->
class_body cty
(* Fully expand the head of a class type *)
let rec scrape_class_type =
function
Cty_constr (_, _, cty) -> scrape_class_type cty
| cty -> cty
let rec class_type_arity =
function
Cty_constr (_, _, cty) -> class_type_arity cty
| Cty_signature _ -> 0
| Cty_arrow (_, _, cty) -> 1 + class_type_arity cty
let rec abbreviate_class_type path params cty =
match cty with
Cty_constr (_, _, _) | Cty_signature _ ->
Cty_constr (path, params, cty)
| Cty_arrow (l, ty, cty) ->
Cty_arrow (l, ty, abbreviate_class_type path params cty)
let self_type cty =
(signature_of_class_type cty).csig_self
let self_type_row cty =
(signature_of_class_type cty).csig_self_row
(* Return the methods of a class signature *)
let methods sign =
Meths.fold
(fun name _ l -> name :: l)
sign.csig_meths []
(* Return the virtual methods of a class signature *)
let virtual_methods sign =
Meths.fold
(fun name (_priv, vr, _ty) l ->
match vr with
| Virtual -> name :: l
| Concrete -> l)
sign.csig_meths []
(* Return the concrete methods of a class signature *)
let concrete_methods sign =
Meths.fold
(fun name (_priv, vr, _ty) s ->
match vr with
| Virtual -> s
| Concrete -> MethSet.add name s)
sign.csig_meths MethSet.empty
(* Return the public methods of a class signature *)
let public_methods sign =
Meths.fold
(fun name (priv, _vr, _ty) l ->
match priv with
| Mprivate _ -> l
| Mpublic -> name :: l)
sign.csig_meths []
(* Return the instance variables of a class signature *)
let instance_vars sign =
Vars.fold
(fun name _ l -> name :: l)
sign.csig_vars []
(* Return the virtual instance variables of a class signature *)
let virtual_instance_vars sign =
Vars.fold
(fun name (_mut, vr, _ty) l ->
match vr with
| Virtual -> name :: l
| Concrete -> l)
sign.csig_vars []
(* Return the concrete instance variables of a class signature *)
let concrete_instance_vars sign =
Vars.fold
(fun name (_mut, vr, _ty) s ->
match vr with
| Virtual -> s
| Concrete -> VarSet.add name s)
sign.csig_vars VarSet.empty
let method_type label sign =
match Meths.find label sign.csig_meths with
| (_, _, ty) -> ty
| exception Not_found -> assert false
let instance_variable_type label sign =
match Vars.find label sign.csig_vars with
| (_, _, ty) -> ty
| exception Not_found -> assert false
(**********************************)
(* Utilities for level-marking *)
(**********************************)
let not_marked_node ty = get_level ty >= lowest_level
(* type nodes with negative levels are "marked" *)
let flip_mark_node ty =
let ty = Transient_expr.repr ty in
Transient_expr.set_level ty (pivot_level - ty.level)
let logged_mark_node ty =
set_level ty (pivot_level - get_level ty)
let try_mark_node ty = not_marked_node ty && (flip_mark_node ty; true)
let try_logged_mark_node ty = not_marked_node ty && (logged_mark_node ty; true)
let rec mark_type ty =
if not_marked_node ty then begin
flip_mark_node ty;
iter_type_expr mark_type ty
end
let mark_type_params ty =
iter_type_expr mark_type ty
let type_iterators =
let it_type_expr it ty =
if try_mark_node ty then it.it_do_type_expr it ty
in
{type_iterators with it_type_expr}
(* Remove marks from a type. *)
let rec unmark_type ty =
if get_level ty < lowest_level then begin
(* flip back the marked level *)
flip_mark_node ty;
iter_type_expr unmark_type ty
end
let unmark_iterators =
let it_type_expr _it ty = unmark_type ty in
{type_iterators with it_type_expr}
let unmark_type_decl decl =
unmark_iterators.it_type_declaration unmark_iterators decl
let unmark_extension_constructor ext =
List.iter unmark_type ext.ext_type_params;
iter_type_expr_cstr_args unmark_type ext.ext_args;
Option.iter unmark_type ext.ext_ret_type
let unmark_class_signature sign =
unmark_type sign.csig_self;
unmark_type sign.csig_self_row;
Vars.iter (fun _l (_m, _v, t) -> unmark_type t) sign.csig_vars;
Meths.iter (fun _l (_m, _v, t) -> unmark_type t) sign.csig_meths
let unmark_class_type cty =
unmark_iterators.it_class_type unmark_iterators cty
(**** Type information getter ****)
let cstr_type_path cstr =
match get_desc cstr.cstr_res with
| Tconstr (p, _, _) -> p
| _ -> assert false