Nominal/nominal_dt_quot.ML
changeset 2628 16ffbc8442ca
parent 2626 d1bdc281be2b
child 2629 ffb5a181844b
--- a/Nominal/nominal_dt_quot.ML	Thu Dec 23 01:05:05 2010 +0000
+++ b/Nominal/nominal_dt_quot.ML	Sun Dec 26 16:35:16 2010 +0000
@@ -42,6 +42,7 @@
   val prove_strong_exhausts: Proof.context -> thm list -> bclause list list list -> thm list -> 
     thm list -> thm list -> thm list -> thm list -> thm list
 
+  val prove_strong_induct: Proof.context -> thm -> thm list -> bclause list list list -> thm
 end
 
 structure Nominal_Dt_Quot: NOMINAL_DT_QUOT =
@@ -608,5 +609,96 @@
     |> ProofContext.export lthy'' lthy
   end
 
+
+
+(** strong induction theorems **)
+
+fun add_c_prop c c_ty trm =
+  let
+    val (P, arg) = dest_comb trm
+    val (P_name, P_ty) = dest_Free P
+    val (ty_args, bool) = strip_type P_ty
+  in
+    Free (P_name, (c_ty :: ty_args) ---> bool) $ c $ arg
+  end
+
+fun add_qnt_c_prop c_name c_ty trm =
+  trm |> HOLogic.dest_Trueprop
+      |> incr_boundvars 1
+      |> add_c_prop (Bound 0) c_ty
+      |> HOLogic.mk_Trueprop 
+      |> mk_all (c_name, c_ty)
+
+fun prep_prem lthy c (c_name, c_ty) bclauses (params, prems, concl) =
+  let
+    val tys = map snd params
+    val binders = get_all_binders bclauses
+   
+    fun prep_binder (opt, i) = 
+      let
+        val t = Bound (length tys - i - 1)
+      in
+        case opt of
+          NONE => setify_ty lthy (nth tys i) t 
+        | SOME bn => to_set_ty (fastype_of1 (tys, bn $ t)) (bn $ t)   
+      end 
+
+    val prems' = prems
+      |> map (incr_boundvars 1) 
+      |> map (add_qnt_c_prop c_name c_ty)
+
+    val prems'' = 
+      case binders of
+        [] => prems'                       (* case: no binders *)
+      | _ => binders                       (* case: binders *) 
+          |> map prep_binder
+          |> fold_union_env tys
+          |> incr_boundvars 1
+          |> (fn t => mk_fresh_star_ty c_ty t (Bound 0))
+          |> (fn t => HOLogic.mk_Trueprop t :: prems')
+
+    val concl' = concl
+      |> HOLogic.dest_Trueprop
+      |> incr_boundvars 1 
+      |> add_c_prop (Bound 0) c_ty  
+      |> HOLogic.mk_Trueprop
+  in  
+    mk_full_horn (params @ [(c_name, c_ty)]) prems'' concl'
+  end
+
+fun prove_strong_induct lthy induct exhausts bclausesss =
+  let
+    val ((insts, [induct']), lthy') = Variable.import true [induct] lthy
+
+    val ([c_name, a], lthy'') = Variable.variant_fixes ["c", "'a"] lthy'
+    val c_ty = TFree (a, @{sort fs})
+    val c = Free (c_name, c_ty)
+
+    val (prems, concl) = induct'
+      |> prop_of
+      |> Logic.strip_horn 
+
+    val concls = concl
+      |> HOLogic.dest_Trueprop
+      |> HOLogic.dest_conj 
+      |> map (add_c_prop c c_ty) 
+      |> map HOLogic.mk_Trueprop
+
+    val prems' = prems
+      |> map strip_full_horn
+      |> map2 (prep_prem lthy'' c (c_name, c_ty)) (flat bclausesss)
+      |> map (Syntax.check_term lthy'') 
+
+    val _ = tracing ("induct:\n" ^ Syntax.string_of_term lthy'' (prop_of induct'))
+    val _ = tracing ("concls:\n" ^ commas (map (Syntax.string_of_term lthy'') concls))  
+    val _ = tracing ("prems':\n" ^ cat_lines (map (Syntax.string_of_term lthy'') prems'))
+
+    val thm = Goal.prove_multi lthy'' [] prems' concls
+      (fn _ => Skip_Proof.cheat_tac (ProofContext.theory_of lthy''))
+  in
+    @{thm TrueI}
+  end
+
+
 end (* structure *)