src/HOL/Tools/Function/partial_function.ML
author haftmann
Tue, 26 Oct 2010 14:11:34 +0200
changeset 40186 fe4a58419d46
parent 40180 c3ef007115a0
child 41117 d6379876ec4c
permissions -rw-r--r--
partial_function is a declaration command
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/Tools/Function/partial_function.ML
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
Partial function definitions based on least fixed points in ccpos.
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     5
*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     6
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     7
signature PARTIAL_FUNCTION =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     8
sig
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
     9
  val setup: theory -> theory
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    10
  val init: term -> term -> thm -> declaration
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    11
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    12
  val add_partial_function: string -> (binding * typ option * mixfix) list ->
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    13
    Attrib.binding * term -> local_theory -> local_theory
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    14
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    15
  val add_partial_function_cmd: string -> (binding * string option * mixfix) list ->
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    16
    Attrib.binding * string -> local_theory -> local_theory
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    17
end;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    18
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    19
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    20
structure Partial_Function: PARTIAL_FUNCTION =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    21
struct
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    22
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    23
(*** Context Data ***)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    24
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    25
structure Modes = Generic_Data
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    26
(
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    27
  type T = ((term * term) * thm) Symtab.table;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    28
  val empty = Symtab.empty;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    29
  val extend = I;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    30
  fun merge (a, b) = Symtab.merge (K true) (a, b);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    31
)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    32
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    33
fun init fixp mono fixp_eq phi =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    34
  let
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    35
    val term = Morphism.term phi;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    36
    val data' = ((term fixp, term mono), Morphism.thm phi fixp_eq);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    37
    val mode = (* extract mode identifier from morphism prefix! *)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    38
      Binding.prefix_of (Morphism.binding phi (Binding.empty))
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    39
      |> map fst |> space_implode ".";
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    40
  in
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    41
    if mode = "" then I
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    42
    else Modes.map (Symtab.update (mode, data'))
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    43
  end
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    44
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    45
val known_modes = Symtab.keys o Modes.get o Context.Proof;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    46
val lookup_mode = Symtab.lookup o Modes.get o Context.Proof;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    47
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    48
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    49
structure Mono_Rules = Named_Thms
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    50
(
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    51
  val name = "partial_function_mono";
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    52
  val description = "monotonicity rules for partial function definitions";
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    53
);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    54
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    55
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    56
(*** Automated monotonicity proofs ***)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    57
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    58
fun strip_cases ctac = ctac #> Seq.map snd;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    59
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    60
(*rewrite conclusion with k-th assumtion*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    61
fun rewrite_with_asm_tac ctxt k =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    62
  Subgoal.FOCUS (fn {context=ctxt', prems, ...} =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    63
    Local_Defs.unfold_tac ctxt' [nth prems k]) ctxt;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    64
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    65
fun dest_case thy t =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    66
  case strip_comb t of
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    67
    (Const (case_comb, _), args) =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    68
      (case Datatype.info_of_case thy case_comb of
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    69
         NONE => NONE
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    70
       | SOME {case_rewrites, ...} =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    71
           let
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    72
             val lhs = prop_of (hd case_rewrites)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    73
               |> HOLogic.dest_Trueprop |> HOLogic.dest_eq |> fst;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    74
             val arity = length (snd (strip_comb lhs));
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    75
             val conv = funpow (length args - arity) Conv.fun_conv
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    76
               (Conv.rewrs_conv (map mk_meta_eq case_rewrites));
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    77
           in
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    78
             SOME (nth args (arity - 1), conv)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    79
           end)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    80
  | _ => NONE;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    81
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    82
(*split on case expressions*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    83
val split_cases_tac = Subgoal.FOCUS_PARAMS (fn {context=ctxt, ...} =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    84
  SUBGOAL (fn (t, i) => case t of
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    85
    _ $ (_ $ Abs (_, _, body)) =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    86
      (case dest_case (ProofContext.theory_of ctxt) body of
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    87
         NONE => no_tac
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    88
       | SOME (arg, conv) =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    89
           let open Conv in
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    90
              if not (null (loose_bnos arg)) then no_tac
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    91
              else ((DETERM o strip_cases o Induct.cases_tac ctxt false [[SOME arg]] NONE [])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    92
                THEN_ALL_NEW (rewrite_with_asm_tac ctxt 0)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    93
                THEN_ALL_NEW etac @{thm thin_rl}
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    94
                THEN_ALL_NEW (CONVERSION
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    95
                  (params_conv ~1 (fn ctxt' =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    96
                    arg_conv (arg_conv (abs_conv (K conv) ctxt'))) ctxt))) i
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    97
           end)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    98
  | _ => no_tac) 1);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
    99
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   100
(*monotonicity proof: apply rules + split case expressions*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   101
fun mono_tac ctxt =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   102
  K (Local_Defs.unfold_tac ctxt [@{thm curry_def}])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   103
  THEN' (TRY o REPEAT_ALL_NEW
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   104
   (resolve_tac (Mono_Rules.get ctxt)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   105
     ORELSE' split_cases_tac ctxt));
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   106
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   107
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   108
(*** Auxiliary functions ***)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   109
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   110
(*positional instantiation with computed type substitution.
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   111
  internal version of  attribute "[of s t u]".*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   112
fun cterm_instantiate' cts thm =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   113
  let
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   114
    val thy = Thm.theory_of_thm thm;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   115
    val vs = rev (Term.add_vars (prop_of thm) [])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   116
      |> map (Thm.cterm_of thy o Var);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   117
  in
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   118
    cterm_instantiate (zip_options vs cts) thm
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   119
  end;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   120
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   121
(*Returns t $ u, but instantiates the type of t to make the
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   122
application type correct*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   123
fun apply_inst ctxt t u =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   124
  let
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   125
    val thy = ProofContext.theory_of ctxt;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   126
    val T = domain_type (fastype_of t);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   127
    val T' = fastype_of u;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   128
    val subst = Type.typ_match (Sign.tsig_of thy) (T, T') Vartab.empty
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   129
      handle Type.TYPE_MATCH => raise TYPE ("apply_inst", [T, T'], [t, u])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   130
  in
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   131
    map_types (Envir.norm_type subst) t $ u
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   132
  end;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   133
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   134
fun head_conv cv ct =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   135
  if can Thm.dest_comb ct then Conv.fun_conv (head_conv cv) ct else cv ct;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   136
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   137
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   138
(*** currying transformation ***)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   139
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   140
fun curry_const (A, B, C) =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   141
  Const (@{const_name Product_Type.curry},
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   142
    [HOLogic.mk_prodT (A, B) --> C, A, B] ---> C);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   143
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   144
fun mk_curry f =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   145
  case fastype_of f of
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   146
    Type ("fun", [Type (_, [S, T]), U]) =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   147
      curry_const (S, T, U) $ f
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   148
  | T => raise TYPE ("mk_curry", [T], [f]);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   149
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   150
(* iterated versions. Nonstandard left-nested tuples arise naturally
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   151
from "split o split o split"*)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   152
fun curry_n arity = funpow (arity - 1) mk_curry;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   153
fun uncurry_n arity = funpow (arity - 1) HOLogic.mk_split;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   154
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   155
val curry_uncurry_ss = HOL_basic_ss addsimps
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   156
  [@{thm Product_Type.curry_split}, @{thm Product_Type.split_curry}]
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   157
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   158
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   159
(*** partial_function definition ***)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   160
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   161
fun gen_add_partial_function prep mode fixes_raw eqn_raw lthy =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   162
  let
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   163
    val ((fixp, mono), fixp_eq) = the (lookup_mode lthy mode)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   164
      handle Option.Option => error (cat_lines ["Unknown mode " ^ quote mode ^ ".",
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   165
        "Known modes are " ^ commas_quote (known_modes lthy) ^ "."]);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   166
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   167
    val ((fixes, [(eq_abinding, eqn)]), _) = prep fixes_raw [eqn_raw] lthy;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   168
    val (_, _, plain_eqn) = Function_Lib.dest_all_all_ctx lthy eqn;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   169
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   170
    val ((f_binding, fT), mixfix) = the_single fixes;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   171
    val fname = Binding.name_of f_binding;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   172
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   173
    val cert = cterm_of (ProofContext.theory_of lthy);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   174
    val (lhs, rhs) = HOLogic.dest_eq (HOLogic.dest_Trueprop plain_eqn);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   175
    val (head, args) = strip_comb lhs;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   176
    val F = fold_rev lambda (head :: args) rhs;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   177
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   178
    val arity = length args;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   179
    val (aTs, bTs) = chop arity (binder_types fT);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   180
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   181
    val tupleT = foldl1 HOLogic.mk_prodT aTs;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   182
    val fT_uc = tupleT :: bTs ---> body_type fT;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   183
    val f_uc = Var ((fname, 0), fT_uc);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   184
    val x_uc = Var (("x", 0), tupleT);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   185
    val uncurry = lambda head (uncurry_n arity head);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   186
    val curry = lambda f_uc (curry_n arity f_uc);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   187
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   188
    val F_uc =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   189
      lambda f_uc (uncurry_n arity (F $ curry_n arity f_uc));
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   190
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   191
    val mono_goal = apply_inst lthy mono (lambda f_uc (F_uc $ f_uc $ x_uc))
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   192
      |> HOLogic.mk_Trueprop
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   193
      |> Logic.all x_uc;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   194
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   195
    val mono_thm = Goal.prove_internal [] (cert mono_goal)
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   196
        (K (mono_tac lthy 1))
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   197
      |> Thm.forall_elim (cert x_uc);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   198
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   199
    val f_def_rhs = curry_n arity (apply_inst lthy fixp F_uc);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   200
    val f_def_binding = Binding.conceal (Binding.name (Thm.def_name fname));
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   201
    val ((f, (_, f_def)), lthy') = Local_Theory.define
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   202
      ((f_binding, mixfix), ((f_def_binding, []), f_def_rhs)) lthy;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   203
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   204
    val eqn = HOLogic.mk_eq (list_comb (f, args),
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   205
        Term.betapplys (F, f :: args))
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   206
      |> HOLogic.mk_Trueprop;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   207
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   208
    val unfold =
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   209
      (cterm_instantiate' (map (SOME o cert) [uncurry, F, curry]) fixp_eq
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   210
        OF [mono_thm, f_def])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   211
      |> Tactic.rule_by_tactic lthy (Simplifier.simp_tac curry_uncurry_ss 1);
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   212
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   213
    val rec_rule = let open Conv in
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   214
      Goal.prove lthy' (map (fst o dest_Free) args) [] eqn (fn _ =>
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   215
        CONVERSION ((arg_conv o arg1_conv o head_conv o rewr_conv) (mk_meta_eq unfold)) 1
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   216
        THEN rtac @{thm refl} 1) end;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   217
  in
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   218
    lthy'
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   219
    |> Local_Theory.note (eq_abinding, [rec_rule])
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   220
    |-> (fn (_, rec') =>
40180
c3ef007115a0 fixed confusion introduced in 008dc2d2c395
krauss
parents: 40172
diff changeset
   221
      Spec_Rules.add Spec_Rules.Equational ([f], rec')
c3ef007115a0 fixed confusion introduced in 008dc2d2c395
krauss
parents: 40172
diff changeset
   222
      #> Local_Theory.note ((Binding.qualify true fname (Binding.name "simps"), []), rec') #> snd)
40107
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   223
  end;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   224
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   225
val add_partial_function = gen_add_partial_function Specification.check_spec;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   226
val add_partial_function_cmd = gen_add_partial_function Specification.read_spec;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   227
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   228
val mode = Parse.$$$ "(" |-- Parse.xname --| Parse.$$$ ")";
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   229
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   230
val _ = Outer_Syntax.local_theory
40186
fe4a58419d46 partial_function is a declaration command
haftmann
parents: 40180
diff changeset
   231
  "partial_function" "define partial function" Keyword.thy_decl
40107
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   232
  ((mode -- (Parse.fixes -- (Parse.where_ |-- Parse_Spec.spec)))
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   233
     >> (fn (mode, (fixes, spec)) => add_partial_function_cmd mode fixes spec));
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   234
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   235
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   236
val setup = Mono_Rules.setup;
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   237
374f3ef9f940 first version of partial_function package
krauss
parents:
diff changeset
   238
end