src/Pure/type_infer.ML
author smolkas
Wed, 28 Nov 2012 12:25:43 +0100
changeset 50278 05f8ec128e83
parent 43329 84472e198515
child 53672 df8068269e90
permissions -rw-r--r--
improved readability
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
     1
(*  Title:      Pure/type_infer.ML
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
     2
    Author:     Stefan Berghofer and Markus Wenzel, TU Muenchen
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
     3
42405
13ecdb3057d8 split Type_Infer into early and late part, after Proof_Context;
wenzelm
parents: 42400
diff changeset
     4
Basic representation of type-inference problems.
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
     5
*)
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
     6
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
     7
signature TYPE_INFER =
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
     8
sig
24504
0edc609e36fd exported is_param;
wenzelm
parents: 24485
diff changeset
     9
  val is_param: indexname -> bool
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    10
  val is_paramT: typ -> bool
42400
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    11
  val param_maxidx: term -> int -> int
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    12
  val param_maxidx_of: term list -> int
14788
9776f0c747c8 incorporate type inference interface from type.ML;
wenzelm
parents: 14695
diff changeset
    13
  val param: int -> string * sort -> typ
40286
b928e3960446 more sharing of operations, without aliases;
wenzelm
parents: 39296
diff changeset
    14
  val mk_param: int -> sort -> typ
39296
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    15
  val anyT: sort -> typ
22771
ce1fe6ca7dbb added paramify_vars;
wenzelm
parents: 22698
diff changeset
    16
  val paramify_vars: typ -> typ
18339
64cb06a0bb50 added mixfixT;
wenzelm
parents: 17496
diff changeset
    17
  val paramify_dummies: typ -> int -> typ * int
39296
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    18
  val deref: typ Vartab.table -> typ -> typ
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    19
  val finish: Proof.context -> typ Vartab.table -> typ list * term list -> typ list * term list
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    20
  val fixate: Proof.context -> term list -> term list
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    21
end;
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    22
37145
01aa36932739 renamed structure TypeInfer to Type_Infer, keeping the old name as legacy alias for some time;
wenzelm
parents: 35013
diff changeset
    23
structure Type_Infer: TYPE_INFER =
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    24
struct
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    25
22698
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    26
(** type parameters and constraints **)
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    27
39286
1f8131a7bcb9 tuned messages;
wenzelm
parents: 37236
diff changeset
    28
(* type inference parameters -- may get instantiated *)
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    29
24504
0edc609e36fd exported is_param;
wenzelm
parents: 24485
diff changeset
    30
fun is_param (x, _: int) = String.isPrefix "?" x;
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    31
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    32
fun is_paramT (TVar (xi, _)) = is_param xi
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    33
  | is_paramT _ = false;
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    34
42400
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    35
val param_maxidx =
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    36
  (Term.fold_types o Term.fold_atyps)
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    37
    (fn (TVar (xi as (_, i), _)) => if is_param xi then Integer.max i else I | _ => I);
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    38
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    39
fun param_maxidx_of ts = fold param_maxidx ts ~1;
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    40
22698
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    41
fun param i (x, S) = TVar (("?" ^ x, i), S);
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    42
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    43
fun mk_param i S = TVar (("?'a", i), S);
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    44
39296
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    45
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    46
(* pre-stage parameters *)
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    47
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    48
fun anyT S = TFree ("'_dummy_", S);
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    49
32002
1a35de4112bb tuned paramify_vars: Term_Subst.map_atypsT_option;
wenzelm
parents: 31977
diff changeset
    50
val paramify_vars =
32146
4937d9836824 paramify_vars: Term_Subst.map_atypsT_same
wenzelm
parents: 32141
diff changeset
    51
  Same.commit
4937d9836824 paramify_vars: Term_Subst.map_atypsT_same
wenzelm
parents: 32141
diff changeset
    52
    (Term_Subst.map_atypsT_same
4937d9836824 paramify_vars: Term_Subst.map_atypsT_same
wenzelm
parents: 32141
diff changeset
    53
      (fn TVar ((x, i), S) => (param i (x, S)) | _ => raise Same.SAME));
22771
ce1fe6ca7dbb added paramify_vars;
wenzelm
parents: 22698
diff changeset
    54
22698
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    55
val paramify_dummies =
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    56
  let
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    57
    fun dummy S maxidx = (param (maxidx + 1) ("'dummy", S), maxidx + 1);
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    58
22698
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    59
    fun paramify (TFree ("'_dummy_", S)) maxidx = dummy S maxidx
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    60
      | paramify (Type ("dummy", _)) maxidx = dummy [] maxidx
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    61
      | paramify (Type (a, Ts)) maxidx =
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    62
          let val (Ts', maxidx') = fold_map paramify Ts maxidx
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    63
          in (Type (a, Ts'), maxidx') end
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    64
      | paramify T maxidx = (T, maxidx);
7e6412e8d64b moved get_sort to sign.ML;
wenzelm
parents: 22688
diff changeset
    65
  in paramify end;
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    66
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    67
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    68
39296
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    69
(** results **)
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    70
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    71
(* dereferenced views *)
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    72
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    73
fun deref tye (T as TVar (xi, _)) =
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    74
      (case Vartab.lookup tye xi of
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    75
        NONE => T
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    76
      | SOME U => deref tye U)
42400
26c8c9cabb24 more precise treatment of existing type inference parameters;
wenzelm
parents: 42386
diff changeset
    77
  | deref _ T = T;
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    78
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    79
fun add_parms tye T =
32146
4937d9836824 paramify_vars: Term_Subst.map_atypsT_same
wenzelm
parents: 32141
diff changeset
    80
  (case deref tye T of
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    81
    Type (_, Ts) => fold (add_parms tye) Ts
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    82
  | TVar (xi, _) => if is_param xi then insert (op =) xi else I
32146
4937d9836824 paramify_vars: Term_Subst.map_atypsT_same
wenzelm
parents: 32141
diff changeset
    83
  | _ => I);
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    84
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    85
fun add_names tye T =
32146
4937d9836824 paramify_vars: Term_Subst.map_atypsT_same
wenzelm
parents: 32141
diff changeset
    86
  (case deref tye T of
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    87
    Type (_, Ts) => fold (add_names tye) Ts
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    88
  | TFree (x, _) => Name.declare x
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    89
  | TVar ((x, i), _) => if is_param (x, i) then I else Name.declare x);
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    90
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    91
39296
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
    92
(* finish -- standardize remaining parameters *)
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
    93
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    94
fun finish ctxt tye (Ts, ts) =
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    95
  let
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    96
    val used =
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    97
      (fold o fold_types) (add_names tye) ts (fold (add_names tye) Ts (Variable.names_of ctxt));
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
    98
    val parms = rev ((fold o fold_types) (add_parms tye) ts (fold (add_parms tye) Ts []));
43329
84472e198515 tuned signature: Name.invent and Name.invent_names;
wenzelm
parents: 42405
diff changeset
    99
    val names = Name.invent used ("?" ^ Name.aT) (length parms);
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   100
    val tab = Vartab.make (parms ~~ names);
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
   101
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   102
    fun finish_typ T =
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   103
      (case deref tye T of
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   104
        Type (a, Ts) => Type (a, map finish_typ Ts)
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   105
      | U as TFree _ => U
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   106
      | U as TVar (xi, S) =>
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   107
          (case Vartab.lookup tab xi of
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   108
            NONE => U
39295
6e8b0672c6a2 Type_Infer.finish: index 0 -- freshness supposedly via Name.invents;
wenzelm
parents: 39294
diff changeset
   109
          | SOME a => TVar ((a, 0), S)));
39294
27fae73fe769 simplified Type_Infer: eliminated separate datatypes pretyp/preterm -- only assign is_paramT TVars;
wenzelm
parents: 39292
diff changeset
   110
  in (map finish_typ Ts, map (Type.strip_constraints o Term.map_types finish_typ) ts) end;
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
   111
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
   112
39296
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   113
(* fixate -- introduce fresh type variables *)
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   114
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   115
fun fixate ctxt ts =
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   116
  let
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   117
    fun subst_param (xi, S) (inst, used) =
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   118
      if is_param xi then
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   119
        let
43329
84472e198515 tuned signature: Name.invent and Name.invent_names;
wenzelm
parents: 42405
diff changeset
   120
          val [a] = Name.invent used Name.aT 1;
39296
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   121
          val used' = Name.declare a used;
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   122
        in (((xi, S), TFree (a, S)) :: inst, used') end
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   123
      else (inst, used);
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   124
    val used = (fold o fold_types) Term.declare_typ_names ts (Variable.names_of ctxt);
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   125
    val (inst, _) = fold_rev subst_param (fold Term.add_tvars ts []) ([], used);
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   126
  in (map o map_types) (Term_Subst.instantiateT inst) ts end;
e275d581a218 tuned signature;
wenzelm
parents: 39295
diff changeset
   127
2957
d35fca99b3be Type inference (isolated from type.ML, completely reimplemented).
wenzelm
parents:
diff changeset
   128
end;