src/HOL/Partial_Function.thy
author krauss
Sat, 23 Oct 2010 23:41:19 +0200
changeset 40107 374f3ef9f940
child 40252 029400b6c893
permissions -rw-r--r--
first version of partial_function package
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
40107
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     1
(* Title:    HOL/Partial_Function.thy
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     2
   Author:   Alexander Krauss, TU Muenchen
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     3
*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     4
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     5
header {* Partial Function Definitions *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     6
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     7
theory Partial_Function
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     8
imports Complete_Partial_Order Option
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     9
uses 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    10
  "Tools/Function/function_lib.ML" 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    11
  "Tools/Function/partial_function.ML" 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    12
begin
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    13
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    14
setup Partial_Function.setup
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    15
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    16
subsection {* Axiomatic setup *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    17
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    18
text {* This techical locale constains the requirements for function
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    19
  definitions with ccpo fixed points.  *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    20
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    21
definition "fun_ord ord f g \<longleftrightarrow> (\<forall>x. ord (f x) (g x))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    22
definition "fun_lub L A = (\<lambda>x. L {y. \<exists>f\<in>A. y = f x})"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    23
definition "img_ord f ord = (\<lambda>x y. ord (f x) (f y))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    24
definition "img_lub f g Lub = (\<lambda>A. g (Lub (f ` A)))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    25
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    26
lemma call_mono[partial_function_mono]: "monotone (fun_ord ord) ord (\<lambda>f. f t)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    27
by (rule monotoneI) (auto simp: fun_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    28
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    29
lemma if_mono[partial_function_mono]: "monotone orda ordb F 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    30
\<Longrightarrow> monotone orda ordb G
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    31
\<Longrightarrow> monotone orda ordb (\<lambda>f. if c then F f else G f)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    32
unfolding monotone_def by simp
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    33
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    34
definition "mk_less R = (\<lambda>x y. R x y \<and> \<not> R y x)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    35
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    36
locale partial_function_definitions = 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    37
  fixes leq :: "'a \<Rightarrow> 'a \<Rightarrow> bool"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    38
  fixes lub :: "'a set \<Rightarrow> 'a"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    39
  assumes leq_refl: "leq x x"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    40
  assumes leq_trans: "leq x y \<Longrightarrow> leq y z \<Longrightarrow> leq x z"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    41
  assumes leq_antisym: "leq x y \<Longrightarrow> leq y x \<Longrightarrow> x = y"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    42
  assumes lub_upper: "chain leq A \<Longrightarrow> x \<in> A \<Longrightarrow> leq x (lub A)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    43
  assumes lub_least: "chain leq A \<Longrightarrow> (\<And>x. x \<in> A \<Longrightarrow> leq x z) \<Longrightarrow> leq (lub A) z"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    44
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    45
lemma partial_function_lift:
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    46
  assumes "partial_function_definitions ord lb"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    47
  shows "partial_function_definitions (fun_ord ord) (fun_lub lb)" (is "partial_function_definitions ?ordf ?lubf")
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    48
proof -
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    49
  interpret partial_function_definitions ord lb by fact
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    50
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    51
  { fix A a assume A: "chain ?ordf A"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    52
    have "chain ord {y. \<exists>f\<in>A. y = f a}" (is "chain ord ?C")
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    53
    proof (rule chainI)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    54
      fix x y assume "x \<in> ?C" "y \<in> ?C"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    55
      then obtain f g where fg: "f \<in> A" "g \<in> A" 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    56
        and [simp]: "x = f a" "y = g a" by blast
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    57
      from chainD[OF A fg]
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    58
      show "ord x y \<or> ord y x" unfolding fun_ord_def by auto
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    59
    qed }
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    60
  note chain_fun = this
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    61
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    62
  show ?thesis
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    63
  proof
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    64
    fix x show "?ordf x x"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    65
      unfolding fun_ord_def by (auto simp: leq_refl)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    66
  next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    67
    fix x y z assume "?ordf x y" "?ordf y z"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    68
    thus "?ordf x z" unfolding fun_ord_def 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    69
      by (force dest: leq_trans)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    70
  next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    71
    fix x y assume "?ordf x y" "?ordf y x"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    72
    thus "x = y" unfolding fun_ord_def
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    73
      by (force intro!: ext dest: leq_antisym)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    74
  next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    75
    fix A f assume f: "f \<in> A" and A: "chain ?ordf A"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    76
    thus "?ordf f (?lubf A)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    77
      unfolding fun_lub_def fun_ord_def
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    78
      by (blast intro: lub_upper chain_fun[OF A] f)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    79
  next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    80
    fix A :: "('b \<Rightarrow> 'a) set" and g :: "'b \<Rightarrow> 'a"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    81
    assume A: "chain ?ordf A" and g: "\<And>f. f \<in> A \<Longrightarrow> ?ordf f g"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    82
    show "?ordf (?lubf A) g" unfolding fun_lub_def fun_ord_def
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    83
      by (blast intro: lub_least chain_fun[OF A] dest: g[unfolded fun_ord_def])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    84
   qed
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    85
qed
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    86
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    87
lemma ccpo: assumes "partial_function_definitions ord lb"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    88
  shows "class.ccpo ord (mk_less ord) lb"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    89
using assms unfolding partial_function_definitions_def mk_less_def
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    90
by unfold_locales blast+
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    91
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    92
lemma partial_function_image:
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    93
  assumes "partial_function_definitions ord Lub"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    94
  assumes inj: "\<And>x y. f x = f y \<Longrightarrow> x = y"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    95
  assumes inv: "\<And>x. f (g x) = x"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    96
  shows "partial_function_definitions (img_ord f ord) (img_lub f g Lub)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    97
proof -
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    98
  let ?iord = "img_ord f ord"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    99
  let ?ilub = "img_lub f g Lub"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   100
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   101
  interpret partial_function_definitions ord Lub by fact
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   102
  show ?thesis
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   103
  proof
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   104
    fix A x assume "chain ?iord A" "x \<in> A"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   105
    then have "chain ord (f ` A)" "f x \<in> f ` A"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   106
      by (auto simp: img_ord_def intro: chainI dest: chainD)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   107
    thus "?iord x (?ilub A)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   108
      unfolding inv img_lub_def img_ord_def by (rule lub_upper)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   109
  next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   110
    fix A x assume "chain ?iord A"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   111
      and 1: "\<And>z. z \<in> A \<Longrightarrow> ?iord z x"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   112
    then have "chain ord (f ` A)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   113
      by (auto simp: img_ord_def intro: chainI dest: chainD)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   114
    thus "?iord (?ilub A) x"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   115
      unfolding inv img_lub_def img_ord_def
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   116
      by (rule lub_least) (auto dest: 1[unfolded img_ord_def])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   117
  qed (auto simp: img_ord_def intro: leq_refl dest: leq_trans leq_antisym inj)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   118
qed
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   119
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   120
context partial_function_definitions
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   121
begin
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   122
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   123
abbreviation "le_fun \<equiv> fun_ord leq"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   124
abbreviation "lub_fun \<equiv> fun_lub lub"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   125
abbreviation "fixp_fun == ccpo.fixp le_fun lub_fun"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   126
abbreviation "mono_body \<equiv> monotone le_fun leq"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   127
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   128
text {* Interpret manually, to avoid flooding everything with facts about
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   129
  orders *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   130
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   131
lemma ccpo: "class.ccpo le_fun (mk_less le_fun) lub_fun"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   132
apply (rule ccpo)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   133
apply (rule partial_function_lift)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   134
apply (rule partial_function_definitions_axioms)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   135
done
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   136
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   137
text {* The crucial fixed-point theorem *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   138
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   139
lemma mono_body_fixp: 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   140
  "(\<And>x. mono_body (\<lambda>f. F f x)) \<Longrightarrow> fixp_fun F = F (fixp_fun F)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   141
by (rule ccpo.fixp_unfold[OF ccpo]) (auto simp: monotone_def fun_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   142
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   143
text {* Version with curry/uncurry combinators, to be used by package *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   144
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   145
lemma fixp_rule_uc:
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   146
  fixes F :: "'c \<Rightarrow> 'c" and
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   147
    U :: "'c \<Rightarrow> 'b \<Rightarrow> 'a" and
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   148
    C :: "('b \<Rightarrow> 'a) \<Rightarrow> 'c"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   149
  assumes mono: "\<And>x. mono_body (\<lambda>f. U (F (C f)) x)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   150
  assumes eq: "f \<equiv> C (fixp_fun (\<lambda>f. U (F (C f))))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   151
  assumes inverse: "\<And>f. C (U f) = f"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   152
  shows "f = F f"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   153
proof -
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   154
  have "f = C (fixp_fun (\<lambda>f. U (F (C f))))" by (simp add: eq)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   155
  also have "... = C (U (F (C (fixp_fun (\<lambda>f. U (F (C f)))))))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   156
    by (subst mono_body_fixp[of "%f. U (F (C f))", OF mono]) (rule refl)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   157
  also have "... = F (C (fixp_fun (\<lambda>f. U (F (C f)))))" by (rule inverse)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   158
  also have "... = F f" by (simp add: eq)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   159
  finally show "f = F f" .
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   160
qed
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   161
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   162
text {* Rules for @{term mono_body}: *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   163
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   164
lemma const_mono[partial_function_mono]: "monotone ord leq (\<lambda>f. c)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   165
by (rule monotoneI) (rule leq_refl)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   166
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   167
declaration {* Partial_Function.init @{term fixp_fun}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   168
  @{term mono_body} @{thm fixp_rule_uc} *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   169
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   170
end
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   171
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   172
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   173
subsection {* Flat interpretation: tailrec and option *}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   174
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   175
definition 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   176
  "flat_ord b x y \<longleftrightarrow> x = b \<or> x = y"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   177
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   178
definition 
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   179
  "flat_lub b A = (if A \<subseteq> {b} then b else (THE x. x \<in> A - {b}))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   180
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   181
lemma flat_interpretation:
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   182
  "partial_function_definitions (flat_ord b) (flat_lub b)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   183
proof
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   184
  fix A x assume 1: "chain (flat_ord b) A" "x \<in> A"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   185
  show "flat_ord b x (flat_lub b A)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   186
  proof cases
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   187
    assume "x = b"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   188
    thus ?thesis by (simp add: flat_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   189
  next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   190
    assume "x \<noteq> b"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   191
    with 1 have "A - {b} = {x}"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   192
      by (auto elim: chainE simp: flat_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   193
    then have "flat_lub b A = x"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   194
      by (auto simp: flat_lub_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   195
    thus ?thesis by (auto simp: flat_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   196
  qed
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   197
next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   198
  fix A z assume A: "chain (flat_ord b) A"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   199
    and z: "\<And>x. x \<in> A \<Longrightarrow> flat_ord b x z"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   200
  show "flat_ord b (flat_lub b A) z"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   201
  proof cases
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   202
    assume "A \<subseteq> {b}"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   203
    thus ?thesis
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   204
      by (auto simp: flat_lub_def flat_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   205
  next
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   206
    assume nb: "\<not> A \<subseteq> {b}"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   207
    then obtain y where y: "y \<in> A" "y \<noteq> b" by auto
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   208
    with A have "A - {b} = {y}"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   209
      by (auto elim: chainE simp: flat_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   210
    with nb have "flat_lub b A = y"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   211
      by (auto simp: flat_lub_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   212
    with z y show ?thesis by auto    
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   213
  qed
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   214
qed (auto simp: flat_ord_def)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   215
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   216
interpretation tailrec!:
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   217
  partial_function_definitions "flat_ord undefined" "flat_lub undefined"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   218
by (rule flat_interpretation)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   219
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   220
interpretation option!:
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   221
  partial_function_definitions "flat_ord None" "flat_lub None"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   222
by (rule flat_interpretation)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   223
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   224
abbreviation "option_ord \<equiv> flat_ord None"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   225
abbreviation "mono_option \<equiv> monotone (fun_ord option_ord) option_ord"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   226
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   227
lemma bind_mono[partial_function_mono]:
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   228
assumes mf: "mono_option B" and mg: "\<And>y. mono_option (\<lambda>f. C y f)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   229
shows "mono_option (\<lambda>f. Option.bind (B f) (\<lambda>y. C y f))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   230
proof (rule monotoneI)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   231
  fix f g :: "'a \<Rightarrow> 'b option" assume fg: "fun_ord option_ord f g"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   232
  with mf
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   233
  have "option_ord (B f) (B g)" by (rule monotoneD[of _ _ _ f g])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   234
  then have "option_ord (Option.bind (B f) (\<lambda>y. C y f)) (Option.bind (B g) (\<lambda>y. C y f))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   235
    unfolding flat_ord_def by auto    
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   236
  also from mg
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   237
  have "\<And>y'. option_ord (C y' f) (C y' g)"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   238
    by (rule monotoneD) (rule fg)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   239
  then have "option_ord (Option.bind (B g) (\<lambda>y'. C y' f)) (Option.bind (B g) (\<lambda>y'. C y' g))"
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   240
    unfolding flat_ord_def by (cases "B g") auto
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   241
  finally (option.leq_trans)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   242
  show "option_ord (Option.bind (B f) (\<lambda>y. C y f)) (Option.bind (B g) (\<lambda>y'. C y' g))" .
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   243
qed
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   244
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   245
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   246
end
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   247