src/HOL/BNF/Tools/coinduction.ML
author Andreas Lochbihler
Thu, 17 Oct 2013 17:14:06 +0200
changeset 54366 13bfdbcfbbfb
parent 54026 82d9b2701a03
permissions -rw-r--r--
swap equations and premises in the coinductive step for better proof automation
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
54026
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     1
(*  Title:      HOL/BNF/Tools/coinduction.ML
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     2
    Author:     Johannes Hölzl, TU Muenchen
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     3
    Author:     Dmitriy Traytel, TU Muenchen
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     4
    Copyright   2013
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     5
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     6
Coinduction method that avoids some boilerplate compared to coinduct.
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     7
*)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     8
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
     9
signature COINDUCTION =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    10
sig
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    11
  val coinduction_tac: Proof.context -> term list -> thm option -> thm list -> cases_tactic
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    12
  val setup: theory -> theory
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    13
end;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    14
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    15
structure Coinduction : COINDUCTION =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    16
struct
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    17
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    18
open BNF_Util
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    19
open BNF_Tactics
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    20
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    21
fun filter_in_out _ [] = ([], [])
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    22
  | filter_in_out P (x :: xs) = (let
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    23
      val (ins, outs) = filter_in_out P xs;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    24
    in
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    25
      if P x then (x :: ins, outs) else (ins, x :: outs)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    26
    end);
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    27
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    28
fun ALLGOALS_SKIP skip tac st =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    29
  let fun doall n = if n = skip then all_tac else tac n THEN doall (n - 1)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    30
  in doall (nprems_of st) st  end;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    31
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    32
fun THEN_ALL_NEW_SKIP skip tac1 tac2 i st =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    33
  st |> (tac1 i THEN (fn st' =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    34
    Seq.INTERVAL tac2 (i + skip) (i + nprems_of st' - nprems_of st) st'));
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    35
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    36
fun DELETE_PREMS_AFTER skip tac i st =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    37
  let
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    38
    val n = nth (prems_of st) (i - 1) |> Logic.strip_assums_hyp |> length;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    39
  in
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    40
    (THEN_ALL_NEW_SKIP skip tac (REPEAT_DETERM_N n o etac thin_rl)) i st
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    41
  end;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    42
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    43
fun coinduction_tac ctxt raw_vars opt_raw_thm prems st =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    44
  let
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    45
    val lhs_of_eq = HOLogic.dest_Trueprop #> HOLogic.dest_eq #> fst;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    46
    fun find_coinduct t = 
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    47
      Induct.find_coinductP ctxt t @
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    48
      (try (Induct.find_coinductT ctxt o fastype_of o lhs_of_eq) t |> the_default [])
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    49
    val raw_thm = case opt_raw_thm
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    50
      of SOME raw_thm => raw_thm
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    51
       | NONE => st |> prems_of |> hd |> Logic.strip_assums_concl |> find_coinduct |> hd;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    52
    val skip = Integer.max 1 (Rule_Cases.get_consumes raw_thm) - 1
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    53
    val cases = Rule_Cases.get raw_thm |> fst
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    54
  in
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    55
    NO_CASES (HEADGOAL (
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    56
      Object_Logic.rulify_tac THEN'
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    57
      Method.insert_tac prems THEN'
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    58
      Object_Logic.atomize_prems_tac THEN'
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    59
      DELETE_PREMS_AFTER skip (Subgoal.FOCUS (fn {concl, context = ctxt, params, prems, ...} =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    60
        let
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    61
          val vars = raw_vars @ map (term_of o snd) params;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    62
          val names_ctxt = ctxt
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    63
            |> fold Variable.declare_names vars
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    64
            |> fold Variable.declare_thm (raw_thm :: prems);
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    65
          val thm_concl = Thm.cprop_of raw_thm |> strip_imp_concl;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    66
          val (rhoTs, rhots) = Thm.match (thm_concl, concl)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    67
            |>> map (pairself typ_of)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    68
            ||> map (pairself term_of);
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    69
          val xs = hd (Thm.prems_of raw_thm) |> HOLogic.dest_Trueprop |> strip_comb |> snd
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    70
            |> map (subst_atomic_types rhoTs);
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    71
          val raw_eqs = map (fn x => (x, AList.lookup op aconv rhots x |> the)) xs;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    72
          val ((names, ctxt), Ts) = map_split (apfst fst o dest_Var o fst) raw_eqs
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    73
            |>> (fn names => Variable.variant_fixes names names_ctxt) ;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    74
          val eqs =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    75
            map3 (fn name => fn T => fn (_, rhs) =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    76
              HOLogic.mk_eq (Free (name, T), rhs))
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    77
            names Ts raw_eqs;
54366
13bfdbcfbbfb swap equations and premises in the coinductive step for better proof automation
Andreas Lochbihler
parents: 54026
diff changeset
    78
          val phi = eqs @ map (HOLogic.dest_Trueprop o prop_of) prems
54026
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    79
            |> try (Library.foldr1 HOLogic.mk_conj)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    80
            |> the_default @{term True}
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    81
            |> list_exists_free vars
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    82
            |> Term.map_abs_vars (Variable.revert_fixed ctxt)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    83
            |> fold_rev Term.absfree (names ~~ Ts)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    84
            |> certify ctxt;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    85
          val thm = cterm_instantiate_pos [SOME phi] raw_thm;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    86
          val e = length eqs;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    87
          val p = length prems;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    88
        in
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    89
          HEADGOAL (EVERY' [rtac thm,
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    90
            EVERY' (map (fn var =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    91
              rtac (cterm_instantiate_pos [NONE, SOME (certify ctxt var)] exI)) vars),
54366
13bfdbcfbbfb swap equations and premises in the coinductive step for better proof automation
Andreas Lochbihler
parents: 54026
diff changeset
    92
            if p = 0 then CONJ_WRAP' (K (rtac refl)) eqs
13bfdbcfbbfb swap equations and premises in the coinductive step for better proof automation
Andreas Lochbihler
parents: 54026
diff changeset
    93
            else REPEAT_DETERM_N e o (rtac conjI THEN' rtac refl) THEN' CONJ_WRAP' rtac prems,
54026
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    94
            K (ALLGOALS_SKIP skip
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    95
               (REPEAT_DETERM_N (length vars) o (etac exE THEN' rotate_tac ~1) THEN'
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    96
               DELETE_PREMS_AFTER 0 (Subgoal.FOCUS (fn {prems, params, context = ctxt, ...} =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    97
                 (case prems of
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    98
                   [] => all_tac
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
    99
                 | inv::case_prems =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   100
                     let
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   101
                       val (init, last) = funpow_yield (p + e - 1) HOLogic.conj_elim inv;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   102
                       val inv_thms = init @ [last];
54366
13bfdbcfbbfb swap equations and premises in the coinductive step for better proof automation
Andreas Lochbihler
parents: 54026
diff changeset
   103
                       val eqs = take e inv_thms;
54026
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   104
                       fun is_local_var t = 
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   105
                         member (fn (t, (_, t')) => t aconv (term_of t')) params t;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   106
                        val (eqs, assms') = filter_in_out (is_local_var o lhs_of_eq o prop_of) eqs;
54366
13bfdbcfbbfb swap equations and premises in the coinductive step for better proof automation
Andreas Lochbihler
parents: 54026
diff changeset
   107
                        val assms = assms' @ drop e inv_thms
54026
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   108
                      in
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   109
                        HEADGOAL (Method.insert_tac (assms @ case_prems)) THEN
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   110
                        unfold_thms_tac ctxt eqs
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   111
                      end)) ctxt)))])
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   112
        end) ctxt) THEN'
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   113
      K (prune_params_tac))) st
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   114
    |> Seq.maps (fn (_, st) =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   115
      CASES (Rule_Cases.make_common (Proof_Context.theory_of ctxt, prop_of st) cases) all_tac st)
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   116
  end;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   117
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   118
local
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   119
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   120
val ruleN = "rule"
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   121
val arbitraryN = "arbitrary"
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   122
fun single_rule [rule] = rule
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   123
  | single_rule _ = error "Single rule expected";
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   124
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   125
fun named_rule k arg get =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   126
  Scan.lift (Args.$$$ k -- Args.colon) |-- Scan.repeat arg :|--
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   127
    (fn names => Scan.peek (fn context => Scan.succeed (names |> map (fn name =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   128
      (case get (Context.proof_of context) name of SOME x => x
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   129
      | NONE => error ("No rule for " ^ k ^ " " ^ quote name))))));
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   130
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   131
fun rule get_type get_pred =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   132
  named_rule Induct.typeN (Args.type_name false) get_type ||
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   133
  named_rule Induct.predN (Args.const false) get_pred ||
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   134
  named_rule Induct.setN (Args.const false) get_pred ||
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   135
  Scan.lift (Args.$$$ ruleN -- Args.colon) |-- Attrib.thms;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   136
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   137
val coinduct_rule = rule Induct.lookup_coinductT Induct.lookup_coinductP >> single_rule;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   138
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   139
fun unless_more_args scan = Scan.unless (Scan.lift
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   140
  ((Args.$$$ arbitraryN || Args.$$$ Induct.typeN ||
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   141
    Args.$$$ Induct.predN || Args.$$$ Induct.setN || Args.$$$ ruleN) -- Args.colon)) scan;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   142
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   143
val arbitrary = Scan.optional (Scan.lift (Args.$$$ arbitraryN -- Args.colon) |--
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   144
  Scan.repeat1 (unless_more_args Args.term)) [];
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   145
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   146
in
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   147
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   148
val setup =
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   149
  Method.setup @{binding coinduction}
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   150
    (arbitrary -- Scan.option coinduct_rule >>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   151
      (fn (arbitrary, opt_rule) => fn ctxt =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   152
        RAW_METHOD_CASES (fn facts =>
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   153
          Seq.DETERM (coinduction_tac ctxt arbitrary opt_rule facts))))
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   154
    "coinduction on types or predicates/sets";
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   155
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   156
end;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   157
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   158
end;
82d9b2701a03 new coinduction method
traytel
parents:
diff changeset
   159