src/HOL/Tools/list_to_set_comprehension.ML
author bulwahn
Mon, 18 Jul 2011 10:34:21 +0200
changeset 43888 ee4be704c2a4
parent 42361 23f352990944
child 44241 7943b69f0188
permissions -rw-r--r--
adapting an experimental setup to changes in quickcheck's infrastructure
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
41467
8fc17c5e11c0 tuned headers;
wenzelm
parents: 41463
diff changeset
     1
(*  Title:      HOL/Tools/list_to_set_comprehension.ML
8fc17c5e11c0 tuned headers;
wenzelm
parents: 41463
diff changeset
     2
    Author:     Lukas Bulwahn, TU Muenchen
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
     3
41467
8fc17c5e11c0 tuned headers;
wenzelm
parents: 41463
diff changeset
     4
Simproc for rewriting list comprehensions applied to List.set to set
8fc17c5e11c0 tuned headers;
wenzelm
parents: 41463
diff changeset
     5
comprehension.
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
     6
*)
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
     7
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
     8
signature LIST_TO_SET_COMPREHENSION =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
     9
sig
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    10
  val simproc : simpset -> cterm -> thm option
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    11
end
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    12
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    13
structure List_to_Set_Comprehension : LIST_TO_SET_COMPREHENSION =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    14
struct
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    15
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    16
(* conversion *)
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    17
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    18
fun all_exists_conv cv ctxt ct =
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    19
  (case Thm.term_of ct of
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    20
    Const (@{const_name HOL.Ex}, _) $ Abs _ =>
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    21
      Conv.arg_conv (Conv.abs_conv (all_exists_conv cv o #2) ctxt) ct
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    22
  | _ => cv ctxt ct)
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    23
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    24
fun all_but_last_exists_conv cv ctxt ct =
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    25
  (case Thm.term_of ct of
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    26
    Const (@{const_name HOL.Ex}, _) $ Abs (_, _, Const (@{const_name HOL.Ex}, _) $ _) =>
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    27
      Conv.arg_conv (Conv.abs_conv (all_but_last_exists_conv cv o #2) ctxt) ct
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    28
  | _ => cv ctxt ct)
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    29
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    30
fun Collect_conv cv ctxt ct =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    31
  (case Thm.term_of ct of
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    32
    Const (@{const_name Set.Collect}, _) $ Abs _ => Conv.arg_conv (Conv.abs_conv cv ctxt) ct
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    33
  | _ => raise CTERM ("Collect_conv", [ct]))
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    34
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    35
fun Trueprop_conv cv ct =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    36
  (case Thm.term_of ct of
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    37
    Const (@{const_name Trueprop}, _) $ _ => Conv.arg_conv cv ct
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    38
  | _ => raise CTERM ("Trueprop_conv", [ct]))
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    39
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    40
fun eq_conv cv1 cv2 ct =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    41
  (case Thm.term_of ct of
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    42
    Const (@{const_name HOL.eq}, _) $ _ $ _ => Conv.combination_conv (Conv.arg_conv cv1) cv2 ct
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    43
  | _ => raise CTERM ("eq_conv", [ct]))
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    44
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    45
fun conj_conv cv1 cv2 ct =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    46
  (case Thm.term_of ct of
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    47
    Const (@{const_name HOL.conj}, _) $ _ $ _ => Conv.combination_conv (Conv.arg_conv cv1) cv2 ct
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    48
  | _ => raise CTERM ("conj_conv", [ct]))
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    49
41618
79dae6b7857d fixing list_to_set_comprehension simproc
bulwahn
parents: 41508
diff changeset
    50
fun rewr_conv' th = Conv.rewr_conv (mk_meta_eq th)
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    51
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    52
fun conjunct_assoc_conv ct =
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    53
  Conv.try_conv
42169
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
    54
    (rewr_conv' @{thm conj_assoc} then_conv conj_conv Conv.all_conv conjunct_assoc_conv) ct
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    55
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    56
fun right_hand_set_comprehension_conv conv ctxt =
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    57
  Trueprop_conv (eq_conv Conv.all_conv
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    58
    (Collect_conv (all_exists_conv conv o #2) ctxt))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    59
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    60
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    61
(* term abstraction of list comprehension patterns *)
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    62
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    63
datatype termlets = If | Case of (typ * int)
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    64
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    65
fun simproc ss redex =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    66
  let
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    67
    val ctxt = Simplifier.the_context ss
42361
23f352990944 modernized structure Proof_Context;
wenzelm
parents: 42169
diff changeset
    68
    val thy = Proof_Context.theory_of ctxt
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    69
    val set_Nil_I = @{thm trans} OF [@{thm set.simps(1)}, @{thm empty_def}]
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    70
    val set_singleton = @{lemma "set [a] = {x. x = a}" by simp}
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    71
    val inst_Collect_mem_eq = @{lemma "set A = {x. x : set A}" by simp}
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    72
    val del_refl_eq = @{lemma "(t = t & P) == P" by simp}
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    73
    fun mk_set T = Const (@{const_name List.set}, HOLogic.listT T --> HOLogic.mk_setT T)
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    74
    fun dest_set (Const (@{const_name List.set}, _) $ xs) = xs
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    75
    fun dest_singleton_list (Const (@{const_name List.Cons}, _)
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    76
          $ t $ (Const (@{const_name List.Nil}, _))) = t
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    77
      | dest_singleton_list t = raise TERM ("dest_singleton_list", [t])
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    78
    (* We check that one case returns a singleton list and all other cases
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    79
       return [], and return the index of the one singleton list case *)
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    80
    fun possible_index_of_singleton_case cases =
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    81
      let
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    82
        fun check (i, case_t) s =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    83
          (case strip_abs_body case_t of
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    84
            (Const (@{const_name List.Nil}, _)) => s
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    85
          | t => (case s of NONE => SOME i | SOME s => NONE))
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    86
      in
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    87
        fold_index check cases NONE
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    88
      end
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    89
    (* returns (case_expr type index chosen_case) option  *)
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    90
    fun dest_case case_term =
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    91
      let
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    92
        val (case_const, args) = strip_comb case_term
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
    93
      in
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    94
        (case try dest_Const case_const of
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    95
          SOME (c, T) =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    96
            (case Datatype_Data.info_of_case thy c of
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    97
              SOME _ =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    98
                (case possible_index_of_singleton_case (fst (split_last args)) of
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
    99
                  SOME i =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   100
                    let
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   101
                      val (Ts, _) = strip_type T
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   102
                      val T' = List.last Ts
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   103
                    in SOME (List.last args, T', i, nth args i) end
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   104
                | NONE => NONE)
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   105
            | NONE => NONE)
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   106
        | NONE => NONE)
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   107
      end
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   108
    (* returns condition continuing term option *)
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   109
    fun dest_if (Const (@{const_name If}, _) $ cond $ then_t $ Const (@{const_name Nil}, _)) =
42169
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   110
          SOME (cond, then_t)
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   111
      | dest_if _ = NONE
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   112
    fun tac _ [] = rtac set_singleton 1 ORELSE rtac inst_Collect_mem_eq 1
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   113
      | tac ctxt (If :: cont) =
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   114
          Splitter.split_tac [@{thm split_if}] 1
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   115
          THEN rtac @{thm conjI} 1
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   116
          THEN rtac @{thm impI} 1
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   117
          THEN Subgoal.FOCUS (fn {prems, context, ...} =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   118
            CONVERSION (right_hand_set_comprehension_conv (K
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   119
              (conj_conv (Conv.rewr_conv (List.last prems RS @{thm Eq_TrueI})) Conv.all_conv
42169
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   120
               then_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   121
               rewr_conv' @{lemma "(True & P) = P" by simp})) context) 1) ctxt 1
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   122
          THEN tac ctxt cont
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   123
          THEN rtac @{thm impI} 1
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   124
          THEN Subgoal.FOCUS (fn {prems, context, ...} =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   125
              CONVERSION (right_hand_set_comprehension_conv (K
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   126
                (conj_conv (Conv.rewr_conv (List.last prems RS @{thm Eq_FalseI})) Conv.all_conv
42169
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   127
                 then_conv rewr_conv' @{lemma "(False & P) = False" by simp})) context) 1) ctxt 1
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   128
          THEN rtac set_Nil_I 1
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   129
      | tac ctxt (Case (T, i) :: cont) =
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   130
          let
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   131
            val info = Datatype.the_info thy (fst (dest_Type T))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   132
          in
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   133
            (* do case distinction *)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   134
            Splitter.split_tac [#split info] 1
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   135
            THEN EVERY (map_index (fn (i', case_rewrite) =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   136
              (if i' < length (#case_rewrites info) - 1 then rtac @{thm conjI} 1 else all_tac)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   137
              THEN REPEAT_DETERM (rtac @{thm allI} 1)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   138
              THEN rtac @{thm impI} 1
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   139
              THEN (if i' = i then
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   140
                (* continue recursively *)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   141
                Subgoal.FOCUS (fn {prems, context, ...} =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   142
                  CONVERSION (Thm.eta_conversion then_conv right_hand_set_comprehension_conv (K
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   143
                      ((conj_conv
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   144
                        (eq_conv Conv.all_conv (rewr_conv' (List.last prems)) then_conv
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   145
                          (Conv.try_conv (Conv.rewrs_conv (map mk_meta_eq (#inject info)))))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   146
                        Conv.all_conv)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   147
                        then_conv (Conv.try_conv (Conv.rewr_conv del_refl_eq))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   148
                        then_conv conjunct_assoc_conv)) context
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   149
                    then_conv (Trueprop_conv (eq_conv Conv.all_conv (Collect_conv (fn (_, ctxt) =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   150
                      Conv.repeat_conv
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   151
                        (all_but_last_exists_conv
42169
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   152
                          (K (rewr_conv'
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   153
                            @{lemma "(EX x. x = t & P x) = P t" by simp})) ctxt)) context)))) 1) ctxt 1
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   154
                THEN tac ctxt cont
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   155
              else
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   156
                Subgoal.FOCUS (fn {prems, context, ...} =>
42169
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   157
                  CONVERSION
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   158
                    (right_hand_set_comprehension_conv (K
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   159
                      (conj_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   160
                        ((eq_conv Conv.all_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   161
                          (rewr_conv' (List.last prems))) then_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   162
                          (Conv.rewrs_conv (map (fn th => th RS @{thm Eq_FalseI}) (#distinct info))))
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   163
                        Conv.all_conv then_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   164
                        (rewr_conv' @{lemma "(False & P) = False" by simp}))) context then_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   165
                      Trueprop_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   166
                        (eq_conv Conv.all_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   167
                          (Collect_conv (fn (_, ctxt) =>
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   168
                            Conv.repeat_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   169
                              (Conv.bottom_conv
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   170
                                (K (rewr_conv'
a7570c66d746 inline lemmas instead of accidental physical addressing -- explicit is better than implicit;
wenzelm
parents: 42168
diff changeset
   171
                                  @{lemma "(EX x. P) = P" by simp})) ctxt)) context))) 1) ctxt 1
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   172
                THEN rtac set_Nil_I 1)) (#case_rewrites info))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   173
          end
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   174
    fun make_inner_eqs bound_vs Tis eqs t =
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   175
      (case dest_case t of
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   176
        SOME (x, T, i, cont) =>
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   177
          let
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   178
            val (vs, body) = strip_abs (Pattern.eta_long (map snd bound_vs) cont)
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   179
            val x' = incr_boundvars (length vs) x
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   180
            val eqs' = map (incr_boundvars (length vs)) eqs
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   181
            val (constr_name, _) = nth (the (Datatype_Data.get_constrs thy (fst (dest_Type T)))) i
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   182
            val constr_t =
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   183
              list_comb
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   184
                (Const (constr_name, map snd vs ---> T), map Bound (((length vs) - 1) downto 0))
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   185
            val constr_eq = Const (@{const_name HOL.eq}, T --> T --> @{typ bool}) $ constr_t $ x'
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   186
          in
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   187
            make_inner_eqs (rev vs @ bound_vs) (Case (T, i) :: Tis) (constr_eq :: eqs') body
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   188
          end
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   189
      | NONE =>
42168
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   190
          (case dest_if t of
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   191
            SOME (condition, cont) => make_inner_eqs bound_vs (If :: Tis) (condition :: eqs) cont
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   192
          | NONE =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   193
            if eqs = [] then NONE (* no rewriting, nothing to be done *)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   194
            else
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   195
              let
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   196
                val Type (@{type_name List.list}, [rT]) = fastype_of1 (map snd bound_vs, t)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   197
                val pat_eq =
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   198
                  (case try dest_singleton_list t of
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   199
                    SOME t' =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   200
                      Const (@{const_name HOL.eq}, rT --> rT --> @{typ bool}) $
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   201
                        Bound (length bound_vs) $ t'
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   202
                  | NONE =>
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   203
                      Const (@{const_name Set.member}, rT --> HOLogic.mk_setT rT --> @{typ bool}) $
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   204
                        Bound (length bound_vs) $ (mk_set rT $ t))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   205
                val reverse_bounds = curry subst_bounds
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   206
                  ((map Bound ((length bound_vs - 1) downto 0)) @ [Bound (length bound_vs)])
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   207
                val eqs' = map reverse_bounds eqs
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   208
                val pat_eq' = reverse_bounds pat_eq
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   209
                val inner_t =
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   210
                  fold (fn (v, T) => fn t => HOLogic.exists_const T $ absdummy (T, t))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   211
                    (rev bound_vs) (fold (curry HOLogic.mk_conj) eqs' pat_eq')
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   212
                val lhs = term_of redex
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   213
                val rhs = HOLogic.mk_Collect ("x", rT, inner_t)
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   214
                val rewrite_rule_t = HOLogic.mk_Trueprop (HOLogic.mk_eq (lhs, rhs))
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   215
              in
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   216
                SOME
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   217
                  ((Goal.prove ctxt [] [] rewrite_rule_t
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   218
                    (fn {context, ...} => tac context (rev Tis))) RS @{thm eq_reflection})
3164e7263b53 tuned white space and indentation;
wenzelm
parents: 41618
diff changeset
   219
              end))
41463
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   220
  in
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   221
    make_inner_eqs [] [] [] (dest_set (term_of redex))
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   222
  end
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   223
edbf0a86fb1c adding simproc to rewrite list comprehensions to set comprehensions; adopting proofs
bulwahn
parents:
diff changeset
   224
end