1 (* Title: Pure/subgoal.ML
4 Tactical operations with explicit subgoal focus, based on canonical
5 proof decomposition. The "visible" part of the text within the
6 context is fixed, the remaining goal may be schematic.
11 type focus = {context: Proof.context, params: (string * cterm) list, prems: thm list,
12 asms: cterm list, concl: cterm, schematics: (ctyp * ctyp) list * (cterm * cterm) list}
13 val focus_params: Proof.context -> int -> thm -> focus * thm
14 val focus_prems: Proof.context -> int -> thm -> focus * thm
15 val focus: Proof.context -> int -> thm -> focus * thm
16 val retrofit: Proof.context -> Proof.context -> (string * cterm) list -> cterm list ->
17 int -> thm -> thm -> thm Seq.seq
18 val FOCUS_PARAMS: (focus -> tactic) -> Proof.context -> int -> tactic
19 val FOCUS_PREMS: (focus -> tactic) -> Proof.context -> int -> tactic
20 val FOCUS: (focus -> tactic) -> Proof.context -> int -> tactic
21 val SUBPROOF: (focus -> tactic) -> Proof.context -> int -> tactic
24 structure Subgoal: SUBGOAL =
29 type focus = {context: Proof.context, params: (string * cterm) list, prems: thm list,
30 asms: cterm list, concl: cterm, schematics: (ctyp * ctyp) list * (cterm * cterm) list};
32 fun gen_focus (do_prems, do_concl) ctxt i raw_st =
34 val st = Simplifier.norm_hhf_protect raw_st;
35 val ((schematic_types, [st']), ctxt1) = Variable.importT [st] ctxt;
36 val ((params, goal), ctxt2) = Variable.focus (Thm.cprem_of st' i) ctxt1;
39 if do_prems then (Drule.strip_imp_prems goal, Drule.strip_imp_concl goal)
41 val text = asms @ (if do_concl then [concl] else []);
43 val ((_, schematic_terms), ctxt3) =
44 Variable.import_inst true (map Thm.term_of text) ctxt2
45 |>> Thm.certify_inst (Thm.theory_of_thm raw_st);
47 val schematics = (schematic_types, schematic_terms);
48 val asms' = map (Thm.instantiate_cterm schematics) asms;
49 val concl' = Thm.instantiate_cterm schematics concl;
50 val (prems, context) = Assumption.add_assumes asms' ctxt3;
52 ({context = context, params = params, prems = prems,
53 asms = asms', concl = concl', schematics = schematics}, Goal.init concl')
56 val focus_params = gen_focus (false, false);
57 val focus_prems = gen_focus (true, false);
58 val focus = gen_focus (true, true);
61 (* lift and retrofit *)
68 fun lift_import idx params th ctxt =
70 val cert = Thm.cterm_of (ProofContext.theory_of ctxt);
71 val ((_, [th']), ctxt') = Variable.importT [th] ctxt;
73 val Ts = map (#T o Thm.rep_cterm) params;
74 val ts = map Thm.term_of params;
76 val prop = Thm.full_prop_of th';
77 val concl_vars = Term.add_vars (Logic.strip_imp_concl prop) [];
78 val vars = rev (Term.add_vars prop []);
79 val (ys, ctxt'') = Variable.variant_fixes (map (Name.clean o #1 o #1) vars) ctxt';
85 if member (op =) concl_vars v then (T, [])
88 in ((Var v, list_comb (u, args)), (u, Var ((x, i + idx), U))) end;
89 val (inst1, inst2) = split_list (map (pairself (pairself cert)) (map2 var_inst vars ys));
91 val th'' = Thm.instantiate ([], inst1) th';
92 in ((inst2, th''), ctxt'') end;
103 fun lift_subgoals params asms th =
105 fun lift ct = fold_rev Thm.all_name params (Drule.list_implies (asms, ct));
107 fold (Thm.elim_implies o Thm.assume) asms o
108 Drule.forall_elim_list (map #2 params) o Thm.assume;
109 val subgoals = map lift (Drule.strip_imp_prems (Thm.cprop_of th));
110 val th' = fold (Thm.elim_implies o unlift) subgoals th;
111 in (subgoals, th') end;
113 fun retrofit ctxt1 ctxt0 params asms i st1 st0 =
115 val idx = Thm.maxidx_of st0 + 1;
116 val ps = map #2 params;
117 val ((subgoal_inst, st2), ctxt2) = lift_import idx ps st1 ctxt1;
118 val (subgoals, st3) = lift_subgoals params asms st2;
121 |> Drule.implies_intr_list asms
122 |> Drule.forall_intr_list ps
123 |> Drule.implies_intr_list subgoals
124 |> fold_rev (Thm.forall_intr o #1) subgoal_inst
125 |> fold (Thm.forall_elim o #2) subgoal_inst
126 |> Thm.adjust_maxidx_thm idx
127 |> singleton (Variable.export ctxt2 ctxt0);
128 in Thm.bicompose false (false, result, Thm.nprems_of st1) i st0 end;
133 fun GEN_FOCUS flags tac ctxt i st =
134 if Thm.nprems_of st < i then Seq.empty
136 let val (args as {context = ctxt', params, asms, ...}, st') = gen_focus flags ctxt i st;
137 in Seq.lifts (retrofit ctxt' ctxt params asms i) (tac args st') st end;
139 val FOCUS_PARAMS = GEN_FOCUS (false, false);
140 val FOCUS_PREMS = GEN_FOCUS (true, false);
141 val FOCUS = GEN_FOCUS (true, true);
143 fun SUBPROOF tac ctxt = FOCUS (Seq.map (tap (Goal.check_finished ctxt)) oo tac) ctxt;
147 val SUBPROOF = Subgoal.SUBPROOF;