src/HOL/W0/W0.thy
author wenzelm
Wed, 01 Feb 2006 22:20:40 +0100
changeset 18888 3b643f81b378
parent 15236 f289e8ba2bb3
child 19380 b808efaa5828
permissions -rw-r--r--
updated;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
     1
(*  Title:      HOL/W0/W0.thy
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
     2
    ID:         $Id$
12945
wenzelm
parents: 12944
diff changeset
     3
    Author:     Dieter Nazareth, Tobias Nipkow, Thomas Stauner, Markus Wenzel
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
     4
*)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
     5
15140
322485b816ac import -> imports
nipkow
parents: 14981
diff changeset
     6
theory W0
322485b816ac import -> imports
nipkow
parents: 14981
diff changeset
     7
imports Main
322485b816ac import -> imports
nipkow
parents: 14981
diff changeset
     8
begin
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
     9
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    10
section {* Universal error monad *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    11
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    12
datatype 'a maybe = Ok 'a | Fail
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    13
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    14
constdefs
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    15
  bind :: "'a maybe \<Rightarrow> ('a \<Rightarrow> 'b maybe) \<Rightarrow> 'b maybe"    (infixl "\<bind>" 60)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    16
  "m \<bind> f \<equiv> case m of Ok r \<Rightarrow> f r | Fail \<Rightarrow> Fail"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    17
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    18
syntax
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    19
  "_bind" :: "patterns \<Rightarrow> 'a maybe \<Rightarrow> 'b \<Rightarrow> 'c"    ("(_ := _;//_)" 0)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    20
translations
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    21
  "P := E; F" == "E \<bind> (\<lambda>P. F)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    22
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    23
lemma bind_Ok [simp]: "(Ok s) \<bind> f = (f s)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    24
  by (simp add: bind_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    25
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    26
lemma bind_Fail [simp]: "Fail \<bind> f = Fail"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    27
  by (simp add: bind_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    28
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    29
lemma split_bind:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    30
    "P (res \<bind> f) = ((res = Fail \<longrightarrow> P Fail) \<and> (\<forall>s. res = Ok s \<longrightarrow> P (f s)))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    31
  by (induct res) simp_all
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    32
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    33
lemma split_bind_asm:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    34
  "P (res \<bind> f) = (\<not> (res = Fail \<and> \<not> P Fail \<or> (\<exists>s. res = Ok s \<and> \<not> P (f s))))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    35
  by (simp split: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    36
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    37
lemmas bind_splits = split_bind split_bind_asm
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    38
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    39
lemma bind_eq_Fail [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    40
  "((m \<bind> f) = Fail) = ((m = Fail) \<or> (\<exists>p. m = Ok p \<and> f p = Fail))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    41
  by (simp split: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    42
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    43
lemma rotate_Ok: "(y = Ok x) = (Ok x = y)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    44
  by (rule eq_sym_conv)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    45
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    46
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    47
section {* MiniML-types and type substitutions *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    48
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    49
axclass type_struct \<subseteq> type
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    50
  -- {* new class for structures containing type variables *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    51
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    52
datatype "typ" = TVar nat | TFun "typ" "typ"    (infixr "->" 70)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    53
  -- {* type expressions *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    54
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    55
types subst = "nat => typ"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    56
  -- {* type variable substitution *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    57
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    58
instance "typ" :: type_struct ..
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    59
instance list :: (type_struct) type_struct ..
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    60
instance fun :: (type, type_struct) type_struct ..
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    61
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    62
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    63
subsection {* Substitutions *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    64
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    65
consts
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    66
  app_subst :: "subst \<Rightarrow> 'a::type_struct \<Rightarrow> 'a::type_struct"    ("$")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    67
  -- {* extension of substitution to type structures *}
13537
f506eb568121 avoid duplicate fact bindings;
wenzelm
parents: 12961
diff changeset
    68
primrec (app_subst_typ)
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    69
  app_subst_TVar: "$s (TVar n) = s n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    70
  app_subst_Fun: "$s (t1 -> t2) = $s t1 -> $s t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    71
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    72
defs (overloaded)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    73
  app_subst_list: "$s \<equiv> map ($s)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    74
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    75
consts
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    76
  free_tv :: "'a::type_struct \<Rightarrow> nat set"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    77
  -- {* @{text "free_tv s"}: the type variables occuring freely in the type structure @{text s} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    78
13537
f506eb568121 avoid duplicate fact bindings;
wenzelm
parents: 12961
diff changeset
    79
primrec (free_tv_typ)
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    80
  "free_tv (TVar m) = {m}"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    81
  "free_tv (t1 -> t2) = free_tv t1 \<union> free_tv t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    82
13537
f506eb568121 avoid duplicate fact bindings;
wenzelm
parents: 12961
diff changeset
    83
primrec (free_tv_list)
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    84
  "free_tv [] = {}"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    85
  "free_tv (x # xs) = free_tv x \<union> free_tv xs"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    86
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    87
constdefs
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    88
  dom :: "subst \<Rightarrow> nat set"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    89
  "dom s \<equiv> {n. s n \<noteq> TVar n}"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    90
  -- {* domain of a substitution *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    91
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    92
  cod :: "subst \<Rightarrow> nat set"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    93
  "cod s \<equiv> \<Union>m \<in> dom s. free_tv (s m)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    94
  -- {* codomain of a substitutions: the introduced variables *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    95
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    96
defs
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    97
  free_tv_subst: "free_tv s \<equiv> dom s \<union> cod s"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    98
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
    99
text {*
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   100
  @{text "new_tv s n"} checks whether @{text n} is a new type variable
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   101
  wrt.\ a type structure @{text s}, i.e.\ whether @{text n} is greater
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   102
  than any type variable occuring in the type structure.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   103
*}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   104
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   105
constdefs
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   106
  new_tv :: "nat \<Rightarrow> 'a::type_struct \<Rightarrow> bool"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   107
  "new_tv n ts \<equiv> \<forall>m. m \<in> free_tv ts \<longrightarrow> m < n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   108
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   109
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   110
subsubsection {* Identity substitution *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   111
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   112
constdefs
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   113
  id_subst :: subst
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   114
  "id_subst \<equiv> \<lambda>n. TVar n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   115
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   116
lemma app_subst_id_te [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   117
  "$id_subst = (\<lambda>t::typ. t)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   118
  -- {* application of @{text id_subst} does not change type expression *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   119
proof
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   120
  fix t :: "typ"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   121
  show "$id_subst t = t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   122
    by (induct t) (simp_all add: id_subst_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   123
qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   124
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   125
lemma app_subst_id_tel [simp]: "$id_subst = (\<lambda>ts::typ list. ts)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   126
  -- {* application of @{text id_subst} does not change list of type expressions *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   127
proof
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   128
  fix ts :: "typ list"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   129
  show "$id_subst ts = ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   130
    by (induct ts) (simp_all add: app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   131
qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   132
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   133
lemma o_id_subst [simp]: "$s o id_subst = s"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   134
  by (rule ext) (simp add: id_subst_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   135
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   136
lemma dom_id_subst [simp]: "dom id_subst = {}"
13890
90611b4e0054 Made empty a translation rather than a constant.
nipkow
parents: 13601
diff changeset
   137
  by (simp add: dom_def id_subst_def)
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   138
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   139
lemma cod_id_subst [simp]: "cod id_subst = {}"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   140
  by (simp add: cod_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   141
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   142
lemma free_tv_id_subst [simp]: "free_tv id_subst = {}"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   143
  by (simp add: free_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   144
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   145
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   146
lemma cod_app_subst [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   147
  assumes free: "v \<in> free_tv (s n)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   148
    and neq: "v \<noteq> n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   149
  shows "v \<in> cod s"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   150
proof -
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   151
  have "s n \<noteq> TVar n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   152
  proof
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   153
    assume "s n = TVar n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   154
    with free have "v = n" by simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   155
    with neq show False ..
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   156
  qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   157
  with free show ?thesis
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   158
    by (auto simp add: dom_def cod_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   159
qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   160
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   161
lemma subst_comp_te: "$g ($f t :: typ) = $(\<lambda>x. $g (f x)) t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   162
  -- {* composition of substitutions *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   163
  by (induct t) simp_all
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   164
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   165
lemma subst_comp_tel: "$g ($f ts :: typ list) = $(\<lambda>x. $g (f x)) ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   166
  by (induct ts) (simp_all add: app_subst_list subst_comp_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   167
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   168
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   169
lemma app_subst_Nil [simp]: "$s [] = []"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   170
  by (simp add: app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   171
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   172
lemma app_subst_Cons [simp]: "$s (t # ts) = ($s t) # ($s ts)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   173
  by (simp add: app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   174
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   175
lemma new_tv_TVar [simp]: "new_tv n (TVar m) = (m < n)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   176
  by (simp add: new_tv_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   177
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   178
lemma new_tv_Fun [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   179
  "new_tv n (t1 -> t2) = (new_tv n t1 \<and> new_tv n t2)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   180
  by (auto simp add: new_tv_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   181
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   182
lemma new_tv_Nil [simp]: "new_tv n []"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   183
  by (simp add: new_tv_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   184
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   185
lemma new_tv_Cons [simp]: "new_tv n (t # ts) = (new_tv n t \<and> new_tv n ts)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   186
  by (auto simp add: new_tv_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   187
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   188
lemma new_tv_id_subst [simp]: "new_tv n id_subst"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   189
  by (simp add: id_subst_def new_tv_def free_tv_subst dom_def cod_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   190
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   191
lemma new_tv_subst:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   192
  "new_tv n s =
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   193
    ((\<forall>m. n \<le> m \<longrightarrow> s m = TVar m) \<and>
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   194
     (\<forall>l. l < n \<longrightarrow> new_tv n (s l)))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   195
  apply (unfold new_tv_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   196
  apply (tactic "safe_tac HOL_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   197
  -- {* @{text \<Longrightarrow>} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   198
    apply (tactic {* fast_tac (HOL_cs addDs [leD] addss (simpset()
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   199
      addsimps [thm "free_tv_subst", thm "dom_def"])) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   200
   apply (subgoal_tac "m \<in> cod s \<or> s l = TVar l")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   201
    apply (tactic "safe_tac HOL_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   202
     apply (tactic {* fast_tac (HOL_cs addDs [UnI2] addss (simpset()
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   203
       addsimps [thm "free_tv_subst"])) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   204
    apply (drule_tac P = "\<lambda>x. m \<in> free_tv x" in subst, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   205
    apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   206
   apply (tactic {* fast_tac (set_cs addss (simpset()
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   207
     addsimps [thm "free_tv_subst", thm "cod_def", thm "dom_def"])) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   208
  -- {* @{text \<Longleftarrow>} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   209
  apply (unfold free_tv_subst cod_def dom_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   210
  apply (tactic "safe_tac set_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   211
   apply (cut_tac m = m and n = n in less_linear)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   212
   apply (tactic "fast_tac (HOL_cs addSIs [less_or_eq_imp_le]) 1")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   213
  apply (cut_tac m = ma and n = n in less_linear)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   214
  apply (fast intro!: less_or_eq_imp_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   215
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   216
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   217
lemma new_tv_list: "new_tv n x = (\<forall>y \<in> set x. new_tv n y)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   218
  by (induct x) simp_all
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   219
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   220
lemma subst_te_new_tv [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   221
  "new_tv n (t::typ) \<longrightarrow> $(\<lambda>x. if x = n then t' else s x) t = $s t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   222
  -- {* substitution affects only variables occurring freely *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   223
  by (induct t) simp_all
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   224
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   225
lemma subst_tel_new_tv [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   226
  "new_tv n (ts::typ list) \<longrightarrow> $(\<lambda>x. if x = n then t else s x) ts = $s ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   227
  by (induct ts) simp_all
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   228
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   229
lemma new_tv_le: "n \<le> m \<Longrightarrow> new_tv n (t::typ) \<Longrightarrow> new_tv m t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   230
  -- {* all greater variables are also new *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   231
proof (induct t)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   232
  case (TVar n)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   233
  thus ?case by (auto intro: less_le_trans)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   234
next
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   235
  case TFun
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   236
  thus ?case by simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   237
qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   238
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   239
lemma [simp]: "new_tv n t \<Longrightarrow> new_tv (Suc n) (t::typ)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   240
  by (rule lessI [THEN less_imp_le [THEN new_tv_le]])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   241
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   242
lemma new_tv_list_le:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   243
  "n \<le> m \<Longrightarrow> new_tv n (ts::typ list) \<Longrightarrow> new_tv m ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   244
proof (induct ts)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   245
  case Nil
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   246
  thus ?case by simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   247
next
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   248
  case Cons
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   249
  thus ?case by (auto intro: new_tv_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   250
qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   251
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   252
lemma [simp]: "new_tv n ts \<Longrightarrow> new_tv (Suc n) (ts::typ list)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   253
  by (rule lessI [THEN less_imp_le [THEN new_tv_list_le]])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   254
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   255
lemma new_tv_subst_le: "n \<le> m \<Longrightarrow> new_tv n (s::subst) \<Longrightarrow> new_tv m s"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   256
  apply (simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   257
  apply clarify
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   258
  apply (rule_tac P = "l < n" and Q = "n <= l" in disjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   259
    apply clarify
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   260
    apply (simp_all add: new_tv_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   261
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   262
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   263
lemma [simp]: "new_tv n s \<Longrightarrow> new_tv (Suc n) (s::subst)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   264
  by (rule lessI [THEN less_imp_le [THEN new_tv_subst_le]])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   265
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   266
lemma new_tv_subst_var:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   267
    "n < m \<Longrightarrow> new_tv m (s::subst) \<Longrightarrow> new_tv m (s n)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   268
  -- {* @{text new_tv} property remains if a substitution is applied *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   269
  by (simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   270
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   271
lemma new_tv_subst_te [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   272
    "new_tv n s \<Longrightarrow> new_tv n (t::typ) \<Longrightarrow> new_tv n ($s t)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   273
  by (induct t) (auto simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   274
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   275
lemma new_tv_subst_tel [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   276
    "new_tv n s \<Longrightarrow> new_tv n (ts::typ list) \<Longrightarrow> new_tv n ($s ts)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   277
  by (induct ts) (fastsimp simp add: new_tv_subst)+
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   278
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   279
lemma new_tv_Suc_list: "new_tv n ts --> new_tv (Suc n) (TVar n # ts)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   280
  -- {* auxilliary lemma *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   281
  by (simp add: new_tv_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   282
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   283
lemma new_tv_subst_comp_1 [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   284
    "new_tv n (s::subst) \<Longrightarrow> new_tv n r \<Longrightarrow> new_tv n ($r o s)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   285
  -- {* composition of substitutions preserves @{text new_tv} proposition *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   286
  by (simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   287
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   288
lemma new_tv_subst_comp_2 [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   289
    "new_tv n (s::subst) \<Longrightarrow> new_tv n r \<Longrightarrow> new_tv n (\<lambda>v. $r (s v))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   290
  by (simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   291
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   292
lemma new_tv_not_free_tv [simp]: "new_tv n ts \<Longrightarrow> n \<notin> free_tv ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   293
  -- {* new type variables do not occur freely in a type structure *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   294
  by (auto simp add: new_tv_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   295
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   296
lemma ftv_mem_sub_ftv_list [simp]:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   297
    "(t::typ) \<in> set ts \<Longrightarrow> free_tv t \<subseteq> free_tv ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   298
  by (induct ts) auto
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   299
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   300
text {*
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   301
  If two substitutions yield the same result if applied to a type
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   302
  structure the substitutions coincide on the free type variables
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   303
  occurring in the type structure.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   304
*}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   305
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   306
lemma eq_subst_te_eq_free:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   307
    "$s1 (t::typ) = $s2 t \<Longrightarrow> n \<in> free_tv t \<Longrightarrow> s1 n = s2 n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   308
  by (induct t) auto
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   309
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   310
lemma eq_free_eq_subst_te:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   311
    "(\<forall>n. n \<in> free_tv t --> s1 n = s2 n) \<Longrightarrow> $s1 (t::typ) = $s2 t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   312
  by (induct t) auto
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   313
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   314
lemma eq_subst_tel_eq_free:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   315
    "$s1 (ts::typ list) = $s2 ts \<Longrightarrow> n \<in> free_tv ts \<Longrightarrow> s1 n = s2 n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   316
  by (induct ts) (auto intro: eq_subst_te_eq_free)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   317
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   318
lemma eq_free_eq_subst_tel:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   319
    "(\<forall>n. n \<in> free_tv ts --> s1 n = s2 n) \<Longrightarrow> $s1 (ts::typ list) = $s2 ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   320
  by (induct ts) (auto intro: eq_free_eq_subst_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   321
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   322
text {*
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   323
  \medskip Some useful lemmas.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   324
*}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   325
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   326
lemma codD: "v \<in> cod s \<Longrightarrow> v \<in> free_tv s"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   327
  by (simp add: free_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   328
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   329
lemma not_free_impl_id: "x \<notin> free_tv s \<Longrightarrow> s x = TVar x"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   330
  by (simp add: free_tv_subst dom_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   331
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   332
lemma free_tv_le_new_tv: "new_tv n t \<Longrightarrow> m \<in> free_tv t \<Longrightarrow> m < n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   333
  by (unfold new_tv_def) fast
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   334
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   335
lemma free_tv_subst_var: "free_tv (s (v::nat)) \<le> insert v (cod s)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   336
  by (cases "v \<in> dom s") (auto simp add: cod_def dom_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   337
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   338
lemma free_tv_app_subst_te: "free_tv ($s (t::typ)) \<subseteq> cod s \<union> free_tv t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   339
  by (induct t) (auto simp add: free_tv_subst_var)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   340
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   341
lemma free_tv_app_subst_tel: "free_tv ($s (ts::typ list)) \<subseteq> cod s \<union> free_tv ts"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   342
  apply (induct ts)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   343
   apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   344
  apply (cut_tac free_tv_app_subst_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   345
  apply fastsimp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   346
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   347
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   348
lemma free_tv_comp_subst:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   349
    "free_tv (\<lambda>u::nat. $s1 (s2 u) :: typ) \<subseteq> free_tv s1 \<union> free_tv s2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   350
  apply (unfold free_tv_subst dom_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   351
  apply (tactic {*
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   352
    fast_tac (set_cs addSDs [thm "free_tv_app_subst_te" RS subsetD,
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   353
    thm "free_tv_subst_var" RS subsetD]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   354
    addss (simpset() delsimps bex_simps
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   355
    addsimps [thm "cod_def", thm "dom_def"])) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   356
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   357
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   358
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   359
subsection {* Most general unifiers *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   360
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   361
consts
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   362
  mgu :: "typ \<Rightarrow> typ \<Rightarrow> subst maybe"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   363
axioms
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   364
  mgu_eq [simp]: "mgu t1 t2 = Ok u \<Longrightarrow> $u t1 = $u t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   365
  mgu_mg [simp]: "mgu t1 t2 = Ok u \<Longrightarrow> $s t1 = $s t2 \<Longrightarrow> \<exists>r. s = $r o u"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   366
  mgu_Ok: "$s t1 = $s t2 \<Longrightarrow> \<exists>u. mgu t1 t2 = Ok u"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   367
  mgu_free [simp]: "mgu t1 t2 = Ok u \<Longrightarrow> free_tv u \<subseteq> free_tv t1 \<union> free_tv t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   368
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   369
lemma mgu_new: "mgu t1 t2 = Ok u \<Longrightarrow> new_tv n t1 \<Longrightarrow> new_tv n t2 \<Longrightarrow> new_tv n u"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   370
  -- {* @{text mgu} does not introduce new type variables *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   371
  by (unfold new_tv_def) (blast dest: mgu_free)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   372
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   373
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   374
section {* Mini-ML with type inference rules *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   375
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   376
datatype
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   377
  expr = Var nat | Abs expr | App expr expr
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   378
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   379
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   380
text {* Type inference rules. *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   381
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   382
consts
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   383
  has_type :: "(typ list \<times> expr \<times> typ) set"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   384
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   385
syntax
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   386
  "_has_type" :: "typ list \<Rightarrow> expr \<Rightarrow> typ \<Rightarrow> bool"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   387
    ("((_) |-/ (_) :: (_))" [60, 0, 60] 60)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   388
translations
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   389
  "a |- e :: t" == "(a, e, t) \<in> has_type"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   390
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   391
inductive has_type
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   392
  intros
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   393
    Var: "n < length a \<Longrightarrow> a |- Var n :: a ! n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   394
    Abs: "t1#a |- e :: t2 \<Longrightarrow> a |- Abs e :: t1 -> t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   395
    App: "a |- e1 :: t2 -> t1 \<Longrightarrow> a |- e2 :: t2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   396
      \<Longrightarrow> a |- App e1 e2 :: t1"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   397
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   398
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   399
text {* Type assigment is closed wrt.\ substitution. *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   400
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   401
lemma has_type_subst_closed: "a |- e :: t ==> $s a |- e :: $s t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   402
proof -
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   403
  assume "a |- e :: t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   404
  thus ?thesis (is "?P a e t")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   405
  proof induct
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   406
    case (Var a n)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   407
    hence "n < length (map ($ s) a)" by simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   408
    hence "map ($ s) a |- Var n :: map ($ s) a ! n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   409
      by (rule has_type.Var)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   410
    also have "map ($ s) a ! n = $ s (a ! n)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   411
      by (rule nth_map)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   412
    also have "map ($ s) a = $ s a"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   413
      by (simp only: app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   414
    finally show "?P a (Var n) (a ! n)" .
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   415
  next
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   416
    case (Abs a e t1 t2)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   417
    hence "$ s t1 # map ($ s) a |- e :: $ s t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   418
      by (simp add: app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   419
    hence "map ($ s) a |- Abs e :: $ s t1 -> $ s t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   420
      by (rule has_type.Abs)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   421
    thus "?P a (Abs e) (t1 -> t2)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   422
      by (simp add: app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   423
  next
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   424
    case App
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   425
    thus ?case by (simp add: has_type.App)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   426
  qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   427
qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   428
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   429
12961
wenzelm
parents: 12945
diff changeset
   430
section {* Correctness and completeness of the type inference algorithm W *}
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   431
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   432
consts
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   433
  W :: "expr \<Rightarrow> typ list \<Rightarrow> nat \<Rightarrow> (subst \<times> typ \<times> nat) maybe"  ("\<W>")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   434
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   435
primrec
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   436
  "\<W> (Var i) a n =
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   437
    (if i < length a then Ok (id_subst, a ! i, n) else Fail)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   438
  "\<W> (Abs e) a n =
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   439
    ((s, t, m) := \<W> e (TVar n # a) (Suc n);
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   440
     Ok (s, (s n) -> t, m))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   441
  "\<W> (App e1 e2) a n =
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   442
    ((s1, t1, m1) := \<W> e1 a n;
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   443
     (s2, t2, m2) := \<W> e2 ($s1 a) m1;
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   444
     u := mgu ($ s2 t1) (t2 -> TVar m2);
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   445
     Ok ($u o $s2 o s1, $u (TVar m2), Suc m2))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   446
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   447
theorem W_correct: "!!a s t m n. Ok (s, t, m) = \<W> e a n ==> $s a |- e :: t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   448
  (is "PROP ?P e")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   449
proof (induct e)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   450
  fix a s t m n
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   451
  {
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   452
    fix i
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   453
    assume "Ok (s, t, m) = \<W> (Var i) a n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   454
    thus "$s a |- Var i :: t" by (simp add: has_type.Var split: if_splits)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   455
  next
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   456
    fix e assume hyp: "PROP ?P e"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   457
    assume "Ok (s, t, m) = \<W> (Abs e) a n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   458
    then obtain t' where "t = s n -> t'"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   459
        and "Ok (s, t', m) = \<W> e (TVar n # a) (Suc n)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   460
      by (auto split: bind_splits)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   461
    with hyp show "$s a |- Abs e :: t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   462
      by (force intro: has_type.Abs)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   463
  next
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   464
    fix e1 e2 assume hyp1: "PROP ?P e1" and hyp2: "PROP ?P e2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   465
    assume "Ok (s, t, m) = \<W> (App e1 e2) a n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   466
    then obtain s1 t1 n1 s2 t2 n2 u where
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   467
          s: "s = $u o $s2 o s1"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   468
        and t: "t = u n2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   469
        and mgu_ok: "mgu ($s2 t1) (t2 -> TVar n2) = Ok u"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   470
        and W1_ok: "Ok (s1, t1, n1) = \<W> e1 a n"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   471
        and W2_ok: "Ok (s2, t2, n2) = \<W> e2 ($s1 a) n1"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   472
      by (auto split: bind_splits simp: that)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   473
    show "$s a |- App e1 e2 :: t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   474
    proof (rule has_type.App)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   475
      from s have s': "$u ($s2 ($s1 a)) = $s a"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   476
        by (simp add: subst_comp_tel o_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   477
      show "$s a |- e1 :: $u t2 -> t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   478
      proof -
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   479
        from W1_ok have "$s1 a |- e1 :: t1" by (rule hyp1)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   480
        hence "$u ($s2 ($s1 a)) |- e1 :: $u ($s2 t1)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   481
          by (intro has_type_subst_closed)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   482
        with s' t mgu_ok show ?thesis by simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   483
      qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   484
      show "$s a |- e2 :: $u t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   485
      proof -
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   486
        from W2_ok have "$s2 ($s1 a) |- e2 :: t2" by (rule hyp2)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   487
        hence "$u ($s2 ($s1 a)) |- e2 :: $u t2"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   488
          by (rule has_type_subst_closed)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   489
        with s' show ?thesis by simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   490
      qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   491
    qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   492
  }
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   493
qed
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   494
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   495
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   496
inductive_cases has_type_casesE:
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   497
  "s |- Var n :: t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   498
  "s |- Abs e :: t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   499
  "s |- App e1 e2 ::t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   500
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   501
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   502
lemmas [simp] = Suc_le_lessD
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   503
  and [simp del] = less_imp_le ex_simps all_simps
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   504
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   505
lemma W_var_ge [simp]: "!!a n s t m. \<W> e a n = Ok (s, t, m) \<Longrightarrow> n \<le> m"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   506
  -- {* the resulting type variable is always greater or equal than the given one *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   507
  apply (atomize (full))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   508
  apply (induct e)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   509
    txt {* case @{text "Var n"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   510
    apply clarsimp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   511
   txt {* case @{text "Abs e"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   512
   apply (simp split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   513
   apply (fast dest: Suc_leD)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   514
  txt {* case @{text "App e1 e2"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   515
  apply (simp (no_asm) split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   516
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   517
  apply (rename_tac s t na sa ta nb sb)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   518
  apply (erule_tac x = a in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   519
  apply (erule_tac x = n in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   520
  apply (erule_tac x = "$s a" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   521
  apply (erule_tac x = s in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   522
  apply (erule_tac x = t in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   523
  apply (erule_tac x = na in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   524
  apply (erule_tac x = na in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   525
  apply (simp add: eq_sym_conv)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   526
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   527
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   528
lemma W_var_geD: "Ok (s, t, m) = \<W> e a n \<Longrightarrow> n \<le> m"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   529
  by (simp add: eq_sym_conv)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   530
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   531
lemma new_tv_W: "!!n a s t m.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   532
  new_tv n a \<Longrightarrow> \<W> e a n = Ok (s, t, m) \<Longrightarrow> new_tv m s & new_tv m t"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   533
  -- {* resulting type variable is new *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   534
  apply (atomize (full))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   535
  apply (induct e)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   536
    txt {* case @{text "Var n"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   537
    apply clarsimp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   538
    apply (force elim: list_ball_nth simp add: id_subst_def new_tv_list new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   539
   txt {* case @{text "Abs e"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   540
   apply (simp (no_asm) add: new_tv_subst new_tv_Suc_list split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   541
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   542
   apply (erule_tac x = "Suc n" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   543
   apply (erule_tac x = "TVar n # a" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   544
   apply (fastsimp simp add: new_tv_subst new_tv_Suc_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   545
  txt {* case @{text "App e1 e2"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   546
  apply (simp (no_asm) split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   547
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   548
  apply (rename_tac s t na sa ta nb sb)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   549
  apply (erule_tac x = n in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   550
  apply (erule_tac x = a in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   551
  apply (erule_tac x = s in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   552
  apply (erule_tac x = t in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   553
  apply (erule_tac x = na in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   554
  apply (erule_tac x = na in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   555
  apply (simp add: eq_sym_conv)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   556
  apply (erule_tac x = "$s a" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   557
  apply (erule_tac x = sa in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   558
  apply (erule_tac x = ta in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   559
  apply (erule_tac x = nb in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   560
  apply (simp add: o_def rotate_Ok)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   561
  apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   562
   apply (rule new_tv_subst_comp_2)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   563
    apply (rule new_tv_subst_comp_2)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   564
     apply (rule lessI [THEN less_imp_le, THEN new_tv_subst_le])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   565
     apply (rule_tac n = na in new_tv_subst_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   566
      apply (simp add: rotate_Ok)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   567
     apply (simp (no_asm_simp))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   568
    apply (fast dest: W_var_geD intro: new_tv_list_le new_tv_subst_tel
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   569
      lessI [THEN less_imp_le, THEN new_tv_subst_le])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   570
   apply (erule sym [THEN mgu_new])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   571
    apply (best dest: W_var_geD intro: new_tv_subst_te new_tv_list_le new_tv_subst_tel
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   572
      lessI [THEN less_imp_le, THEN new_tv_le] lessI [THEN less_imp_le, THEN new_tv_subst_le]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   573
      new_tv_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   574
   apply (tactic {* fast_tac (HOL_cs addDs [thm "W_var_geD"]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   575
     addIs [thm "new_tv_list_le", thm "new_tv_subst_tel", thm "new_tv_le"]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   576
     addss (simpset())) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   577
  apply (rule lessI [THEN new_tv_subst_var])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   578
  apply (erule sym [THEN mgu_new])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   579
    apply (bestsimp intro!: lessI [THEN less_imp_le, THEN new_tv_le] new_tv_subst_te
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   580
      dest!: W_var_geD intro: new_tv_list_le new_tv_subst_tel
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   581
        lessI [THEN less_imp_le, THEN new_tv_subst_le] new_tv_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   582
  apply (tactic {* fast_tac (HOL_cs addDs [thm "W_var_geD"]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   583
    addIs [thm "new_tv_list_le", thm "new_tv_subst_tel", thm "new_tv_le"]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   584
    addss (simpset())) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   585
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   586
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   587
lemma free_tv_W: "!!n a s t m v. \<W> e a n = Ok (s, t, m) \<Longrightarrow>
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   588
    (v \<in> free_tv s \<or> v \<in> free_tv t) \<Longrightarrow> v < n \<Longrightarrow> v \<in> free_tv a"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   589
  apply (atomize (full))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   590
  apply (induct e)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   591
    txt {* case @{text "Var n"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   592
    apply clarsimp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   593
    apply (tactic {* fast_tac (HOL_cs addIs [nth_mem, subsetD, thm "ftv_mem_sub_ftv_list"]) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   594
   txt {* case @{text "Abs e"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   595
   apply (simp add: free_tv_subst split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   596
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   597
   apply (rename_tac s t n1 v)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   598
   apply (erule_tac x = "Suc n" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   599
   apply (erule_tac x = "TVar n # a" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   600
   apply (erule_tac x = s in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   601
   apply (erule_tac x = t in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   602
   apply (erule_tac x = n1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   603
   apply (erule_tac x = v in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   604
   apply (force elim!: allE intro: cod_app_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   605
  txt {* case @{text "App e1 e2"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   606
  apply (simp (no_asm) split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   607
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   608
  apply (rename_tac s t n1 s1 t1 n2 s3 v)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   609
  apply (erule_tac x = n in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   610
  apply (erule_tac x = a in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   611
  apply (erule_tac x = s in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   612
  apply (erule_tac x = t in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   613
  apply (erule_tac x = n1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   614
  apply (erule_tac x = n1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   615
  apply (erule_tac x = v in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   616
  txt {* second case *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   617
  apply (erule_tac x = "$ s a" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   618
  apply (erule_tac x = s1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   619
  apply (erule_tac x = t1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   620
  apply (erule_tac x = n2 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   621
  apply (erule_tac x = v in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   622
  apply (tactic "safe_tac (empty_cs addSIs [conjI, impI] addSEs [conjE])")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   623
   apply (simp add: rotate_Ok o_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   624
   apply (drule W_var_geD)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   625
   apply (drule W_var_geD)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   626
   apply (frule less_le_trans, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   627
   apply (fastsimp dest: free_tv_comp_subst [THEN subsetD] sym [THEN mgu_free] codD
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   628
     free_tv_app_subst_te [THEN subsetD] free_tv_app_subst_tel [THEN subsetD] subsetD elim: UnE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   629
  apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   630
  apply (drule sym [THEN W_var_geD])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   631
  apply (drule sym [THEN W_var_geD])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   632
  apply (frule less_le_trans, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   633
  apply (tactic {* fast_tac (HOL_cs addDs [thm "mgu_free", thm "codD",
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   634
    thm "free_tv_subst_var" RS subsetD,
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   635
    thm "free_tv_app_subst_te" RS subsetD,
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   636
    thm "free_tv_app_subst_tel" RS subsetD, less_le_trans, subsetD]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   637
    addSEs [UnE] addss (simpset() setSolver unsafe_solver)) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   638
      -- {* builtin arithmetic in simpset messes things up *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   639
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   640
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   641
text {*
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   642
  \medskip Completeness of @{text \<W>} wrt.\ @{text has_type}.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   643
*}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   644
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   645
lemma W_complete_aux: "!!s' a t' n. $s' a |- e :: t' \<Longrightarrow> new_tv n a \<Longrightarrow>
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   646
    (\<exists>s t. (\<exists>m. \<W> e a n = Ok (s, t, m)) \<and> (\<exists>r. $s' a = $r ($s a) \<and> t' = $r t))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   647
  apply (atomize (full))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   648
  apply (induct e)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   649
    txt {* case @{text "Var n"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   650
    apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   651
    apply (simp (no_asm) cong add: conj_cong)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   652
    apply (erule has_type_casesE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   653
    apply (simp add: eq_sym_conv app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   654
    apply (rule_tac x = s' in exI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   655
    apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   656
   txt {* case @{text "Abs e"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   657
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   658
   apply (erule has_type_casesE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   659
   apply (erule_tac x = "\<lambda>x. if x = n then t1 else (s' x)" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   660
   apply (erule_tac x = "TVar n # a" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   661
   apply (erule_tac x = t2 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   662
   apply (erule_tac x = "Suc n" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   663
   apply (fastsimp cong add: conj_cong split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   664
  txt {* case @{text "App e1 e2"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   665
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   666
  apply (erule has_type_casesE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   667
  apply (erule_tac x = s' in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   668
  apply (erule_tac x = a in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   669
  apply (erule_tac x = "t2 -> t'" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   670
  apply (erule_tac x = n in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   671
  apply (tactic "safe_tac HOL_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   672
  apply (erule_tac x = r in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   673
  apply (erule_tac x = "$s a" in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   674
  apply (erule_tac x = t2 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   675
  apply (erule_tac x = m in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   676
  apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   677
  apply (tactic "safe_tac HOL_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   678
   apply (tactic {* fast_tac (HOL_cs addIs [sym RS thm "W_var_geD",
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   679
     thm "new_tv_W" RS conjunct1, thm "new_tv_list_le", thm "new_tv_subst_tel"]) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   680
  apply (subgoal_tac
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   681
    "$(\<lambda>x. if x = ma then t' else (if x \<in> free_tv t - free_tv sa then r x
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   682
      else ra x)) ($ sa t) =
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   683
    $(\<lambda>x. if x = ma then t' else (if x \<in> free_tv t - free_tv sa then r x
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   684
      else ra x)) (ta -> (TVar ma))")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   685
   apply (rule_tac [2] t = "$(\<lambda>x. if x = ma then t'
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   686
     else (if x \<in> (free_tv t - free_tv sa) then r x else ra x)) ($sa t)" and
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   687
     s = "($ ra ta) -> t'" in ssubst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   688
    prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   689
    apply (simp add: subst_comp_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   690
    apply (rule eq_free_eq_subst_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   691
    apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   692
    apply (subgoal_tac "na \<noteq> ma")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   693
     prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   694
     apply (fast dest: new_tv_W sym [THEN W_var_geD] new_tv_not_free_tv new_tv_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   695
    apply (case_tac "na \<in> free_tv sa")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   696
     txt {* @{text "na \<notin> free_tv sa"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   697
     prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   698
     apply (frule not_free_impl_id)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   699
     apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   700
    txt {* @{text "na \<in> free_tv sa"} *}
13601
fd3e3d6b37b2 Adapted to new simplifier.
berghofe
parents: 13537
diff changeset
   701
    apply (drule_tac ts1 = "$s a" and r = "$ r ($ s a)" in subst_comp_tel [THEN [2] trans])
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   702
    apply (drule_tac eq_subst_tel_eq_free)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   703
     apply (fast intro: free_tv_W free_tv_le_new_tv dest: new_tv_W)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   704
    apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   705
    apply (case_tac "na \<in> dom sa")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   706
     prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   707
     txt {* @{text "na \<noteq> dom sa"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   708
     apply (simp add: dom_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   709
    txt {* @{text "na \<in> dom sa"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   710
    apply (rule eq_free_eq_subst_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   711
    apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   712
    apply (subgoal_tac "nb \<noteq> ma")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   713
     prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   714
     apply (frule new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   715
     apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   716
     apply (drule new_tv_subst_tel)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   717
      apply (fast intro: new_tv_list_le dest: sym [THEN W_var_geD])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   718
     apply (fastsimp dest: new_tv_W new_tv_not_free_tv simp add: cod_def free_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   719
    apply (fastsimp simp add: cod_def free_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   720
   prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   721
   apply (simp (no_asm))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   722
   apply (rule eq_free_eq_subst_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   723
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   724
   apply (subgoal_tac "na \<noteq> ma")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   725
    prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   726
    apply (frule new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   727
    apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   728
    apply (drule sym [THEN W_var_geD])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   729
    apply (fast dest: new_tv_list_le new_tv_subst_tel new_tv_W new_tv_not_free_tv)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   730
   apply (case_tac "na \<in> free_tv t - free_tv sa")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   731
    prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   732
    txt {* case @{text "na \<notin> free_tv t - free_tv sa"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   733
    apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   734
    defer
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   735
    txt {* case @{text "na \<in> free_tv t - free_tv sa"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   736
    apply simp
13601
fd3e3d6b37b2 Adapted to new simplifier.
berghofe
parents: 13537
diff changeset
   737
    apply (drule_tac ts1 = "$s a" and r = "$ r ($ s a)" in subst_comp_tel [THEN [2] trans])
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   738
    apply (drule eq_subst_tel_eq_free)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   739
     apply (fast intro: free_tv_W free_tv_le_new_tv dest: new_tv_W)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   740
    apply (simp add: free_tv_subst dom_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   741
   prefer 2 apply fast
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   742
  apply (simp (no_asm_simp) split add: split_bind)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   743
  apply (tactic "safe_tac HOL_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   744
   apply (drule mgu_Ok)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   745
   apply fastsimp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   746
  apply (drule mgu_mg, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   747
  apply (erule exE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   748
  apply (rule_tac x = rb in exI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   749
  apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   750
   prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   751
   apply (drule_tac x = ma in fun_cong)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   752
   apply (simp add: eq_sym_conv)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   753
  apply (simp (no_asm) add: o_def subst_comp_tel [symmetric])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   754
  apply (rule subst_comp_tel [symmetric, THEN [2] trans])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   755
  apply (simp add: o_def eq_sym_conv)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   756
  apply (rule eq_free_eq_subst_tel)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   757
  apply (tactic "safe_tac HOL_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   758
  apply (subgoal_tac "ma \<noteq> na")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   759
   prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   760
   apply (frule new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   761
   apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   762
   apply (drule new_tv_subst_tel)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   763
    apply (fast intro: new_tv_list_le dest: sym [THEN W_var_geD])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   764
   apply (frule_tac n = m in new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   765
   apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   766
   apply (drule free_tv_app_subst_tel [THEN subsetD])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   767
   apply (tactic {* fast_tac (set_cs addDs [sym RS thm "W_var_geD", thm "new_tv_list_le",
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   768
     thm "codD", thm "new_tv_not_free_tv"]) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   769
  apply (case_tac "na \<in> free_tv t - free_tv sa")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   770
   prefer 2
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   771
   txt {* case @{text "na \<notin> free_tv t - free_tv sa"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   772
   apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   773
   defer
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   774
   txt {* case @{text "na \<in> free_tv t - free_tv sa"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   775
   apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   776
   apply (drule free_tv_app_subst_tel [THEN subsetD])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   777
   apply (fastsimp dest: codD subst_comp_tel [THEN [2] trans]
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   778
     eq_subst_tel_eq_free simp add: free_tv_subst dom_def)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   779
  apply fast
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   780
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   781
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   782
lemma W_complete: "[] |- e :: t' ==>
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   783
    \<exists>s t. (\<exists>m. \<W> e [] n = Ok (s, t, m)) \<and> (\<exists>r. t' = $r t)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   784
  apply (cut_tac a = "[]" and s' = id_subst and e = e and t' = t' in W_complete_aux)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   785
    apply simp_all
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   786
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   787
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   788
12961
wenzelm
parents: 12945
diff changeset
   789
section {* Equivalence of W and I *}
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   790
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   791
text {*
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   792
  Recursive definition of type inference algorithm @{text \<I>} for
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   793
  Mini-ML.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   794
*}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   795
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   796
consts
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   797
  I :: "expr \<Rightarrow> typ list \<Rightarrow> nat \<Rightarrow> subst \<Rightarrow> (subst \<times> typ \<times> nat) maybe"  ("\<I>")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   798
primrec
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   799
  "\<I> (Var i) a n s = (if i < length a then Ok (s, a ! i, n) else Fail)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   800
  "\<I> (Abs e) a n s = ((s, t, m) := \<I> e (TVar n # a) (Suc n) s;
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   801
    Ok (s, TVar n -> t, m))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   802
  "\<I> (App e1 e2) a n s =
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   803
    ((s1, t1, m1) := \<I> e1 a n s;
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   804
    (s2, t2, m2) := \<I> e2 a m1 s1;
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   805
    u := mgu ($s2 t1) ($s2 t2 -> TVar m2);
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   806
    Ok($u o s2, TVar m2, Suc m2))"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   807
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   808
text {* \medskip Correctness. *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   809
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   810
lemma I_correct_wrt_W: "!!a m s s' t n.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   811
    new_tv m a \<and> new_tv m s \<Longrightarrow> \<I> e a m s = Ok (s', t, n) \<Longrightarrow>
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   812
    \<exists>r. \<W> e ($s a) m = Ok (r, $s' t, n) \<and> s' = ($r o s)"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   813
  apply (atomize (full))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   814
  apply (induct e)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   815
    txt {* case @{text "Var n"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   816
    apply (simp add: app_subst_list split: split_if)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   817
   txt {* case @{text "Abs e"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   818
   apply (tactic {* asm_full_simp_tac
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   819
     (simpset() setloop (split_inside_tac [thm "split_bind"])) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   820
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   821
   apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   822
    apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   823
    apply (erule allE)+
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   824
    apply (erule impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   825
     prefer 2 apply (fastsimp simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   826
    apply (tactic {* fast_tac (HOL_cs addIs [thm "new_tv_Suc_list" RS mp,
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   827
      thm "new_tv_subst_le", less_imp_le, lessI]) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   828
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   829
   apply (erule allE)+
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   830
   apply (erule impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   831
    prefer 2 apply (fastsimp simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   832
   apply (tactic {* fast_tac (HOL_cs addIs [thm "new_tv_Suc_list" RS mp,
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   833
     thm "new_tv_subst_le", less_imp_le, lessI]) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   834
  txt {* case @{text "App e1 e2"} *}
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   835
  apply (tactic {* simp_tac (simpset () setloop (split_inside_tac [thm "split_bind"])) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   836
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   837
  apply (rename_tac s1' t1 n1 s2' t2 n2 sa)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   838
  apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   839
   apply fastsimp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   840
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   841
  apply (rename_tac s1 t1' n1')
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   842
  apply (erule_tac x = a in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   843
  apply (erule_tac x = m in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   844
  apply (erule_tac x = s in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   845
  apply (erule_tac x = s1' in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   846
  apply (erule_tac x = t1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   847
  apply (erule_tac x = n1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   848
  apply (erule_tac x = a in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   849
  apply (erule_tac x = n1 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   850
  apply (erule_tac x = s1' in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   851
  apply (erule_tac x = s2' in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   852
  apply (erule_tac x = t2 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   853
  apply (erule_tac x = n2 in allE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   854
  apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   855
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   856
   apply (rule notI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   857
   apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   858
   apply (erule impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   859
    apply (frule new_tv_subst_tel, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   860
    apply (drule_tac a = "$s a" in new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   861
    apply (fastsimp dest: sym [THEN W_var_geD] new_tv_subst_le new_tv_list_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   862
   apply (fastsimp simp add: subst_comp_tel)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   863
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   864
  apply (rename_tac s2 t2' n2')
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   865
  apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   866
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   867
   apply (rule notI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   868
   apply simp
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   869
   apply (erule impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   870
   apply (frule new_tv_subst_tel, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   871
   apply (drule_tac a = "$s a" in new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   872
    apply (fastsimp dest: sym [THEN W_var_geD] new_tv_subst_le new_tv_list_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   873
   apply (fastsimp simp add: subst_comp_tel subst_comp_te)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   874
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   875
  apply (erule (1) notE impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   876
  apply (erule (1) notE impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   877
  apply (erule exE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   878
  apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   879
  apply (erule impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   880
   apply (frule new_tv_subst_tel, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   881
   apply (drule_tac a = "$s a" in new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   882
   apply (fastsimp dest: sym [THEN W_var_geD] new_tv_subst_le new_tv_list_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   883
  apply (erule (1) notE impE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   884
  apply (erule exE conjE)+
13601
fd3e3d6b37b2 Adapted to new simplifier.
berghofe
parents: 13537
diff changeset
   885
  apply (simp (asm_lr) add: subst_comp_tel subst_comp_te o_def, (erule conjE)+, hypsubst)+
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   886
  apply (subgoal_tac "new_tv n2 s \<and> new_tv n2 r \<and> new_tv n2 ra")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   887
   apply (simp add: new_tv_subst)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   888
  apply (frule new_tv_subst_tel, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   889
  apply (drule_tac a = "$s a" in new_tv_W, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   890
  apply (tactic "safe_tac HOL_cs")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   891
    apply (bestsimp dest: sym [THEN W_var_geD] new_tv_subst_le new_tv_list_le)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   892
   apply (fastsimp dest: sym [THEN W_var_geD] new_tv_subst_le new_tv_list_le)
15236
f289e8ba2bb3 Proofs needed to be updated because induction now preserves name of
nipkow
parents: 15140
diff changeset
   893
  apply (drule_tac e = e1 in sym [THEN W_var_geD])
12944
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   894
  apply (drule new_tv_subst_tel, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   895
  apply (drule_tac ts = "$s a" in new_tv_list_le, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   896
  apply (drule new_tv_subst_tel, assumption)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   897
  apply (bestsimp dest: new_tv_W simp add: subst_comp_tel)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   898
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   899
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   900
lemma I_complete_wrt_W: "!!a m s.
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   901
    new_tv m a \<and> new_tv m s \<Longrightarrow> \<I> e a m s = Fail \<Longrightarrow> \<W> e ($s a) m = Fail"
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   902
  apply (atomize (full))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   903
  apply (induct e)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   904
    apply (simp add: app_subst_list)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   905
   apply (simp (no_asm))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   906
   apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   907
   apply (subgoal_tac "TVar m # $s a = $s (TVar m # a)")
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   908
    apply (tactic {* asm_simp_tac (HOL_ss addsimps
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   909
      [thm "new_tv_Suc_list", lessI RS less_imp_le RS thm "new_tv_subst_le"]) 1 *})
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   910
    apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   911
    apply (drule new_tv_not_free_tv [THEN not_free_impl_id])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   912
    apply (simp (no_asm_simp))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   913
  apply (simp (no_asm_simp))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   914
  apply (intro strip)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   915
  apply (erule exE)+
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   916
  apply (erule conjE)+
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   917
  apply (drule I_correct_wrt_W [COMP swap_prems_rl])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   918
   apply fast
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   919
  apply (erule exE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   920
  apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   921
  apply hypsubst
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   922
  apply (simp (no_asm_simp))
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   923
  apply (erule disjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   924
   apply (rule disjI1)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   925
   apply (simp (no_asm_use) add: o_def subst_comp_tel)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   926
   apply (erule allE, erule allE, erule allE, erule impE, erule_tac [2] impE,
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   927
     erule_tac [2] asm_rl, erule_tac [2] asm_rl)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   928
   apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   929
    apply (fast intro: W_var_ge [THEN new_tv_list_le])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   930
   apply (rule new_tv_subst_comp_2)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   931
    apply (fast intro: W_var_ge [THEN new_tv_subst_le])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   932
   apply (fast intro!: new_tv_subst_tel intro: new_tv_W [THEN conjunct1])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   933
  apply (rule disjI2)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   934
  apply (erule exE)+
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   935
  apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   936
  apply (drule I_correct_wrt_W [COMP swap_prems_rl])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   937
   apply (rule conjI)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   938
   apply (fast intro: W_var_ge [THEN new_tv_list_le])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   939
   apply (rule new_tv_subst_comp_1)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   940
   apply (fast intro: W_var_ge [THEN new_tv_subst_le])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   941
   apply (fast intro!: new_tv_subst_tel intro: new_tv_W [THEN conjunct1])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   942
  apply (erule exE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   943
  apply (erule conjE)
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   944
  apply hypsubst
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   945
  apply (simp add: o_def subst_comp_te [symmetric] subst_comp_tel [symmetric])
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   946
  done
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   947
fa6a3ddec27f converted;
wenzelm
parents:
diff changeset
   948
end